Skip to content

Latest commit

 

History

History
298 lines (211 loc) · 12.1 KB

README.md

File metadata and controls

298 lines (211 loc) · 12.1 KB

ngx_waf

build-master build-dev GitHub release (latest by date including pre-releases) GitHub

English | 简体中文

A web application firewall module for nginx.

Progress & Change Log

Function

  • Anti Challenge Collapsar(IPV4 only), it can automatically block malicious IP.
  • Exceptional allow on specific IP address(IPV4 only).
  • Block the specified IP address (IPV4 only).
  • Block the specified request body.
  • Exceptional allow on specific URL.
  • Block the specified URL.
  • Block the specified request args.
  • Block the specified UserAgent.
  • Block the specified Cookie.
  • Exceptional allow on specific Referer.
  • Block the specified Referer.

Install

On Unix Like

download the source code of nginx

If you want to add a new nginx module, you'll need the nginx source code

cd /usr/local/src
wget http://nginx.org/download/nginx-1.18.0.tar.gz
tar -zxf nginx-1.18.0.tar.gz

It is recommended to use nginx-1.18.0, otherwise this module may not work normally.

download the source code of ngx_waf

cd /usr/local/src
git clone https://github.com/ADD-SP/ngx_waf.git
cd ngx_waf
git clone -b v2.1.0 https://github.com/troydhanson/uthash.git inc/uthash

compile and install

Starting from nginx-1.9.11, nginx began to support dynamic modules.

Using static modules requires all modules to be compiled into binary files, so adding, deleting and updating modules requires recompiling nginx and replacing the old binary files.

Using dynamic modules only needs to load the .so at runtime, without recompiling the entire nginx. Just compile the module into a .so, and then edit nginx.conf to load the corresponding module.


use static modules

cd /usr/local/src/nginx-1.18.0
./configure xxx --add-module=/usr/local/src/ngx_waf
make

If you have already installed nginx, it is recommended to run nginx -V to get the compilation parameters, and then replace xxx with it.

nginx -s stop
mv /usr/local/nginx/sbin/nginx /usr/local/nginx/sbin/nginx.old
cp objs/nginx /usr/local/nginx/sbin/nginx
nginx

If you don’t want to stop the nginx service, you can upgrade through hot deployment, see Official Document for hot deployment method.

If nginx is not installed.

make install

use dynamic modules

cd /usr/local/src/nginx-1.18.0
./configure xxx --add-dynamic-module=/usr/local/src/ngx_waf
make modules

If you have already installed nginx, it is recommended to run nginx -V to get the compilation parameters, and then replace xxx with it.

Now you need to find a directory to store the .so file of the module, this doc assumes it is stored under /usr/local/nginx/modules

cp objs/ngx_http_waf_module.so /usr/local/nginx/modules/ngx_http_waf_module.so

Then edit nginx.conf and add a line at the top.

load_module "/usr/local/nginx/modules/ngx_http_waf_module.so";

How to use?

You should edit nginx.conf as follow.

http {
    ...
    server {
        ...
        waf on;
        waf_rule_path /usr/local/src/ngx_waf/rules/;
        waf_mode STD;
        waf_cc_deny_limit 1000 60;
        ...
    }
    ...
}

waf

  • syntax: waf [ on | off ];
  • default: off
  • context: server

Whether to enable this module.

waf_rule_path

  • syntax: waf_rule_path dir;
  • default: —
  • context: server

The directory where the rule file is located, must end with /.

waf_mult_mount

  • syntax: waf_mult_mount [ on | off ];
  • default: off
  • context: server

Multi-stage inspection. When address rewriting exists in nginx.conf (such as rewrite), it is recommended to enable it, otherwise it is recommended to disable it.

waf_mode

  • syntax: waf_mode mode_type ...;
  • default: —
  • context: server

Set the working mode of the firewall. Specify at least one mode and specify at most eight modes.

mode_type has the following values (not case sensitive):

  • GET: Start the inspection process when Http.Method=GET.
  • HEAD: Start the inspection process when Http.Method=HEAD.
  • POST: Start the inspection process when Http.Method=POST.
  • PUT: Start the inspection process when Http.Method=PUT.
  • DELETE: Start the inspection process when Http.Method=DELETE.
  • MKCOL: Start the check process when Http.Method=MKCOL.
  • COPY: Start the inspection process when Http.Method=COPY.
  • MOVE: Start the inspection process when Http.Method=MOVE.
  • OPTIONS: Start the inspection process when Http.Method=OPTIONS.
  • PROPFIN: Start the inspection process when Http.Method=PROPFIN.
  • PROPPATCH: Start the inspection process when Http.Method=PROPPATCH.
  • LOCK: Start the inspection process when Http.Method=LOCK.
  • UNLOCK: Start the inspection process when Http.Method=UNLOCK.
  • PATCH: Start the inspection process when Http.Method=PATCH.
  • TRAC: Start the inspection process when Http.Method=TRAC.
  • IP: Enable IP address inspecting rules.
  • URL: Enable URL inspecting rules.
  • RBODY: Enable request body inspecting rules.
  • ARGS: Enable ARGS inspecting rules.
  • UA: Enable UA inspecting rules.
  • COOKIE: Enable COOKIE inspecting rules.
  • REFERER: Enable REFERER inspecting rules.
  • CC: Enable 'Anti Challenge Collapsar'.
  • STD: Equivalent to GET POST CC IP URL ARGS RBODY UA.
  • FULL: In any case, the inspection process will be started and all inspection rules will be enabled.

Note: The mode of CC is independent of other modes, and whether it takes effect or not is not affected by other modes. A typical situation such as the URL mode will be affected by the GET mode, because if the GET mode is not used, the check will not be started when Http.Method=GET, and the URL will naturally not be inspected, but CC mode will not be similarly affected.

waf_cc_deny_limit

  • syntax: waf_cc_deny_limit rate duration;
  • default: —
  • context: server

Declare the maximum request rate of the same IP and the blocking time after the rate is exceeded.

rate: The maximum number of requests per minute for the same IP. duration: The number of minutes to block after exceeding the rate.

test

https://example.com/www.bak

If the http status code is 403, it means this module is working normally.

Rule file

All regular expressions follow the PCRE Standard.

The effective order of rule files (from top to bottom, the priority gradually decreases):

  • rules/white-ipv4:IPV4 whitelist, each rule has its own line. Each line can only be an IPV4 address or a CIDR address block. Allow matched IPV4 address.
  • rules/ipv4:IPV4 blacklist, each rule has its own line. Each line can only be an IPV4 address or a CIDR address block. Block the matched IPV4 address and return 403.
  • rules/white-url:URL whitelist, each rule has its own line. One regular expression per line, when the URL is matched by any rule, it will be allowed.
  • rules/url:URL blacklist, each rule has its own line. There is a regular expression per line, and 403 is returned when the URL is matched by any rule.
  • rules/args:Request args blacklist, each rule has its own line. There is one regular expression per line, and 403 is returned when the request args is matched by any rule.
  • rules/user-agent:UserAgent blacklist, each rule has its own line. There is one regular expression per line, and 403 is returned when the user-agent is matched by any rule.
  • rules/white-referer:Referer whitelist, each rule has its own line. One regular expression per line, when the referer is matched by any rule, it will be allowed.
  • rules/referer:Referer blacklist, each rule has its own line. There is a regular expression per line, and 403 is returned when the referer is matched by any rule.
  • rules/cookie:Cookie blacklist, each rule has its own line. There is a regular expression per line, and 403 is returned when the cookie is matched by any rule.
  • rules/post:Request body blacklist, each rule has its own line. There is a regular expression per line, and 403 is returned when the request body is matched by any rule.

Variable

When writing nginx.conf, some variables are inevitably needed. For example, $remote_addr can be used to get the client IP address.

This module adds three available variables.

  • $waf_blocked: Whether this request is intercepted by this module, if intercepted, its value is 'true', otherwise it is 'false'.
  • $waf_rule_type:If current request was blocked by this module, this variable is set to the triggered rule type, otherwise 'null'. The following are possible values.
    • 'BLACK-IPV4'
    • 'BLACK-URL'
    • 'BLACK-ARGS'
    • 'BLACK-USER-AGENT'
    • 'BLACK-REFERER'
    • 'BLACK-COOKIE'
    • 'BLACK-POST'
  • '$waf_rule_details':If this request is blocked by this module, its value is the content of the specific rule triggered. If it is not blocked, its value is 'null'.

Log

The block log is stored in error.log. The log level of the block record is ALERT. The basic format is xxxxx, ngx_waf: [Type][Rule], xxxxx, see the following example for details.

2020/01/20 22:56:30 [alert] 24289#0: *30 ngx_waf: [BLACK-URL][(?i)(?:/\.env$)], client: 192.168.1.1, server: example.com, request: "GET /v1/.env HTTP/1.1", host: "example.com", referrer: "http:/example.com/v1/.env"

2020/01/20 22:58:40 [alert] 24678#0: *11 ngx_waf: [BLACK-POST][(?i)(?:select.+(?:from|limit))], client: 192.168.1.1, server: example.com, request: "POST /xmlrpc.php HTTP/1.1", host: "example.com", referrer: "https://example.com/"

FAQ

Why does the request speed slow down for a while?

It may be because the 'Anti Challenge Collapsar' is enabled, see Performance-Memory Management for details.

Why not check the request body?

For performance reasons, this module will check whether it is in the memory before checking the request body. If it is, it will check normally, otherwise skip the check. You can try to edit nginx.conf.

http {
    ...
    # https://nginx.org/en/docs/http/ngx_http_core_module.html#client_body_buffer_size
    client_body_buffer_size: 10M;
    # https://nginx.org/en/docs/http/ngx_http_core_module.html#client_body_in_file_only
    client_body_in_file_only: off;
    ...
}

fork() failed while spawning "worker process" (12: Cannot allocate memory)

It may be caused by frequent run nginx -s reload. This module will allocate some memory when reading the nginx.conf, but some memory will not be free immediately after run nginx -s reload, so it is frequent in a short time running nginx -s reload will most likely cause this error.

You can kill all of nginx's processes and restart nginx.

Performance

Memory management

When the 'Anti Challenge Collapsar' enabled, this module will free some memory periodically and allocate some memory once, but it will not free all at once, but gradually free. Each request will release a small part of the memory until all the memory is free. Slow down processing time slightly.

Thanks

  • uthash: ngx_waf uses the source code of uthash v2.1.0. The uthash source code and open source license are located at inc/uthash/.
  • ngx_lua_waf: Most of the default rules of this module come from this.
  • nginx-book: Thanks for the tutorial provided by the author.
  • nginx-development-guide: Thanks for the tutorial provided by the author.