- Goal
- Context / Motivation
- How to start
- Additional configuration features
- Solution description
- Troubleshooting
Nginx-Reverse-Proxy project is modern and simple alternative to Nginx-Proxy-Manager. Zero knowledge of nginx or SSL required.
- To have Nginx reverse-proxy for multiple aplication hosted in local network
- Accessible from the WWW by domain name using HTTPS with autorefresh (letsencrypt & certbot)
- Able to work with dynamic public IP (AWS Route53)
- Able to work without ISP NAT-loopback, i.e no access to forwarded domain from local network
- Dockerized and configured by single config file -
nrp.yaml
- Platform -
Mac M1
&Linux
. It should work on Windows, but was not tested - No Static Public IP (I don't have it and don't want to pay €14/mo just for it to Vodafone)
- ISP
NAT loopback
not available, i.e local device can't access services using its public IP address from within the same network. It means that one can't accessyour-domain.tld
proxied service from the same network- If one have public static IP and NAT-loopback the live will be just easier :-)
- Docker setup
-
Install
docker
(usingcolima
ordocker desktop
wrappers) -
Check if
docker
default context uses proper socketcolima status # will output socket path docker context ls #check for default (*) context socket link # if it shows error or socket path is wrong, then update it via DOCKER_HOST vars, e.g. for zshrc echo "export DOCKER_HOST=unix:<path-to-socket>/docker.sock" >> ~/.zshrc # restart shell docker context ls # check socket path again
-
Note: Nginx-Reverse-Proxy = NRP
-
Copy
.env.dist
to.env
and set the values- AWS related values make sense to use only if you plan to use automatic public IP check & update.
- If you plan to use Route53 IP automatic updates for your current public IP, it is recommended to emit credentials with limited to Route53 access only via AWS IAM.
-
Set desired sub/domain(e.g.
your.domain.tld
)A
record pointing to the ISP public IP (portchecker can help to check public IP and open ports) -
Setup port forwarding for NRP on your router:
- Local server network IP, e.g.
192.168.0.111
TCP/UDP 192.168.0.111 :80 → :80
TCP/UDP 192.168.0.111 :443 → :443
- Local server network IP, e.g.
-
Add services to
nrp.yaml
-
Assume your services executed on the same host as nginx, and host has local IP =
192.168.0.111
. Also you had configuredservice1.domain.tld
&service1.domain.tld
redirect to your ISP public IP. And your services listening ports9000
and9001
respectfuly -
Create
nrp.yaml
file in root repo folder, e.g.:letsencrypt: email: "you-name@gmail.com" dryRun: false services: - name: service1 # plain HTTP serviceIp: 192.168.0.111 servicePort: 9000 # your service port domainName: service1.domain.tld cors: true - name: service2 # with HTTPS serviceIp: 192.168.0.111 servicePort: 9100 # your service port domainName: service2.domain.tld cors: true https: use: true force: true hsts: true
Notes:
- use your real email for SSL certificates
- additionaly you can hide sensitive info using env vars (read more in nrp-cli "Adding new service" doc section)
-
-
Start NRP by
make docker-restart
nrp-cli
internally will created necessarynginx
config files and startcertbot
if https needed (read more about config schema options in nrp-cli docs)dnsmasq
andsquid
config wil be generated according tonrp.yaml
certbot
will run every 1d to renew certificates viacron
- test API services will be started using "tuiteraz/fastify-tmpl" docker image
Note: at this point described services should be available locally using their domains in browser
-
Other network clients access to proxied services
- If you added domains in
/etc/hosts
you do not need to usesquid
proxy on that particular host. Otherwise, go to host System Settings → Network → Wi-Fi (or your selected active connection) → Advanced → Proxies.- Add
<local-host-ip>
withport=<SQUID_PORT>
as proxy(e.g.proxy=192.168.0.111, port=3128
).
- Add
- Do the same for Wi-Fi clients, which need to access
your-domain.tld
from local network - Now all local and Wi-Fi DNS and HTTP/S requests will be proxied through the
squid
. Which will resolve your proxied domains viadnsmasq
to local NRP host. Magic!
- If you added domains in
-
In case when no static public IP available, then
public-ip
check can be enabled by config option (check details in nrp-cli doc):public-ip: checkAndUpdate: yes schedule: 1h # default, can be omited dryRun: no # default, can be omited
-
In case you don't need
squid
&dnsmasq
to proxy local requests (check details in nrp-cli doc):squid: use: no useDnsmasq: no
-
In case of possible DNS issues one can enabel dnsmasq logging:
dnsmasq: logs: yes
- Make changes, create PR, merge it.
- Then "release please" will pick it up
- On a new tag "Build Docker image and push to Docker Hub" actions will do the build and publication
With NAT loopback disabled by the ISP, local network devices cannot access the proxied service using the your-domain.tld
name. There are various options available for solving this problem, which are described below for educational purposes. The current solution utilizes a non-root docker-compose
approach with squid
and dnsmasq
. Also, custom image of cadvisor
used since there is no official arm64
version(by 3/10/23).
Also recommended to read nrp-cli docs for more details on config preparation.
Here is the deployment diagram for the solution:
Description:
- External user flow
- (1) - External internet user makes request to
your-domain.tld
- (2) - DNS (AWS in this case) resolve it to the ISP Public IP
- (3) - When request reaches local network ingress, router it redirects it to local NRPs
:80
or:443
ports. Then it resolves to configured local network service according to NRP configuration
- (1) - External internet user makes request to
- Local user flow
- (4) - User from the same residential network make request to the
your-domain.tld
from his browser. And his network already configured to use proxy. Which in fact isSquid
proxy server running in the docker-compose configuration with NRP - (5) -
Squid
configured to resolve domains usingdnsmasq
. Which is aware ofyour-domain.tld
to be forwarded to NRPmulti-proxy
network IP. - (6) - When domain name IP resolved,
squid
proxied requests tonginx
- (4) - User from the same residential network make request to the
Notes
- This is known tradeoff - with
squid
proxy &dnsmasq
enabled on your local devices, all their DNS and HTTP traffic will go through thesquid
&dnsmasq
on host machine. On "Macbook Pro M1" it utilizes ~0.02 cores
-
There is cAdvisor on port
:4082
present in docker-compose in order to monitor resource utilization -
Locally
dig
andnslookup
will resolve your domain to Public IP, but it's okay, because every HTTP/S request will be processed bysquid <-> dnsmasq -> nginx-proxy-manager
when made from local network with proxy setting configured
Described options leaves considerable system footprint, hardly scalable or requires significant manual effort:
Option #1 - when you don't need to access domain locally frequently - use Brave Tor
To access your reverse-proxy resource by domain name you need to access it from different internet connection (if your ISP doesn’t support NAT loopback)
- Open “New Private Window with Tor” (Brave)
- Connect via mobile hotspot from other device
- Use Android “HTTP shortcuts” app with mobile connection (disabled WiFi)
Option #2 - when few hosts in local network need access - update /etc/hosts
Or you can you local domain forward by adding your domain and IP address to the /etc/hosts
file. You may have to use sudo or editor.
echo "127.0.0.1 sub.<your-domain>.com" >> /etc/hosts
dscacheutil -flushcache # Flush the DNS cache for the changes to take effect
Option #3 - when Wi-Fi hosts or many hosts need acces via domain name - use dnsmasq + squid
Setup dnsmasq
-
brew install dnsmasq
-
To start dnsmasq now and restart at startup
sudo brew services start dnsmasq
-
Copy the default configuration file. And set your domain resolution to IP
edit /opt/homebrew/etc/dnsmasq.conf # add "address=/test.my-domain.com/127.0.0.1" # uncomment for logging "log-queries" # add "log-facility=/var/log/dnsmasq.log" # add server "server=8.8.8.8" # uncomment and add "listen-address=127.0.0.1" sudo brew services restart dnsmasq
-
Go to System Settings → Network → Wi-Fi (or your selected active connection) → Advanced → DNS. Then, add
127.0.0.1
to your DNS Servers. -
Flush DNS cache:
sudo killall -HUP mDNSResponder
-
ping your domain to check if it resolved locally:
ping test.my-domain.com
Notes:
-
test resolution
dig example.dev nslookup example.dev ping example.com
Setup squid
The thing is, that your local network wi-fi mobiles still not able to resolve your domain locally (because only rooted Android allowed to change /etc/hosts
). So, let's try local web proxy then with squid
& dnsmasq
- When dnsmasq installed
- Add to dnsmasq config
edit /opt/homebrew/etc/dnsmasq.conf
- dhcp-option=252,”http://127.0.0.1:3128/wpad.dat”
- Now let's setup
squid
brew install squid
cp /opt/homebrew/etc/squid.conf /opt/homebrew/etc/squid.conf.back
edit /opt/homebrew/etc/squid.conf
- Replace config with the following allow-all simple config:
# Squid normally listens to port 3128
http_port 3128
# We setup an ACL that matches all IP addresses
acl all src all
# We allow all of our clients to browse the Internet
http_access allow all
# We strongly recommend the following be uncommented to protect innocent
# web applications running on the proxy server who think the only
# one who can access services on "localhost" is a local user
#http_access deny to_localhost
squid -z
to check confsudo brew services restart dnsmasq
brew services restart squid
- non-root!- check logs
- log file
/opt/homebrew/var/logs/cache.log
- access log
/opt/homebrew/var/logs/access.log
- log file
- Now go to your mobile, open "WiFi settings" -> Proxy -> manual ->
- set IP :
192.168.0.??
(set your squid server ip) - set port:
3128
- set IP :
- Check your domain
test.my-domain.com
from mobile browser, now it should be resolved via squid -> dnsmasq -> NRP -> your local server!!! Note: - By doing this all DNS & HTTP traffic from mobile clients browser (with configured proxy) and local DNS requests will go through
dnsmasq
andsquid
.
-
if certbot failed to make request, try dryRun option for testing and check docker network to be
bridge
without additonal subnets and static ip defined -
if something goes wrong and you miss details - set LOG_LEVEL=debug in
.env
and read the logs.