mproxy
is a tiny HTTP proxy, designed to make it easy to coordinate
zero-downtime deployments. By running a web application behind mproxy
you can
deploy changes to it without interruping any of the traffic that's in progress.
No particular cooperation from the application is required for this to work.
To run an instance of the proxy, use the mproxy run
command. There's no
configuration file, but there are some options you can specify if the defaults
aren't right for your application.
For example, to run the proxy on a port other than 80 (the default) you could:
mproxy run --http-port 8080
Run mproxy help run
to see the full list of options.
To route traffic through the proxy to a web application, you deploy
instances
of the application to the proxy. Deploying an instance makes it available to the
proxy, and replace the instance it was using before (if any).
Use the format hostname:port
when specifying the instance to deploy.
For example:
mproxy deploy web-1:3000
This will instruct the proxy to register web-1:3000
to receive traffic. It
will immediately begin running HTTP health checks to ensure it's reachable and
working and, as soon as those health checks succeed, will start routing traffic
to it.
If the instance fails to become healthy within a reasonable time, the deploy
command will stop the deployment and return a non-zero exit code, so that
deployment scripts can handle the failure appropriately.
Each deployment takes over traffic from the previously deployed instance. As soon as mproxy determines that the new instance is healthy, it will route all new traffic to that instance.
The deploy
command will wait for traffic to drain from the old instance before
returning. This means it's safe to remove the old instance as soon as deploy
returns successfully, without interrupting any in-flight requests.
Because traffic is only routed to a new instance once it's healthy, and traffic is drained from old instances before they are removed, deployments take place with zero downtime.
Host-based routing allows you to run multiple applications on the same server,
using a single instance of mproxy
to route traffic to all of them.
When deploying an instance, you can specify a host that it should serve traffic for:
mproxy deploy web-1:3000 --host app1.example.com
When deployed in this way, the instance will only receive traffic for the specified host. By deploying multiple instances, each with their own host, you can run multiple applications on the same server without port conflicts.
mproxy
can automatically obtain and renew SSL certificates for your
applications. To enable this, add the --ssl
flag when deploying an instance:
mproxy deploy web-1:3000 --host app1.example.com --ssl
To build mproxy
locally, if you have a working Go environment you can:
make
Alternatively, build as a Docker container:
make docker
You can start up a sample environment to try it out using Docker Compose:
docker compose up --build
This will start the proxy, and 4 instances of a simple web server. You can run
proxy commands with docker compose exec proxy ...
, for example:
docker compose exec proxy mproxy deploy mproxy-web-1:3000
And then access the proxy from a browser at http://localhost/.