Re-add old blog posts from Astro
This commit is contained in:
		
							parent
							
								
									076239fa25
								
							
						
					
					
						commit
						051e154c65
					
				
					 178 changed files with 13479 additions and 7 deletions
				
			
		
							
								
								
									
										192
									
								
								source/_posts/automating-sculpin-jenkins.md
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										192
									
								
								source/_posts/automating-sculpin-jenkins.md
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,192 @@ | |||
| --- | ||||
| title: Automating Sculpin Builds with Jenkins CI | ||||
| date: 2015-07-21 | ||||
| excerpt: How to use Jenkins to automate building Sculpin websites. | ||||
| tags: | ||||
|   - jenkins | ||||
|   - sculpin | ||||
| --- | ||||
| 
 | ||||
| As part of re-building this site with [Sculpin](http://sculpin.io), I wanted to | ||||
| automate the deployments, as in I wouldn't need to run a script like | ||||
| [publish.sh](https://raw.githubusercontent.com/sculpin/sculpin-blog-skeleton/master/publish.sh) | ||||
| locally and have that deploy my code onto my server. Not only did that mean that | ||||
| my local workflow was simpler (update, commit and push, rather than update, | ||||
| commit, push and deploy), but if I wanted to make a quick edit or hotfix, I | ||||
| could log into GitHub or Bitbucket (wherever I decided to host the source code) | ||||
| from any computer or my phone, make the change and have it deployed for me. | ||||
| 
 | ||||
| I'd started using [Jenkins CI](http://jenkins-ci.org) during my time at the | ||||
| Drupal Association, and had since built my own Jenkins server to handle | ||||
| deployments of Drupal websites, so that was the logical choice to use. | ||||
| 
 | ||||
| ## Installing Jenkins and Sculpin | ||||
| 
 | ||||
| If you don’t already have Jenkins installed and configured, I'd suggest using | ||||
| [Jeff Geerling](http://jeffgeerling.com/) (aka geerlingguy)'s | ||||
| [Ansible role for Jenkins CI](https://galaxy.ansible.com/list#/roles/440). | ||||
| 
 | ||||
| I've also released an | ||||
| [Ansible role for Sculpin](https://galaxy.ansible.com/list#/roles/4063) that | ||||
| installs the executable so that the Jenkins server can run Sculpin commands. | ||||
| 
 | ||||
| ## Triggering a Build from a Git Commit | ||||
| 
 | ||||
| I created a new Jenkins item for this task, and restricted where it could be run | ||||
| to `master` (i.e. the Jenkins server rather than any of the nodes). | ||||
| 
 | ||||
| ### Polling from Git | ||||
| 
 | ||||
| I entered the url to the | ||||
| [GitHub repo](https://github.com/opdavies/oliverdavies.uk) into the **Source | ||||
| Code Management** section (the Git option _may_ have been added by the | ||||
| [Git plugin](https://wiki.jenkins-ci.org/display/JENKINS/Git+Plugin) that I have | ||||
| installed). | ||||
| 
 | ||||
| As we don’t need any write access back to the repo, using the HTTP URL rather | ||||
| than the SSH one was fine, and I didn’t need to provide any additional | ||||
| credentials. | ||||
| 
 | ||||
| Also, as I knew that I’d be working a lot with feature branches, I entered | ||||
| `*/master` as the only branch to build. This meant that pushing changes or | ||||
| making edits on any other branches would not trigger a build. | ||||
| 
 | ||||
|  | ||||
| 
 | ||||
| I also checked the **Poll SCM** option so that Jenkins would be routinely | ||||
| checking for updated code. This essentially uses the same syntax as cron, | ||||
| specifying minutes, hours etc. I entered `* * * * *` so that Jenkins would poll | ||||
| each minute, knowing that I could make this less frequent if needed. | ||||
| 
 | ||||
| This now that Jenkins would be checking for any updates to the repo each minute, | ||||
| and could execute tasks if needed. | ||||
| 
 | ||||
| ### Building and Deploying | ||||
| 
 | ||||
| Within the **Builds** section of the item, I added an _Execute Shell_ step, | ||||
| where I could enter a command to execute. Here, I pasted a modified version of | ||||
| the original publish.sh script. | ||||
| 
 | ||||
| ```bash | ||||
| #!/bin/bash | ||||
| 
 | ||||
| set -uex | ||||
| 
 | ||||
| sculpin generate --env=prod --quiet | ||||
| if [ $? -ne 0 ]; then echo "Could not generate the site"; exit 1; fi | ||||
| 
 | ||||
| rsync -avze 'ssh' --delete output_prod/ prodwww2:/var/www/html/oliverdavies.uk/htdocs | ||||
| if [ $? -ne 0 ]; then echo "Could not publish the site"; exit 1; fi | ||||
| ``` | ||||
| 
 | ||||
| This essentially is the same as the original file, in that Sculpin generates the | ||||
| site, and uses rsync to deploy it somewhere else. In my case, `prodwww2` is a | ||||
| Jenkins node (this alias is configured in `/var/lib/jenkins/.ssh/config`), and | ||||
| `/var/www/html/oliverdavies.uk/htdocs` is the directory from where my site is | ||||
| served. | ||||
| 
 | ||||
| ## Building Periodically | ||||
| 
 | ||||
| There is some dynamic content on my site, specifically on the Talks page. Each | ||||
| talk has a date assigned to it, and within the Twig template, the talk is | ||||
| positoned within upcoming or previous talks based on whether this date is less | ||||
| or greater than the time of the build. | ||||
| 
 | ||||
| The YAML front matter: | ||||
| 
 | ||||
| ```yaml | ||||
| --- | ||||
| ... | ||||
| talks: | ||||
|     - title: Test Drive Twig with Sculpin | ||||
|             location: DrupalCamp North | ||||
| --- | ||||
| ``` | ||||
| 
 | ||||
| The Twig layout: | ||||
| 
 | ||||
| ```twig | ||||
| {% verbatim %} | ||||
| {% for talk in talks|reverse if talk.date >= now %} | ||||
|     {# Upcoming talks #} | ||||
| {% endfor %} | ||||
| 
 | ||||
| {% for talk in talks if talk.date < now %} | ||||
|     {# Previous talks #} | ||||
| {% endfor %} | ||||
| {% endverbatim %} | ||||
| ``` | ||||
| 
 | ||||
| I also didn’t want to have to push an empty commit or manually trigger a job in | ||||
| Jenkins after doing a talk in order for it to be positioned in the correct place | ||||
| on the page, so I also wanted Jenkins to schedule a regular build regardless of | ||||
| whether or not code had been pushed, so ensure that my talks page would be up to | ||||
| date. | ||||
| 
 | ||||
| After originally thinking that I'd have to split the build steps into a separate | ||||
| item and trigger that from a scheduled item, and amend my git commit item | ||||
| accordingly, I found a **Build periodically** option that I could use within the | ||||
| same item, leaving it intact and not having to make amends. | ||||
| 
 | ||||
| I set this to `@daily` (the same `H H * * *` - `H` is a Jenkins thing), so that | ||||
| the build would be triggered automatically each day without a commit, and deploy | ||||
| any updates to the site. | ||||
| 
 | ||||
|  | ||||
| 
 | ||||
| ## Next Steps | ||||
| 
 | ||||
| This workflow works great for one site, but as I roll out more Sculpin sites, | ||||
| I'd like to reduce duplication. I see this mainly as I’ll end up creating a | ||||
| separate `sculpin_build` item that’s decoupled from the site that it’s building, | ||||
| and instead passing variables such as environment, server name and docroot path | ||||
| as parameters in a parameterized build. | ||||
| 
 | ||||
| I'll probably also take the raw shell script out of Jenkins and save it in a | ||||
| text file that's stored locally on the server, and execute that via Jenkins. | ||||
| This means that I’d be able to store this file in a separate Git repository with | ||||
| my other Jenkins scripts and get the standard advantages of using version | ||||
| control. | ||||
| 
 | ||||
| ## Update | ||||
| 
 | ||||
| Since publishing this post, I've added some more items to the original build | ||||
| script. | ||||
| 
 | ||||
| ### Updating Composer | ||||
| 
 | ||||
| ```bash | ||||
| if [ -f composer.json ]; then | ||||
|     /usr/local/bin/composer install | ||||
| fi | ||||
| ``` | ||||
| 
 | ||||
| Updates project dependencies via | ||||
| [Composer](https://getcomposer.org/doc/00-intro.md#introduction) if | ||||
| composer.json exists. | ||||
| 
 | ||||
| ### Updating Sculpin Dependencies | ||||
| 
 | ||||
| ```bash | ||||
| if [ -f sculpin.json ]; then | ||||
|   sculpin install | ||||
| fi | ||||
| ``` | ||||
| 
 | ||||
| Runs `sculpin install` on each build if the sculpin.json file exists, to ensure | ||||
| that the required custom bundles and dependencies are installed. | ||||
| 
 | ||||
| ### Managing Redirects | ||||
| 
 | ||||
| ```bash | ||||
| if [ -f scripts/redirects.php ]; then | ||||
|     /usr/bin/php scripts/redirects.php | ||||
| fi | ||||
| ``` | ||||
| 
 | ||||
| I've been working on a `redirects.php` script that generates redirects from a | ||||
| .csv file, after seeing similar things in the | ||||
| [Pantheon Documentation](https://github.com/pantheon-systems/documentation) and | ||||
| [That Podcast](https://github.com/thatpodcast/thatpodcast.io) repositories. This | ||||
| checks if that file exists, and if so, runs it and generates the source file | ||||
| containing each redirect. | ||||
		Loading…
	
	Add table
		Add a link
		
	
		Reference in a new issue