doc/ci/examples/deployment/composer-npm-deploy.md
{{< details >}}
{{< /details >}}
This guide covers the building of dependencies of a PHP project while compiling assets via an npm script using GitLab CI/CD.
It is possible to create your own image with custom PHP and Node.js versions. For brevity, this guide uses an existing Docker image with both PHP and Node.js installed.
image: tetraweb/php
The next step is to install zip/unzip packages and make composer available. Place these in the before_script section:
before_script:
- apt-get update
- apt-get install zip unzip
- php -r "copy('https://getcomposer.org/installer', 'composer-setup.php');"
- php composer-setup.php
- php -r "unlink('composer-setup.php');"
This ensures all requirements are ready. Next, run composer install to fetch all PHP dependencies and npm install to load Node.js packages. Then run the npm script. Append the commands to the before_script section:
before_script:
# ...
- php composer.phar install
- npm install
- npm run deploy
In this particular case, the npm deploy script is a Gulp script that does the following:
All these operations put all files into a build folder, which is ready to be deployed to a live server.
You have multiple options such as rsync, SCP, or SFTP. For now, use SCP.
To make this work, you must add a GitLab CI/CD Variable (accessible on gitlab.example/your-project-name/variables). Name this variable STAGING_PRIVATE_KEY and set it to the private SSH key of your server.
Create a user that has access only to the folder that needs to be updated.
After you create that variable, make sure that key is added to the Docker container on run:
before_script:
# - ....
- 'which ssh-agent || ( apt-get update -y && apt-get install openssh-client -y )'
- mkdir -p ~/.ssh
- eval $(ssh-agent -s)
- '[[ -f /.dockerenv ]] && echo -e "Host *\n\tStrictHostKeyChecking no\n\n" > ~/.ssh/config'
This script performs the following actions:
ssh-agent is available and install it if it's not.~/.ssh folder.And this is basically all you need in the before_script section.
To deploy the build folder from the Docker image to your server, create a new job:
stage_deploy:
artifacts:
paths:
- build/
rules:
- if: $CI_COMMIT_BRANCH == "dev"
script:
- ssh-add <(echo "$STAGING_PRIVATE_KEY")
- ssh -p22 server_user@server_host "mkdir htdocs/wp-content/themes/_tmp"
- scp -P22 -r build/* server_user@server_host:htdocs/wp-content/themes/_tmp
- ssh -p22 server_user@server_host "mv htdocs/wp-content/themes/live htdocs/wp-content/themes/_old && mv htdocs/wp-content/themes/_tmp htdocs/wp-content/themes/live"
- ssh -p22 server_user@server_host "rm -rf htdocs/wp-content/themes/_old"
Here's the breakdown:
rules:if: $CI_COMMIT_BRANCH == "dev" means that this build runs only when something is pushed to the dev branch. You can remove this block completely and have everything run on every push (but probably this is something you don't want).ssh-add ... adds that private key you added on the web UI to the Docker container.ssh and create a new _tmp folder.scp and upload the build folder (which was generated by a npm script) to the previously created _tmp folder.ssh and move the live folder to an _old folder, then move _tmp to live._old folder.The artifacts section instructs GitLab CI/CD to keep the build directory (later on, you can download that as needed).
If you're using this only for stage server, you could do this in two steps:
- ssh -p22 server_user@server_host "rm -rf htdocs/wp-content/themes/live/*"
- scp -P22 -r build/* server_user@server_host:htdocs/wp-content/themes/live
The problem is that there's a small period of time when you don't have the app on your server.
Therefore, for a production environment, the additional steps ensure that a functional app is in place at any given time.
Because this was a WordPress project, it includes real code snippets. Some further ideas you can pursue:
The final .gitlab-ci.yml looks like this:
stage_deploy:
image: tetraweb/php
artifacts:
paths:
- build/
rules:
- if: $CI_COMMIT_BRANCH == "dev"
before_script:
- apt-get update
- apt-get install zip unzip
- php -r "copy('https://getcomposer.org/installer', 'composer-setup.php');"
- php composer-setup.php
- php -r "unlink('composer-setup.php');"
- php composer.phar install
- npm install
- npm run deploy
- 'which ssh-agent || ( apt-get update -y && apt-get install openssh-client -y )'
- mkdir -p ~/.ssh
- eval $(ssh-agent -s)
- '[[ -f /.dockerenv ]] && echo -e "Host *\n\tStrictHostKeyChecking no\n\n" > ~/.ssh/config'
script:
- ssh-add <(echo "$STAGING_PRIVATE_KEY")
- ssh -p22 server_user@server_host "mkdir htdocs/wp-content/themes/_tmp"
- scp -P22 -r build/* server_user@server_host:htdocs/wp-content/themes/_tmp
- ssh -p22 server_user@server_host "mv htdocs/wp-content/themes/live htdocs/wp-content/themes/_old && mv htdocs/wp-content/themes/_tmp htdocs/wp-content/themes/live"
- ssh -p22 server_user@server_host "rm -rf htdocs/wp-content/themes/_old"