This project is an example of continuous deployment of a Symfony 4 project.
- Symfony 4
- PHP 7.2 and Apache (mod_php)
- MYSQL 5.7
- Doctrine ORM
The Symfony project is slightly modified with Controller As A Service, Zend Template Renderer, PSR-7 requests and responses, PSR-15 request handlers.
The local docker environment uses the code as a volume, and has a mysql and a minio server for persistance.
We will first deploy the project locally on the local machine using git.
We are assuming the project is currently properly versioned. We will need to create a repository that will act as a remote for us, then edit its hooks to trigger actions when receiving code, and finally set a symlink to handle the webserver configuration.
Assuming ./
as your current directory, empty for now.
Clone this repository in the ./project
folder (git clone git@github.com:tdutrion/continuous-deployment.git project
).
Then create an empty repository using mkdir remote && cd remote && git init --bare
.
Get back to your initial directory (cd ..
).
Finally, let's create our deployment folders (mkdir -p deployment/{blue,green}
).
Copy the post-receive hook sample and edit it:
touch remote/hooks/post-receive && chmod +x remote/hooks/post-receive
#!/bin/bash
while read oldrev newrev ref
do
if [[ $ref =~ .*/master$ ]];
then
cd ..
echo "Master ref received. Deploying master branch to production..."
deployment_dir=$(readlink -- "deployment/current")
echo "Currently deployed: $deployment_dir"
temp_dir="/tmp/$(uuidgen)";
mkdir $temp_dir;
if [ "$deployment_dir" != "./blue/public" ]; then
echo "building for blue";
if [ -d $temp_dir ]; then
rm -r $temp_dir;
fi
git clone remote $temp_dir;
APP_ENV=prod composer install --working-dir=$temp_dir --prefer-dist --no-dev --no-suggest --optimize-autoloader --classmap-authoritative;
# Check whether everything went as planned.
if [ -d deployment/blue ]; then
rm -r deployment/blue;
fi
cp -R $temp_dir deployment/blue
if [ -L deployment/current ]; then
rm deployment/current;
fi
ln -s ./blue/public deployment/current;
else
echo "building for green";
if [ -d $temp_dir ]; then
rm -r $temp_dir;
fi
git clone remote $temp_dir;
APP_ENV=prod composer install --working-dir=$temp_dir --prefer-dist --no-dev --no-suggest --optimize-autoloader --classmap-authoritative;
# Check whether everything went as planned.
if [ -d deployment/green ]; then
rm -r deployment/green;
fi
cp -R $temp_dir deployment/green
if [ -L deployment/current ]; then
rm deployment/current;
fi
ln -s ./green/public deployment/current;
fi
echo "SetEnv APP_ENV prod" > deployment/current/.htaccess;
rm -rf $temp_dir;
else
echo "Ref $ref successfully received. Doing nothing: only the master branch may be deployed on this server."
fi
done
This script can obviously be optimised but you get the idea.
Make sure it is executable (chmod +x remote/hooks/post-receive
).
Then make sure we know which one of green or blue is the current deployment. Because we're on a single server, we will use a symlink. The same concept could be applied using DNS resolution or a reverse proxy configuration.
ln -s ./blue/public deployment/current
After you need to go in project folder
cd ./project
Now you can change the remote of project
git remote rm origin
git remote add origin ../remote
git push origin master
Last thing we need to do here is to set a webserver with a document root pointing on the public directory of the symlink.
mkdir env
touch env/Dockerfile
Add the following content to the Dockerfile:
FROM php:7.2-apache
RUN docker-php-ext-install pdo pdo_mysql \
&& a2enmod rewrite \
&& sed -i 's!/var/www/html!/var/www/html/deployment/current!g' /etc/apache2/sites-available/*.conf
WORKDIR /var/www/html
docker build -t continuous-deployment/server-demo env
docker run --rm -v $(pwd):/var/www/html:cached -p 80:80 continuous-deployment/server-demo
You need to be careful that port 80 is not used already, or change it in the command.
You can go on http://127.0.0.1 and you should see one page with Hello World
Now you can try to edit content of message
vim project/src/Action/Home.php
Now you can do the same things what you did before, and you will see the page change without downtime (you may need to clear the webbrowser http cache)
git status
git add .
git commit -m "second commit"
git push origin master
Launch a docker registry:
docker run -d -p 6000:5000 --name registry registry:2
Then try to push your existing image:
docker tag continuous-deployment/server-demo localhost:6000/continuous-deployment/server-demo
docker push localhost:6000/continuous-deployment/server-demo
When you then try to run it, an error will occur as we haven't added the code yet. Add a new Dockerfile in ./Dockerfile
:
FROM php:7.2-apache
RUN docker-php-ext-install pdo pdo_mysql \
&& a2enmod rewrite \
&& sed -i 's!/var/www/html!/var/www/html/deployment/current!g' /etc/apache2/sites-available/*.conf
ADD . /var/www/html
WORKDIR /var/www/html
In order not to add the whole context, add a ./.dockerignore
:
project/
remote/
env/
docker build -t continuous-deployment/server-demo .
docker run --rm -p 80:80 continuous-deployment/server-demo