capistrano-forkcms 2.0.4 → 3.0.0

Sign up to get free protection for your applications and to get access to all the features.
checksums.yaml CHANGED
@@ -1,7 +1,7 @@
1
1
  ---
2
2
  SHA1:
3
- metadata.gz: 01a1470d79a72c626f0b4d7f320fb9c36da374f3
4
- data.tar.gz: aa70ae7807a44ce30ad23bf4dd7ad80be1900bf9
3
+ metadata.gz: 18c6553a0978a49147150fe626a9e7f2924872bc
4
+ data.tar.gz: 8ad29409a350f8a642933877ed6ff57c4d74fbc9
5
5
  SHA512:
6
- metadata.gz: 3b2e3f53e445fd88fb27b6e1c2e8f6ce16857e673d8cb4b6769f7fec2d3c532a1ebd62bc56acc02847cc83df120fd68a60468f1b1550407ff5a8e7f025600526
7
- data.tar.gz: dbd2d5aac1e63a445f12d27f180148aa5399579726184facb2b610d04a32be62456586d7995b56e1544de3fbfb3549fccde2bde0a31bc066b6de047b87beb6df
6
+ metadata.gz: 0840f3aeafba8b18ac3ac8224bc1f4cc10ca8336eda6bf56fed05fb7f7b7eb229c52321d4c9a3bfaa2cce2f8b52808aef520266ee87b8b3948990ebcaf4494b0
7
+ data.tar.gz: f38e4063de7956cbc0817ca99aad19dea55bcca5f35b864be6c0a40dc220e8adb97c20eb61f25c4523f286a912c12f7b96f19d6269fe1d72ed1e7477dab350f0
data/README.md CHANGED
@@ -24,7 +24,7 @@ Require the module in your Capfile:
24
24
 
25
25
  require "capistrano/forkcms"
26
26
 
27
- The plugin comes with two tasks:
27
+ The plugin comes with some tasks:
28
28
 
29
29
  * `forkcms:configure:composer`, which will configure the `capistrano/composer` plugin.
30
30
  * `forkcms:opcache:reset`, which will reset the opcache.
@@ -43,6 +43,11 @@ Configuration options:
43
43
  required when `:opcache_reset_strategy` is `fcgi`.
44
44
  * `opcache_reset_base_url`, the public url of your website. Required when `:opcache_reset_strategy` is `file`
45
45
 
46
+ ### `:opcache_reset_strategy`
47
+
48
+ If you are using `file` as the strategy, you will need to alter the .htaccess-file to allow 'php-opcache-reset.php' to be accessed directly.
49
+
50
+ Add `RewriteRule ^php-opcache-reset\.php$ - [L]` below `RewriteRule ^index\.php$ - [L]` in the .htaccess-file
46
51
 
47
52
  ## How to use with a fresh Fork install
48
53
 
@@ -58,7 +63,7 @@ require 'capistrano/scm/git'
58
63
  install_plugin Capistrano::SCM::Git
59
64
  require 'capistrano/forkcms'
60
65
 
61
- set :format_options, log_file: 'app/logs/capistrano.log'
66
+ set :format_options, log_file: 'var/logs/capistrano.log'
62
67
 
63
68
  Dir.glob('app/config/capistrano/tasks/*.rake').each { |r| import r }
64
69
  ```
@@ -113,6 +118,25 @@ set :opcache_reset_fcgi_connection_string, "$your-php-fpm-socket-or-connection-s
113
118
  set :branch, "staging"
114
119
  ```
115
120
 
121
+ ## Migrations
122
+
123
+ Built in migrations allow you to easily update your database or locale when deploying a new codebase.
124
+
125
+ ### General
126
+
127
+ Each migration is contained in its own folder in the main `migrations` folder (e.g. `migrations/new-editor-feature`).
128
+ When deploying, this gem will check if `new-editor-feature` has been executed before. If it hasn't, it will be executed and saved.
129
+
130
+ ### Locale
131
+
132
+ To create a migration for the locale, create a file called `locale.xml` in your migration folder (e.g. `migrations/new-editor-feature/locale.xml`).
133
+ This file will be imported automatically using the `bin/console forkcms:locale:import` command.
134
+
135
+ ### Database
136
+
137
+ To create a migration for the database, create a file called `update.sql` in your migration folder (e.g. `migrations/new-editor-feature/update.sql`).
138
+ This file will be imported automatically using `mysql` so make sure you don't accidentally delete your database.
139
+
116
140
  ## Contributing
117
141
 
118
142
  Bug reports and pull requests are welcome on GitHub at [https://github.com/tijsverkoyen/capistrano-forkcms](https://github.com/tijsverkoyen/capistrano-forkcms).
@@ -120,4 +144,4 @@ Bug reports and pull requests are welcome on GitHub at [https://github.com/tijsv
120
144
 
121
145
  ## License
122
146
 
123
- The gem is available as open source under the terms of the [MIT License](http://opensource.org/licenses/MIT).
147
+ The gem is available as open source under the terms of the [MIT License](http://opensource.org/licenses/MIT).
@@ -17,10 +17,15 @@ namespace :deploy do
17
17
  after :starting, 'cachetool:install_executable'
18
18
  after :publishing, 'forkcms:symlink:document_root'
19
19
  after :publishing, 'forkcms:opcache:reset'
20
+ after :updated, 'forkcms:migrations:execute'
21
+ before :reverted, 'forkcms:migrations:rollback'
20
22
  end
21
23
 
22
24
 
23
25
  # Load the tasks
24
26
  load File.expand_path('../../tasks/configure.rake', __FILE__)
27
+ load File.expand_path('../../tasks/database.rake', __FILE__)
28
+ load File.expand_path('../../tasks/maintenance.rake', __FILE__)
29
+ load File.expand_path('../../tasks/migrations.rake', __FILE__)
25
30
  load File.expand_path('../../tasks/opcache.rake', __FILE__)
26
31
  load File.expand_path('../../tasks/symlink.rake', __FILE__)
@@ -1,5 +1,5 @@
1
1
  module Capistrano
2
2
  module Forkcms
3
- VERSION = '2.0.4'
3
+ VERSION = '3.0.0'
4
4
  end
5
5
  end
@@ -21,25 +21,4 @@ namespace :forkcms do
21
21
  SSHKit.config.command_map[:cachetool] = "#{fetch :php_bin_path} #{fetch :deploy_to}/shared/cachetool.phar"
22
22
  end
23
23
  end
24
-
25
- namespace :symlink do
26
- desc <<-DESC
27
- Links the document root to the current folder
28
- The document root is the folder that the webserver will serve, it should link to the current path.
29
- DESC
30
- task :document_root do
31
- on roles(:web) do
32
- if test("[ -L #{fetch :document_root} ]") && capture("readlink -- #{fetch :document_root}") == "#{current_path}/"
33
- # all is well, the symlink is correct
34
- elsif test("[ -d #{fetch :document_root} ]") || test("[ -f #{fetch :document_root} ]")
35
- error "Document root #{fetch :document_root} already exists."
36
- error 'To link it, issue the following command:'
37
- error "ln -sf #{current_path}/ #{fetch :document_root}"
38
- else
39
- execute :mkdir, '-p', File.dirname(fetch(:document_root))
40
- execute :ln, '-sf', "#{current_path}/", fetch(:document_root)
41
- end
42
- end
43
- end
44
- end
45
24
  end
@@ -0,0 +1,92 @@
1
+ namespace :forkcms do
2
+ namespace :database do
3
+ backup_file = 'mysql_backup.sql'
4
+
5
+ desc 'Create a backup of the database'
6
+ task :backup do
7
+ on roles(:web) do
8
+ flags = get_mysql_connection_flags
9
+ execute :mysqldump, "#{flags} --skip-lock-tables > #{release_path}/#{backup_file}"
10
+ end
11
+ end
12
+
13
+ desc 'Restore the database from a backup file'
14
+ task :restore do
15
+ on roles(:web) do
16
+ if not test "[[ -f #{release_path}/#{backup_file} ]]"
17
+ error "No backup file found, create a backup first"
18
+ else
19
+ Rake::Task["forkcms:database:execute"].invoke("#{release_path}/#{backup_file}")
20
+ end
21
+ end
22
+ end
23
+
24
+ desc 'Execute an SQL file'
25
+ task :execute, [:file] do |t, arguments|
26
+ on roles(:web) do
27
+ if not test "[[ -f #{arguments[:file]} ]]"
28
+ raise "File at #{arguments[:file]} does not exist"
29
+ end
30
+
31
+ # Execute the file
32
+ flags = get_mysql_connection_flags
33
+ execute :mysql,"#{flags} < #{arguments[:file]}"
34
+ end
35
+ end
36
+
37
+ private
38
+ def get_parameters
39
+ parameter_path = "#{shared_path}/app/config/parameters.yml"
40
+ if not test "[[ -f #{parameter_path} ]]"
41
+ raise "parameters.yml not found, it should be at #{parameter_path}"
42
+ end
43
+
44
+ # Fetch the content of the parameters
45
+ parameters_content = capture "cat #{parameter_path}"
46
+ # It seems we use invalid yml in our config files
47
+ # Therefore we need to fix some issues with it.
48
+ parameters_content = parameters_content.gsub(/:(\s*)%(.*)/, ':\1"%\2"')
49
+
50
+ parameters_content = YAML::load(parameters_content)
51
+
52
+ # Return them
53
+ return parameters_content.fetch('parameters')
54
+ end
55
+
56
+ def get_mysql_connection_flags
57
+ # Fetch our parameters to reach the database
58
+ parameters = get_parameters
59
+
60
+ # Abort if no parameters are found
61
+ if parameters == nil
62
+ return
63
+ end
64
+
65
+ # Define our mapping
66
+ mapping = {
67
+ 'host' => parameters.fetch('database.host'),
68
+ 'port' => parameters.fetch('database.port'),
69
+ 'user' => parameters.fetch('database.user'),
70
+ 'password' => parameters.fetch('database.password'),
71
+ }
72
+
73
+ # Set the default flags
74
+ flags = "--default-character-set='utf8' "
75
+
76
+ # Append each mapped property to our flags
77
+ mapping.each do |key, value|
78
+ raise "\"#{key}\" is not set in the parameters.yml file" if value.nil?
79
+
80
+ flags << "--#{key}=#{value} "
81
+ end
82
+
83
+ # Append our database
84
+ database = parameters.fetch('database.name')
85
+ raise "\"database.name\" is not set in the parameters.yml file" if database.nil?
86
+
87
+ flags << "#{database}"
88
+
89
+ return flags
90
+ end
91
+ end
92
+ end
@@ -0,0 +1,37 @@
1
+ namespace :forkcms do
2
+ namespace :maintenance do
3
+ desc 'Enable maintenance mode'
4
+ task :enable do
5
+ on roles(:web) do
6
+ create_maintenance_folder
7
+ execute :rm, '-rf', "#{fetch :document_root} && ln -sf #{shared_path}/maintenance #{fetch :document_root}"
8
+ end
9
+ end
10
+
11
+ desc 'Disable maintenance mode'
12
+ task :disable do
13
+ on roles(:web) do
14
+ execute :rm, '-rf', "#{fetch :document_root} && ln -sf #{current_path} #{fetch :document_root}"
15
+ end
16
+ end
17
+
18
+ private
19
+ # Creates the maintenance folder based on the local maintenance folder to display when migrating
20
+ def create_maintenance_folder
21
+ # Stop if the maintenance folder exists
22
+ return if capture("if [ -d #{shared_path}/maintenance ]; then echo 'yes'; fi").chomp == 'yes'
23
+
24
+ lib_path = File.dirname(__FILE__)
25
+ local_maintenance_path = "#{lib_path}/../../maintenance"
26
+
27
+ # Create a maintenance folder containing the index page from our gem
28
+ execute :mkdir, "-p #{shared_path}/maintenance"
29
+
30
+ # copy the contents of the index.html file to our shared folder
31
+ upload! File.open(local_maintenance_path + '/index.html'), "#{shared_path}/maintenance/index.html"
32
+
33
+ # copy the contents of the .htaccess file to our shared folder
34
+ upload! File.open(local_maintenance_path + '/.htaccess'), "#{shared_path}/maintenance/.htaccess"
35
+ end
36
+ end
37
+ end
@@ -0,0 +1,115 @@
1
+ namespace :forkcms do
2
+ namespace :migrations do
3
+ desc 'Prepare the server for running Fork CMS migrations'
4
+ task :prepare do
5
+ on roles(:web) do
6
+ create_migrations_file
7
+ end
8
+ end
9
+
10
+ desc 'Execute pending migrations'
11
+ task :execute do
12
+ on roles(:web) do
13
+ # Prepare for migrations if it's not done already
14
+ Rake::Task["forkcms:migrations:prepare"].invoke()
15
+
16
+ # Abort if no migrations are found
17
+ migration_folders = get_migration_folders
18
+ next if migration_folders.length == 0
19
+
20
+ migrations_to_execute = get_migrations_to_execute
21
+
22
+ # Abort if no new migrations are found
23
+ next if migrations_to_execute.length == 0
24
+
25
+ # As migrations can take a while we show a maintenance page
26
+ Rake::Task["forkcms:maintenance:enable"].invoke()
27
+
28
+ # Back up the database, just in case
29
+ Rake::Task["forkcms:database:backup"].invoke()
30
+
31
+ # Execute all migrations
32
+ migrations_to_execute.each do |dirname|
33
+ migration_path = "#{release_path}/migrations/#{dirname}"
34
+ migration_files = capture("ls -1 #{migration_path}").split(/\r?\n/)
35
+
36
+ migration_files.each do |filename|
37
+ # Execute a MySQL file
38
+ if filename.index('update.sql') != nil
39
+ Rake::Task["forkcms:database:execute"].invoke("#{migration_path}/#{filename}")
40
+
41
+ next
42
+ end
43
+
44
+ # Update the locale through the console command
45
+ if filename.index('locale.xml') != nil
46
+ execute :php, "#{release_path}/bin/console forkcms:locale:import -f #{migration_path}/#{filename}"
47
+
48
+ next
49
+ end
50
+ end
51
+ end
52
+
53
+ # All migrations were executed successfully and we didn't roll back so put them in the executed_migrations file
54
+ migrations_to_execute.each do |dirname|
55
+ execute :echo , "#{dirname} | tee -a #{shared_path}/executed_migrations"
56
+ end
57
+
58
+ # Disable maintenance mode, everything is done
59
+ Rake::Task["forkcms:maintenance:disable"].invoke()
60
+ end
61
+ end
62
+
63
+ desc 'Rollback the migrations'
64
+ task :rollback do
65
+ # Restore the database backup so we undo any executed migrations
66
+ Rake::Task["forkcms:database:restore"].invoke()
67
+
68
+ # Disable the maintenance page so the site is accessible again
69
+ Rake::Task["forkcms:maintenance:disable"].invoke()
70
+ end
71
+
72
+ private
73
+
74
+ # Creates a migration file to hold our executed migrations
75
+ def create_migrations_file
76
+ # Stop if the migrations file exists
77
+ return if test "[[ -f #{shared_path}/executed_migrations ]]"
78
+
79
+ # Create an empty executed_migrations file
80
+ upload! StringIO.new(''), "#{shared_path}/executed_migrations"
81
+
82
+ # If we just created the executed_migrations file, add all existing migrations
83
+ execute_initial_migration
84
+ end
85
+
86
+ # Put all items in the migrations folder in the executed_migrations file
87
+ def execute_initial_migration
88
+ migration_folders = get_migration_folders
89
+
90
+ migration_folders.each do |dirname|
91
+ run "echo #{dirname} | tee -a #{shared_path}/executed_migrations"
92
+ end
93
+ end
94
+
95
+ # Gets the new migrations to execute
96
+ def get_migrations_to_execute
97
+ executed_migrations = capture("cat #{shared_path}/executed_migrations").chomp.split(/\r?\n/)
98
+ migrations_to_execute = Array.new
99
+ migration_folders = get_migration_folders
100
+
101
+ # Fetch all migration directories that aren't executed yet
102
+ migration_folders.each do |dirname|
103
+ if executed_migrations.index(dirname) == nil
104
+ migrations_to_execute.push(dirname)
105
+ end
106
+ end
107
+
108
+ return migrations_to_execute
109
+ end
110
+
111
+ def get_migration_folders
112
+ return capture("if [ -e #{release_path}/migrations ]; then ls -1 #{release_path}/migrations; fi").split(/\r?\n/)
113
+ end
114
+ end
115
+ end
@@ -0,0 +1,10 @@
1
+ # All url's will see the maintenance page (index.html)
2
+ # This way they are back to there original page after maintenance is finished.
3
+ <IfModule mod_rewrite.c>
4
+ # nice urls
5
+ RewriteEngine On
6
+ RewriteBase /
7
+
8
+ # handle urls
9
+ RewriteRule . index.html [NC,L]
10
+ </IfModule>
@@ -0,0 +1,75 @@
1
+ <!DOCTYPE HTML>
2
+ <html lang="en">
3
+ <head>
4
+ <meta charset="utf-8">
5
+ <title>Maintenance break!</title>
6
+ <style type="text/css">
7
+ body
8
+ {
9
+ background-color: #ebf3f9;
10
+ margin: 0;
11
+ padding: 0;
12
+ border: 0;
13
+ font-family: Arial;
14
+ }
15
+ h1
16
+ {
17
+ color: #333333;
18
+ font-size: 20px;
19
+ line-height: 0;
20
+ margin-bottom: 30px;
21
+ }
22
+ p
23
+ {
24
+ color: #666666;
25
+ font-size: 13px;
26
+ }
27
+ #container
28
+ {
29
+ position: absolute;
30
+ left: 50%;
31
+ top: 50%;
32
+ width: 340px;
33
+ height: 471px;
34
+ margin-left: -200px;
35
+ margin-top: -250px;
36
+ }
37
+ #balloon
38
+ {
39
+ background-color: #FFF;
40
+ width: 340px;
41
+ padding: 30px;
42
+ border: 1px solid #c3c3c3;
43
+ -webkit-border-radius: 3px;
44
+ -moz-border-radius: 3px;
45
+ border-radius: 3px;
46
+ }
47
+ #triangle
48
+ {
49
+ background-image: url();
50
+ margin-left: 180px;
51
+ margin-top: -1px;
52
+ width: 40px;
53
+ height: 21px;
54
+ }
55
+ #bottom
56
+ {
57
+ background-image: url();
58
+ width: 400px;
59
+ height: 271px;
60
+ margin-top: 10px;
61
+ }
62
+
63
+ </style>
64
+ </head>
65
+ <body>
66
+ <div id="container">
67
+ <div id="balloon">
68
+ <h1>Maintenance break!</h1>
69
+ <p>Our web wizards are doing some maintenance magic on the site. Go grab a cup of coffee, we'll be back up and running asap.</p>
70
+ </div>
71
+ <div id="triangle">&nbsp;</div>
72
+ <div id="bottom">&nbsp;</div>
73
+ </div>
74
+ </body>
75
+ </html>
metadata CHANGED
@@ -1,97 +1,97 @@
1
1
  --- !ruby/object:Gem::Specification
2
2
  name: capistrano-forkcms
3
3
  version: !ruby/object:Gem::Version
4
- version: 2.0.4
4
+ version: 3.0.0
5
5
  platform: ruby
6
6
  authors:
7
7
  - Tijs Verkoyen
8
8
  autorequire:
9
9
  bindir: bin
10
10
  cert_chain: []
11
- date: 2017-07-27 00:00:00.000000000 Z
11
+ date: 2017-10-05 00:00:00.000000000 Z
12
12
  dependencies:
13
13
  - !ruby/object:Gem::Dependency
14
14
  name: bundler
15
15
  requirement: !ruby/object:Gem::Requirement
16
16
  requirements:
17
- - - "~>"
17
+ - - ~>
18
18
  - !ruby/object:Gem::Version
19
19
  version: '1.14'
20
20
  type: :development
21
21
  prerelease: false
22
22
  version_requirements: !ruby/object:Gem::Requirement
23
23
  requirements:
24
- - - "~>"
24
+ - - ~>
25
25
  - !ruby/object:Gem::Version
26
26
  version: '1.14'
27
27
  - !ruby/object:Gem::Dependency
28
28
  name: rake
29
29
  requirement: !ruby/object:Gem::Requirement
30
30
  requirements:
31
- - - "~>"
31
+ - - ~>
32
32
  - !ruby/object:Gem::Version
33
33
  version: '10.0'
34
34
  type: :development
35
35
  prerelease: false
36
36
  version_requirements: !ruby/object:Gem::Requirement
37
37
  requirements:
38
- - - "~>"
38
+ - - ~>
39
39
  - !ruby/object:Gem::Version
40
40
  version: '10.0'
41
41
  - !ruby/object:Gem::Dependency
42
42
  name: rspec
43
43
  requirement: !ruby/object:Gem::Requirement
44
44
  requirements:
45
- - - "~>"
45
+ - - ~>
46
46
  - !ruby/object:Gem::Version
47
47
  version: '3.0'
48
48
  type: :development
49
49
  prerelease: false
50
50
  version_requirements: !ruby/object:Gem::Requirement
51
51
  requirements:
52
- - - "~>"
52
+ - - ~>
53
53
  - !ruby/object:Gem::Version
54
54
  version: '3.0'
55
55
  - !ruby/object:Gem::Dependency
56
56
  name: capistrano
57
57
  requirement: !ruby/object:Gem::Requirement
58
58
  requirements:
59
- - - "~>"
59
+ - - ~>
60
60
  - !ruby/object:Gem::Version
61
61
  version: '3.8'
62
62
  type: :runtime
63
63
  prerelease: false
64
64
  version_requirements: !ruby/object:Gem::Requirement
65
65
  requirements:
66
- - - "~>"
66
+ - - ~>
67
67
  - !ruby/object:Gem::Version
68
68
  version: '3.8'
69
69
  - !ruby/object:Gem::Dependency
70
70
  name: capistrano-composer
71
71
  requirement: !ruby/object:Gem::Requirement
72
72
  requirements:
73
- - - "~>"
73
+ - - ~>
74
74
  - !ruby/object:Gem::Version
75
75
  version: 0.0.6
76
76
  type: :runtime
77
77
  prerelease: false
78
78
  version_requirements: !ruby/object:Gem::Requirement
79
79
  requirements:
80
- - - "~>"
80
+ - - ~>
81
81
  - !ruby/object:Gem::Version
82
82
  version: 0.0.6
83
83
  - !ruby/object:Gem::Dependency
84
84
  name: capistrano-cachetool
85
85
  requirement: !ruby/object:Gem::Requirement
86
86
  requirements:
87
- - - "~>"
87
+ - - ~>
88
88
  - !ruby/object:Gem::Version
89
89
  version: '1.0'
90
90
  type: :runtime
91
91
  prerelease: false
92
92
  version_requirements: !ruby/object:Gem::Requirement
93
93
  requirements:
94
- - - "~>"
94
+ - - ~>
95
95
  - !ruby/object:Gem::Version
96
96
  version: '1.0'
97
97
  description: Capistrano ForkCMS - Easy deployment of ForkCMS 5+ apps with Ruby over
@@ -102,9 +102,9 @@ executables: []
102
102
  extensions: []
103
103
  extra_rdoc_files: []
104
104
  files:
105
- - ".gitignore"
106
- - ".rspec"
107
- - ".travis.yml"
105
+ - .gitignore
106
+ - .rspec
107
+ - .travis.yml
108
108
  - Gemfile
109
109
  - LICENSE.md
110
110
  - README.md
@@ -115,8 +115,13 @@ files:
115
115
  - lib/capistrano/forkcms/defaults.rb
116
116
  - lib/capistrano/forkcms/version.rb
117
117
  - lib/capistrano/tasks/configure.rake
118
+ - lib/capistrano/tasks/database.rake
119
+ - lib/capistrano/tasks/maintenance.rake
120
+ - lib/capistrano/tasks/migrations.rake
118
121
  - lib/capistrano/tasks/opcache.rake
119
122
  - lib/capistrano/tasks/symlink.rake
123
+ - lib/maintenance/.htaccess
124
+ - lib/maintenance/index.html
120
125
  homepage: https://github.com/tijsverkoyen/capistrano-forkcms
121
126
  licenses:
122
127
  - MIT
@@ -127,17 +132,17 @@ require_paths:
127
132
  - lib
128
133
  required_ruby_version: !ruby/object:Gem::Requirement
129
134
  requirements:
130
- - - ">="
135
+ - - '>='
131
136
  - !ruby/object:Gem::Version
132
137
  version: '0'
133
138
  required_rubygems_version: !ruby/object:Gem::Requirement
134
139
  requirements:
135
- - - ">="
140
+ - - '>='
136
141
  - !ruby/object:Gem::Version
137
142
  version: '0'
138
143
  requirements: []
139
144
  rubyforge_project:
140
- rubygems_version: 2.6.11
145
+ rubygems_version: 2.0.14.1
141
146
  signing_key:
142
147
  specification_version: 4
143
148
  summary: Fork CMS specific Capistrano tasks