Bashisms
As a reminder, this style guide is designed specifically for Bash scripting. When given a choice, ALWAYS prefer Bash built-ins or keywords over external commands or sh(1) syntax.
This section focuses on common Bashisms, which are shell features or commands unique to Bash. Using these Bash-specific constructs can improve your scripts' efficiency, portability (1), readability (2), and robustness.
- Efficiency and Portability: Bashisms provide built-in functionalities that are more efficient and portable than external commands or purely POSIX-compliant syntax. By leveraging these features, you can write scripts that execute faster and run on any system where Bash is available, without relying on external tools.
- Readability: While Bashisms generally improve readability, some features, such as the parameter expansion syntax, may be confusing for beginners. However, the robustness and efficiency they offer make them worth learning.
Conditional Tests
Conditional tests are essential in any programming language, allowing decision-making based on evaluating expressions. Bash provides several constructs for these tests, including [ ... ], [[ ... ]], and the test command.
Guidelines
- Use [[ ... ]]: ALWAYS use the[[ ... ]]construct when performing conditional tests.
- Avoid Using [ ... ]andtest: DO NOT use[ ... ]and thetestcommand for conditional tests.
Advantages of [[ ... ]]
- Regex Support: Enables direct regex matching within conditional expressions, eliminating the need for external tools like grep.
- String Comparison: Provides a consistent and reliable method for string comparison, especially when dealing with variables containing spaces or special characters.
- Compound Conditions: Allows multiple conditions to be combined within a single [[ ... ]]block, simplifying logic and improving readability.
- Safety: Prevents word splitting and globbing on variables, reducing the risk of unexpected behavior.
Examples
Using [ ... ]:
Potential Issues: Without proper quoting, [ ... ] may cause unexpected behaviors due to word splitting, treating the variable as multiple arguments.
Using [[ ... ]]:
Advantage: [[ ... ]] safely handles variables with spaces, whether or not they are quoted.
Using grep for regex matching:
Disadvantage: grep is an external command, which can introduce errors and performance issues (though negligible) due to differing implementations across different operating systems.
Using [[ ... ]] for regex matching:
Advantage: [[ ... ]] directly supports regex matching, eliminating the need for external commands and enhancing script portability.
Using [ ... ] for compound conditions:
Disadvantage: [ ... ] does not support compound conditions directly, requiring additional commands or constructs.
Using [[ ... ]] for compound conditions:
Advantage: [[ ... ]] supports compound conditions, simplifying logic and enhancing readability.
Sequence Iteration
Iterating over sequences is a common task in Bash, allowing you to process elements in a range or list. Bash offers built-in mechanisms for sequence iteration, such as brace expansion and C-style for loops.
Guidelines
- Bash Built-ins: ALWAYS use brace expansion ({start..end}) for fixed ranges and the C-styleforloop for variable limits when iterating over sequences.
- Avoid Using seq: DO NOT useseqfor sequence iteration.
Advantages of Built-in Iteration
- Simplicity: Built-in mechanisms like brace expansion and C-style forloops are native to Bash, providing a straightforward and efficient way to iterate over sequences.
- Reduced Dependencies: Utilizing built-in features minimizes external dependencies, enhancing script reliability and maintainability.
Examples
Using seq:
Using brace expansion:
Using seq:
Using C-style for loop:
Command Substitution
Command substitution allows you to capture the output of a command and use it as part of another command or assignment. Bash provides two syntaxes for command substitution: $(...) and backticks (`...`).
Guidelines
- Use $(...): ALWAYS use$(...)for command substitution instead of backticks.
- Avoid Using Backticks: DO NOT use backticks for command substitution.
Advantages of $(...)
- Improved Readability: $(...)is visually clearer, making scripts easier to read, particularly as commands grow in complexity.
- Easier Nesting: Facilitates nesting multiple commands without the syntactic awkwardness associated with backticks.
- Enhanced Safety: More robust and less prone to errors, especially in complex command substitutions.
Examples
Using backticks:
Using $(...)
Using backticks:
Disadvantage: Nesting commands with backticks can be challenging to read and are often prone to errors.
Using $(...):
Advantage: The $(...) syntax simplifies nested commands, improving readability and reducing the likelihood of errors.
Arithmetic Operations
Arithmetic operations in Bash enable mathematical calculations, comparisons, and other numeric operations. Bash supports these operations using arithmetic expansions $((...)), conditional arithmetic expressions ((...)), and the let command.
Guidelines
- Use $((...))and((...)): ALWAYS use$((...))for arithmetic expansions and((...))for conditional arithmetic expressions.
- Avoid Using let: DO NOT useletfor arithmetic operations.
Advantages of $((...)) and ((...))
- Clarity: Both ((...))and$((...))explicitly indicate arithmetic operations within scripts, making them easier to understand.
- Increased Safety: Unlike let, these methods exclusively evaluate arithmetic expressions and do not risk executing other commands by mistake.
Examples
Using let:
Using $((...)):
Parameter Expansion
Parameter expansion in Bash allows you to manipulate variables and perform string operations directly within the shell. It offers a wide range of options, including substring extraction, string replacement, and transformations.
Guidelines
- Use Parameter Expansion: Use parameter expansion for tasks such as string manipulation, default value assignment, and pattern matching.
- Limit Use of External Tools: Minimize the use of tools like sedandawkfor string manipulation unless advanced text processing is required and cannot be handled by Bash.- Rationale: While powerful, these tools are often unnecessary for tasks that can be performed efficiently using parameter expansion.
 
Advantages of Parameter Expansion
- Streamlined Scripting: Keeps string manipulations inline and shell-native, simplifying the script's logic.
- Enhanced Portability: Enhances script portability across different Unix-like systems by avoiding dependencies on external tools that may not be available or consistent in functionality.
Examples
Using sed
Using Parameter Expansion
Explanation: Parameter expansion offers a more direct and efficient way to extract substrings compared to using external tools like sed.
Using awk
Using Parameter Expansion
Explanation: Parameter expansion simplifies the removal of suffixes from filenames and is more efficient than using external commands like awk.
Using wc
Using Parameter Expansion
Explanation: Parameter expansion provides a more straightforward and efficient method for determining the length of a string compared to using external commands like wc.
Avoid Parsing ls
Parsing the output of ls in Bash scripts can result in errors and unpredictable behavior, especially when dealing with filenames that contain spaces or special characters. Instead, Unix-like systems provide safer and more reliable alternatives for handling files and directories, such as Bash globbing, find, and read.
Guidelines
- Bash Built-ins: ALWAYS use Bash globbing patterns (*) or commands likefindfor file and directory operations to ensure reliable handling.
- Avoid Using ls: DO NOT uselsin scripts where accurate filename interpretation is necessary, such as in loops or complex operations.
Risks of Parsing ls
- Word Splitting: Filenames with spaces or special characters may cause unintended word splitting when processed by ls, leading to errors.
- Command Misinterpretation: Filenames that begin with a hyphen (-) may be interpreted as command-line options, potentially causing unintended behavior.
- Potential Vulnerabilities: Improper handling of filenames can introduce bugs or security vulnerabilities, especially in more complex scripts.
Alternatives to Parsing ls
- Bash Globbing: Use Bash globbing patterns (like *) to list files and directories. This method is safer and more reliable for handling filenames with special characters.
- findCommand: Use the- findcommand for complex file operations, which provides extensive options for searching and handling files.
- readCommand: Use- readwith the appropriate settings to safely process filenames without relying on external commands like- ls.
- statCommand: Use- statto obtain detailed file information, such as size, permissions, and modification times.
- fileCommand: Use- fileto identify the type of a file based on its content, rather than its extension.
Examples
Using ls in a loop:
Using Bash Globing:
Explanation: Bash globbing uses patterns like * to match filenames directly in the specified directory, eliminating the need for external commands like ls. This approach ensures that filenames with spaces, newlines, or special characters are handled correctly by the shell. Using glob patterns (e.g., /path/to/dir/*), the shell expands them into a list of matching filenames, making this method more secure and reliable.
Using ls with wildcards:
Using find:
Explanation: The find command offers robust and flexible options for file searching and operations, making it more suitable for handling complex tasks than ls. It provides precise control over which files are selected and how they are processed.
Using ls with while loop:
Using find and read:
Explanation: Setting the IFS (Internal Field Separator) correctly with read ensures that filenames containing spaces or special characters are processed safely. This method avoids the risks associated with parsing the output of ls.
Using ls -l:
Using stat:
Explanation: stat provides more comprehensive metadata about a file than ls -l, such as size, permissions, and modification times.
Using ls with file extension check:
Using file:
Explanation: The file command examines the contents of a file to determine its type, offering a more accurate method than relying solely on file extensions.
Element Collections
Element collections are a fundamental feature in Bash scripting that allows you to manage groups of items, such as filenames, user inputs, or configuration values. Bash provides three main methods for handling collections: arrays, associative arrays, and space-separated strings.
Guidelines
- Arrays for Collections: ALWAYS use arrays when managing collections of elements to ensure clarity and safety.
- Avoid Space-Separated Strings: DO NOT use space-separated strings for handling collections.
Advantages of Arrays
- Clarity and Safety: Arrays prevent errors caused by word splitting and glob expansion that can occur with space-separated strings.
- Flexibility: Arrays allow for easy manipulation and access to individual elements, as well as simplified expansion in commands that accept multiple arguments.
- Ease of Maintenance: Code utilizing arrays is generally clearer and easier to maintain, particularly as script complexity increases.
Examples
Using Space-Separated Strings
Using Arrays
Advantage: Arrays handle items with spaces or special characters correctly, preventing unintended word splitting.
Using Space-Separated Strings
Using Arrays
Advantage: Arrays simplify command syntax and ensure all arguments are correctly passed, even if filenames contain spaces.
Using Space-Separated Strings
Using Arrays
Advantage: Arrays allow direct access to individual elements without additional parsing.
//// Adding Elements to a Collection
Using Space-Separated Strings
Using Arrays
Advantage: Arrays provide straightforward syntax for adding elements, improving readability and maintainability.
////
Parsing Input into Variables
Parsing input into variables is a common task in Bash scripting, allowing you to extract and process data from user inputs, files, or other sources. Bash provides the read command for these tasks, offering a more efficient and secure alternative to external commands like awk, sed, or cut.
Guidelines
- Bash Built-ins: Use readto safely and efficiently parse user inputs and other data directly into variables.
- Customize with IFS: Adjust the Internal Field Separator (IFS) as needed when using readto ensure that inputs are split according to your specific requirements, enhancing the flexibility and accuracy of data parsing.
- Avoid External Commands: Minimize the use of external commands like awk,sed, orcutfor parsing strings.
Advantages of read
- Efficiency: readoperates within the shell, eliminating the need for slower, resource-intensive external commands.
- Controlled Parsing: readallows for controlled and direct parsing of inputs into variables within the shell, reducing the complexity and potential risks associated with using external commands for input processing.
- Simplicity: Offers an easy-to-understand syntax for directly parsing complex data structures.
- Portability: readis a Bash builtin, ensuring consistent behavior across different systems and environments.
Examples
Using awk
Using read
Advantage: read provides a simpler and more efficient way to parse space-separated strings directly into variables.
Using sed
Using read
Advantage: read directly parses the input into variables, reducing complexity and improving readability.
Efficient Array Population
The mapfile command, also known as readarray, is a Bash built-in that provides an efficient way to populate an array with lines from a file or the output of a command. This command simplifies the process of reading and storing multiline data, making it particularly useful when working with large datasets or when you need to process input line by line.
Guidelines
- Use mapfilefor Multiline Input: When you need to read and store multiline data into an array, prefer mapfile over looping constructs.
- Specify a Delimiter with -d: Use the-doption to specify a custom delimiter if the input data is not newline-separated.
- Avoid Excessive Loops: mapfilecan efficiently replace complex loops that read and store data line by line, improving script readability and performance.
Advantages of mapfile
- Efficiency: mapfilereads an entire stream into an array at once, making it faster and more efficient than reading lines individually in a loop.
- Simplicity: The command reduces the amount of code required to populate arrays, leading to cleaner and more maintainable scripts.
- Flexibility: With options like -t(to remove trailing newlines) and-d(to specify a delimiter), mapfile offers flexibility in handling various input formats.
Examples
Using a loop:
Using mapfile:
Advantage: mapfile simplifies the process of reading a file into an array, reducing the code and improving efficiency.
Using a loop:
Using mapfile:
Advantage: mapfile efficiently reads the output of a command into an array without the need for an explicit loop.