= Coding Standard for Bash Scripts
Perforce Professional Services <consulting-p4@perforce.com>
:revnumber: v2025.2
:revdate: 2026-04-07
:doctype: book
:icons: font
:toc:
:toclevels: 5
:sectnumlevels: 4
:xrefstyle: full
// Attribute for ifdef usage
:unix_doc: true
== DRAFT NOTICE
WARNING: This document is in DRAFT status and should not be relied on yet. It is a preview of a document to be completed in a future release.
== Preface
Welcome to the Perforce P4 Server Deployment Package (SDP) Coding Standard for Bash scripts.
This Standard is intended to provide information useful for Bash script programming in the SDP Environment. This applies to scripts delivered as part of the SDP Package, and can also be applied to custom scripts such as custom triggers or systems integrations for interaction with P4 Servers that are added to the SDP in any given environment in link:../Server/Unix/p4/common/site/ReadMe.md[The Site Directory].
This document provides both _standards_ and _guidelines_ to follow. Standards must be followed for a script to be considered adherent to this standard. Guidelines are suggestions but are not strictly required to achieve compliance. Generally descriptions involving clear words like _must_ indicate standards, while terms like _should_ that allow for variation indicate guidelines.
*Please Give Us Feedback*
Perforce welcomes feedback. Please send suggestions for improving this document or the SDP to consulting-p4@perforce.com.
:sectnums:
== Bash Version
SDP may assume the bash version to be 4.3+ for purposes of deciding which features to use. For example, features like associative arrays can be assumed, as well as syntax like `${var,,}` to return the lowercase form of the value of $var.
The _shebang_ line (the first line of each bash script that starts with the `#!` sequence) must be one of these two options:
#!/bin/bash
or
#!/usr/bin/env bash
While some scripts may run with Bash 3.x, others require Bash 4.x or later, as this is reliably available on all Linux distributions that are not End of Life (EOL). For example, Bash 4+ was introduced in the Red Hat family distributions starting in version 7, in Ubuntu starting with Ubuntu 10.04, and SuSE Enterprise Linux 12.
As a whole, the SDP bash scripts support any version of UNIX/Linux that provides bash 4.0 or later. This includes all Linux and most UNIX environments except for OSX (for which the workaround is installing modern bash and adjusting shebang lines).
This standard applies to SDP on UNIX/Linux. For SDP on Windows, PowerShell, Python and Batch (`.bat`) scripts are used, so this standard for Bash does not apply. (Bash can run on Windows in various ways and some scripts may work, but are not supported due to not being tested.)
== Bash Directives
Immediately after the shebang line should appear the `set -u` directive, requiring variables to be defined before being referenced. The `set +u` directive is allowed as needed for examples, such as in command line processing where variables like `$1` may legitimately be referenced while undefined.
SDP scripts do not use `set -e`. Explicit error handling at each point where errors can occur is preferred instead. See the Error Handling tenet in the Standards section.
== Scripts and Libraries
A _script_ is a bash shell script file intended to be executed directly by users or other automation, and for which the first line is the shebang line. Scripts have the `+x` execute bit set.
A _library_ is a bash shell file intended to be sourced by scripts or other library functions. Library files have a `.lib` suffix, and do not have the `+x` execute bit set. Library files generally define reusable functions and may contribute to the shell environment.
Scripts appear in various directories in the deployed SDP structure.
Libraries should appear only in the `<SDPRoot>/common/lib` directory for scripts that are part of the SDP package, otherwise `<SDPRoot>/common/site/lib` for site-specific libraries.
Legacy exceptions: The `backup_functions.sh` remains in the `<SDPRoot>/common/bin` directory for backward compatibility with customer-side custom scripts. Some older library files may retain the `.sh` rather than `.lib` suffix.
== Script Templates
The bash script template illustrates and adheres to this standard. Here is the link to link:https://workshop.perforce.com/view/p4-sdp/main/Server/Unix/p4/common/bin/templates/template.sh[the bash shell script template].
== Script Sections
Scripts should have whichever of the following sections apply:
=== Header
The Header section contains the shebang line, directives, license info, and
indication of how to get documentation for the script.
.Section Example - Header
[source,bash]
----
#!/bin/bash
set -u
#==============================================================================
# Copyright and license info is available in the LICENSE file included with
# the Server Deployment Package (SDP), and also available online:
# https://swarm.workshop.perforce.com/projects/perforce-software-sdp/view/main/LICENSE
#------------------------------------------------------------------------------
----
=== Declarations and Environment
This section declares global variables. Creating the shell environment for the script also starts in this section.
.Section Example - Declarations and Environment
[source,bash]
----
#==============================================================================
# Declarations and Environment
declare -i ErrorCount=0
----
=== Local Functions
Each script should define a subset of these standard local functions:
.Section Example - Local Functions
[source,bash]
----
#==============================================================================
# Local Functions
function msg () { echo -e "$*"; }
----
=== SDP Library Functions
Most scripts will use SDP libraries. Each library is sourced using the `$SDPCommonLib` variable.
.Section Example - SDP Library Functions
[source,bash]
----
#==============================================================================
# Load SDP Library Functions.
if [[ -d "$SDPCommonLib" ]]; then
# shellcheck disable=SC1090 disable=SC1091
source "$SDPCommonLib/logging.lib" ||\
bail "Failed to load bash lib [$SDPCommonLib/logging.lib]. Aborting."
# shellcheck disable=SC1090 disable=SC1091
source "$SDPCommonLib/run.lib" ||\
bail "Failed to load bash lib [$SDPCommonLib/run.lib]. Aborting."
fi
----
=== Command Line Processing
See the script template for an example of the Command Line Processing block.
=== Command Line Verification
See the script template for an example of the Command Line Verification block.
=== Main Program
See the script template for an example of the Main Program block. Among other things, this section is responsible for starting any log file processing that is to be done.
== SDP Root and Relocatability
In Production, the SDP Root Directory (referenced as the `$SDP_ROOT` shell environment variable or the `$SDPRoot` variable in scripts) will always have a value of `/p4`. However, the standard allows this root to be changed to simplify testing and development. All scripts should behave properly if this value is changed. In any production environment, the default value should apply; a value for SDP_ROOT should only be set for dev/test environments.
== Version Identification
WARNING: This section does not yet apply to the released SDP; it is forward-looking. This requires the SDP to complete its migration from a Classic depot structure to its new Streams structure for versioning to work as described.
All executable shell scripts and libraries must include a _Version ID Block_.
The regular form of the version ID block looks like this:
.Version ID Block (Regular)
[source,bash]
----
# Version ID Block. Relies on +k filetype modifier.
#------------------------------------------------------------------------------
# shellcheck disable=SC2016
declare VersionID='$Id: //p4-sdp/dev_rebrand/Server/Unix/p4/common/sdp_upgrade/sdp_upgrade.sh#6 $ $Change: 31803 $'
declare VersionStream=${VersionID#*//}; VersionStream=${VersionStream#*/}; VersionStream=${VersionStream%%/*};
declare VersionCL=${VersionID##*: }; VersionCL=${VersionCL%% *}
declare Version=${VersionStream}.${VersionCL}
[[ "$VersionStream" == r* ]] || Version="${Version^^}"
----
This Version ID Block has several features:
* It ensures file versions are updated reliably on every submit by taking advantage of the P4 `+k` file type modifier to expand keywords in the file upon submit.
* The SDP major version is determined from the stream name (e.g., r25.2), so the version clarifies what released SDP version the file is part of.
* Non-released "dev branch" versions will have a version identifier that clearly indicates they are not released production code.
* The changelist number gives each file a unique identifier.
There is also a short form of this block intended for library files:
.Version ID Block (Short Form)
[source,bash]
----
# Version ID Block. Relies on +k filetype modifier.
#------------------------------------------------------------------------------
# shellcheck disable=SC2016
declare VersionID='$Id: //p4-sdp/dev_rebrand/Server/Unix/p4/common/sdp_upgrade/sdp_upgrade.sh#6 $ $Change: 31803 $'
declare VersionStream=${VersionID#*//}; VersionStream=${VersionStream#*/}; VersionStream=${VersionStream%%/*};
declare VersionCL=${VersionID##*: }; VersionCL=${VersionCL%% *}
declare Version=${VersionStream}.${VersionCL}
[[ "$VersionStream" == r* ]] || Version="${Version^^}"
----
=== Location
Supported SDP scripts provided by Perforce appear in `/p4/common/bin`.
Custom scripts (inherently unsupported) are expected to appear somewhere under the `/p4/common/site` directory, such as `/p4/common/site/bin` or `/p4/common/site/p4ms`.
[[_logging]]
== Logging
Scripts must be self-logging: all output generated during execution (stdout
and stderr) is automatically captured in a log file without requiring the
caller to use redirection or `tee`.
=== Log File Location and Naming
Log files are written to the `$LOGS` directory, which is set by `p4_vars` to
`/p4/<instance>/logs`. Each script run produces a time-stamped log file:
----
$LOGS/<ScriptName>.<YYYY-MM-DD-HHMMSS>.log
----
For example:
----
/p4/1/logs/daily_checkpoint.2026-04-07-143022.log
----
If two invocations start within the same second (e.g., a cron job and a manual
run), an incrementing integer suffix is appended to guarantee uniqueness:
----
$LOGS/<ScriptName>.<YYYY-MM-DD-HHMMSS>.<N>.log
----
Log files are created atomically using the bash `noclobber` option (`set -C`),
ensuring that concurrent invocations never claim the same filename.
NOTE: Millisecond precision (`%3N`) is intentionally not used in fallback
filenames because it is a GNU `date` extension not available on macOS. The
integer suffix is sufficient to guarantee uniqueness and is fully portable.
=== Log Symlink (LogLink)
In addition to the time-stamped log file, each script maintains a stable
symlink in the same directory:
----
$LOGS/<ScriptName>.log → $LOGS/<ScriptName>.<timestamp>.log
----
This `LogLink` symlink always points to the most recently started log,
providing a predictable name for operators running `tail -f`, monitoring
tools, and any integrations that need to locate the latest log without knowing
the exact timestamp.
If a regular file already exists at the `LogLink` path (e.g., from an older
SDP version that did not use symlinks), it is automatically renamed to a
time-stamped filename before the symlink is created.
=== Log Redirection
After the log file and symlink are established, both stdout and stderr are
redirected to the log via `exec`.
When running interactively (terminal attached, color mode active), a `tee`
process substitution is used so that output appears on both the terminal and
in the log simultaneously. ANSI color codes are stripped from the log copy
so that log files remain clean when viewed with standard text tools.
In silent mode (`-si`), all output goes to the log only — nothing appears on
the terminal. This is the intended mode for crontab invocations, where any
terminal output would trigger an email from the cron daemon.
=== Controlling the Log
Two command-line options govern logging behavior:
`-L <file>`:: Write the log to `<file>` instead of the default time-stamped
file in `$LOGS`. No `LogLink` symlink is created in this case.
`-L off`:: Disable logging entirely. All output goes to the terminal only.
Cannot be combined with `-si`.
== Standards
=== Tenets of Scripting
==== Avoid Requiring Modification
No modification of scripts should be needed in customer environments for normal operation. An appropriate mix of command-line options and configuration files should be used to provide the flexibility required to operate to meet various customer needs.
==== Self Logging
Scripts must be self-logging: all output (stdout and stderr) is captured in a
log file automatically, with no need for the caller to use redirection or
`tee`. See <<_logging>>.
All scripts should support `-h` (short usage synopsis), `-man` (full
documentation), and `-V` (version check, with `--version` alias) options,
with standard meanings.
All scripts must define a `usage()` function per the template. All scripts
must have a `terminate()` function available; this is provided by sourcing
`logging.lib` rather than defined locally in each script.
All scripts should have complete documentation.
==== Error Handling
SDP scripts do not use `set -e`. Instead, errors are handled explicitly at
each point where they can occur, using one of three mechanisms:
`bail`:: Fatal error. Prints a red error message, increments `ErrorCount`,
and exits immediately. Use this when continuing is not meaningful.
[source,bash]
----
some_command || bail "some_command failed; cannot continue."
----
`errmsg`:: Non-fatal error. Prints a red error message and increments
`ErrorCount`, but execution continues. Use this when the script should keep
running and report a failure count at the end.
[source,bash]
----
some_command || errmsg "some_command failed; continuing."
----
`warnmsg`:: Warning. Prints a yellow warning message and increments
`WarningCount`. Use this for conditions that are noteworthy but not errors.
[source,bash]
----
[[ -n "$SomeOptionalVar" ]] || warnmsg "SomeOptionalVar is not set."
----
At the end of the script, `ErrorCount` and `WarningCount` are checked to
produce a final summary message and to set the exit code. The script exits
with a value of `$ErrorCount`, so non-fatal errors still result in a non-zero
exit code. The `terminate()` function (from `logging.lib`) handles this
final exit.
=== Style
==== Indentation
Scripts must use 3-space indentation. Tab characters must not appear in
script or library files, except within here-documents where tab characters
carry semantic meaning.
==== Naming Conventions
===== Shell Environment Variables
Shell environment variables defined outside scripts — set in the surrounding
shell environment or in SDP environment files such as `p4_vars` — must be
all uppercase with underscore word separators, following standard UNIX/POSIX
conventions. Examples: `SDP_ROOT`, `P4PORT`, `LOGS`.
===== Global Script Variables
Variables with global scope (declared outside any function) must use
UpperCamelCase (PascalCase), starting with an uppercase letter.
Examples: `ThisScript`, `ErrorCount`, `LogTimestamp`.
Constants — variables assigned once and not expected to change during the
lifetime of the script — may alternatively use all-uppercase naming with
underscore separators. Examples: `H1`, `H2`, `GREEN`, `RESET`.
===== Library Output Variables
Variables that are written by library functions and read by calling scripts
must use `ALL_UPPERCASE` with underscore separators. This signals that the
variable crosses an ownership boundary: the calling script did not set it —
the library did. Examples: `CMDLAST`, `CMDEXITCODE`, `RCMDLAST`, `RCMDEXITCODE`.
This mirrors the convention for shell environment variables (which are also set
outside the script) and gives a script reader an immediate visual cue: an
all-uppercase name not listed in the script's own Declarations section is a
library output, not a local variable.
===== Summary of Naming Conventions
[cols="1,1,2",options="header"]
|===
|Scope / Origin |Convention |Examples
|Shell environment (set outside the script)
|`ALL_UPPERCASE`
|`SDP_ROOT`, `P4PORT`, `LOGS`
|Global script variable (set by the script)
|`UpperCamelCase`
|`ThisScript`, `ErrorCount`, `LogTimestamp`
|Constant (set once, does not change)
|`ALL_UPPERCASE` or `UpperCamelCase`
|`H1`, `H2`, `GREEN`, `RESET`
|Library output variable (set by a library, read by the script)
|`ALL_UPPERCASE`
|`CMDLAST`, `CMDEXITCODE`
|Function-local variable
|`lowerCamelCase`
|`cmd`, `honorNoOpFlag`, `cmdOut`
|Function name
|`lowercase` or `lower_with_underscores`
|`msg`, `bail`, `get_old_log_timestamp`
|===
===== Function-scoped (Local) Variables
Variables declared inside functions must use the `local` keyword and must
use lowerCamelCase, starting with a lowercase letter.
Examples: `cmd`, `desc`, `honorNoOpFlag`, `cmdOut`.
NOTE: `declare` inside a function is functionally equivalent to `local` in
Bash, but `local` is preferred for function-scoped variables because it more
clearly expresses intent.
===== Function Names
Function names must be all lowercase. Multi-word function names use
underscore separators.
Examples: `msg`, `errmsg`, `bail`, `get_old_log_timestamp`.
==== Quoting
All variable expansions must be double-quoted unless word splitting or glob
expansion is intentionally required. Use `"$var"` rather than `$var`.
==== Conditionals
Use `[[ ]]` (double brackets) rather than `[ ]` (single brackets) for all
conditional tests. Double brackets are a Bash built-in with cleaner behavior
for string comparisons, pattern matching, and avoidance of word-splitting
surprises.
[source,bash]
----
# Correct
[[ -n "$MyVar" ]]
[[ "$Count" -gt 0 ]]
# Avoid
[ -n "$MyVar" ]
[ "$Count" -gt 0 ]
----
==== Line Length
Lines should not exceed 120 characters. Keeping lines to 80 characters is
preferred where practical, especially for documentation-heavy comment blocks.
==== Continuation Lines
When a command spans multiple lines, use `\` for line continuation and indent
the continuation line by 3 additional spaces relative to the opening of the
command.
==== Comments
Use `#` comments to explain intent, especially for logic that is not
immediately apparent from the code. Inline comments are separated from code
by at least two spaces.
Section headings use a standard major divider:
----
#==============================================================================
# Section Name
----
Function headers and sub-section breaks use a minor divider:
----
#------------------------------------------------------------------------------
# Function: function_name
#
# Short description of what the function does.
#
# Input:
# $1 - first_param: Description.
# $2 - second_param: Description.
#
# Returns: 0 on success, non-zero on error.
#------------------------------------------------------------------------------
----
=== ShellCheck Compliance
The ShellCheck utility is a static code analysis tool for bash shell scripts. See: https://www.shellcheck.net
All SDP scripts and library files must pass a ShellCheck scan with ShellCheck version 0.10.0 or later.
Where appropriate, `#shellcheck disable=SC<NNNN>` directives may be used, as may `.shellcheckrc` files, to suppress warnings deemed not of concern.
See <<_shellcheck_appendix>> for cases where ShellCheck guidance conflicts with SDP style.
[[_shellcheck_appendix]]
[appendix]
== ShellCheck Notes
This appendix documents cases where ShellCheck guidance conflicts with SDP
style, and the rationale for the SDP's position.
// Placeholder — to be populated as specific conflicts are identified.
[appendix]
== DRAFT NOTICE
WARNING: This document is in DRAFT status and should not be relied on yet. It is a preview of a document to be completed in a future release.
| # | Change | User | Description | Committed | |
|---|---|---|---|---|---|
| #1 | 32659 | C. Thomas Tyler |
Merge down dev -> dev_rebrand. Local changes to template.sh were discareded; we'll need to redo those. |
||
| //p4-sdp/dev/doc/SDP_CodingStandard_bash.adoc | |||||
| #1 | 32658 | C. Thomas Tyler |
Upkeep merge from Classic to Streams. p4 -s merge -c <CL> -b SDP_Classic_to_Streams p4 -s resolve -as # One file needed override handling. This will undo local changes, but # there shouldn't be any. We'll deal with local changes when we merge # down to dev_rebrand. p4 resolve -at //p4-sdp/dev/Server/Unix/p4/common/bin/templates/template.sh p4 submit -c <CL> |
||
| //guest/perforce_software/sdp/dev/doc/SDP_CodingStandard_bash.adoc | |||||
| #1 | 32539 | bot_Claude_Anthropic | Add DRAFT Bash Coding Standard to SDP (SDP_CodingStandard_bash.adoc). | ||