Run a script on multiple shells?
Solution 1
It would probably be easier to have two separate versions of the script. Since it is short it may not be worth adding extra code to handle the differences between the two formats as jw013 suggested. On the other hand if you had a larger script it would probably be easier to have one and make the script execute different commands depending on where it is running from.
Solution 2
The easiest way by far is to use the same shell on both systems. Just because one shell is preinstalled (there's no such thing as a “default shell” for scripts, the shell is whatever the shebang line says) doesn't mean that you can't install others. You can install bash on AIX (from the toolbox, for example), or ksh93 for Linux (with your distribution's package manager: install the ksh
package).
If you pick ksh, beware that /usr/bin/ksh
on AIX is ksh88. There is no version of ksh88 for Linux, only ksh93, which is available as /usr/bin/ksh93
on AIX.
It is easier if the same shell is installed at the same location on both systems (a symbolic link is fine), so that you can use the same path in the shebang line. If having the same location is too difficult, you can use something like #!/usr/bin/env bash
, as long as you ensure that bash
is in the PATH
on both systems. This may be useful if you have to install bash or ksh in your home directory because you don't have root access.
If you really can't install the same shell on both machines — presumably because there's some silly compliance rule that only complicates people's life but is too entrenched to overturn — you have a few possibilities.
- Use
#!/bin/sh
as your shebang line and program in the intersection of what your systems offer. All modern unix systems provide a POSIX shell as/bin/sh
. On AIX,/bin/sh
is ksh88. On Red Hat,/bin/sh
is bash (which behaves slightly differently when invoked assh
). Beware that on some other operating systems (for example, on many Linux distributions),/bin/sh
is a smaller shell which may not have much more than POSIX features. -
Start your scripts with some boilerplate code that looks for a better shell and executes it.
#!/bin/sh if [ -n "$BASH" ]; then … bash compatibility code … elif type whence >/dev/null 2>/dev/null; then … ksh compatibility code … elif type ksh93 >/dev/null 2>/dev/null; then exec ksh93 "$0" "$@" elif type ksh >/dev/null 2>/dev/null; then exec ksh "$0" "$@" elif type mksh >/dev/null 2>/dev/null; then exec mksh "$0" "$@" elif type bash >/dev/null 2>/dev/null; then exec bash "$0" "$@" else echo 1>&2 "Cannot find ksh or bash, aborting" exit 125 fi
In either case, the intersection of ksh88 and bash provides some useful features beyond POSIX, but you may need a little compatibility code. In particular:
- Array assignment in ksh88 uses
set -A
. See assignation variable under different ksh environment - Local variables are declared by
typeset
. - Ksh88 doesn't have
${VAR/PATTERN/REPLACEMENT}
,$'…'
orFIGNORE
. It does have[[ … ]]
. - To enable
@(…)
and other ksh extended patterns in bash, runshopt -s extglob
.
Solution 3
I had a similar requirement - I needed to use ksh93 on AIX 5.3 to support associative arrays. So I set up the script to start a second instance using ksh93. I also added a safeguard to keep it from respawning itself over and over.
#!/bin/ksh
scr=$0
safe=$1
echo "Check OS to determine if ksh93 is needed..."
if ( `typeset -A testvar > /dev/null 2>&1` ); then
echo "Associative array functions supported."
else
echo "Associative array functions NOT supported. Starting second instance with ksh93..."
if [ "$safe" -eq 1 ]; then
echo "SECOND INSTANCE ALREADY STARTED!"
exit 1
else
echo "BEGIN SECOND INSTANCE USING KSH93"
/bin/ksh93 $scr 1
exit 0
fi
fi
Related videos on Youtube
crkatz
Updated on September 18, 2022Comments
-
crkatz almost 2 years
I have two servers one AIX with default shell ksh, the other RHEL with default shell bash.
I have a script that is mounted on both that will run similar commands but for either AIX or Linux. This script does not work on the bash servers, is there a way to make this script run on both bash and ksh, or would the best option be to create two different scripts?
#!/usr/bin/ksh export OS=`uname -s` echo "OS is "$OS"." case $OS in "AIX") #run AIX commands;; "Linux") #run Linux commands;; "*") echo "Exiting. The OS type is not found.";; esac echo "Done." exit 0
UPDATE
The commands I need to run are for user accounts on each server. An example of unlocking an account. AIX /usr/bin/chuser account_locked=false $USERNAME
Linux /usr/bin/passwd -u $USERNAME
Through further investigation I have found that the the shells location in AIX are located at /usr/bin/sh, while Redhat's are located at /bin/sh.
Can I define the shebang based on the results of "uname"?
-
jw013 almost 10 yearsIf
AIX commands
andLinux commands
are very similar, it may make sense to put them in the same script with some extra code to handle the differences. IfAIX commands
andLinux commands
are completely different, it may make more sense to have two different versions of the script. -
Scott - Слава Україні almost 10 yearsI’m curious whether the issue is (1) the difference between AIX commands and Linux commands, or (2) the difference between ksh scripting style and bash scripting style. In case 1, I would agree with @jw013 and encourage you to assess how much commonality there would be the two scripts, and how much specialized stuff, and decide based on that. But if it’s 2, I would suggest that you back out the ksh-isms and “dumb down” the script to follow POSIX shell conventions, so it works in both shells.
-