Shell coding and style guidelines

Ash, dash, bash and ksh compatible guide.

Code is expected to be presented in the following order:
1. Header
2. Global state and constants
3. Functions
4. Main entrypoint function, when applicable


General rules for file and code formatting.

File name

File name should have the “.sh” extension for POSIX-compliant shell scripts (dash compatible code) and “.bash” for Bash-only shell scripts.

Executable permissions

All shell script programs should have executable permissions set, except for libraries and modules, which are expected not to have a program entrypoint, main or code outside of functions.

Column size

Maximum line length is encouraged to be around 80 and 100 characters. No hard limit.


Four spaces.

Vim settings:

set expandtab
set smarttab
set tabstop=4
set shiftwidth=4
set softtabstop=4

Trailing spaces

Avoid adding unneeded spaces at the end of line.

Vim settings:

set list
set listchars=tab:▸\ ,trail:.

Conditional and loops

Do if [...]; then and ; do on the same line as the initial conditional. Multiple conditional statements should be distributed in multiple lines for clarity.


A given script is expected to have a presentation header and a set of variables followed by a set of functions.

A header is expected to always provide a shebang line and a presentation comment block.

Shebang line

Expected to be the first line of any given script. Always prefer the following for cross-platform support:
#!/usr/bin/env <shellname>
Use #!/usr/bin/env sh for POSIX-compliant scripts or #!/usr/bin/env bash for Bash-only scripts.

This section is optional and usually include a list of authors, copyright holders and general license terms.

Program comments

General program description is advised, with optional usage and instructions information.


Global variables are referred to the ones that are not declared inside a function.


Always start with an underscore and split words with subsequent underscores. Variables should be lower case, except for global variables and constants.


Always add comments to global variables and constants. Examples:

# Set to "1" to allow this and that
# and also that other thing

_DONT_GLOBAL_LIKE_THIS="1" # Don't do this for globals

    _all_good_here="abc" # good "scoped" variable comment

    # another good "scoped" variable comment


Use the readonly modifier for constant data.

Variable expansion

Always quote strings.

Environment variables

Never starts with an underscore. Always upper case, using underscores for splitting words. Prefer reading an environment variable and storing it into another variable before modifying.

Default values

Whenever relevant, have default values set. Example:
In that case, if _HOST_PORT is set to empty _HOST_PORT=, the value will remain null. For setting a default value for cases where a variable might be unset or empty, use the :- instead.

Arithmetic expressions

Use _count=$(( _count + 1 )).



Function names should always be lower case. Private functions should start with an underscore. Do not use “function” as it is an unneeded flourish. Example:




Function description should be straight to the point, with optional usage instructions. Implementation details should be closer to the implementation choices instead. Example, parameters, expected environment variable and return values must be described whenever applicable.

# This is a function description
# long description, multiple lines
# Example:
#   some data in here
# Parameters:
#   $1: user name
#   $2: a second parameter (optional)
# Expects:
#   PATH: an environment variable
#   TERM: expects that global term has been set to known value X or Y
#   GLOBAL_VAR: some global variable
# Returns:
#   0: on success
#   1: on file read error
#   2: no write permissions
#   128: unknown state


Use the shift keyword for accessing arguments. Using shift is less error-prone on function signature changes and mixes well when handling variable arguments i.e. $@. As an added bonus, calling shift when accessing a missing parameter causes an error because $# <= 0. That catches missing arguments and malformed function calls.

Error checking and reporting

Error reporting and general user messages should always go to stderr (i.e. >&2), accompanied by relevant information and available state. : can be used instead of true or as a style to set exit status code to 0.


Avoid usage of && || construct as if it was if then else. Always check for exit status code using $?, where applicable. Use shorthand operators for testing string and conditions in general. Example:

ìf [ -z "${my_str}" ]; then

Prefer using = "1" instead of -eq for comparing public user settings and flags, for instance “0” and “1”. In the event an user provides a non-integer as argument, for example _ENABLE_THIS_SETTING="NO", a comparison using = operator would not break the code.


Avoid polluting stdout. Prefer the default mode to be lean and add the ability to control a debug or verbose option in case the user wants more information. In that case, always send user-relevant information to stderr.


Command substitution

Use $(command) instead of `command` and always separate declaration from assignment. Example:

_output=$(mycommand arg1 arg2)


Prefer single line pipelines whenever possible. Consider splitting complex one-liners for the benefit of clarity. Consider setting set -o pipefail whenever possible.

printf versus echo

Prefer printf instead of echo for portability.

command versus which

Use command -v instead of which for checking for a given program availability.

. (a period) versus source

Use . instead of source for sourcing a file.


Use it with care.


Useful set -o options:
- errexit: exit when a pipeline or compound command fails
- nounset: catches unset variables and parameters as errors during parameter expansion


Use with care. Relevant traps for controlling exit: SIGINT SIGTERM EXIT ERR

Bash-only language constructs


Local defines a new isolated variable, even if the same variable name happens to be declared on outside scope. After function return, the other variable instance is always reestablished (compatible with Dash).
Do not declare multiple local variables in the same line.
Local keyword is not defined by POSIX and its implementation is shell-specific. Some shells will only handle the first variable as local, while exposing the rest as global. Example:

# In the following example, var_one will always be handled as local
# var_two might be handled as global by some shell implementations
local var_one='' var_two=''

# Do this instead
local var_one=''
local var_two=''



Static analysis

Always perform static analysis of code on a regular basis.






Analysis against POSIX rules:

shellcheck --shell=sh

Analysis against Bash-specific constructs:

shellcheck --shell=bash myprogram.bash



Testing is strongly encouraged and should always be provided in separate files inside a sub-directory (e.g. test/


Bashdb can be used as a gdb-like debugger.


Previous: Advanced configuration

Next: Glossary

Edit this page