Differences between revisions 9 and 10
Revision 9 as of 2009-09-29 06:40:04
Size: 3746
Editor: pgas
Comment:
Revision 10 as of 2009-09-29 12:33:28
Size: 3865
Editor: GreyCat
Comment: clean up
Deletions are marked like this. Additions are marked like this.
Line 3: Line 3:
This page is an attempt to list some of the most common bashisms ie features
not defined by POSIX (ie don't work in dash).
It probably won't be exhaustive. Note also we talk about "bashism" because this wiki is largely bash centric
This page is an attempt to list some of the most common bashisms, i.e. features
not defined by POSIX (won't work in dash, or general `/bin/sh`).
It probably won't be exhaustive. Note also we talk about "bashism" because this wiki is largely bash-centric
Line 38: Line 38:
 * [[ is not defined by posix
 * == as an argument of test (aka [) is not defined by posix
 * < > to compare numbers as argument of test are not defined by posix, though dash implements them
 * -nt, -ot, -ef are not defined by posix
 * (( )) is not defined by posix
 * `[[` is not defined by posix
 * `==` as an argument of test (aka [) is not defined by posix
 * `< >` to compare numbers as argument of test are not defined by posix, though dash implements them
 * `-nt`, `-ot`, `-ef` are not defined by posix
 * `(( ))` is not defined by posix (but `$(( ))` substitution is OK)
Line 46: Line 46:
 * ++ -- are not defined by posix
 * "let" is not defined by posix
 * `++ --` are not defined by posix
 * `let` is not defined by posix
Line 50: Line 51:
 * >& and &> are not defined by posix
 * m>n- m<n- ie duplicating and closing a descriptor at the same time is not defined by posix
 * |& (bash4) is not defined by posix
 * `>&` and `&>` are not defined by posix
 * `m>&n- m<&n-` i.e. duplicating and closing a descriptor at the same time is not defined by posix
 * `|&` (bash4) is not defined by posix
Line 56: Line 57:
 * echo. posix doesn't define any options, use printf
 * printf "-v" is not defined by posix. also the %b and %q format are not defined by posix
 * read, the only option defined by posix is "-r"
 * shopt, and therefore all the options it provide, extglob, nullglob, dot glob etc ..are not defined by posix
 * `echo -n` or `-e` -- posix doesn't define any options, use `printf`
 * `printf -v` is not defined by posix. Also the `%b` and `%q` format are not defined by posix
 * `read` -- the only option defined by posix is `-r`
 * `shopt`, and therefore all the options it provides (extglob, nullglob, dotglob, etc.) are not defined by posix
Line 66: Line 67:
|| Get the status of all the commands in a pipeline|| PIPESTATUS || Simplest solution {{{mkfifo fifo; command2 <fifo&;command 1>fifo;echo $?}}} see NamedPipes || see [[ http://shell.cfajohnson.com/cus-faq-2.html#Q11 | this faq ]] and this script [[http://pipestatus.sourceforge.net/ | pipe status for posix shell]] || || Get the status of all the commands in a pipeline|| PIPESTATUS || Simplest solution {{{mkfifo fifo; command2 <fifo & command1 >fifo; echo $?}}} see NamedPipes || see [[ http://shell.cfajohnson.com/cus-faq-2.html#Q11 | this faq ]] and this script [[http://pipestatus.sourceforge.net/ | pipe status for posix shell]] ||
Line 69: Line 70:
== more == == More ==
Line 73: Line 74:
Note that bash in posix mode is only guaranteed to run a shell written according to the posix specification. It doesn't mean that it will fail if you use bashisms in your scripts. Note that bash in posix mode is only guaranteed to run a shell written according to the posix specification. It ''doesn't'' mean that it will fail if you use bashisms in your scripts.

How to make bash scripts work in dash

This page is an attempt to list some of the most common bashisms, i.e. features not defined by POSIX (won't work in dash, or general /bin/sh). It probably won't be exhaustive. Note also we talk about "bashism" because this wiki is largely bash-centric but a number of these extensions work in other shells like ksh or zsh.

Syntax

Works in bash

Change to for dash

Comment

defining functions

function f(){echo hello world}

f() { echo hello world ; }

"function" is not defined by posix, only "name ()" is

numeric C-like for loop

for ((i=0; i<3; i++)); do echo $i ; done

i=0 ; while test $i -lt 3 ; do echo $i ; i=$(($i+1)) ; done

the C-like for syntax (for (( ; ; ));do ..) is not defined by posix

case

;;&

;;& in bash4 is not defined by posix

case

;&

;& in bash4 is not defined by posix

select

select

select is not defined by posix

expand sequences

echo $'hello\tworld'

printf "hello\tworld"

$' ' is not defined by posix

extended glob

( +() @( ) etc...

not defined by posix

Expansions

  • Brace Expansion, eg {1..10} is not defined by posix
  • <( ) >( ) process substitution is not defined by posix

Parameter Expansions

List of expansions not defined by posix:

  • ${name:n:l}
  • ${name/ }
  • ${!name}

Arrays

  • arrays are not defined by posix.

Conditionals

  • [[ is not defined by posix

  • == as an argument of test (aka [) is not defined by posix

  • < > to compare numbers as argument of test are not defined by posix, though dash implements them

  • -nt, -ot, -ef are not defined by posix

  • (( )) is not defined by posix (but $(( )) substitution is OK)

Arithmetic

  • ++ -- are not defined by posix

  • let is not defined by posix

Redirections

  • >& and &> are not defined by posix

  • m>&n- m<&n- i.e. duplicating and closing a descriptor at the same time is not defined by posix

  • |& (bash4) is not defined by posix

Builtins

  • echo -n or -e -- posix doesn't define any options, use printf

  • printf -v is not defined by posix. Also the %b and %q format are not defined by posix

  • read -- the only option defined by posix is -r

  • shopt, and therefore all the options it provides (extglob, nullglob, dotglob, etc.) are not defined by posix

Special Variables

Works in bash

Change to for dash

Comment

keep track of the times

SECONDS

before=$(date +%s) ....seconds=$(( $(date +%s) - $before))

date +%s is not posix see this faq for more info

Generate a random number

RANDOM

random=$(awk 'BEGIN{srand(); printf "%d\n",(rand*256)}') gives a number between 0 and 256

Be sure to learn what srand() and rand() do, ie this method fails if you call awk several times rapidly. Instead generate all the numbers you need inside awk. Some systems also provides /dev/random and /dev/urandom

Get the status of all the commands in a pipeline

PIPESTATUS

Simplest solution mkfifo fifo; command2 <fifo & command1 >fifo; echo $? see NamedPipes

see this faq and this script pipe status for posix shell

More

  • The bash manual has a list of the difference between bash running in posix mode and a normal bash.

Note that bash in posix mode is only guaranteed to run a shell written according to the posix specification. It doesn't mean that it will fail if you use bashisms in your scripts.

Bashism (last edited 2022-10-20 23:13:29 by larryv)