We currently use two hosting services for our production applications: uWSGI and systemd.
- id3c source code
- id3c-customizations source code
Before you get started, you'll need the following:
- a personal admin account to the production ID3C database plus the login information for the
postgres
user account. - a configured
sqitch
environment on your local machine.See Infrastructure → Databases (PostgreSQL) → sqitch configuration
- a public key shared with the
ubuntu
account onbackoffice.seattleflu.org
.
- Merge code changes in id3c or id3c-customizations to each master branch, respectively.
- Run
pipenv update
in theid3c-production
directory of your backoffice checkout. This will lock ID3C and our customizations at the latest state of their master branch on GitHub. Review, commit, and push the changes.
If you have no schema changes to deploy, you may skip this section.
-
Deploy database schema changes via
sqitch
from your local machine.Note: The "refresh-materialized-shipping-views" cronjob locks the shipping views and can cause a delay in the deployment of sqitch changes. Consider commenting out this cronjob on the
backoffice
server before deploying sqitch changes for shipping views.Run the following commands, replacing the curly-bracketed text with your specifications.
First, check that the plan looks good.
PGUSER={your-admin-username} sqitch status {database name}
Tip: check
~/.pg_service.conf
for your admin username and other connection detailsIf everything looks good, run the following using the
postgres
user.PGUSER=postgres sqitch deploy {database name}
-
Grant any newly needed roles to the
backoffice
database automation user viaPGUSER={your-admin-username} PGSERVICE={service name} psql
Once inside of the
psql
prompt, run:grant "{role}" to "backoffice-etl";
If you have no code changes to deploy, you may skip this section.
- Log onto the
backoffice
server. - Navigate to the
/opt/backoffice
directory and rungit pull
. - Add any newly needed secret environment variables under
id3c-production/env.d/…
. (Non-secret environment variables should be committed and pulled in via git.) - Install the latest production environment with
(cd id3c-production; pipenv sync)
. - Install the latest crontabs with
sudo make -C crontabs
. Check the crontabs were successfully installed by inspecting/var/log/syslog
. Look out for errors such as:cron[901]: Error: bad minute; while reading /etc/cron.d/backoffice-id3c-production cron[901]: (*system*backoffice-id3c-production) ERROR (Syntax error, this crontab file will be ignored)
- Reload the web API backend by running
sudo systemctl reload uwsgi@api-production
.See the uWSGI documentation under Infrastructure → Hosts → backoffice.seattleflu.org
- Check the service status by running
sudo systemctl status uwsgi@api-production
. - Check web API log file with
sudo journalctl -fu uwsgi@api-production
.
If you have no data to upload, you may skip this section.
- Upload data to the
receiving
area of the database from your local machine. Run the desiredid3c
command(s) with a prefix ofPGSERVICE={service name}
.
- husky-musher source code
- Log onto the
backoffice
server. - Navigate to the
/opt/husky-musher
directory and rungit pull
. - Install the latest code with
pipenv sync
. - Reload the uWSGI server by running
sudo systemctl reload uwsgi@husky-musher
.See the uWSGI documentation under Infrastructure → Hosts → backoffice.seattleflu.org
- Check the uWSGI server status by running
sudo systemctl status uwsgi@husky-musher
. - Check log file with
sudo journalctl -fu uwsgi@husky-musher
for any errors or warnings.
- sfs-switchboard source code
- Log onto the
backoffice
server. - Navigate to the
/opt/sfs-switchboard
directory and rungit pull
. - Add any newly needed secret environment variables under
/opt/backoffice/id3c-production/env.d/…
. (Non-secret environment variables should be committed and pulled in via git.) - Install the latest code with
./bin/venv-run pip-sync
. - If you've changed the structure of the
record_barcodes
table in the SQLite database, delete the old database file underdata/
. - Restart sfs-switchboard with
sudo systemctl restart sfs-switchboard
There is a crontab that syncs the switchboard. If you have changed something in sfs-switchboard that needs accompanying changes to the crontab, make that change in the backoffice repository and deploy that too.
- specimen-manifests source code
- Log onto the
backoffice
server. - Navigate to the
/opt/specimen-manifests
directory and rungit pull
.
- backoffice-apache2 source code (private)
- Log onto the
backoffice
server with agent forwarding:eval `ssh-agent` ssh-add ~/.ssh/id_rsa ssh -A [email protected]
- Navigate to the
/etc/apache2
directory and runsudo git pull
. - Reload apache by running:
sudo systemctl reload apache2
.
Note: these deployment steps assume you're using Pipenv for dependency management.
-
With
{app-name}
as your desired application name, clone your new repo on the backoffice server under/opt/{app-name}
. This should ideally match the name of the GitHub repository. -
Decide whether the app should be hosted as a uWSGI or systemd service. ASGI apps (like sfs-switchboard cannot use uWSGI).
-
Update the (private) backoffice-apache2 repo. Replacing
{desired-endpoint}
with the desired URL endpoint at https://backoffice.seattleflu.org/ where you want your app to be available, make the following changes:-
uWSGI workflow:
- make sure that the application is available via a variable named
application
. - Update the backoffice apache2 le ssl conf file to add a new
ProxyPass
entry for the app:ProxyPass /{desired-endpoint} unix:/run/uwsgi/app/{desired-endpoint}/socket|uwsgi://{desired-endpoint}/
-
If you want a Shibboleth UW NetID authentication layer, add the following to give access to anyone with a valid UW NetID:
<Location "/{desired-endpoint}"> # Authenticate visitors with Shibboleth (configured elsewhere to use # UW's IdP). Any valid UW NetID will do, so all we need is a valid # Shibboleth session. AuthType shibboleth Require shib-session # Tell Shibboleth that it should always try to establish a session if # none exists, otherwise it might decide not to and the Require rule # above would prevent access. ShibRequestSetting requireSession 1 </Location>
-
If instead you want to restrict UW NetID access to limited NetIDs, instead use
<Location "/{desired-endpoint}"> # Authenticate visitors with Shibboleth (configured elsewhere to use # UW's IdP). Require a UW NetID in one of our predefined groups. AuthType shibboleth AuthGroupFile authorized-users Require group {space-separated group-names} # Tell Shibboleth that it should always try to establish a session if # none exists, otherwise it might decide not to and the Require rule # above would prevent access. ShibRequestSetting requireSession 1 </Location>
The given
AuthGroupFile
should live in the top-level of the (private) backoffice-apache2 repo. Each specified group name should be declared inside the givenAuthGroupFile
file with the following syntax:{group-name}: {netid}@washington.edu {netid}@washington.edu ...
(See an example of the authorized NetIDs).
-
- make sure that the application is available via a variable named
-
systemd workflow:
- Update the backoffice apache2 le ssl conf file to add the following entries for the app. Note that our convention for a new
{port-number}
has been to start at3000
and increment by one for each new application.ProxyPass /{desired-endpoint}/ http://localhost:{port-number}/ ProxyPassReverse /{desired-endpoint}/ http://localhost:{port-number}/ RedirectMatch /{desired-endpoint}$ /{desired-endpoint}/
- Update the backoffice apache2 le ssl conf file to add the following entries for the app. Note that our convention for a new
-
-
Deploy your apache2 changes from the previous step.
-
With
{app-name}
as your desired application name, create a new directory in the backoffice repo named{app-name}
, and add the following to it:-
A README.
-
uWSGI workflow:
- A
uwsgi.ini
file that contains:See the husky-musher uwsgi.ini file as an example. You may want to choose a different number of processes and threads than what is used in the example above.# # This uWSGI configuration file should be used by referencing it from Ubuntu's # app-based configuration layout. # # Put the following in /etc/uwsgi/apps-available/{app-name}.ini: # # [uwsgi] # ini = /opt/backoffice/{app-name}/uwsgi.ini # # and make a symlink to it from /etc/uwsgi/apps-enabled/{app-name}.ini. # # It is assumed that Pipenv is configured to install its virtualenvs in .venv # with PIPENV_VENV_IN_PROJECT=1. # [uwsgi] plugin = python3 envdir = %d/env.d/uwsgi virtualenv = /opt/{app-name}/.venv wsgi-file = /opt/{app-name}/app.py processes = 1 threads = 2 enable-threads = true
- A new envdir diretory at
env.d/uwsgi
. Inside it, add the non-sensitive environment variables.
- A
-
systemd workflow:
- A
Makefile
that contains:See the sfs-switchboard Makefile as an example.SHELL := /bin/bash -euo pipefail apps := \ {app-name}.service install: $(apps:%=/etc/systemd/system/%) /etc/systemd/system/%: % @install -cv $< $@
- A systemd service file named
{app-name}.service
that contains:See the SFS Switchboard service file as an example.[Unit] Description=My new app! After=network.target [Service] WorkingDirectory=/opt/{app-name} User=nobody Environment="PIPENV_VENV_IN_PROJECT=1" ExecStart=/usr/local/bin/pipenv run ./bin/serve --config base_url:/{desired-endpoint}/ Restart=always RestartSec=10 [Install] WantedBy=default.target
- A
-
-
Update backoffice repo documentation README pointing to the newly created directory from the previous step.
-
Deploy the backoffice repo changes.
- From the
/opt/backoffice
directory on the backoffice server, rungit pull
. - Add any secret environment variables under
/opt/backoffice/{app-name}/env.d/
.
- From the
-
Deploy your app for the first time. With
{app-name}
as your desired application name:-
uWSGI workflow:
- On the backoffice server, put the following in
/etc/uwsgi/apps-available/{app-name}.ini
:and make a symlink to it from[uwsgi] ini = /opt/backoffice/{app-name}/uwsgi.ini
/etc/uwsgi/apps-enabled/{app-name}.ini
. - Explicitly grant the www-data user permissions to see the uwsgi envdir with:
chgrp -R www-data /opt/backoffice/*/env.d/uwsgi
chmod -R g=rX /opt/backoffice/*/env.d/uwsgi
- On the backoffice server, put the following in
-
systemd workflow:
- Inside
/opt/backoffice/
, runsudo make -C {app-name}
. - To deploy the app, run
sudo systemctl daemon-reload
. - Then, run
sudo systemctl enable {app-name}
. This command creates a symlink pointing from/etc/systemd/system/default.target.wants/{app-name}.service
→/etc/systemd/system/{app-name}.service
. - Next, start the service via
sudo systemctl start {app-name}
.
- Inside
-
-
Update the recurring deployments documentation on how to update this app in the future.
-
Update infrastructure documentation to add information about your new uWSGI or systemd app.