SpiderLabs / owasp-modsecurity-crs

OWASP ModSecurity Core Rule Set (CRS) Project (Official Repository)
https://modsecurity.org/crs
Apache License 2.0
2.45k stars 727 forks source link

Simple Base64 Post Forbidden #1043

Closed luizbiazus closed 6 years ago

luizbiazus commented 6 years ago

The rules are catching simple base64 uploads, it's not suppose to work without whitelist rules?


---7fZlosHV---A--
[16/Mar/2018:15:23:41 +0000] 152121382116.209189 127.0.0.1 41184 127.0.0.1 4443
---7fZlosHV---B--
POST /api/logo/cliente?cliente_id=11983 HTTP/1.0
User-Agent: Mozilla/5.0 (Macintosh; Intel Mac OS X 10_12_6) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/64.0.3282.186 Safari/537.36
X-Forwarded-Proto: https
Accept: application/json, text/plain, */*
Content-Length: 8524
Connection: close
Authorization: Bearer eyJ0eXAiOiJKV1QiLCJhbGciOiJIUzI1NiJ9.eyJzdWIiOjEsImlzcyI6Imh0dHBzOi8vYXBpLnB1bHNhcnBheS5jb20vYXBpL3VzdWFyaW8vbG9naW4iLCJpYXQiOjE1MjEyMTM4MDMsImV4cCI6MTUyMTIxNzQwMywibmJmIjoxNTIxMjEzODAzLCJqdGkiOiI1SGJFS29MaERyY1hBcHN3In0.QEo_2KI4ZVVZh1s1fmsbUgIxbNbd-JSh9iRa2NUp_Vo
Host: api.XXXXXXXXXXXXXX.com
Origin: http://localhost:4200
X-Forwarded-For: 186.208.81.157
X-Real-IP: 186.208.81.157
Content-Type: application/json
Referer: http://localhost:4200/clientes
Accept-Encoding: gzip, deflate, br
Accept-Language: pt-BR,pt;q=0.9,en-US;q=0.8,en;q=0.7

---7fZlosHV---D--

---7fZlosHV---F--
HTTP/1.0 302
Server: nginx/1.13.9
Date: Fri, 16 Mar 2018 15:23:41 GMT
Content-Length: 161
Content-Type: text/html
Connection: close
Location: https://site.XXXXXXXXXXXXXXX.org/public/error/forbidden_2.html

---7fZlosHV---H--
ModSecurity: Access denied with code 302 (phase 2). Matched "Operator `Rx' with parameter `(?i)[\s\S](?:x(?:link:href|html|mlns)|!ENTITY.*?SYSTEM|data:text\/html|pattern(?=.*?=)|formaction|\@import|base64)\b' against variable `ARGS:json.logo_64' (Value: ` (8410 characters omitted)' ) [file "/usr/local/nginx-modsecurity/conf/rules/REQUEST-941-APPLICATION-ATTACK-XSS.conf"] [line "130"] [id "941130"] [rev "2"] [msg "XSS Filter - Category 3: Attribute Vector"] [data "Matched Data: ;base64 found within ARGS:json.logo_64: "] [severity "2"] [ver "OWASP_CRS/3.0.0"] [maturity "1"] [accuracy "8"] [tag "Host: api.XXXXXXXXXXX.com"] [tag "application-multi"] [tag "language-multi"] [tag "platform-multi"] [tag "attack-xss"] [tag "OWASP_CRS/WEB_ATTACK/XSS"] [tag "WASCTC/WASC-8"] [tag "WASCTC/WASC-22"] [tag "OWASP_TOP_10/A3"] [tag "OWASP_AppSensor/IE1"] [tag "CAPEC-242"] [hostname "127.0.0.1"] [uri "/api/logo/cliente"] [unique_id "152121382116.209189"] [ref "o14,7v13,8510t:utf8toUnicode,t:urlDecodeUni,t:htmlEntityDecode,t:jsDecode,t:cssDecode,t:removeNulls"]
csanders-git commented 6 years ago

This is not the base64 that is triggering the rule, it's the data url passing base64. Here's what i mean.

https://regex101.com/r/D5ogMf/1

This is blocked because such encoding mechanisms (raw data url's in parameters) are frequently used to try and bypass WAFs, see the following: https://www.owasp.org/index.php/XSS_Filter_Evasion_Cheat_Sheet https://www.paladion.net/blogs/bypass-xss-filters-using-data-uris

To resolve the issue you'll first want to make sure that the image isn't included in a way that can cause XSS - be very careful here as depending on how it's used this very well could.

Once you are sure that the issue has been solved you can either add an exception for this parameter

SecUpdateTargetById 941130 !ARGS:json.logo_64

To be even more comfortable it probably makes sense to decode this value in a separate additional rule using base64decodeext action into an external var using setvar. Then you can add the new var to processing.

csanders-git commented 6 years ago

If there are other issues let me know.

spartantri commented 6 years ago

Hi @luizbiazus, at least add a rule to check ARGS:json.logo_64 header to be a PNGfile (and all other formats you support) add a t:base64decode and PNGfile header is \x89\x50\x4e\x47 if I have time this will be included soon but for files that are posted not encoded in arguments however if you modify 914240 from PR #994 as shown below you got the magic you need to do a partial positive validation and white list the affected argument.

Here you have a good starting point ;)

#Check if the file type matches the with the file signature
SecRule ARGS:json.logo_64 "@rx ^data:image/(?:jpeg|jpg|gif|png|bmp|image|ico);base64,(?:\x89\x50\x4e\x47|\xff\xd8\xff(?:\xe1|\xe0|\xfe)|\x47\x49\x46\x38(?:\x37|\x39)\x61|\x42\x4d(?:\xf8\xa9|\x62\x25|\x76\x03)|\x00\x00\x01\x00|\x52\x49\x46\x46)" \
    "id:914245,\
    phase:2,\
    pass,\
    log,\
    noauditlog,\
    msg:'Encoded image type match an image signature, white listing argument',\
    logdata:'Matched Data: %{TX.0} found within %{MATCHED_VAR_NAMES}: %{MATCHED_VARS}',\
    t:none,t:base64decode,\
    tag:'application-multi',\
    tag:'platform-multi',\
    tag:'paranoia-level/1',\
    rev:'1',\
    ver:'OWASP_CRS/3.1.0',\
    severity:'NOTICE',\
    setvar:'tx.msg=%{rule.msg}',\
    ctl:ruleRemoveTargetById=941130;ARGS:json.logo_64"

Good luck, do security not holes and post/contribute feedback ;)

luizbiazus commented 6 years ago

Hello Spartantri,

Thanks for the amazing solution. But i've got this issue:

nginx: [emerg] "modsecurity_rules_file" directive Rules error. File: /usr/local/nginx-modsecurity/conf/rules/whitelist.conf. Line: 24. Column: 85. Expecting a variable, got: : S}: %{MATCHED_VARS}',\ in /usr/local/nginx-modsecurity/conf/sites-enabled/vhost-api.xxxxxxxxx.com:18

can u help-me?

spartantri commented 6 years ago

Hi after a minor change here it is, it should work for your use case, in my setup I used a more generic argument name (logo) and tested with the below curl commands and the payload detection it works fine.

curl 'http://127.0.0.1:80/x?logo=image/png;base64,iVBORw0KGgoA'
curl 'http://127.0.0.1:80/x' -H 'application/json' -d 'logo=image/png;base64,iVBORw0KGgoA'

Notice that this rule is loaded before the rule it white lists 941130.

SecRule ARGS:json.logo_64 "@rx ^image/(?:jpeg|jpg|gif|png|bmp|image|ico);base64,([a-zA-Z0-9\-_]{12})" \
    "id:914245,\
    phase:2,\
    pass,\
    capture,\
    log,\
    noauditlog,\
    msg:'Encoded image type match an image signature, white listing argument',\
    logdata:'Captured header: %{TX.1}, Matched Data: %{TX.0} found within %{MATCHED_VAR_NAMES}: %{MATCHED_VARS}',\
    t:none,\
    tag:'application-multi',\
    tag:'platform-multi',\
    tag:'paranoia-level/2',\
    rev:'1',\
    ver:'OWASP_CRS/3.1.0',\
    severity:'NOTICE',\
    chain"
    SecRule TX:1 "@rx ^(?:\x89\x50\x4e\x47|\xff\xd8\xff(?:\xe1|\xe0|\xfe)|\x47\x49\x46\x38(?:\x37|\x39)\x61|\x42\x4d(?:\xf8\xa9|\x62\x25|\x76\x03)|\x00\x00\x01\x00|\x52\x49\x46\x46)" \
        "t:base64decode,\
        setvar:'tx.msg=%{rule.msg}',\
        ctl:ruleRemoveTargetById=941130;ARGS:json.logo_64"

Also note that I do not use json parsing in this example so I sent plain post args but due to the name of your argument I assume you use json parsing. To test json I use: curl 'http://127.0.0.1:80/x' -H 'application/json' -d '{"json":{"logo_64":"image/png;base64,iVBORw0KGgoA"}}' (or burp)

Switch to json body processor with rule: SecRule REQUEST_HEADERS:Content-Type ^application/json$ "phase:1,id:87,t:lowercase,nolog,pass,ctl:requestBodyProcessor=JSON"

spartantri commented 6 years ago

By the way for help requests use the distribution list, I replied to this just because I'm working in this particular file detection rules PR #1045

luizbiazus commented 6 years ago

Hello spartantri, i've tried and got same issue ... for some reason seens that is not setting the variables in this line: logdata:'Captured header: %{TX.1}, Matched Data: %{TX.0} found within %{MATCHED_VAR_NAMES}: %{MATCHED_VARS}',\

Mar 21 14:23:12 WAF-DDOS-01 nginx[13810]: nginx: [emerg] "modsecurity_rules_file" directive Rules error. File: /usr/local/nginx-modsecurity/conf/rules/whitelist.conf. Line: 28. Column: 111. Expecting a variable, got:  :  S}: %{MATCHED_VARS}',\ in /usr/local/nginx-modsecurity/conf/sites-enabled/vhost-api.xxxxxxxxxxxx.com:18
Mar 21 14:23:12 WAF-DDOS-01 nginx[13810]: nginx: configuration file /usr/local/nginx-modsecurity/conf/nginx.conf test failed
Mar 21 14:23:12 WAF-DDOS-01 systemd[1]: nginx-modsecurity.service: Control process exited, code=exited status=1
spartantri commented 6 years ago

That looks like an NGINIX modsec engine issue, I use Apache and it works as expected, try removing the entire logdata line and let me know if that was the problem, it maybe a problem with the variable %{MATCHED_VAR_NAMES}, this is not required for the rule to work at all, it is only for having some data output for troubleshooting in the log.

luizbiazus commented 6 years ago

Nginx started without that Line but issue remains...


---rftAxnMu---Z--

---w70oy2gH---A--
[21/Mar/2018:20:53:45 +0000] 152166562577.153305 127.0.0.1 43778 127.0.0.1 4443
---w70oy2gH---B--
POST /api/logo HTTP/1.0
User-Agent: Mozilla/5.0 (Macintosh; Intel Mac OS X 10_12_6) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/65.0.3325.162 Safari/537.36
X-Forwarded-Proto: https
Accept: application/json, text/plain, */*
Content-Length: 159504
Connection: close
Authorization: Bearer eyJ0eXAiOiJKV1QiLCJhbGciOiJIUzI1NiJ9.eyJzdWIiOjI1NywiaXNzIjoiaHR0cHM6Ly9hcGkucHVsc2FycGF5LmNvbS9hcGkvdXN1YXJpby9sb2dpbiIsImlhdCI6MTUyMTY2NTU5OCwiZXhwIjoxNTIxNjY5MTk4LCJuYmYiOjE1MjE2NjU1OTgsImp0aSI6IjZqTm5XTnRaVndiNzRNdngifQ.R9Px_7nDSJHm7Hsx-v7oboapV0tuiXuLz_Lfea-LB3Y
Host: api.XXXXXXXXXXXX.com
Origin: https://www2.XXXXXXXXX.com
X-Forwarded-For: 186.208.81.157
X-Real-IP: 186.208.81.157
Content-Type: application/json
Referer: https://www2.XXXXXXXX.com/configuracao
Accept-Encoding: gzip, deflate, br
Accept-Language: pt-BR,pt;q=0.9,en-US;q=0.8,en;q=0.7,es;q=0.6

---w70oy2gH---D--

---w70oy2gH---F--
HTTP/1.0 200
X-RateLimit-Remaining: 59
Server: nginx/1.13.9
X-RateLimit-Limit: 60
expires: -1
pragma: no-cache
Cache-Control: private, must-revalidate
Connection: close
Access-Control-Allow-Origin: *
Content-Type: application/json
Date: Wed, 21 Mar 2018 20:53:46 GMT
Vary: Origin

---w70oy2gH---H--
ModSecurity: Warning. Matched "Operator `Rx' with parameter `(?i)[\s\S](?:x(?:link:href|html|mlns)|!ENTITY.*?SYSTEM|data:text\/html|pattern(?=.*?=)|formaction|\@import|base64)\b' against variable `ARGS:json.logo_64' (Value: ` (159390 characters omitted)' ) [file "/usr/local/nginx-modsecurity/conf/rules/REQUEST-941-APPLICATION-ATTACK-XSS.conf"] [line "130"] [id "941130"] [rev "2"] [msg "XSS Filter - Category 3: Attribute Vector"] [data "Matched Data: ;base64 found within ARGS:json.logo_64: "] [severity "2"] [ver "OWASP_CRS/3.0.0"] [maturity "1"] [accuracy "8"] [tag "Host: api.XXXXXXXXXXX.com"] [tag "application-multi"] [tag "language-multi"] [tag "platform-multi"] [tag "attack-xss"] [tag "OWASP_CRS/WEB_ATTACK/XSS"] [tag "WASCTC/WASC-8"] [tag "WASCTC/WASC-22"] [tag "OWASP_TOP_10/A3"] [tag "OWASP_AppSensor/IE1"] [tag "CAPEC-242"] [hostname "127.0.0.1"] [uri "/api/logo"] [unique_id "152166596846.858825"] [ref "o14,7v13,145306t:utf8toUnicode,t:urlDecodeUni,t:htmlEntityDecode,t:jsDecode,t:cssDecode,t:removeNulls"]

---GhzY8Erb---I--

---GhzY8Erb---J--

---9bLkLQvS---A--
[21/Mar/2018:21:00:08 +0000] 152166600841.913440 127.0.0.1 44302 127.0.0.1 4443
---9bLkLQvS---B--
POST /api/logo HTTP/1.0
User-Agent: Mozilla/5.0 (Macintosh; Intel Mac OS X 10_12_6) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/65.0.3325.162 Safari/537.36
X-Forwarded-Proto: https
Accept: application/json, text/plain, */*
Content-Length: 145320
Connection: close
Authorization: Bearer eyJ0eXAiOiJKV1QiLCJhbGciOiJIUzI1NiJ9.eyJzdWIiOjI1NywiaXNzIjoiaHR0cHM6Ly9hcGkucHVsc2FycGF5LmNvbS9hcGkvdXN1YXJpby9sb2dpbiIsImlhdCI6MTUyMTY2NTU5OCwiZXhwIjoxNTIxNjY5MTk4LCJuYmYiOjE1MjE2NjU1OTgsImp0aSI6IjZqTm5XTnRaVndiNzRNdngifQ.R9Px_7nDSJHm7Hsx-v7oboapV0tuiXuLz_Lfea-LB3Y
Host: api.XXXXXXXXX.com
Origin: https://www2.XXXXXXXXX.com
X-Forwarded-For: 186.208.81.157
X-Real-IP: 186.208.81.157
Content-Type: application/json
Referer: https://www2.XXXXXXXXXXXX.com/configuracao
Accept-Encoding: gzip, deflate, br
Accept-Language: pt-BR,pt;q=0.9,en-US;q=0.8,en;q=0.7,es;q=0.6

---9bLkLQvS---D--

---9bLkLQvS---F--
HTTP/1.0 302
Server: nginx/1.13.9
Date: Wed, 21 Mar 2018 21:00:08 GMT
Content-Length: 161
Content-Type: text/html
Connection: close
Location: https://XXXXXXXXXX.bmsoftware.org/public/error/forbidden_2.html

---9bLkLQvS---H--
ModSecurity: Access denied with code 302 (phase 2). Matched "Operator `Rx' with parameter `(?i)[\s\S](?:x(?:link:href|html|mlns)|!ENTITY.*?SYSTEM|data:text\/html|pattern(?=.*?=)|formaction|\@import|base64)\b' against variable `ARGS:json.logo_64' (Value: ` (145206 characters omitted)' ) [file "/usr/local/nginx-modsecurity/conf/rules/REQUEST-941-APPLICATION-ATTACK-XSS.conf"] [line "130"] [id "941130"] [rev "2"] [msg "XSS Filter - Category 3: Attribute Vector"] [data "Matched Data: ;base64 found within ARGS:json.logo_64: "] [severity "2"] [ver "OWASP_CRS/3.0.0"] [maturity "1"] [accuracy "8"] [tag "Host: api.XXXXXXXXXXXX.com"] [tag "application-multi"] [tag "language-multi"] [tag "platform-multi"] [tag "attack-xss"] [tag "OWASP_CRS/WEB_ATTACK/XSS"] [tag "WASCTC/WASC-8"] [tag "WASCTC/WASC-22"] [tag "OWASP_TOP_10/A3"] [tag "OWASP_AppSensor/IE1"] [tag "CAPEC-242"] [hostname "127.0.0.1"] [uri "/api/logo"] [unique_id "152166624999.899350"] [ref "o14,7v13,152474t:utf8toUnicode,t:urlDecodeUni,t:htmlEntityDecode,t:jsDecode,t:cssDecode,t:removeNulls"]
spartantri commented 6 years ago

The rule loading order is important, you have reversed the order of the includes crs and this whitelisting rule, some exceptions are handled before loading the offending rules and some others are after. This rule should run before 941130.

luizbiazus commented 6 years ago

i've tried place the rule right over rule 941130 in REQUEST-941 file but the issue persists...

---kamIElfo---F--
HTTP/1.0 302
Server: nginx/1.13.9
Date: Thu, 22 Mar 2018 22:30:19 GMT
Content-Length: 161
Content-Type: text/html
Connection: close
Location: https://XXXXXXXXXXXX.org/public/error/forbidden_2.html

---kamIElfo---H--
ModSecurity: Access denied with code 302 (phase 2). Matched "Operator `Rx' with parameter `(?i)[\s\S](?:x(?:link:href|html|mlns)|!ENTITY.*?SYSTEM|data:text\/html|pattern(?=.*?=)|formaction|\@import|base64)\b' against variable `ARGS:json.logo_64' (Value: ` (28234 characters omitted)' ) [file "/usr/local/nginx-modsecurity/conf/rules/REQUEST-941-APPLICATION-ATTACK-XSS.conf"] [line "150"] [id "941130"] [rev "2"] [msg "XSS Filter - Category 3: Attribute Vector"] [data "Matched Data: ;base64 found within ARGS:json.logo_64: "] [severity "2"] [ver "OWASP_CRS/3.0.0"] [maturity "1"] [accuracy "8"] [tag "Host: api.XXXXXXXXXX.com"] [tag "application-multi"] [tag "language-multi"] [tag "platform-multi"] [tag "attack-xss"] [tag "OWASP_CRS/WEB_ATTACK/XSS"] [tag "WASCTC/WASC-8"] [tag "WASCTC/WASC-22"] [tag "OWASP_TOP_10/A3"] [tag "OWASP_AppSensor/IE1"] [tag "CAPEC-242"] [hostname "127.0.0.1"] [uri "/api/logo"] [unique_id "15217578191.995242"] [ref "o14,7v13,28334t:utf8toUnicode,t:urlDecodeUni,t:htmlEntityDecode,t:jsDecode,t:cssDecode,t:removeNulls"]
spartantri commented 6 years ago

The rule is may not be executing, chack the error log to confirm, if it is not ARGS:json.logo_64 in the request is not exactly image/png;base64,base64stuff, may you add log part C to see the full payload you send on the log not only the headers it is possible that you have an encoding and if that is the case the rule will need a transform to decode the payload.

Another simple approach to make it work blindly without checking what is inside the argument is using after the CRS rules SecRuleUpdateTargetById 941130 !ARGS:json.logo_64

luizbiazus commented 6 years ago

can i use ur REQUEST-914-FILE-DECTION.conf instead doing it?

spartantri commented 6 years ago

As it is, 914 is generic so you may have to add your specific argument name to the inspections or setup the location.

In that case you may also use the rule below to add your content to whatever rule that matches your need (replace the ??? for the rule id) SecRuleUpdateTargetById 914??? ARGS:json.logo_64

GRB099 commented 4 years ago

Hi, i am using modsecurity in nginx level and now i need to whitelist the some of special characters can anyone help me in this. Thank You.

dune73 commented 4 years ago

Hi @GRB099, this is the wrong place to ask for help. But you may want to look into the tutorial at https://www.netnea.com/cms/apache-tutorial-8_handling-false-positives-modsecurity-core-rule-set/ to figure it out yourself.