This document revisits the essential security configurations for Nginx HTTP Server from a practical perspective.
By following the key recommendations outlined below, you can avoid common configuration errors and prevent security vulnerabilities.
- 1. Ensure Proper Permissions for Nginx Process Account
- 2. Ensure Nginx Version Information is Not Exposed
- 3. Ensure the autoindex Feature is removed or disabled
- 4. Set Proper Permissions for Critical Directories and Files
- 5. Disabling Unused HTTP Methods
- 6. Ensure Removal of Insecure and Outdated SSL Protocols and Ciphers
- 7. Ensure Proper Permissions on SSL Certificate Files
- 8. Ensure Blocking of Arbitrary Host Connections Not Configured in the Web Server
- 9. Limit Simultaneous Connections per IP
- 10. Limit Requests per IP
- 11. Inspect and Control Request Headers
- 12. Utilizing the Location Block
Nginx HTTP Server typically runs under the default nginx
, www-data
user account (or another non-privileged account).
It is important to ensure that Nginx is not running under a privileged account, such as root
. If it is, it must be changed to a non-privileged user to prevent potential security risks.
Audit:
- Verify the current user account under which Nginx is running, execute the following command:
[root@localhost ~]# ps -ef | grep nginx root 626653 1 0 Apr08 ? 00:00:00 nginx: master process /usr/sbin/nginx nginx 626654 626653 0 Apr08 ? 00:00:00 nginx: worker process nginx 626655 626653 0 Apr08 ? 00:00:01 nginx: worker process nginx 626656 626653 0 Apr08 ? 00:00:00 nginx: worker process nginx 626657 626653 0 Apr08 ? 00:00:00 nginx: worker process
Remediation:
- If Nginx is running under the
root
account or any other account with elevated privileges, modify the configuration to run under a non-privileged account such asnginx
,nobody
,daemon
:[root@localhost ~]# vim /etc/nginx/nginx.conf ... user nginx; # <== # <== Set to a non-privileged account e.g. nginx, nobody, daemon ...
(1) Check sudo privileges:
- The Nginx process account must not have sudo privileges:
[root@localhost ~]# sudo -l -U nginx User nginx is not allowed to run sudo on pg-sec-mqrelay01
(2) Verify shell access:
- The Nginx process account should not have access to a login shell:
[root@localhost ~]# cat /etc/passwd | grep -i nginx nginx:x:498:498:Nginx web server:/var/lib/nginx:/sbin/nologin
(3) Ensure password is locked:
- The password for the Nginx process account should be locked:
[root@localhost ~]# passwd -S nginx nginx LK 2023-10-16 -1 -1 -1 -1 (Password locked.)
By default, Nginx HTTP Server displays the version information in the HTTP response header. This can provide attackers with insights into potential vulnerabilities. Hiding the version information mitigates this risk.
Audit:
- Verify that Nginx version information is exposed in the HTTP response header:
[root@localhost ~]# curl -i -k http://127.0.0.1 HTTP/1.1 200 Server: nginx/1.26.1 ...
Remediation:
- To prevent the version information from being exposed, modify the following settings in the Nginx configuration:
[root@localhost ~]# vim /etc/nginx/nginx.conf http { ... server_tokens off; # <== Set to hide version information ...
The autoindex feature provides directory listing functionality. Directory listing can expose critical information about the web application and server, and should be disabled by default.
Audit:
- Verify whether directory listing is displayed or the autoindex feature is enabled.
[root@localhost ~]# vim /etc/nginx/nginx.conf ... location / { root /usr/local/nginx/html/html; autoindex on; index index.html index.htm; }
Remediation:
- To disable the autoindex feature, remove or comment out the autoindex directive in the Nginx configuration:
[root@localhost ~]# vim /etc/nginx/nginx.conf ... location / { root /usr/local/nginx/html/html; autoindex on; # <== Remove or comment out index index.html index.htm; # <== Remove if necessary }
Ensure that the Nginx installation directory and configuration files are accessible only to the "root" user. The permissions for these directories and files should be checked and modified if any other users have access rights.
Audit:
- Verify the permissions of Nginx's installation directory and configuration files.
[root@localhost ~]# ls -al /etc/nginx/ total 88 drwxr-xr-x. 4 root root 4096 Aug 12 11:28 . drwxr-xr-x. 87 root root 8192 Aug 6 11:46 .. drwxr-xr-x. 4 root root 68 Aug 12 11:06 conf.d drwxr-xr-x. 2 root root 6 Jun 10 2021 default.d -rw-r--r--. 1 root root 1077 Jun 10 2021 fastcgi.conf -rw-r--r--. 1 root root 1077 Jun 10 2021 fastcgi.conf.default -rw-r--r--. 1 root root 1007 Jun 10 2021 fastcgi_params -rw-r--r--. 1 root root 1007 Jun 10 2021 fastcgi_params.default -rw-r--r--. 1 root root 2837 Jun 10 2021 koi-utf -rw-r--r--. 1 root root 2223 Jun 10 2021 koi-win -rw-r--r--. 1 root root 5170 Jun 10 2021 mime.types -rw-r--r--. 1 root root 5170 Jun 10 2021 mime.types.default -rw-r--r--. 1 root root 927 Oct 16 2023 nginx.conf -rw-r--r--. 1 root root 2656 Jun 10 2021 nginx.conf.default -rw-r--r--. 1 root root 636 Jun 10 2021 scgi_params -rw-r--r--. 1 root root 636 Jun 10 2021 scgi_params.default -rw-r--r--. 1 root root 664 Jun 10 2021 uwsgi_params -rw-r--r--. 1 root root 664 Jun 10 2021 uwsgi_params.default -rw-r--r--. 1 root root 3610 Jun 10 2021 win-utf
Remediation:
- Ensure that all Nginx directories and configuration files are owned by root:root.
- Adjust the permissions so that others (non-owners) cannot access the directories and files.
[root@localhost ~]# chown root:root -R /etc/nginx [root@localhost ~]# chmod o-rwx -R /etc/nginx
When unused HTTP methods are enabled on a web service, they can expose vulnerabilities or unintended behaviors that may lead to abuse and become an attack vector.
It is important to disable any methods that are not in use, particularly the TRACE
method, which is used for debugging and is generally unnecessary for normal web service operations.
HTTP Methods:
Name | Description |
---|---|
OPTIONS | Used to check the list of supported methods by the web server. |
HEAD | The server sends only the header information in the response. It is often used to check the existence of a page by receiving the HTTP status code. |
GET | Similar to POST, but does not handle form input; typically used to retrieve information such as lists in forums via URI transmission. |
POST | Used when sending data from a client to a web application for processing. |
PUT | Similar to POST, used to save content specified by the client to the server. |
DELETE | Used by the client to delete files on the server. |
TRACE | Used to invoke a loopback message on the web server to trace the data transmission path. |
CONNECT | Used to request proxy functionality from the web server. |
Audit:
- Verify that which HTTP methods are available on your web server:
[root@localhost ~]# curl -i -k -X OPTIONS http://example.com HTTP/1.1 200 OK Server: nginx Date: Mon, 12 Aug 2024 03:00:29 GMT Content-Type: text/html; charset=utf-8 Content-Length: 0 Connection: keep-alive Allow: GET, HEAD, OPTIONS, PUT, DELETE, POST, PATCH ...
Remediation:
- Disable TRACE Method
- Enable Only Necessary HTTP Methods:
[root@localhost ~]# vim /etc/nginx/nginx.conf ... server { ... # Allow only specified methods (e.g., HEAD, GET, POST) if ($request_method !~ ^(HEAD|GET|POST)$ ){ return 403; } } ...
Note: The $request_method directive can be applied at the server or location block level as needed.
When configuring SSL, outdated and insecure protocols or algorithms should be avoided.
Major browsers like Chrome, Microsoft Edge, and Firefox have dropped support for TLS 1.0 and TLS 1.1 due to security vulnerabilities such as POODLE and BEAST. These older protocols are known to be vulnerable to various attacks.
No. | CVE-ID | Vulnerability Name | Description |
---|---|---|---|
1 | CVE-2014-3566 | POODLE (Padding Oracle On Downgraded Legacy Encryption) | Exploits outdated encryption methods, enabling protocol downgrade attacks. |
2 | CVE-2011-3389 | BEAST (Browser Exploit Against SSL/TLS) | Allows attackers to decrypt HTTPS cookies and hijack sessions. |
Deprecated Cipher Algorithms
- The following cipher algorithms are considered insecure and should not be used:
- RC4
- DES
- 3DES
- MD5
- aNULL
- eNULL
- EXPORT
Audit:
- Verify the SSL settings for
ssl_protocols
andssl_ciphers
configurations.[root@localhost ~]# vim /etc/nginx/nginx.conf ... server { listen 443 ssl; server_name example.com; ... ssl_protocols TLSv1 TLSv1.1 TLSv1.2; ssl_ciphers HIGH:MEDIUM:aNULL:MD5:EXPORT:RC4:eNULL:DES:3DES; }
Remediation:
ssl_protocols
should use TLS 1.2 or later.ssl_ciphers
to utilize secure algorithms supported in TLS 1.2 or later.
Recommended Configuration (General Use):
- For general services (not bound by PCI-DSS), use the following configuration:
ssl_protocols TLSv1.2 TLSv1.3; ssl_ciphers ECDHE-ECDSA-AES128-GCM-SHA256:ECDHE-RSA-AES128-GCM-SHA256:ECDHE-ECDSA-AES256-GCM-SHA384:ECDHE-RSA-AES256-GCM-SHA384:ECDHE-ECDSA-CHACHA20-POLY1305:ECDHE-RSA-CHACHA20-POLY1305:DHE-RSA-AES128-GCM-SHA256:DHE-RSA-AES256-GCM-SHA384:DHE-RSA-CHACHA20-POLY1305;
- After configuration, verify the SSL security status using SSL Labs' SSL Test.
- Refer to Mozilla's SSL Configuration Generator for specific configurations based on your web server and OpenSSL version.
In cases where a web service vulnerability allows remote access to web server permissions, attackers may attempt to access SSL certificate files, potentially leading to the exposure of the private key. If the SSL certificate's private key is leaked, an attacker could create a spoofed website that appears legitimate or use the key to distribute malware. This becomes especially dangerous when SSL certificate pinning is used, or if the certificate is a wildcard certificate, as it could necessitate replacing certificates across all servers and clients.
Ensure that SSL certificate files are secured by verifying and applying proper file permissions.
Audit:
- Verify the file ownership and permissions for the
ssl_certificate
andssl_certificate_key
files.[root@localhost ~]# vim /etc/nginx/nginx.conf ... server { listen 443 ssl; server_name example.com; ... ssl_certificate /etc/nginx/conf.d/cert/example.com_ssl.crt; ssl_certificate_key /etc/nginx/conf.d/cert/example.com_ssl.key; ... } [root@localhost ~]# ls -al /etc/nginx/conf.d/cert total 12 drwxr-x---. 2 root root 64 Apr 8 14:47 . drwxr-x---. 4 root root 68 Aug 12 11:06 .. -rw-r--r--. 1 nobody root 7869 Apr 8 14:45 example.com_ssl.crt -rw-r--r--. 1 nobody root 1679 Apr 8 14:44 example.com_ssl.key
Remediation:
- Set the ownership of certificate and key files to
root:root
. - Adjust the file permissions to
600
or400
to prevent leaked.[root@localhost ~]# chown root:root -R /etc/nginx/conf.d/cert/example.com_ssl.crt [root@localhost ~]# chown root:root -R /etc/nginx/conf.d/cert/example.com_ssl.key [root@localhost ~]# chmod 400 /etc/nginx/conf.d/cert/example.com_ssl.crt [root@localhost ~]# chmod 400 /etc/nginx/conf.d/cert/example.com_ssl.key
In a typical web server configuration, if someone knows the server's IP address, they might still be able to access the web service without knowing the valid hostname (domain). By isolating and blocking access to requests made to non-service hosts, you can reduce unnecessary connections that shouldn't lead to your service, thereby reducing server load and improving performance.
Blocking such arbitrary connections also enhances security by preventing various types of web service attacks, such as host header injection and IP address-based scanning.
Ensure that your web server is configured to only respond to requests made to the correct service host, while blocking all others.
Audit:
- Verify that unknown host requests are accepting when accessing the web server.
(Vulnerable) Web server accepting connections with unknown host requests:
[root@localhost ~]# curl -k -v https://192.168.130.23 -H 'Host: invalid.host.com'
* Trying 192.168.130.23:443...
* Connected to 192.168.130.23 (192.168.130.23) port 443
* ALPN: curl offers h2,http/1.1
* TLSv1.3 (OUT), TLS handshake, Client hello (1):
* TLSv1.3 (IN), TLS handshake, Server hello (2):
* TLSv1.2 (IN), TLS handshake, Certificate (11):
* TLSv1.2 (IN), TLS handshake, Server key exchange (12):
* TLSv1.2 (IN), TLS handshake, Server finished (14):
* TLSv1.2 (OUT), TLS handshake, Client key exchange (16):
* TLSv1.2 (OUT), TLS change cipher, Change cipher spec (1):
* TLSv1.2 (OUT), TLS handshake, Finished (20):
* TLSv1.2 (IN), TLS handshake, Finished (20):
* SSL connection using TLSv1.2 / ECDHE-RSA-AES256-GCM-SHA384 / prime256v1 / rsaEncryption
* ALPN: server did not agree on a protocol. Uses default.
* Server certificate:
* subject: CN=*.example.com
* start date: Jan 2 00:00:00 2024 GMT
* expire date: Jan 20 23:59:59 2025 GMT
* issuer: C=GB; ST=Greater Manchester; L=Salford; O=Sectigo Limited; CN=Sectigo RSA Domain Validation Secure Server CA
* SSL certificate verify result: unable to get local issuer certificate (20), continuing anyway.
* Certificate level 0: Public key type RSA (2048/112 Bits/secBits), signed using sha256WithRSAEncryption
* Certificate level 1: Public key type RSA (2048/112 Bits/secBits), signed using sha384WithRSAEncryption
* Certificate level 2: Public key type RSA (4096/152 Bits/secBits), signed using sha384WithRSAEncryption
* using HTTP/1.x
> GET / HTTP/1.1
> Host: invalid.host.com
> User-Agent: curl/8.5.0
> Accept: */*
>
< HTTP/1.1 200 OK
< Date: Mon, 12 Aug 2024 05:35:40 GMT
< Server: Nginx
< Cache-Control: no-cache, no-store, must-revalidate
< Expires: Thu, 01 Jan 1970 09:00:00 KST
...
(Not Vulnerable) Web server not accepting connections with unknown host requests:
[root@localhost ~]# curl -k -v https://192.168.130.236 -H 'Host: invalid.host.com'
* Trying 192.168.130.236:443...
* Connected to 192.168.130.236 (192.168.130.236) port 443 (#0)
* ALPN: offers h2
* ALPN: offers http/1.1
* (304) (OUT), TLS handshake, Client hello (1):
* (304) (IN), TLS handshake, Server hello (2):
* (304) (IN), TLS handshake, Unknown (8):
* (304) (IN), TLS handshake, Certificate (11):
* (304) (IN), TLS handshake, CERT verify (15):
* (304) (IN), TLS handshake, Finished (20):
* (304) (OUT), TLS handshake, Finished (20):
* SSL connection using TLSv1.3 / AEAD-CHACHA20-POLY1305-SHA256
* ALPN: server accepted http/1.1
* Server certificate:
* subject: CN=*.example.com
* start date: Jan 2 00:00:00 2024 GMT
* expire date: Jan 20 23:59:59 2025 GMT
* issuer: C=GB; ST=Greater Manchester; L=Salford; O=Sectigo Limited; CN=Sectigo RSA Domain Validation Secure Server CA
* SSL certificate verify result: unable to get local issuer certificate (20), continuing anyway.
* Certificate level 0: Public key type RSA (2048/112 Bits/secBits), signed using sha256WithRSAEncryption
* Certificate level 1: Public key type RSA (2048/112 Bits/secBits), signed using sha384WithRSAEncryption
* Certificate level 2: Public key type RSA (4096/152 Bits/secBits), signed using sha384WithRSAEncryption
* using HTTP/1.x
> GET / HTTP/1.1
> Host: invalid.host.com
> User-Agent: curl/7.84.0
> Accept: */*
>
* Mark bundle as not supporting multiuse
< HTTP/1.1 403 Forbidden
< Server: Nginx
< Date: Mon, 12 Aug 2024 05:32:49 GMT
< Content-Type: text/html
< Content-Length: 162
< Connection: keep-alive
<
<html>
<head><title>403 Forbidden</title></head>
<body bgcolor="white">
<center><h1>403 Forbidden</h1></center>
<hr><center>Nginx</center>
</body>
</html>
* Connection #0 to host 192.168.130.236 left intact
Remediation:
- Create a virtual host that processes and blocks requests with host headers that do not match the intended service hosts.
[root@localhost ~]# vim /etc/nginx/nginx.conf ... # VirtualHost to handle and block requests with unmatched host headers server { listen 80 default_server; listen 443 default_server ssl; error_log /var/log/nginx/http.unknown-header.error.log; access_log /var/log/nginx/http.unknown-header.access.log main; ssl_certificate /etc/nginx/conf.d/cert/example.com_ssl.crt; ssl_certificate_key /etc/nginx/conf.d/cert/example.com_ssl.key; ssl_session_cache shared:SSL:1m; ssl_session_timeout 10m; ssl_protocols TLSv1.2 TLSv1.3; ssl_ciphers ECDHE-ECDSA-AES128-GCM-SHA256:ECDHE-RSA-AES128-GCM-SHA256:ECDHE-ECDSA-AES256-GCM-SHA384:ECDHE-RSA-AES256-GCM-SHA384:ECDHE-ECDSA-CHACHA20-POLY1305:ECDHE-RSA-CHACHA20-POLY1305:DHE-RSA-AES128-GCM-SHA256:DHE-RSA-AES256-GCM-SHA384:DHE-RSA-CHACHA20-POLY1305; location / { deny all; } } # VirtualHost for the valid service hostname (example.com) server { listen 80; server_name example.com; location / { return 301 https://example.com$request_uri; } ... } server { listen 443 ssl; server_name example.com; error_log /var/log/nginx/example.com.error.log; access_log /var/log/nginx/example.access.log main; ssl_certificate /etc/nginx/conf.d/cert/example.com_ssl.crt; ssl_certificate_key /etc/nginx/conf.d/cert/example.com_ssl.key; ssl_session_cache shared:SSL:1m; ssl_session_timeout 10m; ssl_protocols TLSv1.2 TLSv1.3; ssl_ciphers ECDHE-ECDSA-AES128-GCM-SHA256:ECDHE-RSA-AES128-GCM-SHA256:ECDHE-ECDSA-AES256-GCM-SHA384:ECDHE-RSA-AES256-GCM-SHA384:ECDHE-ECDSA-CHACHA20-POLY1305:ECDHE-RSA-CHACHA20-POLY1305:DHE-RSA-AES128-GCM-SHA256:DHE-RSA-AES256-GCM-SHA384:DHE-RSA-CHACHA20-POLY1305; ... }
Notes:
- Ensuring that requests with incorrect host header are blocked is a critical part of web server security.
Many web servers are not configured this way by default
, making this an essential configuration for system administrators to apply.
Limiting the number of simultaneous connections from a single IP address is a critical step in preventing resource exhaustion and ensuring the availability of services. By implementing connection limits, you can mitigate the risk of slow denial-of-service (DoS) attacks and maintain stable web server performance.
Audit:
- Verifying Simultaneous Connections
- To audit the current simultaneous connections per IP, you can monitor server logs or analyze connection counts using tools like Nginx’s built-in logging or external monitoring systems.
Remediation:
- Configuring Connection Limits
(1) Define a Shared Memory Zone
- Use the limit_conn_zone directive in the http block to create a shared memory zone that tracks connections by IP. Example:
http { limit_conn_zone $binary_remote_addr zone=limitperip:10m; ... }
Memory Allocation:
10MB can store approximately 160,000 unique IPs.Tracking Mechanism:
$binary_remote_addr tracks client IPs efficiently.
(2) Apply Connection Limits
- Use the limit_conn directive to enforce restrictions at the server or location level. Examples:
server { listen 443 ssl; server_name auth.example.com; # Allow a maximum of 10 simultaneous connections per IP limit_conn limitperip 10; } server { listen 443 ssl; server_name download.example.com; location /download/ { # Restrict simultaneous connections to 3 per IP for this location limit_conn limitperip 3; limit_conn_status 429; # Return HTTP 429 (Too Many Requests) } }
Notes:
- The
ngx_http_limit_conn_module
module must be enabled. - Fine-tune memory allocation and limits based on your server’s capacity and traffic profile.
- The configuration ensures that excessive connections from a single IP do not degrade server performance or accessibility for other users.
By enforcing simultaneous connection limits, you can protect your server from potential abuse and maintain service quality under high-load scenarios.
Rate limiting per IP address in Nginx helps maintain stable web services by controlling the number of requests a client can make within a given timeframe. This configuration prevents abuse by users generating excessive requests and protects the server from being overwhelmed.
Audit:
- Verifying Request Limits
- To check if rate-limiting is configured:
- Review the Nginx configuration file for
limit_req_zone
andlimit_req
directives. - Analyze the server logs to identify patterns of excessive requests or potential abuse.
- Review the Nginx configuration file for
Remediation:
- Configuring Rate Limits
(1) Define a Shared Memory Zone
- Use the
limit_req_zone
directive to allocate a shared memory zone that tracks request rates by IP. Example:http { limit_req_zone $binary_remote_addr zone=ratelimit:10m rate=5r/s; ... }
Memory Allocation
: 10MB can store approximately 160,000 unique IPs.Rate:
Limits each IP to 5 requests per second (5r/s
).
(2) Apply Request Limits
- Use the
limit_req
directive in thelocation
block to enforce the rate limits. Example:server { listen 443 ssl; server_name auth.example.com; location / { limit_req zone=ratelimit burst=10 nodelay; limit_req_status 429; # Return HTTP 429 (Too Many Requests) } }
Burst:
Allows up to 10 requests to be queued during traffic spikes.nodelay:
Ensures all requests in the burst are processed immediately without delay.
Notes:
- The
ngx_http_limit_conn_module
module must be enabled. - Adjust the rate and burst settings to align with your application's traffic profile.
- Rate limits help maintain consistent performance by preventing abuse while accommodating occasional bursts of traffic.
By implementing IP-based rate limits, you can protect your server from misuse and ensure a reliable experience for legitimate users.
Web servers can inspect and control HTTP request headers to determine the presence or value of specific headers. This capability enables conditional access, such as blocking specific clients or permitting predefined user agents (e.g., mobile apps). It is commonly applied for:
- Blocking known attack tools or abusive scanning software.
- Allowing access only to predefined clients or bots.
- Securing APIs by validating authorization headers.
Audit:
- Verifying Header Restrictions is applied
- Review the Nginx configuration for if directives that check headers like User-Agent or Authorization.
- Check server logs for unauthorized requests and verify appropriate error responses are sent.
Remediation:
- Configuring Header Restrictions
(1) Block Known Vulnerability Scanners
- Example configuration to block common web vulnerability scanning tools:
server { listen 443 ssl; server_name download.example.com; # Block requests with known attack tool User-Agent strings if ($http_user_agent ~* "Paros|ZmEu|nikto|dirbuster|sqlmap|openvas|w3af|Morfeus|Zollard|Arachni|Brutus|bsqlbf|Grendel-Scan|Havij|Hydra|N-Stealth|Netsparker|Pangolin|pmafind|webinspect") { return 404; # Return HTTP 404 Not Found } }
(2) Restrict Specific Headers for Certain URLs
- Example to enforce valid authorization headers for API access:
server { listen 443 ssl; server_name download.example.com; location /api { proxy_http_version 1.1; # Only allow requests with Authorization header starting with "Bearer" if ($http_authorization !~ "^Bearer") { return 401; # Return HTTP 401 Unauthorized } proxy_pass http://app:3000/; } }
(3) Allow Only Specific Clients
- Example to permit access to predefined User-Agent strings:
http { if ($http_user_agent !~ "Agent.MyApp") { return 404; # Return HTTP 404 Not Found } } server { listen 443 ssl; server_name auth.example.com; } server { listen 443 ssl; server_name download.example.com; }
Notes:
- The
$http_
variables in Nginx provide access to HTTP headers. - Regular expressions (
~
or~*
) are used to match patterns in headers. - Use conditional blocks (
if
) judiciously to avoid unintended behavior or performance issues. - Return appropriate HTTP status codes (e.g.,
404
,401
) to signal blocked or unauthorized access.
The location block in Nginx is used to define how specific URIs are handled. Each virtual host can use multiple location blocks, which can be configured with different priorities. This allows fine-grained control over request handling.
Basic Usage
- The syntax for a
location
block is as follows:location [modifier] [URI] { ... }
Matching Priorities
Priority | Modifier | Description | Example |
---|---|---|---|
1 | = |
Matches exact URIs. | location = /favicon.ico { # do something } |
2 | ^~ |
Matches URIs with a preferred prefix. | location ^~ /images/ { # do something } |
3 | ~ |
Matches URIs using case-sensitive regex. | location ~ \.php$ { # do something } |
4 | ~* |
Matches URIs using case-insensitive regex. | location ~* \.(jpg|jpeg|png)$ { # do something } |
5 | / |
Matches all other URIs (fallback). | location /docs/ { # do something } |
Remediation:
- Example Use Cases of the location Block
(1) Serve Specific File for Dynamic URLs
- Serve
/cafe-detail/index.html
for requests like/cafes/12345
location ~ ^/cafes/[1-9]+$ { try_files $uri /cafes/cafe-detail/index.html; }
(2) Block Access to Hidden(.Dot) Files
- Deny access to hidden files (starting with a dot):
location ~ /\. { deny all; return 404; }
(3) Serve Static Files with Caching
- Serve
css
andjs
files with a 30-day cache:location ~* \.(css|js)$ { root /var/www/static; expires 30d; }
(4) Restrict Access to Sensitive Files
- Block access to
.sql
and.bak
files:location ~* \.(sql|bak)$ { deny all; return 404; }
(5) Add Content-Type for API Responses
- Add a Content-Type header for
.json
or.xml
responses:location /api/ { location ~* \.(json|xml)$ { add_header Content-Type application/json; } }
(6) Redirect Mobile Users
- Redirect mobile users to a mobile-specific site version:
location / { set $mobile_rewrite do_not_perform; if ($http_user_agent ~* "(android|bb\d+|meego).+mobile|avantgo|bada\/|blackberry|blazer|compal|elaine|fennec|...)") { set $mobile_rewrite perform; } if ($mobile_rewrite = perform) { rewrite ^ /mobile$uri redirect; } }
(7) Enable File Downloads
- Set response headers for file downloads:
location ~* ^/download/.*\.(zip|tar\.gz)$ { root /var/www/files; add_header Content-Disposition "attachment"; }
(8) Rate Limiting for Login Attempts
- Limit request rates for WordPress login pages:
location = /wp-login.php { limit_req zone=one burst=1 nodelay; include fastcgi_params; fastcgi_pass unix:/var/run/php/php7.4-fpm.sock; } location ~* /wp-admin/.*\.php$ { auth_basic "Restricted Access"; auth_basic_user_file /etc/nginx/.htpasswd; include fastcgi_params; fastcgi_pass unix:/var/run/php/php7.4-fpm.sock; }
(9) Rewrite API Versions
- Process API requests with version-specific rewrites:
location /api/ { rewrite ^/api/v1/(.*)$ /api.php?version=1&request=$1 break; rewrite ^/api/v2/(.*)$ /api.php?version=2&request=$1 break; }
(10) Redirect Old Blog URLs
- Rewrite legacy blog URLs to a new format:
location /blog/ { rewrite ^/blog/(\d{4})/(\d{2})/(.*)$ /articles/$1-$2-$3 permanent; }
(11) Disable Caching for Sensitive Pages
- Prevent caching for certain pages:
location ~ ^/payments/v[1-2]/daily/download { add_header Cache-Control 'no-cache, no-store, must-revalidate, proxy-revalidate, max-age=0'; add_header Pragma no-cache; expires 0; }
Using the location block effectively enhances web server performance, security, and user experience.