Welcome to the CRS Linter documentation.
To run this tool, you need Python 3.7 or later.
To install, run:
pip3 install crs-linter
The script expects an argument at least - this would be a single file or a file list, eg: /path/to/coreruleset/*.conf
.
First, an attempt is made to parse each file specified on the command line. This is a "pre-check", and runs on all files before the other tests.
- Parsing check - try to parse the structure, this is a syntax check note: this script is a bit more strict than mod_security. There are some cases, where mod_security allows the syntax, but msc_pyparser not.
Second, the script loops over each of the parsed structures. Each iteration consists of the following steps:
- Casing check - checks operators, actions, transformations and ctl names for proper casing
e.g.,
@beginsWith
is allowed,@beginswith
is not. In this step, the script also ensures that an operator is present, egSecRule ARGS "^.*"
isn't allowed without@rx
operator. - Action order check - This step verifies that actions are specified in the correct order - see the wiki
- Format check CRS has a good reference for indentation and other formatting.
msc_pyparser
follows these rules when it creates the config file(s) from parsed structure(s). After the re-build is done, it runs a compare between the original file and the built one with help ofdifflib
. The script reports all non-compliant formatting. Note, thatdifflib
is a part of the standard Python library, you don't need to install it. - Deprecation check - This step checks for use of deprecated features. The following features are deprecated:
ctl:auditLogParts
is no longer supported by CRS
- Duplicate ID's check - This step checks that each rule has a unique ID.
- paranoia-level/N tag and its value - This step checks that the
paranoia-level/N
tag is present when required and whether it has the correct valueN
for its context. Specifically:- if a rule is activated for a specific paranoia level
L
and does not have thenolog
action, theparanoia-level/N
tag must be set and the value ofN
must beL
- if a rule is activated outside of any paranoia level, or has the
nolog
action, theparanoia-level/N
tag must not be set
- if a rule is activated for a specific paranoia level
- Anomaly scoring check - This step checks that rules are configured properly for the anomaly scoring mechanism:
- every rule must update the correct scoring variable with the correct severity related score, for example:
setvar:inbound_anomaly_score_pl2=+%{tx.critical_anomaly_score}
- every rule must update the correct scoring variable with the correct severity related score, for example:
setvar:inbound_anomaly_score_pl2=+%{tx.critical_anomaly_score}
- every rule must update the correct scoring variable with the correct severity related score, for example:
- Initialization of used transaction (TX) variables - all used TX variables must be initialised before their first use. Using a TX variable means one of the following:
- the variable is a target of a rule, e.g.,
SecRule TX.foo ...
- the variable is an operator argument, eg
SecRule ARGS "@rx %{TX.foo}"...
- the variable is a right hand side operand in a
setvar
action, egsetvar:tx.bar=%{tx.foo}
- the variable is in an expansion, e.g., as part of the value of a
msg
action:msg:'Current value of variable: %{tx.foo}
- the variable is a target of a rule, e.g.,
- Check rule tags - only tags listed in
util/APPROVED_TAGS
may be used as tags in rules- to use a new tag on a rule, it must first be registered in the util/APPROVED_TAGS file
- Check t:lowercase and (?i) flag - No combination of t:lowercase and (?i) should appear in the same rule.
- Check rule has a tag with value
OWASP_CRS
- Every rule must have a tag with valueOWASP_CRS
- Check rule has a
ver
action with correct version - Every rule must havever
action with correct value- script accepts
-v
or--version
argument if you want to pass it manually - if no
-v
was given, the script tries to extract the version from result ofgit describe --tags
- script accepts
- Check if the rule uses any
TX:N
target in a chained rule then there must be acapture
action - Consider the rule is a chained rule and not the first rule uses theTX:1
target- this means we want to check the previously rule's result
- which is produced by
capture
- if there is no previously
capture
, then it means the nextTX:1
will uses a previously produced captured value
Finally, the script prints a report of all unused TX variables. Usually, unused TX variables occur when a rule creates a TX variable (e.g., setvar:tx.foo=1
) but the value of the variable is never used anywhere else. This will only be revealed after the script has checked all rules.
If script finds any parser error, it stops immediately. In case of other error, shows it (rule-by-rule). Finally, the script returns a non-zero value.
If everything is fine, rule returns with 0.
Normally, you should run the script:
crs-linter -r crs-setup.conf.example -r rules/*.conf
Optionally, you can add the option --output=github
(default value is native
):
crs-linter --output=github -r crs-setup.conf.example -r rules/*.conf
In this case, each line will have a prefix, which could be ::debug
or ::error
. See this.
To run these samples, see the files in examples
directory.
SecRule &ARGS_GET "@eq 3" \
"id:1,\
phase:2,\
pass,\
t:none,\
nolog,\
chain
SecRule ARGS_GET:foo "@rx bar" \
"t:none,t:urlDecodeUni,t:lowercase,\
setvar:'tx.some_vars=1'
As you can see, there are two "
missing above: the first one after the chain
, and the other one from the end of the chained rule. Mod_security allows this, but this isn't well formed. (See #2184)
Check it:
crs-linter -r examples/test1.conf
Config file: examples/test1.conf
Can't parse config file: examples/test1.conf
file=examples/test1.conf, line=8, endLine=8, title=Parser error: can't parse file
$ echo $?
1
SecRule REQUEST_URI "@beginswith /index.php" \
"id:1,\
phase:1,\
deny,\
t:none,\
nolog"
In this rule the operator is lowercase. Mod_security allows both form.
crs-linter -r examples/test2.conf
Config file: examples/test2.conf
Parsing ok.
Ignore case check found error(s)
file=examples/test2.conf, line=1, endLine=1, title=Case check: Operator case mismatch: @beginswith (rule: 1)
Action order check ok.
Indentation check ok.
$ echo $?
1
SecRule REQUEST_URI "@beginsWith /index.php" \
"phase:1,\
id:1,\
deny,\
t:none,\
nolog"
In this rule, the phase
and id
are interchanged. As documentation says, the first action must be the id
, the second one is the phase
.
crs-linter -r examples/test3.conf
Config file: examples/test3.conf
Parsing ok.
Ignore case check ok.
Action order check found error(s)
file=examples/test3.conf, line=3, endLine=3, title=Action order check: action 'phase' at pos 0 is wrong place against 'id' at pos 1 (rule: 1)
Indentation check ok.
$ echo $?
1
SecRule ARGS "@rx foo" \
"id:1,\
phase:1,\
pass,\
nolog"
SecRule ARGS "@rx foo" \
"id:2,\
phase:1,\
pass,\
nolog"
SecRule ARGS "@rx foo" \
"id:3,\
phase:1,\
pass,\
nolog"
In this rule set, the first line and the rule with id:3
first action have an extra leading space. As documentation describes, CRS has a strict indentation rules. The script checks the indentation with help of Python's difflib.
crs-linter -r examples/test4.conf
Config file: examples/test4.conf
Parsing ok.
Ignore case check ok.
Action order check ok.
Indentation check found error(s)
---
+++
file=examples/test4.conf, line=1, endLine=6, title=Indentation error: an indetation error has found
@@ -1,5 +1,5 @@
- SecRule ARGS "@rx foo" \
- "id:1,\
+SecRule ARGS "@rx foo" \
+ "id:1,\
phase:1,\
pass,\
nolog"
file=examples/test4.conf, line=11, endLine=18, title=Indentation error: an indetation error has found
@@ -11,7 +11,7 @@
nolog"
SecRule ARGS "@rx foo" \
- "id:3,\
+ "id:3,\
phase:1,\
pass,\
nolog"
SecRule REQUEST_URI "index.php" \
"phase:1,\
id:1,\
deny,\
t:none,\
nolog"
In this rule, the operator is missing. As ModSecurity documentation says "the rules that do not explicitly specify an operator default to @rx". In CRS, this isn't allowed.
$ crs-linter -r examples/test5.conf
Config file: examples/test5.conf
Parsing ok.
Ignore case check found error(s)
file=examples/test5.conf, line=1, endLine=1, title=Case check: Empty operator isn't allowed (rule: 1)
Action order check ok.
Indentation check ok.
$ echo $?
1
SecRule TX:sql_error_match "@eq 1" \
"id:1,\
phase:4,\
block,\
capture,\
t:none,\
ctl:auditLogParts=+E"
The ctl:auditLogParts=+E
(or any kind of ctl:auditLogParts
) is not allowed in CRS.
See the CRS PR #3034
$ util/crs-rules-check/rules-check.py -r util/crs-rules-check/examples/test6.conf
Config file: util/crs-rules-check/examples/test6.conf
Parsing ok.
Ignore case check ok.
Action order check ok.
Indentation check ok.
Found 'ctl:auditLogParts' action is in wrong place.
file=util/crs-rules-check/examples/test6.conf, line=7, endLine=7, title='ctl:auditLogParts' action in wrong place: action can only be placed in last part of a chained rule (rule: 1)
$ echo $?
1
SecRule ARGS "@rx foo" \
"id:1001,\
phase:2,\
block,\
capture,\
t:none"
SecRule ARGS_NAMES "@rx bar" \
"id:1001,\
phase:2,\
block,\
capture,\
t:none"
In this rule file, there are two rules with same id
.
crs-linter -r util/crs-rules-check/examples/test7.conf
Config file: util/crs-rules-check/examples/test7.conf
Parsing ok.
Checking parsed rules...
util/crs-rules-check/examples/test7.conf
Ignore case check ok.
Action order check ok.
Indentation check ok.
'ctl:auditLogParts' actions are in right place.
Found duplicated id('s)
file=util/crs-rules-check/examples/test7.conf, line=10, endLine=10, title='id' is duplicated: id 1001 is duplicated, previous place: util/crs-rules-check/examples/test7.conf:3
paranoia-level tags are correct.
PL anomaly_scores are correct.
All TX variables are set
End of checking parsed rules
$ echo $?
1
SecRule &TX:blocking_paranoia_level "@eq 0" \
"id:901120,\
phase:1,\
pass,\
nolog,\
ver:'OWASP_CRS/4.0.0-rc1',\
setvar:'tx.blocking_paranoia_level=1'"
SecRule &TX:detection_paranoia_level "@eq 0" \
"id:901125,\
phase:1,\
pass,\
nolog,\
ver:'OWASP_CRS/4.0.0-rc1',\
setvar:'tx.detection_paranoia_level=%{TX.blocking_paranoia_level}'"
SecRule &TX:error_anomaly_score "@eq 0" \
"id:901141,\
phase:1,\
pass,\
nolog,\
ver:'OWASP_CRS/4.0.0-rc1',\
setvar:'tx.error_anomaly_score=4'"
SecRule TX:DETECTION_PARANOIA_LEVEL "@lt 1" "id:920011,phase:1,pass,nolog,skipAfter:END-REQUEST-920-PROTOCOL-ENFORCEMENT"
SecRule TX:DETECTION_PARANOIA_LEVEL "@lt 1" "id:920012,phase:2,pass,nolog,skipAfter:END-REQUEST-920-PROTOCOL-ENFORCEMENT"
SecRule REQUEST_HEADERS:Content-Length "!@rx ^\d+$" \
"id:920160,\
phase:1,\
block,\
t:none,\
tag:'paranoia-level/2',\
severity:'CRITICAL',\
setvar:'tx.inbound_anomaly_score_pl1=+%{tx.error_anomaly_score}'"
SecRule REQUEST_HEADERS:Content-Length "!@rx ^\d+$" \
"id:920161,\
phase:1,\
block,\
t:none,\
tag:'paranoia-level/1',\
setvar:'tx.inbound_anomaly_score_pl1=+%{tx.error_anomaly_score}'"
SecRule REQUEST_HEADERS:Content-Length "!@rx ^\d+$" \
"id:920162,\
phase:1,\
block,\
t:none,\
tag:'paranoia-level/1',\
severity:'CRITICAL',\
setvar:'tx.inbound_anomaly_score_pl2=+%{tx.critical_anomaly_score}'"
SecMarker "END-REQUEST-920-PROTOCOL-ENFORCEMENT"
In this rule file, there are more problems:
- rule 920160 is activated on PL1, but the
tag
value is PL2 - at rule 920160, the TX variable gets error_anomaly_score, but the severity is CRITICAL
- at rule 920161 there is no severity action
- rule 920162 increments anomaly_score_pl2, but it's in PL1
crs-linter -r examples/test8.conf
Config file: examples/test8.conf
Parsing ok.
Checking parsed rules...
examples/test8.conf
Ignore case check ok.
Action order check ok.
Indentation check ok.
'ctl:auditLogParts' actions are in right place.
no duplicate id's
Found incorrect paranoia-level/N tag(s)
file=examples/test8.conf, line=34, endLine=34, title=wrong or missing paranoia-level/N tag: tag 'paranoia-level/2' on PL 1, rule id: 920160
Found incorrect (inbound|outbout)_anomaly_score value(s)
file=examples/test8.conf, line=36, endLine=36, title=wrong (inbound|outbout)_anomaly_score variable or value: invalid value for anomaly_score_pl1: tx.error_anomaly_score with severity critical, rule id: 920160
file=examples/test8.conf, line=44, endLine=44, title=wrong (inbound|outbout)_anomaly_score variable or value: missing severity action, rule id: 920161
file=examples/test8.conf, line=53, endLine=53, title=wrong (inbound|outbout)_anomaly_score variable or value: variable inbound_anomaly_score_pl2 on PL 1, rule id: 920162
There are one or more unset TX variables.
file=examples/test8.conf, line=53, endLine=53, title=unset TX variable: TX variable 'critical_anomaly_score' not set / later set (rvar) in rule 920162
End of checking parsed rules
Cumulated report about unused TX variables
No unused TX variable
$ echo $?
1
SecRule TX:foo "@rx bar" \
"id:1001,\
phase:1,\
pass,\
nolog"
SecRule ARGS "@rx ^.*$" \
"id:1002,\
phase:1,\
pass,\
nolog,\
setvar:tx.bar=1"
In this rule file, there are more problems:
- rule 1001 used an uninitialized variable (
TX:foo
) - rule 1002 sets a TX variable which never used
SecRule ARGS "@rx (?i)foo" \
"id:1,\
phase:1,\
pass,\
t:lowercase,\
nolog"
Rule 1 uses a combination of t:lowercase and the (?i) in the regex
crs-linter -r examples/test10.conf
Config file: examples/test10.conf
Parsing ok.
Checking parsed rules...
examples/test10.conf
Ignore case check ok.
Action order check ok.
Indentation check ok.
no 'ctl:auditLogParts' action found.
no duplicate id's
paranoia-level tags are correct.
PL anomaly_scores are correct.
All TX variables are set.
No new tags added.
There are one or more combinations of t:lowercase and (?i) flag.
file=examples/test10.conf, line=5, endLine=5, title=t:lowercase and (?i): rule uses (?i) in combination with t:lowercase: 'lowercase'; rule id: 1
End of checking parsed rules
Cumulated report about unused TX variables
No unused TX variable
# no tag with OWASP_CRS
SecRule REQUEST_URI "@rx index.php" \
"id:1,\
phase:1,\
deny,\
t:none,\
nolog,\
tag:attack-xss"
Rule 1 does not have tag:OWASP_CRS
crs-linter -r examples/test11.conf -t ../APPROVED_TAGS
Config file: examples/test11.conf
Parsing ok.
Checking parsed rules...
examples/test11.conf
Ignore case check ok.
Action order check ok.
Indentation check ok.
no 'ctl:auditLogParts' action found.
no duplicate id's
paranoia-level tags are correct.
PL anomaly_scores are correct.
All TX variables are set.
No new tags added.
No t:lowercase and (?i) flag used.
There are one or more rules without OWASP_CRS tag.
file=examples/test11.conf, line=8, endLine=8, title=tag:OWASP_CRS is missing: rule does not have tag with value 'OWASP_CRS'; rule id: 1
There are one or more rules without ver action.
file=examples/test11.conf, line=8, endLine=8, title=ver is missing / incorrect: rule does not have 'ver' action; rule id: 1
End of checking parsed rules
Cumulated report about unused TX variables
No unused TX variable
# no 'ver' action
SecRule REQUEST_URI "@rx index.php" \
"id:1,\
phase:1,\
deny,\
t:none,\
nolog,\
tag:OWASP_CRS"
# 'ver' action has invalid value
SecRule REQUEST_URI "@rx index.php" \
"id:1,\
phase:1,\
deny,\
t:none,\
nolog,\
tag:OWASP_CRS,\
ver:OWASP_CRS/1.0.0-dev"
Rule 1 does not have ver
.
Rule 2 has incorrect ver
value.
crs-linter -r examples/test12.conf -t ../APPROVED_TAGS
Config file: examples/test12.conf
Parsing ok.
Checking parsed rules...
examples/test12.conf
Ignore case check ok.
Action order check ok.
Indentation check ok.
no 'ctl:auditLogParts' action found.
no duplicate id's
paranoia-level tags are correct.
PL anomaly_scores are correct.
All TX variables are set.
No new tags added.
No t:lowercase and (?i) flag used.
No rule without OWASP_CRS tag.
There are one or more rules without ver action.
file=examples/test12.conf, line=8, endLine=8, title=ver is missing / incorrect: rule does not have 'ver' action; rule id: 1
file=examples/test12.conf, line=18, endLine=18, title=ver is missing / incorrect: rule's 'ver' action has incorrect value; rule id: 2, version: 'OWASP_CRS/1.0.0-dev', expected: 'OWASP_CRS/4.6.0-dev'
End of checking parsed rules
Cumulated report about unused TX variables
No unused TX variable
# no need 'capture' action because the TX:1, but there is no chain action
SecRule ARGS "@rx TX:1" \
"id:1,\
phase:2,\
deny,\
t:none,\
nolog,\
tag:OWASP_CRS,\
ver:'OWASP_CRS/4.7.0-dev'"
# normal use
SecRule ARGS "@rx attack" \
"id:2,\
phase:2,\
deny,\
capture,\
t:none,\
nolog,\
tag:OWASP_CRS,\
ver:'OWASP_CRS/4.7.0-dev',\
chain"
SecRule TX:1 "@eq attack"
# invalid use
SecRule ARGS "@rx attack" \
"id:3,\
phase:2,\
deny,\
t:none,\
nolog,\
tag:OWASP_CRS,\
ver:'OWASP_CRS/4.7.0-dev',\
chain"
SecRule TX:0 "@eq attack"
Rule 1 is a "regular" rule, it can use TX:1
without any restriction.
Rule 2 is the valid form.
Rule 3 is a chained rule and it uses TX:0
in second rule, but first rule does not have capture
.
crs-linter -r examples/test13.conf -t ../APPROVED_TAGS -v "4.7.0-dev"
Config file: examples/test13.conf
Parsing ok.
Checking parsed rules...
examples/test13.conf
Ignore case check ok.
Action order check ok.
Indentation check ok.
no 'ctl:auditLogParts' action found.
no duplicate id's
paranoia-level tags are correct.
PL anomaly_scores are correct.
All TX variables are set.
No new tags added.
No t:lowercase and (?i) flag used.
No rule without OWASP_CRS tag.
No rule without correct ver action.
There are one or more rules using TX.N without capture action.
file=examples/test13.conf, line=34, endLine=34, title=capture is missing: rule uses TX.N without capture; rule id: 3'
End of checking parsed rules
Cumulated report about unused TX variables
No unused TX variable