Add Dublin Fabric talk
This commit is contained in:
		
							parent
							
								
									45ee78b607
								
							
						
					
					
						commit
						6165e4a372
					
				
					 7 changed files with 299 additions and 154 deletions
				
			
		
										
											Binary file not shown.
										
									
								
							| Before Width: | Height: | Size: 42 KiB | 
										
											Binary file not shown.
										
									
								
							| After Width: | Height: | Size: 38 KiB | 
										
											Binary file not shown.
										
									
								
							| After Width: | Height: | Size: 162 KiB | 
|  | @ -1,8 +1,11 @@ | ||||||
| autoscale: true | autoscale: true | ||||||
| build-lists: true | build-lists: true | ||||||
| theme: poster, 7 | theme: next, 9 | ||||||
| 
 | 
 | ||||||
| # [fit] *Deploying* PHP<br>applications<br>*with* Fabric | # [fit] Deploying Drupal <br>with Fabric | ||||||
|  | 
 | ||||||
|  | ### Oliver Davies | ||||||
|  | ### bit.ly/deploying-drupal-fabric | ||||||
| 
 | 
 | ||||||
| --- | --- | ||||||
| 
 | 
 | ||||||
|  | @ -16,26 +19,30 @@ theme: poster, 7 | ||||||
| 
 | 
 | ||||||
| [.build-lists: false] | [.build-lists: false] | ||||||
| 
 | 
 | ||||||
|  |  | ||||||
|  | 
 | ||||||
| - Senior Developer at Microserve | - Senior Developer at Microserve | ||||||
| - Part-time freelance Developer & Sysadmin | - Part-time freelance Developer & System Administrator | ||||||
|  | - Drupal, Symfony, Silex, Laravel, Sculpin | ||||||
| - Drupal Bristol, PHPSW, DrupalCamp Bristol | - Drupal Bristol, PHPSW, DrupalCamp Bristol | ||||||
|  | - Sticker collector, herder of elePHPants | ||||||
| - @opdavies | - @opdavies | ||||||
| - oliverdavies.uk | - oliverdavies.uk | ||||||
| 
 | 
 | ||||||
|  |  | ||||||
| 
 | 
 | ||||||
| ^ Drupal (7 & 8), Symfony, Silex, Laravel, Sculpin | ^ Not a Python Developer | ||||||
| Not a Python Developer |  | ||||||
| 
 | 
 | ||||||
| --- | --- | ||||||
| 
 | 
 | ||||||
| ## *What is* Fabric*?* |  | ||||||
|  | 
 | ||||||
|  | ## [fit] What is <br>Fabric? | ||||||
| 
 | 
 | ||||||
| --- | --- | ||||||
| 
 | 
 | ||||||
| ## What is Fabric? | ## What is Fabric? | ||||||
| 
 | 
 | ||||||
| *Fabric is a* Python (2.5-2.7) library and command-line tool for streamlining the use of SSH *for application deployment or systems administration tasks.* | Fabric is a Python (2.5-2.7) library and command-line tool for streamlining the use of SSH for application deployment or systems administration tasks. | ||||||
| 
 | 
 | ||||||
| --- | --- | ||||||
| 
 | 
 | ||||||
|  | @ -63,7 +70,7 @@ It provides a basic suite of operations for executing local or remote shell comm | ||||||
| 
 | 
 | ||||||
| - Powerful | - Powerful | ||||||
| - Flexible | - Flexible | ||||||
| - Easier to write than bash | - Easier to read and write than bash | ||||||
| 
 | 
 | ||||||
| ^ - Can be used for different languages, frameworks | ^ - Can be used for different languages, frameworks | ||||||
| - Can be used for small, simple deployments or large, complicated ones | - Can be used for small, simple deployments or large, complicated ones | ||||||
|  | @ -86,7 +93,7 @@ $ apt-get install python-fabric | ||||||
| 
 | 
 | ||||||
| --- | --- | ||||||
| 
 | 
 | ||||||
| ## Writing your <br>first *fabfile* | ## [fit] Writing your <br>first fabfile | ||||||
| 
 | 
 | ||||||
| --- | --- | ||||||
| 
 | 
 | ||||||
|  | @ -159,10 +166,10 @@ from fabric.contrib.files import * | ||||||
| 
 | 
 | ||||||
| ```python | ```python | ||||||
| 
 | 
 | ||||||
| def build(): | def main(): | ||||||
|     with cd('/var/www/html'): |   with cd('/var/www/html'): | ||||||
|         run('git pull') |     run('git pull') | ||||||
|         run('composer install') |     run('composer install') | ||||||
| ``` | ``` | ||||||
| 
 | 
 | ||||||
| --- | --- | ||||||
|  | @ -170,12 +177,12 @@ def build(): | ||||||
| ## Task arguments | ## Task arguments | ||||||
| 
 | 
 | ||||||
| ```python, [.highlight: 1, 4-6] | ```python, [.highlight: 1, 4-6] | ||||||
| def build(run_composer=True): | def main(run_composer=True): | ||||||
|     with cd('/var/www/html'): |   with cd('/var/www/html'): | ||||||
|         run('git pull') |     run('git pull') | ||||||
| 
 | 
 | ||||||
|         if run_composer: |     if run_composer: | ||||||
|             run('composer install') |       run('composer install') | ||||||
| ``` | ``` | ||||||
| 
 | 
 | ||||||
| --- | --- | ||||||
|  | @ -183,22 +190,22 @@ def build(run_composer=True): | ||||||
| ## Task arguments | ## Task arguments | ||||||
| 
 | 
 | ||||||
| ```python, [.highlight: 1, 4-15] | ```python, [.highlight: 1, 4-15] | ||||||
| def build(run_composer=True, env='prod', build_type): | def main(run_composer=True, env='prod', build_type): | ||||||
|     with cd('/var/www/html'): |   with cd('/var/www/html'): | ||||||
|         run('git pull') |     run('git pull') | ||||||
| 
 | 
 | ||||||
|         if run_composer: |     if run_composer: | ||||||
|             if env == 'prod': |       if env == 'prod': | ||||||
|                 run('composer install --no-dev') |         run('composer install --no-dev') | ||||||
|             else: |       else: | ||||||
|                 run('composer install') |         run('composer install') | ||||||
| 
 | 
 | ||||||
|         if build_type == 'drupal': |       if build_type == 'drupal': | ||||||
|             ... |         ... | ||||||
|         elif build_type == 'symfony': |       elif build_type == 'symfony': | ||||||
|             ... |         ... | ||||||
|         elif build_type == 'sculpin': |       elif build_type == 'sculpin': | ||||||
|             ... |         ... | ||||||
| ``` | ``` | ||||||
| 
 | 
 | ||||||
| --- | --- | ||||||
|  | @ -207,20 +214,20 @@ def build(run_composer=True, env='prod', build_type): | ||||||
| 
 | 
 | ||||||
| ```python, [.highlight: 4-15] | ```python, [.highlight: 4-15] | ||||||
| @task | @task | ||||||
| def build(): | def main(): | ||||||
|     with cd('/var/www/html'): |   with cd('/var/www/html'): | ||||||
|         build() |     build() | ||||||
|         post_install() |     post_install() | ||||||
| 
 | 
 | ||||||
| def build(): | def build(): | ||||||
|     run('git pull') |   run('git pull') | ||||||
|     run('composer install') |   run('composer install') | ||||||
| 
 | 
 | ||||||
| def post_install(): | def post_install(): | ||||||
|     with prefix('drush'): |   with prefix('drush'): | ||||||
|         run('updatedb -y') |     run('updatedb -y') | ||||||
|         run('entity-updates -y') |     run('entity-updates -y') | ||||||
|         run('cache-rebuild') |     run('cache-rebuild') | ||||||
| ``` | ``` | ||||||
| 
 | 
 | ||||||
| ^ Better organised code | ^ Better organised code | ||||||
|  | @ -269,7 +276,7 @@ Disconnecting from production... done. | ||||||
| 
 | 
 | ||||||
| [.build-lists: false] | [.build-lists: false] | ||||||
| 
 | 
 | ||||||
| ## *Not* Building on Prod | ## Not Building on Prod | ||||||
| 
 | 
 | ||||||
| 1. Build locally and deploy. | 1. Build locally and deploy. | ||||||
| 
 | 
 | ||||||
|  | @ -303,14 +310,14 @@ local('composer install') | ||||||
| from fabric.api import cd | from fabric.api import cd | ||||||
| 
 | 
 | ||||||
| with cd('themes/custom/drupalbristol'): | with cd('themes/custom/drupalbristol'): | ||||||
|     ... |   ... | ||||||
| 
 | 
 | ||||||
| # Runs locally. | # Runs locally. | ||||||
| 
 | 
 | ||||||
| from fabric.api import lcd | from fabric.api import lcd | ||||||
| 
 | 
 | ||||||
| with lcd('themes/custom/drupalbristol'): | with lcd('themes/custom/drupalbristol'): | ||||||
|     ... |   ... | ||||||
| ``` | ``` | ||||||
| 
 | 
 | ||||||
| --- | --- | ||||||
|  | @ -323,12 +330,12 @@ from fabric.contrib.project import rsync_project | ||||||
| ... | ... | ||||||
| 
 | 
 | ||||||
| def deploy(): | def deploy(): | ||||||
|     rsync_project( |   rsync_project( | ||||||
|         local_dir='./', |     local_dir='./', | ||||||
|         remote_dir='/var/www/html' |     remote_dir='/var/www/html' | ||||||
|         default_opts='-vzcrSLh', |     default_opts='-vzcrSLh', | ||||||
|         exclude=('.git', 'node_modules/', '.sass-cache/') |     exclude=('.git', 'node_modules/', '.sass-cache/') | ||||||
|     ) |   ) | ||||||
| ``` | ``` | ||||||
| 
 | 
 | ||||||
| --- | --- | ||||||
|  | @ -353,7 +360,7 @@ Done. | ||||||
| 
 | 
 | ||||||
| [.build-lists: false] | [.build-lists: false] | ||||||
| 
 | 
 | ||||||
| ## *Not* Building on Prod | ## Not Building on Prod | ||||||
| 
 | 
 | ||||||
| 1. ~~Build locally and deploy.~~ | 1. ~~Build locally and deploy.~~ | ||||||
| 1. Build in a separate directory and switch after build. | 1. Build in a separate directory and switch after build. | ||||||
|  | @ -362,25 +369,25 @@ Done. | ||||||
| 
 | 
 | ||||||
| --- | --- | ||||||
| 
 | 
 | ||||||
| ## Deploying into a *different directory* | ## Deploying into a different directory | ||||||
| 
 | 
 | ||||||
| ```python | ```python | ||||||
| from fabric.api import * | from fabric.api import * | ||||||
| from time import time | from time import time | ||||||
| 
 | 
 | ||||||
| project_dir = '/var/www/html' | project_dir = '/var/www/html' | ||||||
| next_release = "%(time).0f" % { 'time': time() } # timestamp | next_release = "%(time).0f" % { 'time': time() } # Current timestamp | ||||||
| 
 | 
 | ||||||
| def init(): | def init(): | ||||||
|     if not exists(project_dir): |   if not exists(project_dir): | ||||||
|         run('mkdir -p %s/backups' % project_dir) |     run('mkdir -p %s/backups' % project_dir) | ||||||
|         run('mkdir -p %s/shared' % project_dir) |     run('mkdir -p %s/shared' % project_dir) | ||||||
|         run('mkdir -p %s/releases' % project_dir) |     run('mkdir -p %s/releases' % project_dir) | ||||||
| ``` | ``` | ||||||
| 
 | 
 | ||||||
| --- | --- | ||||||
| 
 | 
 | ||||||
| ## Deploying into a *different directory* | ## Deploying into a different directory | ||||||
| 
 | 
 | ||||||
| ```python | ```python | ||||||
| current_release = '%s/%s' % (releases_dir, next_release) | current_release = '%s/%s' % (releases_dir, next_release) | ||||||
|  | @ -388,10 +395,10 @@ current_release = '%s/%s' % (releases_dir, next_release) | ||||||
| run('git clone %s %s' % (git_repo, current_release)) | run('git clone %s %s' % (git_repo, current_release)) | ||||||
| 
 | 
 | ||||||
| def build(): | def build(): | ||||||
|     with cd(current_release): |   with cd(current_release): | ||||||
|         pre_tasks() |     pre_tasks() | ||||||
|         build() |     build() | ||||||
|         post_tasks() |     post_tasks() | ||||||
| ``` | ``` | ||||||
| 
 | 
 | ||||||
| ^ - Clone the repository into a different directory | ^ - Clone the repository into a different directory | ||||||
|  | @ -399,25 +406,25 @@ def build(): | ||||||
| 
 | 
 | ||||||
| --- | --- | ||||||
| 
 | 
 | ||||||
| ## Deploying into a *different directory* | ## Deploying into a different directory | ||||||
| 
 | 
 | ||||||
| ```python | ```python | ||||||
| def pre_build(build_number): | def pre_build(build_number): | ||||||
|     with cd('current'): |   with cd('current'): | ||||||
|         print '==> Dumping the DB (just in case)...' |     print '==> Dumping the DB (just in case)...' | ||||||
|         backup_database() |     backup_database() | ||||||
| 
 | 
 | ||||||
| def backup_database(): | def backup_database(): | ||||||
|     cd('drush sql-dump --gzip > ../backups/%s.sql.gz' % build_number) |   cd('drush sql-dump --gzip > ../backups/%s.sql.gz' % build_number) | ||||||
| ``` | ``` | ||||||
| 
 | 
 | ||||||
| --- | --- | ||||||
| 
 | 
 | ||||||
| ## Deploying into a *different directory* | ## Deploying into a different directory | ||||||
| 
 | 
 | ||||||
| ```python | ```python | ||||||
| def update_symlinks(): | def update_symlinks(): | ||||||
|     run('ln -nfs %s/releases/%s %s/current' |   run('ln -nfs %s/releases/%s %s/current' | ||||||
|     % (project_dir, next_release, project_dir)) |     % (project_dir, next_release, project_dir)) | ||||||
| 
 | 
 | ||||||
| # /var/www/html/current | # /var/www/html/current | ||||||
|  | @ -429,9 +436,9 @@ def update_symlinks(): | ||||||
| [production] Executing task 'main' | [production] Executing task 'main' | ||||||
| [production] run: git clone https://github.com/opdavies/oliverdavies.uk.git | [production] run: git clone https://github.com/opdavies/oliverdavies.uk.git | ||||||
|   /var/www/html/releases/1505865600 |   /var/www/html/releases/1505865600 | ||||||
| ===> Installing Composer dependencies... | Installing Composer dependencies... | ||||||
| [production] run: composer install --no-dev | [production] run: composer install --no-dev | ||||||
| ===> Update the symlink to the new release... | Update the symlink to the new release... | ||||||
| [production] run: ln -nfs /var/www/html/releases/1505865600 | [production] run: ln -nfs /var/www/html/releases/1505865600 | ||||||
|   /var/www/html/current |   /var/www/html/current | ||||||
| 
 | 
 | ||||||
|  | @ -443,7 +450,7 @@ Done. | ||||||
| ```bash | ```bash | ||||||
| # /var/www/html | # /var/www/html | ||||||
| 
 | 
 | ||||||
| shared | shared # settings.local.php, sites.php, files etc. | ||||||
| releases/1502323200 | releases/1502323200 | ||||||
| releases/1505692800 | releases/1505692800 | ||||||
| releases/1505696400 | releases/1505696400 | ||||||
|  | @ -469,7 +476,7 @@ current -> releases/1505865600 # symlink | ||||||
| def main(builds_to_keep=3): | def main(builds_to_keep=3): | ||||||
|   with cd('%s/releases' % project_dir): |   with cd('%s/releases' % project_dir): | ||||||
|     run("ls -1tr | head -n -%d | xargs -d '\\n' rm -fr" |     run("ls -1tr | head -n -%d | xargs -d '\\n' rm -fr" | ||||||
|     % builds_to_keep) |       % builds_to_keep) | ||||||
| ``` | ``` | ||||||
| 
 | 
 | ||||||
| ^ - Find directory names | ^ - Find directory names | ||||||
|  | @ -479,7 +486,91 @@ def main(builds_to_keep=3): | ||||||
| 
 | 
 | ||||||
| --- | --- | ||||||
| 
 | 
 | ||||||
| ## Is the site still running? | ## [fit] Does the code still <br>merge cleanly? | ||||||
|  | 
 | ||||||
|  | ^ Pre-task | ||||||
|  | 
 | ||||||
|  | --- | ||||||
|  | 
 | ||||||
|  | ```python | ||||||
|  | def check_for_merge_conflicts(target_branch): | ||||||
|  |   with settings(warn_only=True): | ||||||
|  |     print('Ensuring that this can be merged into the main branch.') | ||||||
|  | 
 | ||||||
|  |     if local('git fetch && git merge --no-ff origin/%s' | ||||||
|  |       % target_branch).failed: | ||||||
|  |       abort('Cannot merge into target branch.') | ||||||
|  | ``` | ||||||
|  | 
 | ||||||
|  | ^ Define target branch in Jenkins/when Fabric is run. | ||||||
|  | 
 | ||||||
|  | --- | ||||||
|  | 
 | ||||||
|  |  | ||||||
|  | 
 | ||||||
|  | ## [fit] Do our tests <br>still pass? | ||||||
|  | 
 | ||||||
|  | [.footer: http://nashvillephp.org/images/testing-workshop-banner-1600x800.jpg] | ||||||
|  | 
 | ||||||
|  | --- | ||||||
|  | 
 | ||||||
|  | ```python | ||||||
|  | with settings(warn_only=True): | ||||||
|  |   with lcd('%s/docroot/core' % project_dir): | ||||||
|  |     if local('../../vendor/bin/phpunit ../modules/custom').failed: | ||||||
|  |       abort('Tests failed!') | ||||||
|  | ``` | ||||||
|  | 
 | ||||||
|  | --- | ||||||
|  | 
 | ||||||
|  | ``` | ||||||
|  | [localhost] run: ../../vendor/bin/phpunit ../modules/custom | ||||||
|  | [localhost] out: PHPUnit 4.8.35 by Sebastian Bergmann and contributors. | ||||||
|  | [localhost] out: | ||||||
|  | [localhost] out: ....... | ||||||
|  | [localhost] out: | ||||||
|  | [localhost] out: Time: 1.59 minutes, Memory: 6.00MB | ||||||
|  | [localhost] out: | ||||||
|  | [localhost] out: OK (7 tests, 42 assertions) | ||||||
|  | [localhost] out: | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | Done. | ||||||
|  | ``` | ||||||
|  | 
 | ||||||
|  | --- | ||||||
|  | 
 | ||||||
|  | ``` | ||||||
|  | [localhost] run: ../../vendor/bin/phpunit ../modules/custom | ||||||
|  | [localhost] out: PHPUnit 4.8.35 by Sebastian Bergmann and contributors. | ||||||
|  | [localhost] out: | ||||||
|  | [localhost] out: E | ||||||
|  | [localhost] out: | ||||||
|  | [localhost] out: Time: 18.67 seconds, Memory: 6.00MB | ||||||
|  | [localhost] out: | ||||||
|  | [localhost] out: There was 1 error: | ||||||
|  | [localhost] out: | ||||||
|  | [localhost] out: 1) Drupal\Tests\broadbean\Functional\AddJobTest::testNodesAreCreated | ||||||
|  | [localhost] out: Behat\Mink\Exception\ExpectationException: Current response status code is 200, but 201 expected. | ||||||
|  | [localhost] out: | ||||||
|  | [localhost] out: /var/www/html/vendor/behat/mink/src/WebAssert.php:770 | ||||||
|  | [localhost] out: /var/www/html/vendor/behat/mink/src/WebAssert.php:130 | ||||||
|  | [localhost] out: /var/www/html/docroot/modules/custom/broadbean/tests/src/Functional/AddJobTest.php:66 | ||||||
|  | [localhost] out: | ||||||
|  | [localhost] out: FAILURES! | ||||||
|  | [localhost] out: Tests: 1, Assertions: 6, Errors: 1. | ||||||
|  | [localhost] out: | ||||||
|  | 
 | ||||||
|  | Warning: run() received nonzero return code 2 while executing '../../vendor/bin/phpunit ../modules/custom/broadbean'! | ||||||
|  | 
 | ||||||
|  | Fatal error: Tests failed! | ||||||
|  | 
 | ||||||
|  | Aborting. | ||||||
|  | ``` | ||||||
|  | 
 | ||||||
|  | --- | ||||||
|  | 
 | ||||||
|  | ## [fit] Is the site <br>still running? | ||||||
| 
 | 
 | ||||||
| --- | --- | ||||||
| 
 | 
 | ||||||
|  | @ -487,13 +578,13 @@ def main(builds_to_keep=3): | ||||||
| 
 | 
 | ||||||
| ```python | ```python | ||||||
| run(command).failed: | run(command).failed: | ||||||
|     # Fail |   # Fail | ||||||
| 
 |  | ||||||
| run(command).return_code == 0: |  | ||||||
|     # Pass |  | ||||||
| 
 | 
 | ||||||
| run(command).return_code == 1: | run(command).return_code == 1: | ||||||
|     # Fail |   # Fail | ||||||
|  | 
 | ||||||
|  | run(command).return_code == 0: | ||||||
|  |   # Pass | ||||||
| ``` | ``` | ||||||
| 
 | 
 | ||||||
| ^ Works for local and remote. | ^ Works for local and remote. | ||||||
|  | @ -501,12 +592,13 @@ run(command).return_code == 1: | ||||||
| --- | --- | ||||||
| 
 | 
 | ||||||
| ```python | ```python | ||||||
| def post_tasks(): | print 'Checking the site is alive...' | ||||||
|     print '===> Checking the site is alive.' | if run('drush status | egrep "Connected|Successful"').failed: | ||||||
|     if run('drush status | egrep "Connected|Successful"').failed: |   # Revert back to previous build. | ||||||
|         # Revert back to previous build. |  | ||||||
| ``` | ``` | ||||||
| 
 | 
 | ||||||
|  | ^ egrep is an acronym for "Extended Global Regular Expressions Print". It is a program which scans a specified file line by line, returning lines that contain a pattern matching a given regular expression. | ||||||
|  | 
 | ||||||
| --- | --- | ||||||
| 
 | 
 | ||||||
| ```bash, [.highlight 3] | ```bash, [.highlight 3] | ||||||
|  | @ -527,30 +619,31 @@ PHP configuration               :  /etc/php5/cli/php.ini | ||||||
| ... | ... | ||||||
| ``` | ``` | ||||||
| 
 | 
 | ||||||
| ^ "Database" to "PHP configuration" missing if cannot connect. | ^ Successful | ||||||
| 
 | 
 | ||||||
| --- | --- | ||||||
| 
 | 
 | ||||||
| ## Does the code still merge cleanly? | ```bash, [.highlight 3] | ||||||
|  | $ drush status | ||||||
| 
 | 
 | ||||||
| ^ Pre-task | Drupal version                  :  8.3.7 | ||||||
| 
 | Site URI                        :  http://default | ||||||
| --- | Database driver                 :  mysql | ||||||
| 
 | Database hostname               :  db | ||||||
| ```python | Database username               :  user | ||||||
| def check_for_merge_conflicts(target_branch): | Database name                   :  default | ||||||
|     with settings(warn_only=True): | PHP configuration               :  /etc/php5/cli/php.ini | ||||||
|         print('===> Ensuring that this can be merged into the main branch.') | ... | ||||||
| 
 |  | ||||||
|         if local('git fetch && git merge --no-ff origin/%s' % target_branch).failed: |  | ||||||
|             abort('Cannot merge into target branch.') |  | ||||||
| ``` | ``` | ||||||
| 
 | 
 | ||||||
|  | ^ Failed. | ||||||
|  | "Database" to "PHP configuration" missing if cannot connect or DB is empty. | ||||||
|  | 
 | ||||||
| --- | --- | ||||||
| 
 | 
 | ||||||
|  |  | ||||||
| 
 | 
 | ||||||
| ## Making fabric smarter | ## [fit] Making Fabric <br>Smarter | ||||||
| 
 | 
 | ||||||
| --- | --- | ||||||
| 
 | 
 | ||||||
|  | @ -560,9 +653,9 @@ def check_for_merge_conflicts(target_branch): | ||||||
| drupal_version = None | drupal_version = None | ||||||
| 
 | 
 | ||||||
| if exists('composer.json') and exists('core'): | if exists('composer.json') and exists('core'): | ||||||
|     drupal_version = 8 |   drupal_version = 8 | ||||||
| else: | else: | ||||||
|     drupal_version = 7 |   drupal_version = 7 | ||||||
| ``` | ``` | ||||||
| 
 | 
 | ||||||
| --- | --- | ||||||
|  | @ -571,16 +664,16 @@ else: | ||||||
| 
 | 
 | ||||||
| ```python | ```python | ||||||
| if exists('composer.json'): | if exists('composer.json'): | ||||||
|     run('composer install') |   run('composer install') | ||||||
| 
 | 
 | ||||||
| with cd('themes/custom/example'): | with cd('themes/custom/example'): | ||||||
|     if exists('package.json') and not exists('node_modules'): |   if exists('package.json') and not exists('node_modules'): | ||||||
|         run('yarn --pure-lockfile') |     run('yarn --pure-lockfile') | ||||||
| 
 | 
 | ||||||
|     if exists('gulpfile.js'): |   if exists('gulpfile.js'): | ||||||
|         run('node_modules/.bin/gulp --production') |     run('node_modules/.bin/gulp --production') | ||||||
|     elif exists('gruntfile.js'): |   elif exists('gruntfile.js'): | ||||||
|         run('node_modules/.bin/grunt build') |     run('node_modules/.bin/grunt build') | ||||||
| ``` | ``` | ||||||
| 
 | 
 | ||||||
| --- | --- | ||||||
|  | @ -591,21 +684,24 @@ with cd('themes/custom/example'): | ||||||
| # app.yml | # app.yml | ||||||
| 
 | 
 | ||||||
| drupal: | drupal: | ||||||
|     version: 8 |   version: 8 | ||||||
|     root: web |   root: web | ||||||
|     config: |   config: | ||||||
|         import: yes |     import: yes | ||||||
|         name: sync |     name: sync | ||||||
|         cmi_tools: no |     cmi_tools: no | ||||||
|     theme: |   tests: | ||||||
|         path: 'themes/custom/drupalbristol' |     simpletest: false | ||||||
|         build: |     phpunit: true | ||||||
|             npm: no |   theme: | ||||||
|             type: gulp |     path: 'themes/custom/drupalbristol' | ||||||
|             yarn: yes |     build: | ||||||
|  |       type: gulp | ||||||
|  |       npm: no | ||||||
|  |       yarn: yes | ||||||
| 
 | 
 | ||||||
| composer: | composer: | ||||||
|     install: true |   install: true | ||||||
| ``` | ``` | ||||||
| 
 | 
 | ||||||
| --- | --- | ||||||
|  | @ -618,7 +714,10 @@ composer: | ||||||
| from fabric.api import * | from fabric.api import * | ||||||
| import yaml | import yaml | ||||||
| 
 | 
 | ||||||
| with open('app.yml', 'r') as file: | config = [] | ||||||
|  | 
 | ||||||
|  | if exists('app.yml'): | ||||||
|  |   with open('app.yml', 'r') as file: | ||||||
|     config = yaml.load(file.read()) |     config = yaml.load(file.read()) | ||||||
| ``` | ``` | ||||||
| 
 | 
 | ||||||
|  | @ -630,7 +729,7 @@ with open('app.yml', 'r') as file: | ||||||
| # fabfile.py | # fabfile.py | ||||||
| 
 | 
 | ||||||
| if config['composer']['install'] == True: | if config['composer']['install'] == True: | ||||||
|     local('composer install') |   local('composer install') | ||||||
| ``` | ``` | ||||||
| 
 | 
 | ||||||
| --- | --- | ||||||
|  | @ -641,18 +740,61 @@ if config['composer']['install'] == True: | ||||||
| # fabfile.py | # fabfile.py | ||||||
| 
 | 
 | ||||||
| if build_type == 'drupal': | if build_type == 'drupal': | ||||||
|     drupal = config['drupal'] |   drupal = config['drupal'] | ||||||
|  | ``` | ||||||
| 
 | 
 | ||||||
|     with cd(drupal['root']): | --- | ||||||
|         if drupal['version'] == 8: |  | ||||||
|             if drupal['config']['import'] == True: |  | ||||||
|                 if drupal['config']['cmi_tools']: |  | ||||||
|                     run('drush cim -y %s' % drupal['config']['import']['name']) |  | ||||||
|                 else: |  | ||||||
|                     run('drush cimy -y %s' % drupal['config']['import']['name']) |  | ||||||
| 
 | 
 | ||||||
|         if drupal['version'] == 7: | ## Project settings file | ||||||
|             ... | 
 | ||||||
|  | ```python, [.highlight: 6-9] | ||||||
|  | # fabfile.py | ||||||
|  | 
 | ||||||
|  | if build_type == 'drupal': | ||||||
|  |   drupal = config['drupal'] | ||||||
|  | 
 | ||||||
|  |   with cd(drupal['root']): | ||||||
|  |     if drupal['version'] == 8: | ||||||
|  | 
 | ||||||
|  |     if drupal['version'] == 7: | ||||||
|  | ``` | ||||||
|  | 
 | ||||||
|  | --- | ||||||
|  | 
 | ||||||
|  | ## Project settings file | ||||||
|  | 
 | ||||||
|  | ```python, [.highlight: 8-10] | ||||||
|  | # fabfile.py | ||||||
|  | 
 | ||||||
|  | if build_type == 'drupal': | ||||||
|  |   drupal = config['drupal'] | ||||||
|  | 
 | ||||||
|  |   with cd(drupal['root']): | ||||||
|  |     if drupal['version'] == 8: | ||||||
|  |       if drupal['config']['import'] == True: | ||||||
|  |         # Import the staged configuration. | ||||||
|  |         run('drush cim -y %s' % drupal['config']['name']) | ||||||
|  | ``` | ||||||
|  | 
 | ||||||
|  | --- | ||||||
|  | 
 | ||||||
|  | ## Project settings file | ||||||
|  | 
 | ||||||
|  | ```python, [.highlight: 9-15] | ||||||
|  | # fabfile.py | ||||||
|  | 
 | ||||||
|  | if build_type == 'drupal': | ||||||
|  |   drupal = config['drupal'] | ||||||
|  | 
 | ||||||
|  |   with cd(drupal['root']): | ||||||
|  |     if drupal['version'] == 8: | ||||||
|  |       if drupal['config']['import'] == True: | ||||||
|  |         if drupal['config']['cmi_tools'] == True: | ||||||
|  |           # Use Drush CMI Tools. | ||||||
|  |           run('drush cimy -y %s' % drupal['config']['name']) | ||||||
|  |         else: | ||||||
|  |           # Use core. | ||||||
|  |           run('drush cim -y %s' % drupal['config']['name']) | ||||||
| ``` | ``` | ||||||
| 
 | 
 | ||||||
| ^ - Less hard-coded values | ^ - Less hard-coded values | ||||||
|  | @ -665,14 +807,16 @@ if build_type == 'drupal': | ||||||
| ## Project settings file | ## Project settings file | ||||||
| 
 | 
 | ||||||
| ```python | ```python | ||||||
|  | # fabfile.py | ||||||
|  | 
 | ||||||
| theme = config['theme'] | theme = config['theme'] | ||||||
| 
 | 
 | ||||||
| with cd(theme['path']): | with cd(theme['path']): | ||||||
|     if theme['build']['gulp'] == True: |   if theme['build']['gulp'] == True: | ||||||
|         if env == 'prod': |     if env == 'prod': | ||||||
|             run('node_modules/.bin/gulp --production') |       run('node_modules/.bin/gulp --production') | ||||||
|         else: |     else: | ||||||
|             run('node_modules/.bin/gulp') |       run('node_modules/.bin/gulp') | ||||||
| ``` | ``` | ||||||
| 
 | 
 | ||||||
| --- | --- | ||||||
|  | @ -683,14 +827,14 @@ with cd(theme['path']): | ||||||
| # app.yml | # app.yml | ||||||
| 
 | 
 | ||||||
| commands: | commands: | ||||||
|     build: | |   build: | | ||||||
|         cd web/themes/custom/drupalbristol |     cd web/themes/custom/drupalbristol | ||||||
|         yarn --pure-lockfile |     yarn --pure-lockfile | ||||||
|         node_modules/.bin/gulp --production |     node_modules/.bin/gulp --production | ||||||
| 
 | 
 | ||||||
|     deploy: | |   deploy: | | ||||||
|         cd web |     cd web | ||||||
|         drush cache-rebuild -y |     drush cache-rebuild -y | ||||||
| ``` | ``` | ||||||
| 
 | 
 | ||||||
| --- | --- | ||||||
|  | @ -701,19 +845,20 @@ commands: | ||||||
| # fabfile.py | # fabfile.py | ||||||
| 
 | 
 | ||||||
| for hook in config['commands'].get('build', '').split("\n"): | for hook in config['commands'].get('build', '').split("\n"): | ||||||
|     run(hook) |   run(hook) | ||||||
| 
 | 
 | ||||||
| ... | ... | ||||||
| 
 | 
 | ||||||
| for hook in config['commands'].get('deploy', '').split("\n"): | for hook in config['commands'].get('deploy', '').split("\n"): | ||||||
|     run(hook) |   run(hook) | ||||||
| ``` | ``` | ||||||
| 
 | 
 | ||||||
| --- | --- | ||||||
| 
 | 
 | ||||||
| ## Other things | ## Other things | ||||||
| 
 | 
 | ||||||
| - Run Drush/console/artisan commands | - Run Drush commands | ||||||
|  | - Run automated tests | ||||||
| - Verify file permissions | - Verify file permissions | ||||||
| - Restart services | - Restart services | ||||||
| - Anything you can do on the command line... | - Anything you can do on the command line... | ||||||
|  | @ -732,17 +877,17 @@ for hook in config['commands'].get('deploy', '').split("\n"): | ||||||
| 
 | 
 | ||||||
| [.build-lists: false] | [.build-lists: false] | ||||||
| 
 | 
 | ||||||
| - https://www.oliverdavies.uk/talks/deploying-php-fabric | - https://www.oliverdavies.uk/talks/deploying-drupal-fabric | ||||||
| - http://fabfile.org | - http://fabfile.org | ||||||
| - https://github.com/opdavies/fabric-example-sculpin |  | ||||||
| - https://github.com/opdavies/fabric-example-drupal | - https://github.com/opdavies/fabric-example-drupal | ||||||
|  | - https://github.com/opdavies/fabric-example-sculpin | ||||||
| - https://deploy.serversforhackers.com (~~$129~~ $79) | - https://deploy.serversforhackers.com (~~$129~~ $79) | ||||||
| 
 | 
 | ||||||
| --- | --- | ||||||
| 
 | 
 | ||||||
| ## *joind.in/talk/*4e35d | ## Thanks! | ||||||
| 
 | 
 | ||||||
| --- | # Questions? | ||||||
| 
 | 
 | ||||||
| ## @opdavies | ### @opdavies | ||||||
| ## *oliverdavies.uk* | ### oliverdavies.uk | ||||||
|  |  | ||||||
										
											Binary file not shown.
										
									
								
							
							
								
								
									
										
											BIN
										
									
								
								me-phpnw.png
									
										
									
									
									
								
							
							
						
						
									
										
											BIN
										
									
								
								me-phpnw.png
									
										
									
									
									
								
							
										
											Binary file not shown.
										
									
								
							| Before Width: | Height: | Size: 873 KiB After Width: | Height: | Size: 177 KiB | 
							
								
								
									
										
											BIN
										
									
								
								me-phpnw2.png
									
										
									
									
									
								
							
							
						
						
									
										
											BIN
										
									
								
								me-phpnw2.png
									
										
									
									
									
								
							
										
											Binary file not shown.
										
									
								
							| Before Width: | Height: | Size: 172 KiB | 
		Loading…
	
	Add table
		Add a link
		
	
		Reference in a new issue