Configuration
Overview
This document explains how to use F5 NGINX Ingress Controller to configure NGINX App Protect WAF v5.
Note:
There are complete NGINX Ingress Controller with NGINX App Protect WAF example resources on GitHub.
F5 recommends recompiling your NGINX AppProtect WAF Policy Bundles with each release of NGINX Ingress Controller. This ensures Policies remain compatible and are compiled with the latest attack signatures, bot signatures, and Ttreat campaigns.
Global configuration
NGINX Ingress Controller has global configuration parameters that match those in NGINX App Protect WAF. They are found in the ConfigMap resource: the NGINX App Protect WAF parameters are prefixed with app-protect*
.
Enable NGINX App Protect WAF v5
NGINX App Protect WAF v5 can be enabled and configured for custom resources only(VirtualServer, VirtualServerRoute). You need to create a Policy Custom Resource referencing a policy bundle, then add it to the VirtualServer/VirtualServerRoute definition. Additional detail can be found in the Policy Resource documentation.
NGINX App Protect WAF Bundles
App Protect WAF bundles for VirtualServer custom resources are defined by creating policy bundles and putting them on a mounted volume accessible from NGINX Ingress Controller.
Before applying a policy, a WAF policy bundle must be created, then copied to a volume mounted to /etc/app_protect/bundles
.
Note:
NGINX Ingress Controller supportssecurityLogs
for policy bundles. Log bundles must also be copied to a volume mounted to/etc/app_protect/bundles
.
This example shows how a policy is configured by referencing a generated WAF Policy Bundle:
apiVersion: k8s.nginx.org/v1
kind: Policy
metadata:
name: <policy_name>
spec:
waf:
enable: true
apBundle: "<policy_bundle_name>.tgz"
This example shows the same policy as above but with a log bundle used for security log configuration:
apiVersion: k8s.nginx.org/v1
kind: Policy
metadata:
name: <policy_name>
spec:
waf:
enable: true
apBundle: "<policy_bundle_name>.tgz"
securityLogs:
- enable: true
apLogBundle: "<log_bundle_name>.tgz"
logDest: "syslog:server=syslog-svc.default:514"
Configure NGINX Plus Ingress Controller using Virtual Server resources
This example shows how to deploy NGINX Ingress Controller with NGINX Plus and NGINX App Protect WAF v5, deploy a simple web application, and then configure load balancing and WAF protection for that application using the VirtualServer resource.
Note:
You can find the files for this example on GitHub.
Prerequisites
-
Follow the installation instructions to deploy NGINX Ingress Controller with NGINX Plus and NGINX App Protect WAF version 5.
-
Save the public IP address of NGINX Ingress Controller into a shell variable:
IC_IP=XXX.YYY.ZZZ.III
-
Save the HTTP port of NGINX Ingress Controller into a shell variable:
IC_HTTP_PORT=<port number>
Deploy a web application
Create the application deployment and service:
kubectl apply -f https://raw.githubusercontent.com/nginx/kubernetes-ingress/v4.0.0/examples/custom-resources/app-protect-waf-v5/webapp.yaml
Create the Syslog service
Create the syslog service and pod for the NGINX App Protect WAF security logs:
kubectl apply -f https://raw.githubusercontent.com/nginx/kubernetes-ingress/v4.0.0/examples/custom-resources/app-protect-waf-v5/syslog.yaml
Deploy the WAF Policy
Note:
Configuration settings in the Policy resource enable WAF protection by configuring NGINX App Protect WAF with the log configuration created in the previous step. The policy bundle referenced asyour_policy_bundle_name.tgz
need to be created and placed in the/etc/app_protect/bundles
volume first.
Create and deploy the WAF policy.
kubectl apply -f https://raw.githubusercontent.com/nginx/kubernetes-ingress/v4.0.0/examples/custom-resources/app-protect-waf-v5/waf.yaml
Configure load balancing
Note:
VirtualServer references thewaf-policy
created in Step 3.
-
Create the VirtualServer Resource:
kubectl apply -f https://raw.githubusercontent.com/nginx/kubernetes-ingress/v4.0.0/examples/custom-resources/app-protect-waf-v5/virtual-server.yaml
Test the application
To access the application, curl the coffee and the tea services. Use the --resolve
option to set the Host header of a request with webapp.example.com
- Send a request to the application:
curl --resolve webapp.example.com:$IC_HTTP_PORT:$IC_IP http://webapp.example.com:$IC_HTTP_PORT/
Server address: 10.12.0.18:80
Server name: webapp-7586895968-r26zn
- Try to send a request with a suspicious URL:
curl --resolve webapp.example.com:$IC_HTTP_PORT:$IC_IP "http://webapp.example.com:$IC_HTTP_PORT/<script>"
<html><head><title>Request Rejected</title></head><body>
- Check the security logs in the syslog pod:
kubectl exec -it <SYSLOG_POD> -- cat /var/log/messages
Example VirtualServer configuration
The GitHub repository has a full VirtualServer example.
apiVersion: k8s.nginx.org/v1
kind: VirtualServer
metadata:
name: webapp
spec:
host: webapp.example.com
policies:
- name: waf-policy
upstreams:
- name: webapp
service: webapp-svc
port: 80
routes:
- path: /
action:
pass: webapp