Skip to content
This repository has been archived by the owner on Mar 21, 2020. It is now read-only.

Latest commit

 

History

History
126 lines (93 loc) · 3.19 KB

README.md

File metadata and controls

126 lines (93 loc) · 3.19 KB

How it works

This follows a very simple concept:

We have a running instance 'A' and another one 'B' standing by.
Once the deployment on 'B' is completed, traffic is switched over and 'A' can be shut down.

A load balancer (such as nginx) is responsible for switching traffic.

graph

Usage

./substitute deploy [COMMIT]
./substitute rollback
./substitute stop <APP_NAME>
./substitute restart

config

Make sure that you copy deploy/config.sample to deploy/config.
And don't forget to adjust it for your needs :)

Script hooks

You can create custom scripts that will run on various events.
Scripts are searched for in deploy/scripts/{name}.sh

Script names (in their typical order for a deployment):

  • begin
  • pre_checkout
  • pre_install
  • post_install
  • pre_start
  • post_start
  • running
  • pre_stop
  • post_stop
  • end

These scripts run in the context of the substitute script and you have access to its variables.
Please note that the working directory varies. Refer to the source for details.
You can use them for whatever you want, such as creating required folders or symlinks, updating your load balancer, or sending notifications. Be creative! 😎

Details

Let's assume A is currently active. All traffic will go to port 3001, where it's running.
Deployment works as follows:

  1. B (not running) is updated, built, and then started.
  2. After making sure B is running, the load balancer has to send new connections to port 3002.
  3. A receives a SIGTERM.
    That tells it to close the server and no longer accept new connections.
    (This has to be implemented in the application, see below)
  4. At this point, all new connections are routed to B
  5. A is still running until all running requests are answered. It is then shut down.
  6. next_deploy is set to A

For the next deployment, repeat the same steps with a and b swapped.

Rollbacks

In case of trouble with the newly deployed app, we can just spin up A again and send SIGTERM to B to undo the whole thing.

SIGTERM

Your application needs to handle SIGTERM in the way it is expected by substitute.

That is, your app will reject new connections but complete running requests.
Your app must shut down at some time.

Here is a sample node.js implementation:

var server = http.createServer(requestHandler).listen();
process.on("SIGTERM", function() {
  server.close(function() {
    // all connections were closed
    process.exit();
  });
  
  setTimeout(function() {
    // force quit after 30 seconds
    process.exit(1);
  }, 30000);
});

Example

nginx config

include /var/lib/my_app/nginx-upstream.conf;

server {
    listen 80;
    server_name my_app.com;
    location / {
        proxy_pass http://my_app;
        proxy_set_header Host $host;
        …
    }
}

Script Hook

Create deploy/scripts/running.sh:

#!/usr/bin/env bash

info "Switching nginx upstream to port ${port}"
cat <<EOF > /var/lib/my_app/nginx-upstream.conf
upstream my_app {
    server 127.0.0.1:${port};
}
EOF
info "Reloading nginx..."
serivice nginx reload
info "nginx reload successful."