capistrano 2.8.0 → 3.19.0
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- checksums.yaml +7 -0
- data/.docker/Dockerfile +7 -0
- data/.docker/ssh_key_rsa +49 -0
- data/.docker/ssh_key_rsa.pub +1 -0
- data/.docker/ubuntu_setup.sh +23 -0
- data/.github/issue_template.md +19 -0
- data/.github/pull_request_template.md +22 -0
- data/.github/release-drafter.yml +25 -0
- data/.github/workflows/ci.yml +80 -0
- data/.github/workflows/release-drafter.yml +18 -0
- data/.gitignore +23 -8
- data/.rubocop.yml +62 -0
- data/CHANGELOG.md +1 -0
- data/CONTRIBUTING.md +63 -0
- data/DEVELOPMENT.md +112 -0
- data/Gemfile +42 -9
- data/LICENSE.txt +21 -0
- data/README.md +221 -0
- data/RELEASING.md +17 -0
- data/Rakefile +17 -8
- data/UPGRADING-3.7.md +86 -0
- data/bin/cap +2 -3
- data/bin/capify +7 -89
- data/capistrano.gemspec +29 -43
- data/docker-compose.yml +8 -0
- data/features/configuration.feature +28 -0
- data/features/deploy.feature +92 -0
- data/features/deploy_failure.feature +17 -0
- data/features/doctor.feature +11 -0
- data/features/installation.feature +21 -0
- data/features/sshconnect.feature +11 -0
- data/features/stage_failure.feature +9 -0
- data/features/step_definitions/assertions.rb +162 -0
- data/features/step_definitions/cap_commands.rb +21 -0
- data/features/step_definitions/setup.rb +91 -0
- data/features/subdirectory.feature +9 -0
- data/features/support/docker_gateway.rb +53 -0
- data/features/support/env.rb +1 -0
- data/features/support/remote_command_helpers.rb +29 -0
- data/features/support/remote_ssh_helpers.rb +33 -0
- data/lib/Capfile +3 -0
- data/lib/capistrano/all.rb +17 -0
- data/lib/capistrano/application.rb +153 -0
- data/lib/capistrano/configuration/empty_filter.rb +9 -0
- data/lib/capistrano/configuration/filter.rb +26 -0
- data/lib/capistrano/configuration/host_filter.rb +29 -0
- data/lib/capistrano/configuration/null_filter.rb +9 -0
- data/lib/capistrano/configuration/plugin_installer.rb +51 -0
- data/lib/capistrano/configuration/question.rb +76 -0
- data/lib/capistrano/configuration/role_filter.rb +29 -0
- data/lib/capistrano/configuration/scm_resolver.rb +149 -0
- data/lib/capistrano/configuration/server.rb +137 -0
- data/lib/capistrano/configuration/servers.rb +56 -96
- data/lib/capistrano/configuration/validated_variables.rb +110 -0
- data/lib/capistrano/configuration/variables.rb +79 -94
- data/lib/capistrano/configuration.rb +178 -33
- data/lib/capistrano/console.rb +1 -0
- data/lib/capistrano/defaults.rb +36 -0
- data/lib/capistrano/deploy.rb +3 -0
- data/lib/capistrano/doctor/environment_doctor.rb +19 -0
- data/lib/capistrano/doctor/gems_doctor.rb +45 -0
- data/lib/capistrano/doctor/output_helpers.rb +79 -0
- data/lib/capistrano/doctor/servers_doctor.rb +105 -0
- data/lib/capistrano/doctor/variables_doctor.rb +74 -0
- data/lib/capistrano/doctor.rb +6 -0
- data/lib/capistrano/dotfile.rb +2 -0
- data/lib/capistrano/dsl/env.rb +43 -0
- data/lib/capistrano/dsl/paths.rb +89 -0
- data/lib/capistrano/dsl/stages.rb +31 -0
- data/lib/capistrano/dsl/task_enhancements.rb +61 -0
- data/lib/capistrano/dsl.rb +95 -0
- data/lib/capistrano/framework.rb +2 -0
- data/lib/capistrano/i18n.rb +46 -0
- data/lib/capistrano/immutable_task.rb +30 -0
- data/lib/capistrano/install.rb +1 -0
- data/lib/capistrano/plugin.rb +95 -0
- data/lib/capistrano/proc_helpers.rb +13 -0
- data/lib/capistrano/scm/git.rb +105 -0
- data/lib/capistrano/scm/hg.rb +55 -0
- data/lib/capistrano/scm/plugin.rb +13 -0
- data/lib/capistrano/scm/svn.rb +56 -0
- data/lib/capistrano/scm/tasks/git.rake +84 -0
- data/lib/capistrano/scm/tasks/hg.rake +53 -0
- data/lib/capistrano/scm/tasks/svn.rake +53 -0
- data/lib/capistrano/scm.rb +115 -0
- data/lib/capistrano/setup.rb +36 -0
- data/lib/capistrano/tasks/console.rake +25 -0
- data/lib/capistrano/tasks/deploy.rake +280 -0
- data/lib/capistrano/tasks/doctor.rake +24 -0
- data/lib/capistrano/tasks/framework.rake +67 -0
- data/lib/capistrano/tasks/install.rake +41 -0
- data/lib/capistrano/templates/Capfile +38 -0
- data/lib/capistrano/templates/deploy.rb.erb +39 -0
- data/lib/capistrano/templates/stage.rb.erb +61 -0
- data/lib/capistrano/upload_task.rb +9 -0
- data/lib/capistrano/version.rb +1 -14
- data/lib/capistrano/version_validator.rb +32 -0
- data/lib/capistrano.rb +0 -3
- data/spec/integration/dsl_spec.rb +632 -0
- data/spec/integration_spec_helper.rb +5 -0
- data/spec/lib/capistrano/application_spec.rb +60 -0
- data/spec/lib/capistrano/configuration/empty_filter_spec.rb +17 -0
- data/spec/lib/capistrano/configuration/filter_spec.rb +109 -0
- data/spec/lib/capistrano/configuration/host_filter_spec.rb +71 -0
- data/spec/lib/capistrano/configuration/null_filter_spec.rb +17 -0
- data/spec/lib/capistrano/configuration/plugin_installer_spec.rb +98 -0
- data/spec/lib/capistrano/configuration/question_spec.rb +92 -0
- data/spec/lib/capistrano/configuration/role_filter_spec.rb +80 -0
- data/spec/lib/capistrano/configuration/scm_resolver_spec.rb +56 -0
- data/spec/lib/capistrano/configuration/server_spec.rb +309 -0
- data/spec/lib/capistrano/configuration/servers_spec.rb +331 -0
- data/spec/lib/capistrano/configuration_spec.rb +357 -0
- data/spec/lib/capistrano/doctor/environment_doctor_spec.rb +44 -0
- data/spec/lib/capistrano/doctor/gems_doctor_spec.rb +67 -0
- data/spec/lib/capistrano/doctor/output_helpers_spec.rb +47 -0
- data/spec/lib/capistrano/doctor/servers_doctor_spec.rb +86 -0
- data/spec/lib/capistrano/doctor/variables_doctor_spec.rb +89 -0
- data/spec/lib/capistrano/dsl/paths_spec.rb +228 -0
- data/spec/lib/capistrano/dsl/task_enhancements_spec.rb +108 -0
- data/spec/lib/capistrano/dsl_spec.rb +125 -0
- data/spec/lib/capistrano/immutable_task_spec.rb +31 -0
- data/spec/lib/capistrano/plugin_spec.rb +84 -0
- data/spec/lib/capistrano/scm/git_spec.rb +194 -0
- data/spec/lib/capistrano/scm/hg_spec.rb +109 -0
- data/spec/lib/capistrano/scm/svn_spec.rb +137 -0
- data/spec/lib/capistrano/scm_spec.rb +103 -0
- data/spec/lib/capistrano/upload_task_spec.rb +19 -0
- data/spec/lib/capistrano/version_validator_spec.rb +118 -0
- data/spec/lib/capistrano_spec.rb +7 -0
- data/spec/spec_helper.rb +29 -0
- data/spec/support/matchers.rb +5 -0
- data/spec/support/tasks/database.rake +11 -0
- data/spec/support/tasks/fail.rake +8 -0
- data/spec/support/tasks/failed.rake +5 -0
- data/spec/support/tasks/plugin.rake +6 -0
- data/spec/support/tasks/root.rake +11 -0
- data/spec/support/test_app.rb +205 -0
- metadata +234 -208
- data/.rvmrc +0 -1
- data/CHANGELOG +0 -954
- data/README.mdown +0 -76
- data/lib/capistrano/callback.rb +0 -45
- data/lib/capistrano/cli/execute.rb +0 -85
- data/lib/capistrano/cli/help.rb +0 -125
- data/lib/capistrano/cli/help.txt +0 -81
- data/lib/capistrano/cli/options.rb +0 -243
- data/lib/capistrano/cli/ui.rb +0 -40
- data/lib/capistrano/cli.rb +0 -47
- data/lib/capistrano/command.rb +0 -286
- data/lib/capistrano/configuration/actions/file_transfer.rb +0 -51
- data/lib/capistrano/configuration/actions/inspect.rb +0 -46
- data/lib/capistrano/configuration/actions/invocation.rb +0 -298
- data/lib/capistrano/configuration/callbacks.rb +0 -148
- data/lib/capistrano/configuration/connections.rb +0 -230
- data/lib/capistrano/configuration/execution.rb +0 -143
- data/lib/capistrano/configuration/loading.rb +0 -197
- data/lib/capistrano/configuration/namespaces.rb +0 -197
- data/lib/capistrano/configuration/roles.rb +0 -73
- data/lib/capistrano/errors.rb +0 -19
- data/lib/capistrano/ext/string.rb +0 -5
- data/lib/capistrano/extensions.rb +0 -57
- data/lib/capistrano/logger.rb +0 -59
- data/lib/capistrano/processable.rb +0 -53
- data/lib/capistrano/recipes/compat.rb +0 -32
- data/lib/capistrano/recipes/deploy/assets.rb +0 -57
- data/lib/capistrano/recipes/deploy/dependencies.rb +0 -44
- data/lib/capistrano/recipes/deploy/local_dependency.rb +0 -54
- data/lib/capistrano/recipes/deploy/remote_dependency.rb +0 -111
- data/lib/capistrano/recipes/deploy/scm/accurev.rb +0 -169
- data/lib/capistrano/recipes/deploy/scm/base.rb +0 -196
- data/lib/capistrano/recipes/deploy/scm/bzr.rb +0 -86
- data/lib/capistrano/recipes/deploy/scm/cvs.rb +0 -153
- data/lib/capistrano/recipes/deploy/scm/darcs.rb +0 -96
- data/lib/capistrano/recipes/deploy/scm/git.rb +0 -282
- data/lib/capistrano/recipes/deploy/scm/mercurial.rb +0 -137
- data/lib/capistrano/recipes/deploy/scm/none.rb +0 -44
- data/lib/capistrano/recipes/deploy/scm/perforce.rb +0 -138
- data/lib/capistrano/recipes/deploy/scm/subversion.rb +0 -121
- data/lib/capistrano/recipes/deploy/scm.rb +0 -19
- data/lib/capistrano/recipes/deploy/strategy/base.rb +0 -88
- data/lib/capistrano/recipes/deploy/strategy/checkout.rb +0 -20
- data/lib/capistrano/recipes/deploy/strategy/copy.rb +0 -224
- data/lib/capistrano/recipes/deploy/strategy/export.rb +0 -20
- data/lib/capistrano/recipes/deploy/strategy/remote.rb +0 -52
- data/lib/capistrano/recipes/deploy/strategy/remote_cache.rb +0 -57
- data/lib/capistrano/recipes/deploy/strategy.rb +0 -19
- data/lib/capistrano/recipes/deploy/templates/maintenance.rhtml +0 -53
- data/lib/capistrano/recipes/deploy.rb +0 -568
- data/lib/capistrano/recipes/standard.rb +0 -37
- data/lib/capistrano/recipes/templates/maintenance.rhtml +0 -53
- data/lib/capistrano/role.rb +0 -102
- data/lib/capistrano/server_definition.rb +0 -56
- data/lib/capistrano/shell.rb +0 -260
- data/lib/capistrano/ssh.rb +0 -101
- data/lib/capistrano/task_definition.rb +0 -75
- data/lib/capistrano/transfer.rb +0 -216
- data/rvmrc.sample +0 -1
- data/test/cli/execute_test.rb +0 -132
- data/test/cli/help_test.rb +0 -165
- data/test/cli/options_test.rb +0 -329
- data/test/cli/ui_test.rb +0 -28
- data/test/cli_test.rb +0 -17
- data/test/command_test.rb +0 -289
- data/test/configuration/actions/file_transfer_test.rb +0 -61
- data/test/configuration/actions/inspect_test.rb +0 -65
- data/test/configuration/actions/invocation_test.rb +0 -247
- data/test/configuration/callbacks_test.rb +0 -220
- data/test/configuration/connections_test.rb +0 -420
- data/test/configuration/execution_test.rb +0 -175
- data/test/configuration/loading_test.rb +0 -132
- data/test/configuration/namespace_dsl_test.rb +0 -311
- data/test/configuration/roles_test.rb +0 -144
- data/test/configuration/servers_test.rb +0 -183
- data/test/configuration/variables_test.rb +0 -190
- data/test/configuration_test.rb +0 -88
- data/test/deploy/local_dependency_test.rb +0 -76
- data/test/deploy/remote_dependency_test.rb +0 -135
- data/test/deploy/scm/accurev_test.rb +0 -23
- data/test/deploy/scm/base_test.rb +0 -55
- data/test/deploy/scm/bzr_test.rb +0 -51
- data/test/deploy/scm/darcs_test.rb +0 -37
- data/test/deploy/scm/git_test.rb +0 -184
- data/test/deploy/scm/mercurial_test.rb +0 -134
- data/test/deploy/scm/none_test.rb +0 -35
- data/test/deploy/scm/subversion_test.rb +0 -32
- data/test/deploy/strategy/copy_test.rb +0 -321
- data/test/extensions_test.rb +0 -69
- data/test/fixtures/cli_integration.rb +0 -5
- data/test/fixtures/config.rb +0 -5
- data/test/fixtures/custom.rb +0 -3
- data/test/logger_test.rb +0 -123
- data/test/recipes_test.rb +0 -25
- data/test/role_test.rb +0 -11
- data/test/server_definition_test.rb +0 -121
- data/test/shell_test.rb +0 -90
- data/test/ssh_test.rb +0 -113
- data/test/task_definition_test.rb +0 -116
- data/test/transfer_test.rb +0 -160
- data/test/utils.rb +0 -37
|
@@ -0,0 +1,95 @@
|
|
|
1
|
+
require "capistrano/all"
|
|
2
|
+
require "rake/tasklib"
|
|
3
|
+
|
|
4
|
+
# IMPORTANT: The Capistrano::Plugin system is not yet considered a stable,
|
|
5
|
+
# public API, and is subject to change without notice. Eventually it will be
|
|
6
|
+
# officially documented and supported, but for now, use it at your own risk.
|
|
7
|
+
#
|
|
8
|
+
# Base class for Capistrano plugins. Makes building a Capistrano plugin as easy
|
|
9
|
+
# as writing a `Capistrano::Plugin` subclass and overriding any or all of its
|
|
10
|
+
# three template methods:
|
|
11
|
+
#
|
|
12
|
+
# * set_defaults
|
|
13
|
+
# * register_hooks
|
|
14
|
+
# * define_tasks
|
|
15
|
+
#
|
|
16
|
+
# Within the plugin you can use any methods of the Rake or Capistrano DSLs, like
|
|
17
|
+
# `fetch`, `invoke`, etc. In cases when you need to use SSHKit's backend outside
|
|
18
|
+
# of an `on` block, use the `backend` convenience method. E.g. `backend.test`,
|
|
19
|
+
# `backend.execute`, or `backend.capture`.
|
|
20
|
+
#
|
|
21
|
+
# Package up and distribute your plugin class as a gem and you're good to go!
|
|
22
|
+
#
|
|
23
|
+
# To use a plugin, all a user has to do is install it in the Capfile, like this:
|
|
24
|
+
#
|
|
25
|
+
# # Capfile
|
|
26
|
+
# require "capistrano/superfancy"
|
|
27
|
+
# install_plugin Capistrano::Superfancy
|
|
28
|
+
#
|
|
29
|
+
# Or, to install the plugin without its hooks:
|
|
30
|
+
#
|
|
31
|
+
# # Capfile
|
|
32
|
+
# require "capistrano/superfancy"
|
|
33
|
+
# install_plugin Capistrano::Superfancy, load_hooks: false
|
|
34
|
+
#
|
|
35
|
+
class Capistrano::Plugin < Rake::TaskLib
|
|
36
|
+
include Capistrano::DSL
|
|
37
|
+
|
|
38
|
+
# Implemented by subclasses to provide default values for settings needed by
|
|
39
|
+
# this plugin. Typically done using the `set_if_empty` Capistrano DSL method.
|
|
40
|
+
#
|
|
41
|
+
# Example:
|
|
42
|
+
#
|
|
43
|
+
# def set_defaults
|
|
44
|
+
# set_if_empty :my_plugin_option, true
|
|
45
|
+
# end
|
|
46
|
+
#
|
|
47
|
+
def set_defaults; end
|
|
48
|
+
|
|
49
|
+
# Implemented by subclasses to hook into Capistrano's deployment flow using
|
|
50
|
+
# using the `before` and `after` DSL methods. Note that `register_hooks` will
|
|
51
|
+
# not be called if the user has opted-out of hooks when installing the plugin.
|
|
52
|
+
#
|
|
53
|
+
# Example:
|
|
54
|
+
#
|
|
55
|
+
# def register_hooks
|
|
56
|
+
# after "deploy:updated", "my_plugin:do_something"
|
|
57
|
+
# end
|
|
58
|
+
#
|
|
59
|
+
def register_hooks; end
|
|
60
|
+
|
|
61
|
+
# Implemented by subclasses to define Rake tasks. Typically a plugin will call
|
|
62
|
+
# `eval_rakefile` to load Rake tasks from a separate .rake file.
|
|
63
|
+
#
|
|
64
|
+
# Example:
|
|
65
|
+
#
|
|
66
|
+
# def define_tasks
|
|
67
|
+
# eval_rakefile File.expand_path("../tasks.rake", __FILE__)
|
|
68
|
+
# end
|
|
69
|
+
#
|
|
70
|
+
# For simple tasks, you can define them inline. No need for a separate file.
|
|
71
|
+
#
|
|
72
|
+
# def define_tasks
|
|
73
|
+
# desc "Do something fantastic."
|
|
74
|
+
# task "my_plugin:fantastic" do
|
|
75
|
+
# ...
|
|
76
|
+
# end
|
|
77
|
+
# end
|
|
78
|
+
#
|
|
79
|
+
def define_tasks; end
|
|
80
|
+
|
|
81
|
+
private
|
|
82
|
+
|
|
83
|
+
# Read and eval a .rake file in such a way that `self` within the .rake file
|
|
84
|
+
# refers to this plugin instance. This gives the tasks in the file access to
|
|
85
|
+
# helper methods defined by the plugin.
|
|
86
|
+
def eval_rakefile(path)
|
|
87
|
+
contents = IO.read(path)
|
|
88
|
+
instance_eval(contents, path, 1)
|
|
89
|
+
end
|
|
90
|
+
|
|
91
|
+
# Convenience to access the current SSHKit backend outside of an `on` block.
|
|
92
|
+
def backend
|
|
93
|
+
SSHKit::Backend.current
|
|
94
|
+
end
|
|
95
|
+
end
|
|
@@ -0,0 +1,13 @@
|
|
|
1
|
+
module Capistrano
|
|
2
|
+
module ProcHelpers
|
|
3
|
+
module_function
|
|
4
|
+
|
|
5
|
+
# Tests whether the given object appears to respond to `call` with
|
|
6
|
+
# zero parameters. In Capistrano, such a proc is used to represent a
|
|
7
|
+
# "deferred value". That is, a value that is resolved by invoking `call` at
|
|
8
|
+
# the time it is first needed.
|
|
9
|
+
def callable_without_parameters?(x)
|
|
10
|
+
x.respond_to?(:call) && (!x.respond_to?(:arity) || x.arity.zero?)
|
|
11
|
+
end
|
|
12
|
+
end
|
|
13
|
+
end
|
|
@@ -0,0 +1,105 @@
|
|
|
1
|
+
require "capistrano/scm/plugin"
|
|
2
|
+
require "cgi"
|
|
3
|
+
require "securerandom"
|
|
4
|
+
require "shellwords"
|
|
5
|
+
require "uri"
|
|
6
|
+
|
|
7
|
+
class Capistrano::SCM::Git < Capistrano::SCM::Plugin
|
|
8
|
+
def set_defaults
|
|
9
|
+
set_if_empty :git_shallow_clone, false
|
|
10
|
+
set_if_empty :git_wrapper_path, lambda {
|
|
11
|
+
# Use a unique name that won't collide with other deployments, and
|
|
12
|
+
# that cannot be guessed by other processes that have access to /tmp.
|
|
13
|
+
"#{fetch(:tmp_dir)}/git-ssh-#{SecureRandom.hex(10)}.sh"
|
|
14
|
+
}
|
|
15
|
+
set_if_empty :git_environmental_variables, lambda {
|
|
16
|
+
{
|
|
17
|
+
git_askpass: "/bin/echo",
|
|
18
|
+
git_ssh: fetch(:git_wrapper_path)
|
|
19
|
+
}
|
|
20
|
+
}
|
|
21
|
+
set_if_empty :git_max_concurrent_connections, 10
|
|
22
|
+
set_if_empty :git_wait_interval, 0
|
|
23
|
+
end
|
|
24
|
+
|
|
25
|
+
def register_hooks
|
|
26
|
+
after "deploy:new_release_path", "git:create_release"
|
|
27
|
+
before "deploy:check", "git:check"
|
|
28
|
+
before "deploy:set_current_revision", "git:set_current_revision"
|
|
29
|
+
before "deploy:set_current_revision_time", "git:set_current_revision_time"
|
|
30
|
+
end
|
|
31
|
+
|
|
32
|
+
def define_tasks
|
|
33
|
+
eval_rakefile File.expand_path("../tasks/git.rake", __FILE__)
|
|
34
|
+
end
|
|
35
|
+
|
|
36
|
+
def repo_mirror_exists?
|
|
37
|
+
backend.test " [ -f #{repo_path}/HEAD ] "
|
|
38
|
+
end
|
|
39
|
+
|
|
40
|
+
def check_repo_is_reachable
|
|
41
|
+
git :'ls-remote', git_repo_url, "HEAD"
|
|
42
|
+
end
|
|
43
|
+
|
|
44
|
+
def clone_repo
|
|
45
|
+
if (depth = fetch(:git_shallow_clone))
|
|
46
|
+
git :clone, "--mirror", "--depth", depth, "--no-single-branch", git_repo_url, repo_path.to_s
|
|
47
|
+
else
|
|
48
|
+
git :clone, "--mirror", git_repo_url, repo_path.to_s
|
|
49
|
+
end
|
|
50
|
+
end
|
|
51
|
+
|
|
52
|
+
def update_mirror
|
|
53
|
+
# Update the origin URL if necessary.
|
|
54
|
+
git :remote, "set-url", "origin", git_repo_url
|
|
55
|
+
|
|
56
|
+
# Note: Requires git version 1.9 or greater
|
|
57
|
+
if (depth = fetch(:git_shallow_clone))
|
|
58
|
+
git :fetch, "--depth", depth, "origin", fetch(:branch)
|
|
59
|
+
else
|
|
60
|
+
git :remote, :update, "--prune"
|
|
61
|
+
end
|
|
62
|
+
end
|
|
63
|
+
|
|
64
|
+
def verify_commit
|
|
65
|
+
git :"verify-commit", fetch_revision
|
|
66
|
+
end
|
|
67
|
+
|
|
68
|
+
def archive_to_release_path
|
|
69
|
+
if (tree = fetch(:repo_tree))
|
|
70
|
+
tree = tree.slice %r#^/?(.*?)/?$#, 1
|
|
71
|
+
components = tree.split("/").size
|
|
72
|
+
git :archive, fetch(:branch), tree, "| #{SSHKit.config.command_map[:tar]} -x --strip-components #{components} -f - -C", release_path
|
|
73
|
+
else
|
|
74
|
+
git :archive, fetch(:branch), "| #{SSHKit.config.command_map[:tar]} -x -f - -C", release_path
|
|
75
|
+
end
|
|
76
|
+
end
|
|
77
|
+
|
|
78
|
+
def fetch_revision
|
|
79
|
+
backend.capture(:git, "rev-list --max-count=1 #{fetch(:branch)}")
|
|
80
|
+
end
|
|
81
|
+
|
|
82
|
+
def fetch_revision_time
|
|
83
|
+
backend.capture(:git, "log -1 --pretty=format:\"%ct\" #{fetch(:branch)}")
|
|
84
|
+
end
|
|
85
|
+
|
|
86
|
+
def git(*args)
|
|
87
|
+
args.unshift :git
|
|
88
|
+
backend.execute(*args)
|
|
89
|
+
end
|
|
90
|
+
|
|
91
|
+
def git_repo_url
|
|
92
|
+
if fetch(:git_http_username) && fetch(:git_http_password)
|
|
93
|
+
URI.parse(repo_url).tap do |repo_uri|
|
|
94
|
+
repo_uri.user = fetch(:git_http_username)
|
|
95
|
+
repo_uri.password = CGI.escape(fetch(:git_http_password))
|
|
96
|
+
end.to_s
|
|
97
|
+
elsif fetch(:git_http_username)
|
|
98
|
+
URI.parse(repo_url).tap do |repo_uri|
|
|
99
|
+
repo_uri.user = fetch(:git_http_username)
|
|
100
|
+
end.to_s
|
|
101
|
+
else
|
|
102
|
+
repo_url
|
|
103
|
+
end
|
|
104
|
+
end
|
|
105
|
+
end
|
|
@@ -0,0 +1,55 @@
|
|
|
1
|
+
require "capistrano/scm/plugin"
|
|
2
|
+
require "securerandom"
|
|
3
|
+
|
|
4
|
+
class Capistrano::SCM::Hg < Capistrano::SCM::Plugin
|
|
5
|
+
def register_hooks
|
|
6
|
+
after "deploy:new_release_path", "hg:create_release"
|
|
7
|
+
before "deploy:check", "hg:check"
|
|
8
|
+
before "deploy:set_current_revision", "hg:set_current_revision"
|
|
9
|
+
end
|
|
10
|
+
|
|
11
|
+
def define_tasks
|
|
12
|
+
eval_rakefile File.expand_path("../tasks/hg.rake", __FILE__)
|
|
13
|
+
end
|
|
14
|
+
|
|
15
|
+
def hg(*args)
|
|
16
|
+
args.unshift(:hg)
|
|
17
|
+
backend.execute(*args)
|
|
18
|
+
end
|
|
19
|
+
|
|
20
|
+
def repo_mirror_exists?
|
|
21
|
+
backend.test " [ -d #{repo_path}/.hg ] "
|
|
22
|
+
end
|
|
23
|
+
|
|
24
|
+
def check_repo_is_reachable
|
|
25
|
+
hg "id", repo_url
|
|
26
|
+
end
|
|
27
|
+
|
|
28
|
+
def clone_repo
|
|
29
|
+
hg "clone", "--noupdate", repo_url, repo_path.to_s
|
|
30
|
+
end
|
|
31
|
+
|
|
32
|
+
def update_mirror
|
|
33
|
+
hg "pull"
|
|
34
|
+
end
|
|
35
|
+
|
|
36
|
+
def archive_to_release_path
|
|
37
|
+
if (tree = fetch(:repo_tree))
|
|
38
|
+
tree = tree.slice %r#^/?(.*?)/?$#, 1
|
|
39
|
+
components = tree.split("/").size
|
|
40
|
+
temp_tar = "#{fetch(:tmp_dir)}/#{SecureRandom.hex(10)}.tar"
|
|
41
|
+
|
|
42
|
+
hg "archive -p . -I", tree, "--rev", fetch(:branch), temp_tar
|
|
43
|
+
|
|
44
|
+
backend.execute :mkdir, "-p", release_path
|
|
45
|
+
backend.execute :tar, "-x --strip-components #{components} -f", temp_tar, "-C", release_path
|
|
46
|
+
backend.execute :rm, temp_tar
|
|
47
|
+
else
|
|
48
|
+
hg "archive", release_path, "--rev", fetch(:branch)
|
|
49
|
+
end
|
|
50
|
+
end
|
|
51
|
+
|
|
52
|
+
def fetch_revision
|
|
53
|
+
backend.capture(:hg, "log --rev #{fetch(:branch)} --template \"{node}\n\"")
|
|
54
|
+
end
|
|
55
|
+
end
|
|
@@ -0,0 +1,13 @@
|
|
|
1
|
+
require "capistrano/plugin"
|
|
2
|
+
require "capistrano/scm"
|
|
3
|
+
|
|
4
|
+
# Base class for all built-in and third-party SCM plugins. Notice that this
|
|
5
|
+
# class doesn't really do anything other than provide an `scm?` predicate. This
|
|
6
|
+
# tells Capistrano that the plugin provides SCM functionality. All other plugin
|
|
7
|
+
# features are inherited from Capistrano::Plugin.
|
|
8
|
+
#
|
|
9
|
+
class Capistrano::SCM::Plugin < Capistrano::Plugin
|
|
10
|
+
def scm?
|
|
11
|
+
true
|
|
12
|
+
end
|
|
13
|
+
end
|
|
@@ -0,0 +1,56 @@
|
|
|
1
|
+
require "capistrano/scm/plugin"
|
|
2
|
+
|
|
3
|
+
class Capistrano::SCM::Svn < Capistrano::SCM::Plugin
|
|
4
|
+
def register_hooks
|
|
5
|
+
after "deploy:new_release_path", "svn:create_release"
|
|
6
|
+
before "deploy:check", "svn:check"
|
|
7
|
+
before "deploy:set_current_revision", "svn:set_current_revision"
|
|
8
|
+
end
|
|
9
|
+
|
|
10
|
+
def define_tasks
|
|
11
|
+
eval_rakefile File.expand_path("../tasks/svn.rake", __FILE__)
|
|
12
|
+
end
|
|
13
|
+
|
|
14
|
+
def svn(*args)
|
|
15
|
+
args.unshift(:svn)
|
|
16
|
+
args.push "--username #{fetch(:svn_username)}" if fetch(:svn_username)
|
|
17
|
+
args.push "--password #{fetch(:svn_password)}" if fetch(:svn_password)
|
|
18
|
+
args.push "--revision #{fetch(:svn_revision)}" if fetch(:svn_revision)
|
|
19
|
+
backend.execute(*args)
|
|
20
|
+
end
|
|
21
|
+
|
|
22
|
+
def repo_mirror_exists?
|
|
23
|
+
backend.test " [ -d #{repo_path}/.svn ] "
|
|
24
|
+
end
|
|
25
|
+
|
|
26
|
+
def check_repo_is_reachable
|
|
27
|
+
svn_username = fetch(:svn_username) ? "--username #{fetch(:svn_username)}" : ""
|
|
28
|
+
svn_password = fetch(:svn_password) ? "--password #{fetch(:svn_password)}" : ""
|
|
29
|
+
backend.test :svn, :info, repo_url, svn_username, svn_password
|
|
30
|
+
end
|
|
31
|
+
|
|
32
|
+
def clone_repo
|
|
33
|
+
svn :checkout, repo_url, repo_path.to_s
|
|
34
|
+
end
|
|
35
|
+
|
|
36
|
+
def update_mirror
|
|
37
|
+
# Switch the repository URL if necessary.
|
|
38
|
+
repo_mirror_url = fetch_repo_mirror_url
|
|
39
|
+
svn :switch, repo_url unless repo_mirror_url == repo_url
|
|
40
|
+
svn :update
|
|
41
|
+
end
|
|
42
|
+
|
|
43
|
+
def archive_to_release_path
|
|
44
|
+
svn :export, "--force", ".", release_path
|
|
45
|
+
end
|
|
46
|
+
|
|
47
|
+
def fetch_revision
|
|
48
|
+
backend.capture(:svnversion, repo_path.to_s)
|
|
49
|
+
end
|
|
50
|
+
|
|
51
|
+
def fetch_repo_mirror_url
|
|
52
|
+
backend.capture(:svn, :info, repo_path.to_s).each_line do |line|
|
|
53
|
+
return $1 if /\AURL: (.*)\n\z/ =~ line
|
|
54
|
+
end
|
|
55
|
+
end
|
|
56
|
+
end
|
|
@@ -0,0 +1,84 @@
|
|
|
1
|
+
# This trick lets us access the Git plugin within `on` blocks.
|
|
2
|
+
git_plugin = self
|
|
3
|
+
|
|
4
|
+
namespace :git do
|
|
5
|
+
desc "Upload the git wrapper script, this script guarantees that we can script git without getting an interactive prompt"
|
|
6
|
+
task :wrapper do
|
|
7
|
+
on release_roles(:all), in: :groups, limit: fetch(:git_max_concurrent_connections), wait: fetch(:git_wait_interval) do
|
|
8
|
+
execute :mkdir, "-p", File.dirname(fetch(:git_wrapper_path)).shellescape
|
|
9
|
+
upload! StringIO.new("#!/bin/sh -e\nexec /usr/bin/env ssh -o PasswordAuthentication=no -o StrictHostKeyChecking=no \"$@\"\n"), fetch(:git_wrapper_path)
|
|
10
|
+
execute :chmod, "700", fetch(:git_wrapper_path).shellescape
|
|
11
|
+
end
|
|
12
|
+
end
|
|
13
|
+
|
|
14
|
+
desc "Check that the repository is reachable"
|
|
15
|
+
task check: :'git:wrapper' do
|
|
16
|
+
fetch(:branch)
|
|
17
|
+
on release_roles(:all), in: :groups, limit: fetch(:git_max_concurrent_connections), wait: fetch(:git_wait_interval) do
|
|
18
|
+
with fetch(:git_environmental_variables) do
|
|
19
|
+
git_plugin.check_repo_is_reachable
|
|
20
|
+
end
|
|
21
|
+
end
|
|
22
|
+
end
|
|
23
|
+
|
|
24
|
+
desc "Clone the repo to the cache"
|
|
25
|
+
task clone: :'git:wrapper' do
|
|
26
|
+
on release_roles(:all), in: :groups, limit: fetch(:git_max_concurrent_connections), wait: fetch(:git_wait_interval) do
|
|
27
|
+
if git_plugin.repo_mirror_exists?
|
|
28
|
+
info t(:mirror_exists, at: repo_path)
|
|
29
|
+
else
|
|
30
|
+
within deploy_path do
|
|
31
|
+
with fetch(:git_environmental_variables) do
|
|
32
|
+
git_plugin.clone_repo
|
|
33
|
+
end
|
|
34
|
+
end
|
|
35
|
+
end
|
|
36
|
+
end
|
|
37
|
+
end
|
|
38
|
+
|
|
39
|
+
desc "Update the repo mirror to reflect the origin state"
|
|
40
|
+
task update: :'git:clone' do
|
|
41
|
+
on release_roles(:all), in: :groups, limit: fetch(:git_max_concurrent_connections), wait: fetch(:git_wait_interval) do
|
|
42
|
+
within repo_path do
|
|
43
|
+
with fetch(:git_environmental_variables) do
|
|
44
|
+
git_plugin.update_mirror
|
|
45
|
+
git_plugin.verify_commit if fetch(:git_verify_commit)
|
|
46
|
+
end
|
|
47
|
+
end
|
|
48
|
+
end
|
|
49
|
+
end
|
|
50
|
+
|
|
51
|
+
desc "Copy repo to releases"
|
|
52
|
+
task create_release: :'git:update' do
|
|
53
|
+
on release_roles(:all), in: :groups, limit: fetch(:git_max_concurrent_connections), wait: fetch(:git_wait_interval) do
|
|
54
|
+
with fetch(:git_environmental_variables) do
|
|
55
|
+
within repo_path do
|
|
56
|
+
execute :mkdir, "-p", release_path
|
|
57
|
+
git_plugin.archive_to_release_path
|
|
58
|
+
end
|
|
59
|
+
end
|
|
60
|
+
end
|
|
61
|
+
end
|
|
62
|
+
|
|
63
|
+
desc "Determine the revision that will be deployed"
|
|
64
|
+
task :set_current_revision do
|
|
65
|
+
on release_roles(:all), in: :groups, limit: fetch(:git_max_concurrent_connections), wait: fetch(:git_wait_interval) do
|
|
66
|
+
within repo_path do
|
|
67
|
+
with fetch(:git_environmental_variables) do
|
|
68
|
+
set :current_revision, git_plugin.fetch_revision
|
|
69
|
+
end
|
|
70
|
+
end
|
|
71
|
+
end
|
|
72
|
+
end
|
|
73
|
+
|
|
74
|
+
desc "Determine the unix timestamp that the revision that will be deployed was created"
|
|
75
|
+
task :set_current_revision_time do
|
|
76
|
+
on release_roles(:all), in: :groups, limit: fetch(:git_max_concurrent_connections), wait: fetch(:git_wait_interval) do
|
|
77
|
+
within repo_path do
|
|
78
|
+
with fetch(:git_environmental_variables) do
|
|
79
|
+
set :current_revision_time, git_plugin.fetch_revision_time
|
|
80
|
+
end
|
|
81
|
+
end
|
|
82
|
+
end
|
|
83
|
+
end
|
|
84
|
+
end
|
|
@@ -0,0 +1,53 @@
|
|
|
1
|
+
# TODO: this is nearly identical to git.rake. DRY up?
|
|
2
|
+
|
|
3
|
+
# This trick lets us access the Hg plugin within `on` blocks.
|
|
4
|
+
hg_plugin = self
|
|
5
|
+
|
|
6
|
+
namespace :hg do
|
|
7
|
+
desc "Check that the repo is reachable"
|
|
8
|
+
task :check do
|
|
9
|
+
on release_roles :all do
|
|
10
|
+
hg_plugin.check_repo_is_reachable
|
|
11
|
+
end
|
|
12
|
+
end
|
|
13
|
+
|
|
14
|
+
desc "Clone the repo to the cache"
|
|
15
|
+
task :clone do
|
|
16
|
+
on release_roles :all do
|
|
17
|
+
if hg_plugin.repo_mirror_exists?
|
|
18
|
+
info t(:mirror_exists, at: repo_path)
|
|
19
|
+
else
|
|
20
|
+
within deploy_path do
|
|
21
|
+
hg_plugin.clone_repo
|
|
22
|
+
end
|
|
23
|
+
end
|
|
24
|
+
end
|
|
25
|
+
end
|
|
26
|
+
|
|
27
|
+
desc "Pull changes from the remote repo"
|
|
28
|
+
task update: :'hg:clone' do
|
|
29
|
+
on release_roles :all do
|
|
30
|
+
within repo_path do
|
|
31
|
+
hg_plugin.update_mirror
|
|
32
|
+
end
|
|
33
|
+
end
|
|
34
|
+
end
|
|
35
|
+
|
|
36
|
+
desc "Copy repo to releases"
|
|
37
|
+
task create_release: :'hg:update' do
|
|
38
|
+
on release_roles :all do
|
|
39
|
+
within repo_path do
|
|
40
|
+
hg_plugin.archive_to_release_path
|
|
41
|
+
end
|
|
42
|
+
end
|
|
43
|
+
end
|
|
44
|
+
|
|
45
|
+
desc "Determine the revision that will be deployed"
|
|
46
|
+
task :set_current_revision do
|
|
47
|
+
on release_roles :all do
|
|
48
|
+
within repo_path do
|
|
49
|
+
set :current_revision, hg_plugin.fetch_revision
|
|
50
|
+
end
|
|
51
|
+
end
|
|
52
|
+
end
|
|
53
|
+
end
|
|
@@ -0,0 +1,53 @@
|
|
|
1
|
+
# TODO: this is nearly identical to git.rake. DRY up?
|
|
2
|
+
|
|
3
|
+
# This trick lets us access the Svn plugin within `on` blocks.
|
|
4
|
+
svn_plugin = self
|
|
5
|
+
|
|
6
|
+
namespace :svn do
|
|
7
|
+
desc "Check that the repo is reachable"
|
|
8
|
+
task :check do
|
|
9
|
+
on release_roles :all do
|
|
10
|
+
svn_plugin.check_repo_is_reachable
|
|
11
|
+
end
|
|
12
|
+
end
|
|
13
|
+
|
|
14
|
+
desc "Clone the repo to the cache"
|
|
15
|
+
task :clone do
|
|
16
|
+
on release_roles :all do
|
|
17
|
+
if svn_plugin.repo_mirror_exists?
|
|
18
|
+
info t(:mirror_exists, at: repo_path)
|
|
19
|
+
else
|
|
20
|
+
within deploy_path do
|
|
21
|
+
svn_plugin.clone_repo
|
|
22
|
+
end
|
|
23
|
+
end
|
|
24
|
+
end
|
|
25
|
+
end
|
|
26
|
+
|
|
27
|
+
desc "Pull changes from the remote repo"
|
|
28
|
+
task update: :'svn:clone' do
|
|
29
|
+
on release_roles :all do
|
|
30
|
+
within repo_path do
|
|
31
|
+
svn_plugin.update_mirror
|
|
32
|
+
end
|
|
33
|
+
end
|
|
34
|
+
end
|
|
35
|
+
|
|
36
|
+
desc "Copy repo to releases"
|
|
37
|
+
task create_release: :'svn:update' do
|
|
38
|
+
on release_roles :all do
|
|
39
|
+
within repo_path do
|
|
40
|
+
svn_plugin.archive_to_release_path
|
|
41
|
+
end
|
|
42
|
+
end
|
|
43
|
+
end
|
|
44
|
+
|
|
45
|
+
desc "Determine the revision that will be deployed"
|
|
46
|
+
task :set_current_revision do
|
|
47
|
+
on release_roles :all do
|
|
48
|
+
within repo_path do
|
|
49
|
+
set :current_revision, svn_plugin.fetch_revision
|
|
50
|
+
end
|
|
51
|
+
end
|
|
52
|
+
end
|
|
53
|
+
end
|
|
@@ -0,0 +1,115 @@
|
|
|
1
|
+
module Capistrano
|
|
2
|
+
# Base class for SCM strategy providers.
|
|
3
|
+
#
|
|
4
|
+
# @abstract
|
|
5
|
+
#
|
|
6
|
+
# @attr_reader [Rake] context
|
|
7
|
+
#
|
|
8
|
+
# @author Hartog de Mik
|
|
9
|
+
#
|
|
10
|
+
class SCM
|
|
11
|
+
attr_reader :context
|
|
12
|
+
|
|
13
|
+
# Provide a wrapper for the SCM that loads a strategy for the user.
|
|
14
|
+
#
|
|
15
|
+
# @param [Rake] context The context in which the strategy should run
|
|
16
|
+
# @param [Module] strategy A module to include into the SCM instance. The
|
|
17
|
+
# module should provide the abstract methods of Capistrano::SCM
|
|
18
|
+
#
|
|
19
|
+
def initialize(context, strategy)
|
|
20
|
+
@context = context
|
|
21
|
+
singleton = class << self; self; end
|
|
22
|
+
singleton.send(:include, strategy)
|
|
23
|
+
end
|
|
24
|
+
|
|
25
|
+
# Call test in context
|
|
26
|
+
def test!(*args)
|
|
27
|
+
context.test(*args)
|
|
28
|
+
end
|
|
29
|
+
|
|
30
|
+
# The repository URL according to the context
|
|
31
|
+
def repo_url
|
|
32
|
+
context.repo_url
|
|
33
|
+
end
|
|
34
|
+
|
|
35
|
+
# The repository path according to the context
|
|
36
|
+
def repo_path
|
|
37
|
+
context.repo_path
|
|
38
|
+
end
|
|
39
|
+
|
|
40
|
+
# The release path according to the context
|
|
41
|
+
def release_path
|
|
42
|
+
context.release_path
|
|
43
|
+
end
|
|
44
|
+
|
|
45
|
+
# Fetch a var from the context
|
|
46
|
+
# @param [Symbol] variable The variable to fetch
|
|
47
|
+
# @param [Object] default The default value if not found
|
|
48
|
+
#
|
|
49
|
+
def fetch(*args)
|
|
50
|
+
context.fetch(*args)
|
|
51
|
+
end
|
|
52
|
+
|
|
53
|
+
# @abstract
|
|
54
|
+
#
|
|
55
|
+
# Your implementation should check the existence of a cache repository on
|
|
56
|
+
# the deployment target
|
|
57
|
+
#
|
|
58
|
+
# @return [Boolean]
|
|
59
|
+
#
|
|
60
|
+
def test
|
|
61
|
+
raise NotImplementedError, "Your SCM strategy module should provide a #test method"
|
|
62
|
+
end
|
|
63
|
+
|
|
64
|
+
# @abstract
|
|
65
|
+
#
|
|
66
|
+
# Your implementation should check if the specified remote-repository is
|
|
67
|
+
# available.
|
|
68
|
+
#
|
|
69
|
+
# @return [Boolean]
|
|
70
|
+
#
|
|
71
|
+
def check
|
|
72
|
+
raise NotImplementedError, "Your SCM strategy module should provide a #check method"
|
|
73
|
+
end
|
|
74
|
+
|
|
75
|
+
# @abstract
|
|
76
|
+
#
|
|
77
|
+
# Create a (new) clone of the remote-repository on the deployment target
|
|
78
|
+
#
|
|
79
|
+
# @return void
|
|
80
|
+
#
|
|
81
|
+
def clone
|
|
82
|
+
raise NotImplementedError, "Your SCM strategy module should provide a #clone method"
|
|
83
|
+
end
|
|
84
|
+
|
|
85
|
+
# @abstract
|
|
86
|
+
#
|
|
87
|
+
# Update the clone on the deployment target
|
|
88
|
+
#
|
|
89
|
+
# @return void
|
|
90
|
+
#
|
|
91
|
+
def update
|
|
92
|
+
raise NotImplementedError, "Your SCM strategy module should provide a #update method"
|
|
93
|
+
end
|
|
94
|
+
|
|
95
|
+
# @abstract
|
|
96
|
+
#
|
|
97
|
+
# Copy the contents of the cache-repository onto the release path
|
|
98
|
+
#
|
|
99
|
+
# @return void
|
|
100
|
+
#
|
|
101
|
+
def release
|
|
102
|
+
raise NotImplementedError, "Your SCM strategy module should provide a #release method"
|
|
103
|
+
end
|
|
104
|
+
|
|
105
|
+
# @abstract
|
|
106
|
+
#
|
|
107
|
+
# Identify the SHA of the commit that will be deployed. This will most likely involve SshKit's capture method.
|
|
108
|
+
#
|
|
109
|
+
# @return void
|
|
110
|
+
#
|
|
111
|
+
def fetch_revision
|
|
112
|
+
raise NotImplementedError, "Your SCM strategy module should provide a #fetch_revision method"
|
|
113
|
+
end
|
|
114
|
+
end
|
|
115
|
+
end
|
|
@@ -0,0 +1,36 @@
|
|
|
1
|
+
require "capistrano/doctor"
|
|
2
|
+
require "capistrano/immutable_task"
|
|
3
|
+
include Capistrano::DSL
|
|
4
|
+
|
|
5
|
+
namespace :load do
|
|
6
|
+
task :defaults do
|
|
7
|
+
load "capistrano/defaults.rb"
|
|
8
|
+
end
|
|
9
|
+
end
|
|
10
|
+
|
|
11
|
+
require "airbrussh/capistrano"
|
|
12
|
+
# We don't need to show the "using Airbrussh" banner announcement since
|
|
13
|
+
# Airbrussh is now the built-in formatter. Also enable command output by
|
|
14
|
+
# default; hiding the output might be confusing to users new to Capistrano.
|
|
15
|
+
Airbrussh.configure do |airbrussh|
|
|
16
|
+
airbrussh.banner = false
|
|
17
|
+
airbrussh.command_output = true
|
|
18
|
+
end
|
|
19
|
+
|
|
20
|
+
stages.each do |stage|
|
|
21
|
+
Rake::Task.define_task(stage) do
|
|
22
|
+
set(:stage, stage.to_sym)
|
|
23
|
+
|
|
24
|
+
invoke "load:defaults"
|
|
25
|
+
Rake.application["load:defaults"].extend(Capistrano::ImmutableTask)
|
|
26
|
+
env.variables.untrusted! do
|
|
27
|
+
load deploy_config_path
|
|
28
|
+
load stage_config_path.join("#{stage}.rb")
|
|
29
|
+
end
|
|
30
|
+
configure_scm
|
|
31
|
+
I18n.locale = fetch(:locale, :en)
|
|
32
|
+
configure_backend
|
|
33
|
+
end
|
|
34
|
+
end
|
|
35
|
+
|
|
36
|
+
require "capistrano/dotfile"
|