Skip to content
Snippets Groups Projects
Forked from CompileNix / docker-nginx-http3
This fork has diverged from the upstream repository.

What is this?

A smol Nginx container image, plus:

The nginx binary is built from source (using fedora) into a FROM scratch container image.

Project Links

Supported Container Image Tags

  • 1.27.2, 1.27, 1, latest
  • 1.27.2-extras, 1.27-extras, 1-extras, latest-extras

Container Image Variants

Default (1.27.2, 1.27, 1, latest)

This container image has all features and modules built-in that will suit most regular use-cases, while maintaining a low image size.

The following notable features are available:

The following nginx modules are NOT available:

  • --without-http_empty_gif_module
  • --without-http_geo_module
  • --without-http_grpc_module
  • --without-http_memcached_module
  • --without-http_mirror_module
  • --without-http_scgi_module
  • --without-http_ssi_module
  • --without-http_uwsgi_module
  • --without-mail_imap_module
  • --without-mail_pop3_module
  • --without-mail_smtp_module
  • --without-stream_geo_module

Extras (1.27.2-extras, 1.27-extras, 1-extras, latest-extras)

This container image has all features and modules built-in, that I was able to build, without the focus on low image size.

The following notable features are available:

  • Everything that is also part of the Default image variant
  • Nginx Perl Module
  • --with-http_dav_module
  • --with-http_gunzip_module
  • --with-http_image_filter_module
  • --with-http_random_index_module
  • --with-http_secure_link_module
  • --with-http_slice_module
  • --with-http_xslt_module
  • --with-mail
  • --with-stream_geoip_module

Table of Contents

How to use this image

Also have a look at the following sites to view some additional examples and guidance on how to use nginx:

Hosting some simple static content on port 80 and 443

docker run -v "/some/content:/var/www/html:ro,z" -p 80:80 -p 443:443 compilenix/nginx

Provide Your Own Nginx Configuration

You can either add or replace individual nginx config files to the existing default configs by mapping them as a volume to /config into the container. For example would a mapped config file to /config/nginx.conf replace the default nginx.conf that is present in the container image.

For information on the syntax of nginx configuration files, see the official documentation and the Beginner's Guide.

Add A New HTTP Server Config Example

sites/domain.tld.conf:

# vim: sw=2 et filetype=nginx

server {
  listen 80;
  server_name domain.tld www.domain.tld;

  location / {
    return 307 https://www.domain.tld$request_uri;
  }
}

server {
  listen 443 ssl;
  http2 on;
  server_name domain.tld www.domain.tld;
  root '/var/www/domain.tld';

  include 'cfg/header_referrer_policy.conf';
  ssl_certificate 'ssl/domain.tld/fullchain.pem';
  ssl_certificate_key 'ssl/domain.tld/privkey.pem';

  location / {
    allow all;
  }
}

Remember to place the required ssl certificates into ./ssl and add the website content to ./webroot.

docker run \
  -v "$PWD/sites/domain.tld.conf:/config/sites/domain.tld.conf:ro,z" \
  -v "$PWD/ssl:/config/ssl/domain.tld:ro,z" \
  -v "$PWD/webroot:/var/www/domain.tld:ro,z" \
  -p 80:80 \
  -p 443:443 \
  compilenix/nginx

Using environment variables in nginx configuration

Environment variables can be used in any nginx config file, regardless if they are mapped as a volume, are part of the original container image or copied into custom container images which are based on this one.

Environment variable substitution / template-ing is performed by 900-envsubst-on-templates.sh on container start.

Mapped nginx config files directly to /etc/nginx (instead of /config) will be overwritten / process by this substitution process!

Only config files whose name ends with a certain suffix will be processed.

This behavior can be changed via the NGINX_ENVSUBST_TEMPLATE_SUFFIX environment variable. The default value is defined as ".conf".

Here is an example using docker-compose.yml:

# vim: sw=2 et

services:
  nginx:
    image: compilenix/nginx
    volumes:
      - "./domain.tld.conf:/config/sites/domain.tld.conf:ro,z"
      - "./ssl:/config/ssl/domain.tld:ro,z"
      - "./webroot:/var/www/html:ro,z"
    ports:
      - "80:80"
      - "443:443"
    environment:
      NGINX_HOST: "domain.tld"
      NGINX_HOST_CERT_PATH: "/etc/nginx/ssl/domain.tld"
      NGINX_HTTP_REDIRECT_LOCATION: "https://domain.tld$$request_uri"

Also have a look at the corresponding default nginx http server config: default.conf.

Running nginx using different UID and GID

Set the USER_ID and GROUP_ID env variables to the desired uid and gid. The entrypoint 110-add-user.sh will create a corresponding nginx user and group at container start.

Defaults are:

USER_NAME=nginx
USER_ID=101
GROUP_NAME=nginx
GROUP_ID=101

NJS

  1. Override njs.conf via /config/njs.conf.
  2. Map / copy njs code into container or image.

Example njs.conf:

# vim: sw=2 et filetype=nginx

js_path '/etc/nginx/njs/';
js_import http.js;

See also:

Docker Compose Example:

# vim: sw=2 et

services:
  nginx:
    image: compilenix/nginx:${NGINX_VERSION}
    volumes:
      - "./njs/njs.conf:/config/njs.conf:ro,z"
      - "./njs/http.js:/config/njs/http.js:ro,z"
      - "./njs/localhost.conf:/config/sites/localhost.conf:ro,z"
    ports:
      - "0.0.0.0:42661:80"
      - "0.0.0.0:42663:443"

Testing:

curl -vk 'https://127.0.0.1:42663/njs'
# Hello world from nginx njs!

Nginx config test

This can be accomplished by creating and starting a new container with the same parameters, environment variables and mapped volumes as the currently running container and by overriding the start command.

Example using docker run:

source .env
docker run \
  -it \
  --rm \
  --env-file .env \
  -v "$PWD/webroot:/var/www/html:ro,z" \
  -v "/some/nginx/config:/config:ro,z" \
  compilenix/nginx \
    /usr/bin/nginx -t

Set a custom timezone

Simply set the environment variable TZ to the desired timezone.

Example:

TZ="Europe/Berlin"

Default value: UTC

Set a custom amount of nginx worker processes

Set the environment variable NGINX_WORKER_PROCESSES to the desired amount.

Example:

NGINX_WORKER_PROCESSES=4

Default value: auto

Set a custom nginx http server response header value

Set the environment variable NGINX_SERVER_HEADER to the desired value.

Example:

NGINX_SERVER_HEADER="nginx"

Default value: (empty string)

Change the DNS resolver nginx will use

Set the environment variable DNS_RESOLVER to the desired ip address.

This dns server wll be used by nginx to perform dns lookups for dns based upstream targets and OCSP stapling queries.

Example:

DNS_RESOLVER="8.8.8.8"

Default value: 1.1.1.1

Log the current nginx version and buildinfo on container start

Set the environment variable NGINX_LOG_INFO_ON_START to the desired value.

The following values are valid:

  • false (or any other value): show nothing
  • version: show version which translates to nginx -v
  • buildinfo: show version and configure options which translates to nginx -V

Example:

NGINX_LOG_INFO_ON_START="version"

Default value: buildinfo

Nginx Access Log Format

There are two built-in log formats configured:

  • main
  • json

Switch the default nginx log format to the json log format by setting the environment variable NGINX_LOG_FORMAT_NAME.

Example:

NGINX_LOG_FORMAT_NAME="json"

Default value: main

main

log_format main '[$time_iso8601] status:$status domain:$host port:$server_port request_time:$request_time upstream_response_time:$upstream_response_time request_length:$request_length bytes_sent:$body_bytes_sent client_ip:$remote_addr request:"$request" referer:"$http_referer" user_agent:"$http_user_agent"';

Example "main" Log Message

[2022-09-27T20:50:58+02:00] status:200 domain:127.0.0.1 port:443 request_time:0.000 upstream_response_time:- request_length:26 bytes_sent:25 client_ip:172.18.0.1 request:"GET /test.html HTTP/2.0" referer:"-" user_agent:"Mozilla/5.0 (X11; Linux x86_64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/105.0.0.0 Safari/537.36"

GROK Pattern

upstream_response_time, user_agent and referer are omitted if the value is equal to "-".

\[%{TIMESTAMP_ISO8601:time_local:date}\] status:%{INT:status:short} domain:%{HOSTNAME:domain:text} port:%{INT:port:integer} request_time:%{NUMBER:request_time:float} upstream_response_time:(-|%{NUMBER:upstream_response_time:float}) request_length:%{INT:request_length:integer} bytes_sent:%{INT:bytes_sent:integer} client_ip:%{IP:client_ip:ip} request:\"%{WORD:method:text} %{DATA:request:text} %{DATA:http_protocol_version:text}\" referer:\"(-|%{DATA:referer:text})\" user_agent:\"(-|%{DATA:user_agent:text})\"

GROK Pattern Example result

{
  "bytes_sent": 20,
  "client_ip": "172.17.0.1",
  "domain": "127.0.0.1",
  "http_protocol_version": "HTTP/2.0",
  "method": "GET",
  "port": 443,
  "request": "/test.html",
  "request_length": 39,
  "request_time": 0,
  "status": 200,
  "time_local": "2023-04-05T11:53:34+02:00",
  "user_agent": "curl/7.85.0"
}

JSON

This log format contains the following properties:

  • body_bytes_sent
  • brotli_ratio
  • bytes_sent
  • connection
  • connection_requests
  • connection_time
  • content_length
  • content_type
  • document_root
  • fastcgi_path_info
  • fastcgi_script_name
  • gzip_ratio
  • host
  • hostname
  • http2
  • http_host
  • http_name
  • http_referer
  • http_user_agent
  • http_x_forwarded_for
  • https
  • invalid_referer
  • limit_conn_status
  • limit_rate
  • limit_req_status
  • msec
  • nginx_version
  • pid
  • pipe
  • proxy_add_x_forwarded_for
  • proxy_host
  • proxy_port
  • query_string
  • realip_remote_addr
  • realip_remote_port
  • realpath_root
  • remote_addr
  • remote_port
  • remote_user
  • request
  • request_body
  • request_body_file
  • request_completion
  • request_filename
  • request_id
  • request_length
  • request_method
  • request_time
  • request_uri
  • scheme
  • server_name
  • server_port
  • server_protocol
  • ssl_alpn_protocol
  • ssl_cipher
  • ssl_ciphers
  • ssl_client_escaped_cert
  • ssl_client_fingerprint
  • ssl_client_i_dn
  • ssl_client_raw_cert
  • ssl_client_s_dn
  • ssl_client_serial
  • ssl_client_v_end
  • ssl_client_v_remain
  • ssl_client_v_start
  • ssl_client_verify
  • ssl_curve
  • ssl_curves
  • ssl_early_data
  • ssl_protocol
  • ssl_server_name
  • ssl_session_id
  • ssl_session_reused
  • status
  • tcpinfo_rcv_space
  • tcpinfo_rtt
  • tcpinfo_rttvar
  • tcpinfo_snd_cwnd
  • time_iso8601
  • time_local
  • uid_got
  • uid_reset
  • uid_set
  • upstream
  • upstream_addr
  • upstream_bytes_received
  • upstream_bytes_sent
  • upstream_cache_status
  • upstream_connect_time
  • upstream_header_time
  • upstream_response_length
  • upstream_response_time
  • upstream_status
  • uri

binary_remote_addr is intentionally not included because the nginx built-in json escape system does weird stuff.

There are a couple more values you could add, see here ./src/etc/nginx/nginx.conf#L23.

sent_http_name

Include a arbitrary response header field; the last part of a variable name is the field name converted to lower case with dashes replaced by underscores.

Example: sent_http_x_powered_by

upstream_http_name

Include a arbitrary server response header fields. For example, the "Server" response header field is available through the

upstream_http_server variable. The rules of converting header field names to variable names are the same as for the variables that start with the "
http_" prefix. Only the header fields from the response of the last server are saved.

Example: sent_http_server

cookie_name

Include a arbitrary Cookie with the specified name sent by this server in the "Set-Cookie" response header field (1.7.1).

Example: cookie_sid

upstream_cookie_name

Include a arbitrary Cookie with the specified name sent by the upstream server in the "Set-Cookie" response header field (1.7.1). Only the cookies from the response of the last server are saved.

Example: upstream_cookie_sid

http3

Negotiated protocol identifier: "h3" for HTTP/3 over TLS or an empty string.

Example: http3

Available variables as part of the commercial subscription

This list might be incomplete.

session_log_binary_id

Current session ID in binary form (16 bytes).

Example: session_log_binary_id

session_log_id

Current session ID;

Example: session_log_id

upstream_queue_time

Keeps time the request spent in the upstream queue (1.13.9); the time is kept in seconds with millisecond resolution. Times of several responses are separated by commas and colons like addresses in the $upstream_addr variable.

Example: upstream_queue_time

Example JSON Log Message

{
  "body_bytes_sent": "25",
  "brotli_ratio": "",
  "bytes_sent": "260",
  "connection": "3",
  "connection_requests": "3",
  "connection_time": "6.019",
  "content_length": "",
  "content_type": "",
  "document_root": "/var/www/html",
  "fastcgi_path_info": "",
  "fastcgi_script_name": "/test.html",
  "gzip_ratio": "",
  "host": "127.0.0.1",
  "hostname": "76656407ead8",
  "http_host": "127.0.0.1:42662",
  "http_name": "",
  "http_referer": "",
  "http_user_agent": "Mozilla/5.0 (X11; Linux x86_64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/105.0.0.0 Safari/537.36",
  "http_x_forwarded_for": "",
  "http2": "h2",
  "https": "on",
  "invalid_referer": "",
  "limit_conn_status": "",
  "limit_rate": "0",
  "limit_req_status": "",
  "msec": "1664306240.619",
  "nginx_version": "1.27.2",
  "pid": "46",
  "pipe": ".",
  "proxy_add_x_forwarded_for": "172.18.0.1",
  "proxy_host": "",
  "proxy_port": "",
  "query_string": "",
  "realip_remote_addr": "172.18.0.1",
  "realip_remote_port": "41190",
  "realpath_root": "/var/www/html",
  "remote_addr": "172.18.0.1",
  "remote_port": "41190",
  "remote_user": "",
  "request": "GET /test.html HTTP/2.0",
  "request_body": "",
  "request_body_file": "",
  "request_completion": "OK",
  "request_filename": "/var/www/html/test.html",
  "request_id": "035c89c2558c2048bab75b655fecebd3",
  "request_length": "26",
  "request_method": "GET",
  "request_time": "0.000",
  "request_uri": "/test.html",
  "scheme": "https",
  "server_name": "localhost",
  "server_port": "2443",
  "server_protocol": "HTTP/2.0",
  "ssl_alpn_protocol": "h2",
  "ssl_cipher": "TLS_AES_256_GCM_SHA384",
  "ssl_ciphers": "0x2a2a:TLS_AES_128_GCM_SHA256:TLS_AES_256_GCM_SHA384:TLS_CHACHA20_POLY1305_SHA256: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:ECDHE-RSA-AES128-SHA:ECDHE-RSA-AES256-SHA:AES128-GCM-SHA256:AES256-GCM-SHA384:AES128-SHA:AES256-SHA",
  "ssl_client_escaped_cert": "",
  "ssl_client_fingerprint": "",
  "ssl_client_i_dn": "",
  "ssl_client_raw_cert": "",
  "ssl_client_s_dn": "",
  "ssl_client_serial": "",
  "ssl_client_v_end": "",
  "ssl_client_v_remain": "",
  "ssl_client_v_start": "",
  "ssl_client_verify": "NONE",
  "ssl_curve": "X25519",
  "ssl_curves": "0xfafa:X25519:prime256v1:secp384r1",
  "ssl_early_data": "",
  "ssl_protocol": "TLSv1.3",
  "ssl_server_name": "",
  "ssl_session_id": "0bd1ef5889b3b13db90eb56f616f3f450e17f651acb63bf3a75696fe945c5da8",
  "ssl_session_reused": ".",
  "status": "200",
  "tcpinfo_rcv_space": "14600",
  "tcpinfo_rtt": "7915",
  "tcpinfo_rttvar": "14084",
  "tcpinfo_snd_cwnd": "10",
  "time_iso8601": "2022-09-27T21:17:20+02:00",
  "time_local": "27/Sep/2022:21:17:20 +0200",
  "uid_got": "",
  "uid_reset": "",
  "uid_set": "",
  "upstream": "",
  "upstream_addr": "",
  "upstream_bytes_received": "",
  "upstream_bytes_sent": "",
  "upstream_cache_status": "",
  "upstream_connect_time": "",
  "upstream_header_time": "",
  "upstream_response_length": "",
  "upstream_response_time": "",
  "upstream_status": "",
  "uri": "/test.html"
}

Building

Build Requirements

  • docker
  • git
  • internet connection (HTTP/S)
  • openssl

Build Steps

git clone https://git.compilenix.org/CompileNix/docker-nginx
cd docker-nginx
cp example.env .env
$EDITOR .env
./tools/build-with-logs.sh ./docker/latest.Dockerfile
./tools/build-with-logs.sh ./docker/latest-extras.Dockerfile extras
./tools/tests.sh

Making Updates & Changes

If you want to change any versions used to build the container image take a look into .env.

Checklist

  • Update or create .env:
    • cp example.env .env
  • Run ./tools/build-with-logs.sh ./docker/latest.Dockerfile
  • Run ./tools/build-with-logs.sh ./docker/latest-extras.Dockerfile extras
  • Run ./tools/tests.sh
  • Upload build logs (printed out at the end of the build-with-logs.sh command)
  • Update Supported Container Image Tags
  • Update CHANGELOG.md
  • Remove Old Docker Image Tags
  • Create / Update Docker Image Tags
    • docker image tag "${IMAGE_NAME}:${NGINX_VERSION}" "${IMAGE_NAME}:latest"
      docker image tag "${IMAGE_NAME}:${NGINX_VERSION}" "${IMAGE_NAME}:1.27"
      docker image tag "${IMAGE_NAME}:${NGINX_VERSION}" "${IMAGE_NAME}:1"
      docker image tag "${IMAGE_NAME}:${NGINX_VERSION}-extras" "${IMAGE_NAME}:latest-extras"
      docker image tag "${IMAGE_NAME}:${NGINX_VERSION}-extras" "${IMAGE_NAME}:1.27-extras"
      docker image tag "${IMAGE_NAME}:${NGINX_VERSION}-extras" "${IMAGE_NAME}:1-extras"
      # inspect image tags
      docker image ls "${IMAGE_NAME}"
  • Run ./tools/push-image-tags.sh
  • Update Supported Container Image Tags on hub.docker.com