# Reverse Proxying Learn what a reverse proxy is, and put it in practice using nginx. This material is part of [web development courses](https://github.com/MediaComem/comem-webdev) for [Media Engineering](https://heig-vd.ch/formations/bachelor/filieres/ingenierie-des-medias). **You will need** * A Unix CLI * A server with an Ubuntu operating system and a public IP address **Recommended reading** * [Unix Administration](../unix-admin/) * [Unix Networking](../unix-networking/) * [APT](../apt/) --- class: center, middle ## What is a proxy? .breadcrumbs[
Reverse Proxying
]
> A [**proxy server**][proxy] is a computer or application that acts as an **intermediary** for requests from clients seeking resources from other servers. --- ### Types of proxy servers .breadcrumbs[
Reverse Proxying
>
What is a proxy?
] There are 3 main kinds of proxy servers: * A [**tunneling proxy**][tunneling-proxy] or [**gateway**][gateway] passes **unmodified requests and responses** from a client to a server. * A [**forward proxy**][open-proxy] is used to retrieve data from a server usually on the Internet. * A [**reverse proxy**][reverse-proxy] is an **internal-facing proxy** used to control and protect access to servers in a **private network**. --- #### Tunneling proxy .breadcrumbs[
Reverse Proxying
>
What is a proxy?
>
Types of proxy servers
] A [tunneling proxy][tunneling-proxy] can pass **unmodified requests and responses** from one network to another. It can also be used to encapsulate a protocol into another, such as running IPv6 over IPv4. .grid-50[ For example, an SSH connection may be relayed by a proxy server to a different target server. The proxy server simply passes the packets through, with no ability to compromise the security of the communication. ] .grid-50[
] .container[ [**Virtual private networks (VPN)**][vpn] use tunneling protocols, usually with an additional layer of encryption. ] --- #### Forward proxy .breadcrumbs[
Reverse Proxying
>
What is a proxy?
>
Types of proxy servers
] A **forward proxy** retrieves data from a server on behalf of a client.
.grid-100[ An [**open proxy**][open-proxy] is a forward proxy accessible by any Internet user. It can be either **anonymous** or **transparent**. ] --- ##### Anonymous or transparent .breadcrumbs[
Reverse Proxying
>
What is a proxy?
>
Types of proxy servers
>
Forward proxy
] .grid-50[ An **anonymous forward proxy** reveals its identity as a server but conceals that of the client. Since the target server does not know who the original client is, it can be used to protect privacy. [VPNs][vpn] are often used in combination with this type of proxy server. A **transparent forward proxy** identifies both itself and the original client through the use of HTTP headers. It can be used to cache websites. Schools often use this kind of proxy to restrict access to particular websites (e.g. Facebook). ] .grid-50[
] .container[ When many clients go through the same forward proxy server, its IP address may get banned, since the target server only sees one computer (the proxy) making too many requests at the same time. ] --- #### Reverse proxy .breadcrumbs[
Reverse Proxying
>
What is a proxy?
>
Types of proxy servers
] A [**reverse proxy**][reverse-proxy] is a server that **appears to clients to be an ordinary server**, but actually **transmit their requests to** one or more **other servers in an internal private network** which handle the requests.
.grid-50[ The response from the private server is returned as if it was coming from the proxy server itself, leaving the client with no knowledge of the structure of the internal network. ] .grid-50[
] --- class: center, middle ## Why use a reverse proxy? .breadcrumbs[
Reverse Proxying
]
--- ### Hiding internal architecture .breadcrumbs[
Reverse Proxying
>
Why use a reverse proxy?
] Reverse proxies can **hide the existence and characteristics of an internal network's private servers**. Since the client only sees the proxy server, it is unaware of the complexity of the internal architecture and does not have to worry about it.
In a scenario where you have only a single public IP address available, a reverse proxy allows you to make multiple private servers accessible on that IP address through the proxy server. --- ### Hiding multi-component websites .breadcrumbs[
Reverse Proxying
>
Why use a reverse proxy?
] Modern websites can be complex applications, often with a **separate frontend and backend** developed by different teams with different technologies. Putting a reverse proxy in front can make it **appear as one single website** on a single domain name, avoiding [CORS][cors] issues.
--- ### SSL termination or authentication .breadcrumbs[
Reverse Proxying
>
Why use a reverse proxy?
] Managing SSL certificates to provide websites over HTTPS is rather complex. It can be hard to configure some frameworks or tools to ensure they are only using secure communications.
A reverse proxy can help by being the **secure endpoint** with all the SSL certificates, then **forwarding unencrypted requests** to the servers or applications in the private network. Similarly, a reverse proxy could also require **authentication** before letting a client access an insecure application, adding security without having to modify the application itself. --- ### Scalability .breadcrumbs[
Reverse Proxying
>
Why use a reverse proxy?
] [Scalability][scalability] is the capability of a computer system to handle a growing amount of work, such as many clients making requests to an application at the same time. .grid-40[ There are [2 broad ways][horizontal-and-vertical-scaling] of adding more resources for a particular application. **Vertical scaling** consists in using a more powerful computer, with more CPU, RAM, throughput, etc. The added power will allow the server to serve more clients. ] .grid-60[
] .container[ **Horizontal scaling** means adding more computers to handle the same work. For example, 3 instances of a web application can probably handle 3 times as many clients at the same time. Computers or applications can be combined in [clusters][cluster] to improve performance. ] --- #### Load balancing .breadcrumbs[
Reverse Proxying
>
Why use a reverse proxy?
>
Scalability
] A common function of reverse proxies is to perform [**load balancing**][load-balancing], i.e. the distribution of workloads across multiple servers to achieve **horizontal scaling**.
As multiple clients arrive simultaneously, the reverse proxy will distribute requests to different servers, spreading the load between them. --- ### Other uses .breadcrumbs[
Reverse Proxying
>
Why use a reverse proxy?
] Reverse proxies also have [other uses][reverse-proxy-uses] such as: * Cache static content and dynamic content to reduce load on internal servers. This is known as [web acceleration][web-acceleration]. * Optimize content by transparently compressing it to speed up loading times. * *Spoon-feeding*, a technique where the reverse proxy temporarily stores a dynamically generated page, then serves it to the client a little bit at a time. This avoids the internal server having to wait for slow clients such as mobile applications. * Protect against [denial-of-service (DoS) attacks][dos] and distributed denial-of-service (DDoS) attacks. * Allow [A/B testing][ab-testing] by dynamically modifying served content. --- class: center, middle, image-header ## nginx .breadcrumbs[
Reverse Proxying
]
--- ### What is nginx? .breadcrumbs[
Reverse Proxying
>
nginx
] [nginx][nginx] is an HTTP and reverse proxy server used by more than 25% of the busiest sites in December 2018. It was developed to solve the [C10k problem][c10k], i.e. the capability of a computer system to handle ten thousand concurrent connections, thanks to its [event-driven architecture][nginx-performance]. It also has [many other features][nginx-features] to serve modern web applications. It is a concurrent of the well-known [Apache HTTP server][apache].
--- #### Apache vs. nginx .breadcrumbs[
Reverse Proxying
>
nginx
>
What is nginx?
] Although Apache is still used to serve more websites, nginx leads the pack in web performance, and is used more for the busiest websites.
--- class: center, image-header ### Installing nginx .breadcrumbs[
Reverse Proxying
>
nginx
]
It's as simple as installing it with APT: ```bash $> sudo apt install nginx ``` --- #### Making sure it's working .breadcrumbs[
Reverse Proxying
>
nginx
>
Installing nginx
] APT should automatically launch nginx, which will start listening on port 80 right away. ```bash $> ss -tlpn State Recv-Q Send-Q Local Address:Port LISTEN 0 128 0.0.0.0:`80` ... users:(("`nginx`",...) LISTEN 0 128 [::]:`80` ... users:(("`nginx`",...) ... ``` You can check that the service is running with the following command: ```bash $> sudo systemctl status nginx ``` .grid-50[ You should be able to access your server's public IP address in a browser and see nginx's welcome message: ] .grid-50[
] --- ### Nginx configuration files .breadcrumbs[
Reverse Proxying
>
nginx
] Nginx stores its configuration in the `/etc/nginx` directory. .grid-40[ ```bash $> ls -1 /etc/nginx *conf.d fastcgi.conf fastcgi_params koi-utf koi-win mime.types modules-available modules-enabled *nginx.conf proxy_params scgi_params *sites-available *sites-enabled snippets uwsgi_params win-utf ``` ] .grid-60[ The most important files are highlighted: * `nginx.conf` is the main configuration file. Everything starts from there. * `conf.d` is a directory to store reusable configuration fragments. * `sites-available` is a directory where website configuration files are stored. * `sites-enabled` is a directory containing symbolic links to the configurations in `sites-available`. ] --- #### The main configuration file .breadcrumbs[
Reverse Proxying
>
nginx
>
Nginx configuration files
] The main `/etc/nginx/nginx.conf` file is a tree-like structure formed of **contexts** delimited by braces (`{` and `}`). Each context contains configuration related to a specific area of concern. The basic structure is as follows: ```nginx # Main context: global nginx options events { # Events context: connection handling options } http { # HTTP context: web server & reverse proxy configuration server { # Server context: website configuration } server { # Another server... } } ``` These contexts can contain a number of [directives][nginx-directives]. --- #### Includes .breadcrumbs[
Main context: global nginx options
] Nginx has an [`include` directive][nginx-include] which allows you to split the configuration into several files instead of having one big file. As you can see, nginx already includes several other files out of the box: ``` $> grep include /etc/nginx/nginx.conf include /etc/nginx/modules-enabled/*.conf; include /etc/nginx/mime.types; include /etc/nginx/conf.d/*.conf; include /etc/nginx/sites-enabled/*; ``` The most important include is `/etc/nginx/sites-enabled/*`. Any file in there will be loaded in the `http` context, which means they a good place to define `server` contexts in. --- #### `sites-available` vs. `sites-enabled` .breadcrumbs[
Main context: global nginx options
] By convention, nginx has two directories for website configuration files: * **`sites-available`** contains the actual configuration files. For example, you could create a file at `/etc/nginx/sites-available/example-site` to configure nginx to serve your website. This directory is **not** included by nginx, so configuration files in it are not automatically picked up. * **`sites-enabled`** contains **symbolic links** to the configuration files in `sites-available`. This directory is included by nginx. Configuration files in it (or links to configuration files) are automatically loaded. Things are set up like this so that you can work on configuration files in `sites-available` but not actually enable them until you put the symbolic link in `sites-enabled`. It also allows you to **quickly disable** a website without deleting its configuration file. ``` /etc/nginx/sites-available/example-site /etc/nginx/sites-enabled/example-site -> /etc/nginx/sites-available/example-site ``` --- #### Reloading nginx's configuration .breadcrumbs[
Main context: global nginx options
] When you put new symbolic links in `sites-enabled`, nginx will not automatically reload them. You can restart nginx, which will make it reload the configuration, but you can also tell it to **reload its configuration without shutting down**. Nginx reacts to [some Unix signals][nginx-signals] in certain ways. Notably, it reloads its configuration when receiving a `HUP` signal. You can send the signal to the nginx master process yourself with the `kill` command, or you can simply reload the associated systemd service, which will send the signal for you: ```bash $> sudo systemctl reload nginx ``` --- #### Common nginx directives .breadcrumbs[
Main context: global nginx options
] Nginx has many [directives][nginx-directives] that can be put in various contexts. These are the most commonly used ones in the `server` context, used to serve websites (static or dynamic): Directive | Example value | Description :--- | :--- | :--- [`listen`][nginx-directive-listen] | `80`, `localhost:8000` | Port (and optional address) on which to listen and accept requests from. [`server_name`][nginx-server-names] | `example.com, www.example.com` | Comma-separated list of domain names. Determine which `server` block handles client requests. [`root`][nginx-directive-root] | `/home/john_doe/www/site`, `/var/www/site` | The root directory for requests. [`index`][nginx-directive-index] | `index.html` | Define files that will be used as an index, i.e. served by default if there is no URL path. [`location`][nginx-directive-location] | `/api`, `~* \.jpg$` | Vary configuration based on the URL. --- class: center, middle ## nginx configuration examples .breadcrumbs[
Main context: global nginx options
]
--- ### How to use these examples .breadcrumbs[
Main context: global nginx options
>
nginx configuration examples
] Each of the following examples can be put in a separate file in the `/etc/nginx/sites-available` directory: * Copy a website configuration example. * Save it in `/etc/nginx/sites-available/example`. * Create a symbolic link to it in `/etc/nginx/sites-enabled` with this command (this is one command on 3 lines): ```bash $> sudo ln -s \ /etc/nginx/sites-available/example \ /etc/nginx/sites-enabled/example ``` * Check that you have not made a mistake: ```bash $> sudo nginx -t ``` * Instruct nginx to reload its configuration with this command: ```bash $> sudo systemctl nginx reload ``` --- ### Static website configuration .breadcrumbs[
Main context: global nginx options
>
nginx configuration examples
] This configuration tells nginx to serve the static files in the directory `/home/john_doe/example` when a request arrives on port 80 for the domain `example.com`. This demonstrates nginx's capabilities as a basic **web server**. ```nginx server { listen `80`; server_name `example.com`; root `/home/john_doe/example`; index index.html; # Cache images, icons, video, audio, HTC, etc. location ~* \.(?:jpg|jpeg|gif|png|ico|cur|gz|svg|mp4|ogg|ogv|webm|htc)$ { access_log off; add_header Cache-Control "max-age=2592000"; } } ``` > The `Cache-Control` header is sent to the client's browser to tell it that > these media files (`jpg`, `png`, etc) can be cached, i.e. there's no need to > redownload them the next time. --- ### Reverse proxy configuration .breadcrumbs[
Main context: global nginx options
>
nginx configuration examples
] This configuration tells nginx to forward any request on port 80 for the domain `dynamic.example.com` to the application or server at address `http://127.0.0.1:3000`, making it behave as a **reverse proxy**. ```nginx server { listen 80; server_name dynamic.example.com; root /path/to/static/files; # Proxy requests for dynamic content to another server/application. location / { `proxy_pass http://127.0.0.1:3000;` } } ``` > In this example, the `proxy_pass` address is a local address, > but it could well be the IP address of another server in an internal network. --- ### Load balancing configuration .breadcrumbs[
Main context: global nginx options
>
nginx configuration examples
] This configuration tells nginx to perform **load balancing** on incoming requests on port 80 for the domain `big.example.com`. With the [`upstream` directive][nginx-upstream], it defines a cluster of 4 servers among which to distribute the load. ```nginx # Cluster of applications or servers to handle requests. upstream `big_server_com` { server 127.0.0.3:8000 weight=5; # local application server 127.0.0.3:8001 weight=5; # local application server 10.192.67.3:8000; # separate server server 10.192.56.12:8001; # separate server } server { listen 80; server_name big.example.com; # Proxy requests to the cluster. location / { `proxy_pass http://big_server_com;` } } ``` > By default, nginx uses a [weighted round robin][weighted-round-robin] algorithm to decide > to which server each request is proxied. A `weight` parameter can be specified to direct > more requests to specific servers. --- ## References .breadcrumbs[
Cluster of applications or servers to handle requests.
] * [Proxy server][proxy] * [Tunneling proxy][tunneling-proxy] * [Open proxy][open-proxy] * [Reverse proxy][reverse-proxy] * [Load balancing][load-balancing] * [Scalability][scalability] * [Nginx Documentation][nginx-docs] * [Nginx Beginner's Guide](http://nginx.org/en/docs/beginners_guide.html) * [Digital Ocean][digital-ocean] * [How to Install Nginx on Ubuntu 18.04](https://www.digitalocean.com/community/tutorials/how-to-install-nginx-on-ubuntu-18-04) * [Understanding the Nginx Configuration File Structure and Configuration Contexts](https://www.digitalocean.com/community/tutorials/understanding-the-nginx-configuration-file-structure-and-configuration-contexts) --- ## TODO .breadcrumbs[
Cluster of applications or servers to handle requests.
] * cors configuration [ab-testing]: https://en.wikipedia.org/wiki/A/B_testing [apache]: https://httpd.apache.org/ [c10k]: https://en.wikipedia.org/wiki/C10k_problem [cluster]: https://en.wikipedia.org/wiki/Computer_cluster [cors]: https://en.wikipedia.org/wiki/Cross-origin_resource_sharing [digital-ocean]: https://www.digitalocean.com/ [dos]: https://en.wikipedia.org/wiki/Denial-of-service_attack [gateway]: https://en.wikipedia.org/wiki/Gateway_(telecommunications) [horizontal-and-vertical-scaling]: https://en.wikipedia.org/wiki/Scalability#Horizontal_and_vertical_scaling [load-balancing]: https://en.wikipedia.org/wiki/Load_balancing_(computing) [nginx]: http://nginx.org/ [nginx-directive-index]: http://nginx.org/en/docs/http/ngx_http_index_module.html#index [nginx-directive-listen]: http://nginx.org/en/docs/http/ngx_http_core_module.html#listen [nginx-directive-location]: http://nginx.org/en/docs/http/ngx_http_core_module.html#location [nginx-directive-root]: http://nginx.org/en/docs/http/ngx_http_core_module.html#root [nginx-directives]: http://nginx.org/en/docs/dirindex.html [nginx-docs]: http://nginx.org/en/docs/ [nginx-features]: http://nginx.org/en/#basic_http_features [nginx-include]: http://nginx.org/en/docs/ngx_core_module.html#include [nginx-performance]: https://www.nginx.com/blog/inside-nginx-how-we-designed-for-performance-scale/ [nginx-server-names]: http://nginx.org/en/docs/http/server_names.html [nginx-signals]: http://nginx.org/en/docs/control.html [nginx-upstream]: http://nginx.org/en/docs/http/ngx_http_upstream_module.html [open-proxy]: https://en.wikipedia.org/wiki/Open_proxy [proxy]: https://en.wikipedia.org/wiki/Proxy_server [reverse-proxy]: https://en.wikipedia.org/wiki/Reverse_proxy [reverse-proxy-uses]: https://en.wikipedia.org/wiki/Reverse_proxy#Uses_of_reverse_proxies [scalability]: https://en.wikipedia.org/wiki/Scalability [tunneling-proxy]: https://en.wikipedia.org/wiki/Tunneling_protocol [vpn]: https://en.wikipedia.org/wiki/Virtual_private_network [web-acceleration]: https://en.wikipedia.org/wiki/Web_accelerator [weighted-round-robin]: https://en.wikipedia.org/wiki/Weighted_round_robin