irulescan

security analyzer for iRules

View on GitHub

irulescan
security analyzer for iRules


build container image size releases

irulescan is a tool to scan iRules for unexpected/unsafe expressions that may have undesirable effects like double substitution and additional issues.

irulescan would not exist without tclscan.

It is available as a docker/container image as well as a Github Action irulescan-action.

Usage

It is easiest to use the irulescan container to scan your iRules, available via docker hub as well as ghcr.io.

The container will recursively scan files in the /scandir directory within the container and return results in JSON format. Only files with the (case insensitive) extensions .tcl, .irul and .irule will be scanned.

Command line

Scanning all iRules in a directory recursively ($PWD/tests/basic):

docker run --rm -v "$PWD/tests/basic:/scandir" simonkowallik/irulescan
[
  {
    "filepath": "dangerous.tcl",
    "warning": [
      "Unquoted expr at `1` in `expr 1 + $one`",
      "Unquoted expr at `+` in `expr 1 + $one`"
    ],
    "dangerous": ["Dangerous unquoted expr at `$one` in `expr 1 + $one`"]
  },
  {
    "filepath": "ok.tcl",
    "warning": [],
    "dangerous": []
  },
  {
    "filepath": "warning.tcl",
    "warning": [
      "Unquoted expr at `1` in `expr 1 + 1`",
      "Unquoted expr at `+` in `expr 1 + 1`",
      "Unquoted expr at `1` in `expr 1 + 1`"
    ],
    "dangerous": []
  }
]

If you plan to use irulescan frequently, consider adding a shell function to wrap the container execution and add it in your shell, eg. in your .bashrc or similar.

irulescan(){ docker run --rm -i -v "$PWD:$PWD" -w "$PWD" simonkowallik/irulescan:latest "${@:--help}"; }
# invoke ephemeral (--rm) container interactively (-i)
# bind mount (-v) PWD to the same path and use it as workdir (-w)
# pass any parameters, if none/empty, pass "help" ("${@:--help}")
irulescan check --exclude-empty-findings ./tests/basic/ | jq
[
  {
    "filepath": "tests/basic/dangerous.tcl",
    "warning": [
      "Unquoted expr at `1` in `expr 1 + $one`",
      "Unquoted expr at `+` in `expr 1 + $one`"
    ],
    "dangerous": ["Dangerous unquoted expr at `$one` in `expr 1 + $one`"]
  },
  {
    "filepath": "tests/basic/warning.tcl",
    "warning": [
      "Unquoted expr at `1` in `expr 1 + 1`",
      "Unquoted expr at `+` in `expr 1 + 1`",
      "Unquoted expr at `1` in `expr 1 + 1`"
    ],
    "dangerous": []
  }
]

[!NOTE] Please note the differences in "filepath" between the two invocation methods. This is important to consider when comparing a new scan to existing results. --exclude-empty-findings removes the entry for the file "ok.tcl" as it has no findings. cd tests/basic; irulescan check . | jq would have provided the same results as the docker command from the previous example.

When specifying a file, irulescan will try to scan the file regardless of the file extension.

irulescan check --no-warn tests/basic/dangerous.txt
[{"filepath":"tests/basic/dangerous.txt","warning":[],"dangerous":["Dangerous unquoted expr at `$one` in `expr 1 + $one`"]}]
irulescan --help
irulescan is a tool to scan iRules for unexpected/unsafe expressions that may have undesirable effects like double substitution.
home: https://github.com/simonkowallik/irulescan

Usage: irulescan <COMMAND>

Commands:
  check     Scan all iRules in a directory (recursively) or the specified file
  checkref  Scan all iRules in reference file (JSON) and compare to reference
  parsestr  Parse given string or stdin
  help      Print this message or the help of the given subcommand(s)

Options:
  -h, --help
          Print help (see a summary with '-h')

  -V, --version
          Print version

Using irulescan in CI

The container uses irulescan as the entry point with the default CMD set to check . which will scan /scandir recursively.

When using a CI system with custom mount points (eg. /custom/path), this can be easily supported by passing a new CMD like below. Note that, like outlined above, this will change the prefix of "filepath" for each iRule file.

docker run --rm \
  -v "$PWD/tests/tcl/:/custom/path" \
  simonkowallik/irulescan check /custom/path

Also have a look at GitHub Action: irulescan-action, which can be used when using GitHub.

Alternatives are using the API Server (see below) or invoke the irulescan container as outlined above. Consider creating a scan result file to compare against known findings and known iRules. Any change in findings could then be used to fail the CI run.

Check results against a reference (previous scan result)

The checkref command can be used to check each file listed in that reference file. The reference file is just a saved scan. See examples below.

irulescan/ $ cd ./tests/basic

Using a previous scan result (irulescan.json) and passing it to checkref via STDIN. If the check is successful, the exist code will be 0.

$ cat irulescan.json | irulescan checkref -
OK

$ echo $?
0

Re-checking irulescan_nowarn.json with no additional options.

$ irulescan checkref irulescan_nowarn.json
Failed reference check!
Extra in scan_results: .[0].warning.[0].("Unquoted expr at `+` in `expr 1 + $one`")
Extra in scan_results: .[0].warning.[1].("Unquoted expr at `1` in `expr 1 + $one`")

$ echo $?
1

The above example produces a failed check as the scan results produce additional findings. The exit code is 1.

The previous scan result irulescan_nowarn.json does not contain any warnings, using the --no-warn option the check succeeds.

$ irulescan checkref --no-warn irulescan_nowarn.json
OK

$ echo $?
0

CLI exit codes

When using the irulescan check command:

When using the irulescan checkref command:

API Server

The irulescan container tag :apiserver provides a simple Swagger / OpenAPI server for scanning iRule code or multiple files.

Start the API server:

docker run -t --rm -p 8000:8000 simonkowallik/irulescan:apiserver

Scanning a single file / iRule code:

curl -s http://localhost/scan/ \
  --data-binary '@tests/basic/dangerous.tcl' | jq
{
  "warning": [
    "Unquoted expr at `1` in `expr 1 + $one`",
    "Unquoted expr at `+` in `expr 1 + $one`"
  ],
  "dangerous": [
    "Dangerous unquoted expr at `$one` in `expr 1 + $one`"
  ]
}

Scanning multiple files:

curl -s http://localhost/scanfiles/ \
  -F 'file=@tests/basic/warning.tcl' \
  -F 'file=@tests/basic/ok.tcl' \
  | jq
[
  {
    "filepath": "ok.tcl",
    "warning": [],
    "dangerous": []
  },
  {
    "filepath": "warning.tcl",
    "warning": [
      "Unquoted expr at `1` in `expr 1 + 1`",
      "Unquoted expr at `+` in `expr 1 + 1`",
      "Unquoted expr at `1` in `expr 1 + 1`"
    ],
    "dangerous": []
  }
]

Here is a demo of the Swagger UI:

simonkowallik/irulescan:apiserver

Additional resources

For safer authoring the VS Code iRules Extension is highly recommended: