jordillonch / deploy-bundle
A Symfony2 deploy bundle
Installs: 191
Dependents: 0
Suggesters: 0
Security: 0
Stars: 7
Watchers: 3
Forks: 4
Open Issues: 1
Type:symfony-bundle
Requires
- php: >=5.4.0
- yriveiro/php-fslock: 0.1
Requires (Dev)
- guzzle/guzzle: 3.4.*@dev
- mockery/mockery: dev-master@dev
- phpunit/phpunit: 3.7.*
- symfony/symfony: 2.3.*
This package is not auto-updated.
Last update: 2025-01-18 17:22:46 UTC
README
WIP
What it is?
This bundle aims to be a deploy system for your projects.
It is a Symfony2 Bundle but it can be used to deploy several kind of projects.
The bundle provides some commands to automatize deploy process. Here are main commands:
- Initialize: Prepare deployer and remote servers creating a directories structure to host new code.
- Download: Download code from repository, adapt, warn up… and ship it to remote servers in order to put new code to production.
- Code to production. Deploy new code to production atomically and reload web server, app…
- Rollback. Return back to previous deployed version.
Deployer have zones configured to deploy new code.
Zones are a project and environment (e.g. prod_api, our project Api for the production environment).
Deployer uses GitHub repository, a configured branch, and HEAD as a target to deploy.
You can use this bundle adding it to your projects via composer (see installation section) but my recommendation is that you create a new project to deploy because you may be want to not have your production configurations in your repository project. So it is a good idea to delegate add productions configuration to the deploy system.
Furthermore this bundle offers helpers to automatize common operations like install composer dependencies, do a cache warm up for Symfony2 projects, refresh php-fpm after put code to production, get url to GitHub with diffs of new deployed code...
How it works?
There are two basic ideas that allows to deploy several code versions and put one of them to production and rollback between them.
When you want to deploy new code you have to do a "download" operation. That operation clones code from your git repository to a new directory (e.g. 20130704_180131_a618a56b08549794ec4c9d5db29058a01a58977f) then do adaptations to the code. Adaptations are the step where configurations are added, app cache is warned up, dependencies are downloaded and installed, symlinks are created to shared directories…
Shared directories are directories where there are data that you want to keep between deploys. (e.g. logs, reports, generated images...)
After that, code is copied to configured servers. Ssh authorized keys are used to allow copy and execute commands to remote servers.
Then, when you want to use last downloaded code to production you have to execute "code to production" operation. This operation modify a symlink to the directory where last version are downloaded.
Usually, after that you should restart webserver or php-fpm.
Note: These ideas are taken from Capistrano.
Quick start
Create a new Symfony 2.3 project
php composer.phar create-project symfony/framework-standard-edition path/ 2.3.0
Install the bundle
Add following lines to your composer.json
file:
"require": {
"jordillonch/deploy-bundle": "dev-master"
},
"minimum-stability": "dev",
Execute:
php composer.phar update
Add it to the AppKernel.php
class:
new JordiLlonch\Bundle\DeployBundle\JordiLlonchDeployBundle(),
Configure general settings and a zone
app/config/parameters.yml
jordi_llonch_deploy:
config:
project: MyProject
vcs: git
servers_parameter_file: app/config/parameters_deployer_servers.yml
local_repository_dir: /home/deploy/local_repository
clean_max_deploys: 7
ssh:
user: myuser
public_key_file: '/home/myuser/.ssh/id_rsa.pub'
private_key_file: '/home/myuser/.ssh/id_rsa'
private_key_file_pwd: 'mypassword'
zones:
prod_myproj:
deployer: myproj
environment: prod
checkout_url: 'git@github.com:myrepo/myproj.git'
checkout_branch: master
repository_dir: /var/www/production/myproj/deploy
production_dir: /var/www/production/myproj/code
app/config/parameters_deployer_servers.yml
prod_myproj:
urls:
- deploy@testserver1:8822
- deploy@testserver2:8822
- Note: Servers urls are in other file (
parameters_deployers_servers.yml
) because this file contains a dynamic configuration that would be changed in a scalable environment like AWS.
Create a deploy class to myproj
- First create your own bundle:
php app/console generate:bundle --namespace=MyProj/DeployBundle --dir=src --no-interaction
- Create your own class to deploy:
src/MyProj/DeployBundle/Service/Test.php:
<?php
namespace MyProj/DeployBundle/Service;
use JordiLlonch\Bundle\DeployBundle\Service\BaseDeployer;
class Test extends BaseDeployer
{
public function downloadCode()
{
$this->logger->debug('Downloading code...');
$this->output->writeln('<info>Downloading code...</info>');
$this->downloadCodeVcs();
$this->logger->debug('Adapting code');
$this->output->writeln('<info>Adapting code...</info>');
// Here you can download vendors, add productions configuration,
// do cache warm up, set shared directories...
$this->logger->debug('Copying code to zones...');
$this->output->writeln('<info>Copying code to zones...</info>');
$this->code2Servers();
}
public function downloadCodeRollback()
{
}
protected function runClearCache()
{
$this->logger->debug('Clearing cache...');
$this->output->writeln('<info>Clearing cache...</info>');
$this->execRemoteServers('sudo pkill -USR2 -o php-fpm');
}
}
- Add your deploy class as a service with tag:
jordi_llonch_deploy
<service id="myproj.deployer.test" class="MyProj/DeployBundle/Service/Test">
<tag name="jordi_llonch_deploy" deployer="myproj"/>
</service>
myproj
is used in the configuration asdeployer
value.
Allow ssh access to remote servers
It is necessary to add the public key of the deploy user to .ssh/authorized_keys
in remote servers in order to allow access to the deploy system.
Initialize
After configure the deployer you have to do the initialization.
app/console deployer:initialize --zones=prod_myproj
Download
Now you can download code from your repository and copy to your servers.
app/console deployer:download --zones=prod_myproj
Code to production
After download you just need to put code into production.
app/console deployer:code2production --zones=prod_myproj
Rollback
If there is any problem you can roll back to a previous version. See rollback command help.
Commands
initialize
Prepare deployer and remote servers creating a directories structure to host new code.
app/console deployer:initialize --zones=[zone1,zone2...]
download
Download code from repository, adapt, warn up… and ship it to remote servers in order to put new code to production.
app/console deployer:download --zones=[zone1,zone2...]
code2production
Deploy new code to production atomically and reload web server, app...
app/console deployer:code2production --zones=[zone1,zone2...]
syncronize
Ensure that all downloaded versions of code are copied to all servers. Useful when you add a new server to a zone. If you not syncronize the new server the rollback operation will break the code in the new server.
app/console deployer:syncronize --zones=prod_myproj
rollback
If there is any problem and you need to roll back to a previous version you have two options:
Using number of steps to roll back
app/console deployer:rollback execute [steps_backward] --zones=[zone1]
- If you want to roll back to a previous version
steps_backward
should be1
.
Referencing to a concrete version
- Ask deploy for available versions to rollback.
app/console deployer:rollback list --zones=[zone1]
- Execute rollback to specific version
app/console deployer:rollback execute [version] --zones=[zone1]
status
Shows running version and last downloaded version prepared to put to production.
app/console deployer:status [--zones=[zone1,zone2...]]
configure
Configure remote servers for zones. Useful for automatize scaling.
app/console deployer:configure zone [add, set, rm, list, list_json] [url]
clean
Remove old code. Left clean_max_deploys
deploys.
app/console deployer:clean
exec2servers
Executes command passed as argument to all configured servers.
app/console deployer:exec2servers [command]
Configuration
Deployer configurations are set in parameters.yml
.
You must set general configurations and zones.
General configuration
app/config/parameters.yml
jordi_llonch_deploy:
config:
project: MyProject
vcs: git
local_repository_dir: /home/deploy/deploy_repository
clean_max_deploys: 7
sudo: true
ssh:
user: myuser
public_key_file: '/home/myuser/.ssh/id_rsa.pub'
private_key_file: '/home/myuser/.ssh/id_rsa'
private_key_file_pwd: 'mypassword'
project
Your project name.
mail_from
Mail from.
mail_to
Mail to send emails about deployments. It is an array.
servers_parameter_file
Path where the servers parameters file is. A common configuration could be app/config/parameters_deployer_servers.yml
.
Servers urls are in this file because it contains a dynamic configuration that would be changed in a scalable environment like AWS by some script.
local_repository_dir
Directory where deployer clone your repositories, adapt code and save data about versions in the deploy system.
clean_max_deploys
Used in the clean command to remove previous downloaded versions. Left clean_max_deploys
downloaded versions.
sudo
Add sudo
to all commands send to remote servers. If you want to use, you should set your deploy user to sudoers on all remote servers with NOPASSWD:
/etc/sudoers
:
deployer_user ALL=(ALL) NOPASSWD: ALL
ssh
Ssh configuration to establish connections on remote servers to execute commands.
Here an example of configuration:
ssh:
proxy: cli
user: jllonch
password: 'mypassword'
public_key_file: '/home/myuser/.ssh/id_rsa.pub'
private_key_file: '/home/myuser/.ssh/id_rsa'
private_key_file_pwd: 'mykeypassword'
proxy
Client to use to execute remote commands.
It could be:
pecl
: php-ssh2 extensioncli
: console ssh command
If it is not defined it will try to use pecl
if it is installed, otherwise it will use cli
.
user
Username used in auth by password and auth by public key.
password
Password used in auth by password.
public_key_file
Public key file path.
private_key_file
Private key file path.
private_key_file_pwd
Private key password.
helper
Helper parameters that can be get in your deploy class.
Zones configuration
You need to set a minimum of one zone. Here is created your zone prod_myproj
:
app/config/parameters.yml
jordi_llonch_deploy:
...
zones:
prod_myproj:
deployer: myproj
environment: prod
checkout_url: 'git@github.com:myrepo/myproj.git'
checkout_branch: master
checkout_proxy: true
repository_dir: /var/www/production/myproj/deploy
production_dir: /var/www/production/myproj/code
custom:
my_key1: value1
my_key2: value2
app/config/parameters_deployer_servers.yml
prod_myproj:
urls:
- deploy@testserver1:8822
- deploy@testserver2:8822
deployer
Name of the service used to deploy the zone.
<service id="myproj.deployer.test" class="MyProj/DeployBundle/Service/Test">
<tag name="jordi_llonch_deploy" deployer="myproj"/>
</service>
environment
Environment.
urls
Remote servers where deployed code will be copied and set to production.
Format is: [user]@[server]:[port]
checkout_url
Url to git repository.
checkout_branch
Git branch to clone.
checkout_proxy
Deployer always clone a repository for every deploy. If you want to avoid to download from remote server you can clone your repository locally, then set checkout_url
to your local local repository (file:///home/deploy/proxy_repositories/myproj
). Then before download operation, deployer execute a git pull
to your local repository.
repository_dir
Path on remote servers where to copy new deployed code.
production_dir
Path that is updated by deployer when new deployed code is set to production. So you must set this path to your webserver as a root path.
custom
Custom parameters that can be get in your deploy class.
helper
Helper parameters that must be set to use some helpers.
If helper parameters are set in general configuration those configuration are merged here.
Helpers
Hepers automatize common operations like to install composer dependencies, to do a cache warm up for Symfony2 projects, to restart php-fpm after to put code to production, to get url to GitHub with differences of new deployed code...
How use it?
You can use Helpers
in your Deployer classes to execute common operartions.
Here and example:
src/MyProj/DeployBundle/Service/Test.php:
<?php
namespace MyProj/DeployBundle/Service;
use JordiLlonch\Bundle\DeployBundle\Service\BaseDeployer;
class Test extends BaseDeployer
{
public function initialize()
{
parent::initialize();
// Shared dirs
$this->getHelper('shared_dirs')->initialize('logs');
}
public function downloadCode()
{
$this->logger->debug('Downloading code...');
$this->output->writeln('<info>Downloading code...</info>');
$this->downloadCodeVcs();
$this->logger->debug('Adapting code');
$this->output->writeln('<info>Adapting code...</info>');
// composer
$this->getHelper('composer')->install();
$this->getHelper('composer')->executeInstall();
// cache warm:up
$this->getHelper('symfony2')->cacheWarmUp();
// shared directories
$this->getHelper('shared_dirs')->set('app/logs', 'logs');
$this->logger->debug('Copying code to zones...');
$this->output->writeln('<info>Copying code to zones...</info>');
$this->code2Servers();
}
public function downloadCodeRollback()
{
}
protected function runClearCache()
{
$this->logger->debug('Clearing cache...');
$this->output->writeln('<info>Clearing cache...</info>');
$this->getHelper('phpfpm')->refresh();
}
}
- As you can see we used some methods prefixed by
helper…
to install composer dependencies, doing cache warm up, setting shared directories and restarting php-fpm.
Provided helpers
SharedDirs
Easy way to manage shared directories.
$this->getHelper('shared_dirs')->initialize($path)
- Initialize given path in the shared directory.
$this->getHelper('shared_dirs')->set($pathInAppToLink, $pathInSharedDir)
- Set shared directory for a given path in the project that will be removed and replaced by a symlink to given path to shared directory.
PhpFpm
Provides methods to restart php-fpm gracefully.
$this->getHelper('phpfpm')->refresh()
- Refresh php-fpm gracefully but you must configure your webserver (e.g. Nginx) to retry the request.
$this->getHelper('phpfpm')->refreshCommand()
- Command used to reload php-fpm
Composer
Helpers to manage composer installation and some commands.
$this->getHelper('composer')->install()
- Install composer.phar in the new repository dir.
$this->getHelper('composer')->executeInstall()
- Executes composer install in the new repository dir.
- If environment is dev or test --dev parameter is added to composer install
- If environment is prod --no-dev parameter is added to composer install
Symfony2
For now only provides a method to do a cache warm up.
$this->getHelper('symfony2')->cacheWarmupOnServers()
- Do a cache:warmup for production environment. This operation should be executed after code2Servers() operation.
Deprecated:
$this->getHelper('symfony2')->cacheWarmUp()
- This warmup it is not a secure operation because serialized data could be corrupted.
GitHub
Useful methods to have feedback of your deploy in GitHub.
$this->getHelper('github')->getCompareUrl($gitUidFrom, $gitUidTo)
- Give an url comparing two commits
- You must set your http url to your GitHub repository in jordi_llonch_deploy.zones parameters:
helper:
github:
url: https://github.com/YourUser/Repository
$this->getHelper('github')->getCompareUrlFromCurrentCodeToNewRepository()
- Give an url comparing commits between current running code and the new downloaded code.
HipChat
Provides a method to send messages to a room in a HipChat.
$this->getHelper('hipchat')->send($msg, $color='purple')
- Send a message to a given room
- You must set your token and room_id to your HipChat in jordi_llonch_deploy.general parameters:
helper:
hipchat:
token: your_token
room_id: your_room_id
TODO
- Tests
- Ssh with Process component
Author
Jordi Llonch - llonch.jordi at gmail dot com
License
DeployBundle is licensed under the MIT License. See the LICENSE file for full details.