Code Enigma's custom Fabric and shell scripts for deploying various PHP apps. We predominantly do this using Jenkins CI, however in principle, any system capable of triggering a shell command on a server with Fabric installed can be used as a trigger for these scripts.
These deployment scripts currently support:
Here is an overview of the kinds of things these build scripts allow you to do:
- specify target server(s) in a repository config file
- stand up initial builds of WordPress and Drupal automatically
- provision of a "seed" database
- supports clusters of app servers
- optional automatic SSL configuration
- optional HTTP AUTH password protection
- build hooks - developers can insert their own Fabric and shell scripts into the process at various points
- rollback - automatically restore the previous version of an app if build fails
- StatusCake integration allowing alert pausing during build
Because as a company we mostly specialise in Drupal, a lot of the best features of the current scripts are actually Drupal specific:
- first install from a specified install profile
- inject environment-specific Drupal settings safely
- Behat tests - run Behat tests automatically using Selenium
- Simpletest unit tests - run Simpletest automatically
- Coder module support - automatically check code style when building
- Features module support - often used for configuration in code in Drupal 7
- Read Only Mode module support - for deployments without going offline
- Environment Indicator module support - for colour-coded environment notification for administrators
- automated feature branching support
- Drupal 8 configuration importing and exporting
- Drupal 8 deployments with Composer
- database update handling
- automatic provisioning of Drush aliases
- automatic configuration of Drupal cron
- synchronise databases between environments
- obfuscate sensitive data
We are continually looking to reduce this list of assumptions and dependencies, but for now in order to use these scripts you must meet the following criteria:
- Linux CI server with Fabric installed
- a 'jenkins' user on the CI server which is used to trigger commands
- Linux target app server(s) (Debian will work best without modification)
- a 'jenkins' user with a home directory and public key of the CI user on target app server(s)
- passwordless sudo for the 'jenkins' user on the target app server(s)
- Git installed on the target app server(s)
- cachetool for clearing the PHP opcode cache
- composer (Drupal > 8 or Symfony)
- drush (Drupal) - installed at
/usr/local/bin/drush
- wp-cli (WordPress)
- Ruby (Drupal data obfuscation)
If you use a MySQL (or similar) database (so clearly Drupal, WordPress and some Symfony apps) you also need:
- mysql client on the target app server(s)
- mysql 'defaults' file we can pass to prepare scripts (default location
/etc/mysql/debian.cnf
)
Clone this repository into a location of your choice on your CI server (usually Jenkins).
Create a means to trigger the Fabric scripts (usually a "Freestyle project" in Jenkins with a trigger based on repository change).
Ensure your application is either in a www
repository in the repository or has a link to ./www
committed to the repository and pointing at the relative application location within the repository (the scripts expect to find an app in www
).
Ensure your application has a config.ini
file in the repository root, which contains at least the lines necessary to direct the deployment to a server, for example:
[drupal8]
example-drupal=dev1.codeenigma.com
'drupal8' is the build type (you may choose a string), 'example-drupal' is the repo name, as used below in the call to the script and 'dev1.codeenignma.com' is the hostname of the server to deploy to. We are building more comprehensive examples of the config.ini
file in our example-drupal project.
An example trigger command, typically placed in an Execute Shell box in Jenkins, looks like this - example uses Drupal:
fab -f /path/to/these/scripts/drupal/fabfile.py
main:repo=myreponame,[email protected]:myreponame.git,build=build_${BUILD_NUMBER},branch=master,buildtype=master,keepbuilds=5
For any given type of application, you can look at the main()
function in fabfile.py
in the appropriate sub-directory, in this case ./drupal/fabfile.py
.
Drupal scripts are the most mature, as our specialism is as a Drupal development company, though we have WordPress and Symfony projects as well, and we also do some work with Magento.
- our
_sshagent_run()
now supports a path to a private key, however some deploy scripts need updating to allow it to be passed to themain()
function - Nginx is the preferred web server, Apache support isn there and it mostly works, but the initial build parts for WordPress and Drupal may be shaky
- some manual manipulation of vhosts might be required if you do not like the default URL format
- we're working around some challenges with regard to supporting AWS RDS for MySQL initial builds and initial builds in cluster environments generally
- the
drush status
check does not use the--format
option, and ceased to work with Drush 9 (we'll fix this pretty quick!) - Drupal 8 multisites are a work in progress
- need to improve support for prebuilt Drupal apps, where the 'vendor' directory exists already
- the scripts work well for less complex WordPress sites, but are untested on more complex implementations
- need to parse
parameters.yml
and allow developers to specify parameters to use for database manipulation / backups
These scripts are continually improved, specific enhancements on our radar are:
- Magento support
- unit tests
- support for front-end framework compilers
- better microservice support