mbailey-capistrano 2.5.5

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.
Files changed (105) hide show
  1. data/CHANGELOG.rdoc +761 -0
  2. data/Manifest +104 -0
  3. data/README.rdoc +66 -0
  4. data/Rakefile +34 -0
  5. data/bin/cap +4 -0
  6. data/bin/capify +78 -0
  7. data/examples/sample.rb +14 -0
  8. data/lib/capistrano/callback.rb +45 -0
  9. data/lib/capistrano/cli/execute.rb +84 -0
  10. data/lib/capistrano/cli/help.rb +125 -0
  11. data/lib/capistrano/cli/help.txt +75 -0
  12. data/lib/capistrano/cli/options.rb +224 -0
  13. data/lib/capistrano/cli/ui.rb +40 -0
  14. data/lib/capistrano/cli.rb +47 -0
  15. data/lib/capistrano/command.rb +283 -0
  16. data/lib/capistrano/configuration/actions/file_transfer.rb +47 -0
  17. data/lib/capistrano/configuration/actions/inspect.rb +46 -0
  18. data/lib/capistrano/configuration/actions/invocation.rb +293 -0
  19. data/lib/capistrano/configuration/callbacks.rb +148 -0
  20. data/lib/capistrano/configuration/connections.rb +200 -0
  21. data/lib/capistrano/configuration/execution.rb +132 -0
  22. data/lib/capistrano/configuration/loading.rb +197 -0
  23. data/lib/capistrano/configuration/namespaces.rb +197 -0
  24. data/lib/capistrano/configuration/roles.rb +73 -0
  25. data/lib/capistrano/configuration/servers.rb +85 -0
  26. data/lib/capistrano/configuration/variables.rb +127 -0
  27. data/lib/capistrano/configuration.rb +43 -0
  28. data/lib/capistrano/errors.rb +15 -0
  29. data/lib/capistrano/extensions.rb +57 -0
  30. data/lib/capistrano/logger.rb +59 -0
  31. data/lib/capistrano/processable.rb +53 -0
  32. data/lib/capistrano/recipes/compat.rb +32 -0
  33. data/lib/capistrano/recipes/deploy/dependencies.rb +44 -0
  34. data/lib/capistrano/recipes/deploy/local_dependency.rb +54 -0
  35. data/lib/capistrano/recipes/deploy/remote_dependency.rb +105 -0
  36. data/lib/capistrano/recipes/deploy/scm/accurev.rb +169 -0
  37. data/lib/capistrano/recipes/deploy/scm/base.rb +196 -0
  38. data/lib/capistrano/recipes/deploy/scm/bzr.rb +83 -0
  39. data/lib/capistrano/recipes/deploy/scm/cvs.rb +152 -0
  40. data/lib/capistrano/recipes/deploy/scm/darcs.rb +85 -0
  41. data/lib/capistrano/recipes/deploy/scm/git.rb +271 -0
  42. data/lib/capistrano/recipes/deploy/scm/mercurial.rb +137 -0
  43. data/lib/capistrano/recipes/deploy/scm/none.rb +44 -0
  44. data/lib/capistrano/recipes/deploy/scm/perforce.rb +133 -0
  45. data/lib/capistrano/recipes/deploy/scm/subversion.rb +121 -0
  46. data/lib/capistrano/recipes/deploy/scm.rb +19 -0
  47. data/lib/capistrano/recipes/deploy/strategy/base.rb +79 -0
  48. data/lib/capistrano/recipes/deploy/strategy/checkout.rb +20 -0
  49. data/lib/capistrano/recipes/deploy/strategy/copy.rb +210 -0
  50. data/lib/capistrano/recipes/deploy/strategy/export.rb +20 -0
  51. data/lib/capistrano/recipes/deploy/strategy/remote.rb +52 -0
  52. data/lib/capistrano/recipes/deploy/strategy/remote_cache.rb +56 -0
  53. data/lib/capistrano/recipes/deploy/strategy.rb +19 -0
  54. data/lib/capistrano/recipes/deploy/templates/maintenance.rhtml +53 -0
  55. data/lib/capistrano/recipes/deploy.rb +562 -0
  56. data/lib/capistrano/recipes/standard.rb +37 -0
  57. data/lib/capistrano/recipes/templates/maintenance.rhtml +53 -0
  58. data/lib/capistrano/recipes/upgrade.rb +33 -0
  59. data/lib/capistrano/role.rb +102 -0
  60. data/lib/capistrano/server_definition.rb +56 -0
  61. data/lib/capistrano/shell.rb +260 -0
  62. data/lib/capistrano/ssh.rb +99 -0
  63. data/lib/capistrano/task_definition.rb +70 -0
  64. data/lib/capistrano/transfer.rb +216 -0
  65. data/lib/capistrano/version.rb +18 -0
  66. data/lib/capistrano.rb +2 -0
  67. data/setup.rb +1346 -0
  68. data/test/cli/execute_test.rb +132 -0
  69. data/test/cli/help_test.rb +165 -0
  70. data/test/cli/options_test.rb +317 -0
  71. data/test/cli/ui_test.rb +28 -0
  72. data/test/cli_test.rb +17 -0
  73. data/test/command_test.rb +286 -0
  74. data/test/configuration/actions/file_transfer_test.rb +61 -0
  75. data/test/configuration/actions/inspect_test.rb +65 -0
  76. data/test/configuration/actions/invocation_test.rb +224 -0
  77. data/test/configuration/callbacks_test.rb +220 -0
  78. data/test/configuration/connections_test.rb +349 -0
  79. data/test/configuration/execution_test.rb +175 -0
  80. data/test/configuration/loading_test.rb +132 -0
  81. data/test/configuration/namespace_dsl_test.rb +311 -0
  82. data/test/configuration/roles_test.rb +144 -0
  83. data/test/configuration/servers_test.rb +121 -0
  84. data/test/configuration/variables_test.rb +184 -0
  85. data/test/configuration_test.rb +88 -0
  86. data/test/deploy/local_dependency_test.rb +76 -0
  87. data/test/deploy/remote_dependency_test.rb +114 -0
  88. data/test/deploy/scm/accurev_test.rb +23 -0
  89. data/test/deploy/scm/base_test.rb +55 -0
  90. data/test/deploy/scm/git_test.rb +167 -0
  91. data/test/deploy/scm/mercurial_test.rb +129 -0
  92. data/test/deploy/strategy/copy_test.rb +258 -0
  93. data/test/extensions_test.rb +69 -0
  94. data/test/fixtures/cli_integration.rb +5 -0
  95. data/test/fixtures/config.rb +5 -0
  96. data/test/fixtures/custom.rb +3 -0
  97. data/test/logger_test.rb +123 -0
  98. data/test/role_test.rb +11 -0
  99. data/test/server_definition_test.rb +121 -0
  100. data/test/shell_test.rb +90 -0
  101. data/test/ssh_test.rb +104 -0
  102. data/test/task_definition_test.rb +101 -0
  103. data/test/transfer_test.rb +160 -0
  104. data/test/utils.rb +38 -0
  105. metadata +205 -0
@@ -0,0 +1,90 @@
1
+ require "utils"
2
+ require 'capistrano/configuration'
3
+ require 'capistrano/shell'
4
+
5
+ class ShellTest < Test::Unit::TestCase
6
+ def setup
7
+ @config = Capistrano::Configuration.new
8
+ @shell = Capistrano::Shell.new(@config)
9
+ @shell.stubs(:puts)
10
+ end
11
+
12
+ def test_readline_fallback_prompt_should_write_to_stdout_and_read_from_stdin
13
+ STDOUT.expects(:print).with("prompt> ")
14
+ STDOUT.expects(:flush)
15
+ STDIN.expects(:gets).returns("hi\n")
16
+ assert_equal "hi\n", Capistrano::Shell::ReadlineFallback.readline("prompt> ")
17
+ end
18
+
19
+ def test_question_mark_as_input_should_trigger_help
20
+ @shell.expects(:read_line).returns("?")
21
+ @shell.expects(:help)
22
+ assert @shell.read_and_execute
23
+ end
24
+
25
+ def test_help_as_input_should_trigger_help
26
+ @shell.expects(:read_line).returns("help")
27
+ @shell.expects(:help)
28
+ assert @shell.read_and_execute
29
+ end
30
+
31
+ def test_quit_as_input_should_cause_read_and_execute_to_return_false
32
+ @shell.expects(:read_line).returns("quit")
33
+ assert !@shell.read_and_execute
34
+ end
35
+
36
+ def test_exit_as_input_should_cause_read_and_execute_to_return_false
37
+ @shell.expects(:read_line).returns("exit")
38
+ assert !@shell.read_and_execute
39
+ end
40
+
41
+ def test_set_should_parse_flag_and_value_and_call_set_option
42
+ @shell.expects(:read_line).returns("set -v 5")
43
+ @shell.expects(:set_option).with("v", "5")
44
+ assert @shell.read_and_execute
45
+ end
46
+
47
+ def test_text_without_with_or_on_gets_processed_verbatim
48
+ @shell.expects(:read_line).returns("hello world")
49
+ @shell.expects(:process_command).with(nil, nil, "hello world")
50
+ assert @shell.read_and_execute
51
+ end
52
+
53
+ def test_text_with_with_gets_processed_with_with # lol
54
+ @shell.expects(:read_line).returns("with app,db hello world")
55
+ @shell.expects(:process_command).with("with", "app,db", "hello world")
56
+ assert @shell.read_and_execute
57
+ end
58
+
59
+ def test_text_with_on_gets_processed_with_on
60
+ @shell.expects(:read_line).returns("on app,db hello world")
61
+ @shell.expects(:process_command).with("on", "app,db", "hello world")
62
+ assert @shell.read_and_execute
63
+ end
64
+
65
+ def test_task_command_with_bang_gets_processed_by_exec_tasks
66
+ while_testing_post_exec_commands do
67
+ @shell.expects(:read_line).returns("!deploy")
68
+ @shell.expects(:exec_tasks).with(["deploy"])
69
+ assert @shell.read_and_execute
70
+ end
71
+ end
72
+
73
+ def test_normal_command_gets_processed_by_exec_command
74
+ while_testing_post_exec_commands do
75
+ @shell.expects(:read_line).returns("uptime")
76
+ @shell.expects(:exec_command).with("uptime",nil)
77
+ @shell.expects(:connect)
78
+ assert @shell.read_and_execute
79
+ end
80
+ end
81
+
82
+
83
+ private
84
+
85
+ def while_testing_post_exec_commands(&block)
86
+ @shell.instance_variable_set(:@mutex,Mutex.new)
87
+ yield
88
+ end
89
+
90
+ end
data/test/ssh_test.rb ADDED
@@ -0,0 +1,104 @@
1
+ require "utils"
2
+ require 'capistrano/ssh'
3
+
4
+ class SSHTest < Test::Unit::TestCase
5
+ def setup
6
+ Capistrano::ServerDefinition.stubs(:default_user).returns("default-user")
7
+ @options = { :password => nil,
8
+ :auth_methods => %w(publickey hostbased),
9
+ :config => false }
10
+ @server = server("capistrano")
11
+ Net::SSH.stubs(:configuration_for).returns({})
12
+ end
13
+
14
+ def test_connect_with_bare_server_without_options_or_config_with_public_key_succeeding_should_only_loop_once
15
+ Net::SSH.expects(:start).with(@server.host, "default-user", @options).returns(success = Object.new)
16
+ assert_equal success, Capistrano::SSH.connect(@server)
17
+ end
18
+
19
+ def test_connect_with_bare_server_without_options_with_public_key_failing_should_try_password
20
+ Net::SSH.expects(:start).with(@server.host, "default-user", @options).raises(Net::SSH::AuthenticationFailed)
21
+ Net::SSH.expects(:start).with(@server.host, "default-user", @options.merge(:password => "f4b13n", :auth_methods => %w(password keyboard-interactive))).returns(success = Object.new)
22
+ assert_equal success, Capistrano::SSH.connect(@server, :password => "f4b13n")
23
+ end
24
+
25
+ def test_connect_with_bare_server_without_options_public_key_and_password_failing_should_raise_error
26
+ Net::SSH.expects(:start).with(@server.host, "default-user", @options).raises(Net::SSH::AuthenticationFailed)
27
+ Net::SSH.expects(:start).with(@server.host, "default-user", @options.merge(:password => "f4b13n", :auth_methods => %w(password keyboard-interactive))).raises(Net::SSH::AuthenticationFailed)
28
+ assert_raises(Net::SSH::AuthenticationFailed) do
29
+ Capistrano::SSH.connect(@server, :password => "f4b13n")
30
+ end
31
+ end
32
+
33
+ def test_connect_with_bare_server_and_user_via_public_key_should_pass_user_to_net_ssh
34
+ Net::SSH.expects(:start).with(@server.host, "jamis", @options).returns(success = Object.new)
35
+ assert_equal success, Capistrano::SSH.connect(@server, :user => "jamis")
36
+ end
37
+
38
+ def test_connect_with_bare_server_and_user_via_password_should_pass_user_to_net_ssh
39
+ Net::SSH.expects(:start).with(@server.host, "jamis", @options).raises(Net::SSH::AuthenticationFailed)
40
+ Net::SSH.expects(:start).with(@server.host, "jamis", @options.merge(:password => "f4b13n", :auth_methods => %w(password keyboard-interactive))).returns(success = Object.new)
41
+ assert_equal success, Capistrano::SSH.connect(@server, :user => "jamis", :password => "f4b13n")
42
+ end
43
+
44
+ def test_connect_with_bare_server_with_explicit_port_should_pass_port_to_net_ssh
45
+ Net::SSH.expects(:start).with(@server.host, "default-user", @options.merge(:port => 1234)).returns(success = Object.new)
46
+ assert_equal success, Capistrano::SSH.connect(@server, :port => 1234)
47
+ end
48
+
49
+ def test_connect_with_server_with_user_should_pass_user_to_net_ssh
50
+ server = server("jamis@capistrano")
51
+ Net::SSH.expects(:start).with(server.host, "jamis", @options).returns(success = Object.new)
52
+ assert_equal success, Capistrano::SSH.connect(server)
53
+ end
54
+
55
+ def test_connect_with_server_with_port_should_pass_port_to_net_ssh
56
+ server = server("capistrano:1235")
57
+ Net::SSH.expects(:start).with(server.host, "default-user", @options.merge(:port => 1235)).returns(success = Object.new)
58
+ assert_equal success, Capistrano::SSH.connect(server)
59
+ end
60
+
61
+ def test_connect_with_server_with_user_and_port_should_pass_user_and_port_to_net_ssh
62
+ server = server("jamis@capistrano:1235")
63
+ Net::SSH.expects(:start).with(server.host, "jamis", @options.merge(:port => 1235)).returns(success = Object.new)
64
+ assert_equal success, Capistrano::SSH.connect(server)
65
+ end
66
+
67
+ def test_connect_with_server_with_other_ssh_options_should_pass_ssh_options_to_net_ssh
68
+ server = server("jamis@capistrano:1235", :ssh_options => { :keys => %w(some_valid_key), :auth_methods => %w(a_method), :hmac => 'none' })
69
+ Net::SSH.expects(:start).with(server.host, "jamis", @options.merge(:port => 1235, :keys => %w(some_valid_key), :auth_methods => %w(a_method), :hmac => 'none' )).returns(success = Object.new)
70
+ assert_equal success, Capistrano::SSH.connect(server)
71
+ end
72
+
73
+ def test_connect_with_ssh_options_should_use_ssh_options
74
+ ssh_options = { :username => "JamisMan", :port => 8125, :config => false }
75
+ Net::SSH.expects(:start).with(@server.host, "JamisMan", @options.merge(:port => 8125, :config => false)).returns(success = Object.new)
76
+ assert_equal success, Capistrano::SSH.connect(@server, {:ssh_options => ssh_options})
77
+ end
78
+
79
+ def test_connect_with_options_and_ssh_options_should_see_options_override_ssh_options
80
+ ssh_options = { :username => "JamisMan", :port => 8125, :forward_agent => true }
81
+ Net::SSH.expects(:start).with(@server.host, "jamis", @options.merge(:port => 1235, :forward_agent => true)).returns(success = Object.new)
82
+ assert_equal success, Capistrano::SSH.connect(@server, :ssh_options => ssh_options, :user => "jamis", :port => 1235)
83
+ end
84
+
85
+ def test_connect_with_ssh_options_should_see_server_options_override_ssh_options
86
+ ssh_options = { :username => "JamisMan", :port => 8125, :forward_agent => true }
87
+ server = server("jamis@capistrano:1235")
88
+ Net::SSH.expects(:start).with(server.host, "jamis", @options.merge(:port => 1235, :forward_agent => true, :config => false)).returns(success = Object.new)
89
+ assert_equal success, Capistrano::SSH.connect(server, {:ssh_options => ssh_options})
90
+ end
91
+
92
+ def test_connect_should_add_xserver_accessor_to_connection
93
+ Net::SSH.expects(:start).with(@server.host, "default-user", @options).returns(success = Object.new)
94
+ assert_equal success, Capistrano::SSH.connect(@server)
95
+ assert success.respond_to?(:xserver)
96
+ assert success.respond_to?(:xserver)
97
+ assert_equal success.xserver, @server
98
+ end
99
+
100
+ def test_connect_should_not_retry_if_custom_auth_methods_are_given
101
+ Net::SSH.expects(:start).with(@server.host, "default-user", @options.merge(:auth_methods => %w(publickey))).raises(Net::SSH::AuthenticationFailed)
102
+ assert_raises(Net::SSH::AuthenticationFailed) { Capistrano::SSH.connect(@server, :ssh_options => { :auth_methods => %w(publickey) }) }
103
+ end
104
+ end
@@ -0,0 +1,101 @@
1
+ require "utils"
2
+ require 'capistrano/task_definition'
3
+
4
+ class TaskDefinitionTest < Test::Unit::TestCase
5
+ def setup
6
+ @namespace = namespace
7
+ end
8
+
9
+ def test_fqn_at_top_level_should_be_task_name
10
+ task = new_task(:testing)
11
+ assert_equal "testing", task.fully_qualified_name
12
+ end
13
+
14
+ def test_fqn_in_namespace_should_include_namespace_fqn
15
+ ns = namespace("outer:inner")
16
+ task = new_task(:testing, ns)
17
+ assert_equal "outer:inner:testing", task.fully_qualified_name
18
+ end
19
+
20
+ def test_fqn_at_top_level_when_default_should_be_default
21
+ task = new_task(:default)
22
+ assert_equal "default", task.fully_qualified_name
23
+ end
24
+
25
+ def test_fqn_in_namespace_when_default_should_be_namespace_fqn
26
+ ns = namespace("outer:inner")
27
+ task = new_task(:default, ns)
28
+ ns.stubs(:default_task => task)
29
+ assert_equal "outer:inner", task.fully_qualified_name
30
+ end
31
+
32
+ def test_task_should_require_block
33
+ assert_raises(ArgumentError) do
34
+ Capistrano::TaskDefinition.new(:testing, @namespace)
35
+ end
36
+ end
37
+
38
+ def test_description_should_return_empty_string_if_not_given
39
+ assert_equal "", new_task(:testing).description
40
+ end
41
+
42
+ def test_description_should_return_desc_attribute
43
+ assert_equal "something", new_task(:testing, @namespace, :desc => "something").description
44
+ end
45
+
46
+ def test_description_should_strip_leading_and_trailing_whitespace
47
+ assert_equal "something", new_task(:testing, @namespace, :desc => " something ").description
48
+ end
49
+
50
+ def test_description_should_normalize_newlines
51
+ assert_equal "a\nb\nc", new_task(:testing, @namespace, :desc => "a\nb\r\nc").description
52
+ end
53
+
54
+ def test_description_should_detect_and_remove_indentation
55
+ desc = <<-DESC
56
+ Here is some indented text \
57
+ and I want all of this to \
58
+ run together on a single line, \
59
+ without any extraneous spaces.
60
+
61
+ additional indentation will
62
+ be preserved.
63
+ DESC
64
+
65
+ task = new_task(:testing, @namespace, :desc => desc)
66
+ assert_equal "Here is some indented text and I want all of this to run together on a single line, without any extraneous spaces.\n\n additional indentation will\n be preserved.", task.description
67
+ end
68
+
69
+ def test_description_munging_should_be_sensitive_to_code_blocks
70
+ desc = <<-DESC
71
+ Here is a line \
72
+ wrapped with spacing in it.
73
+
74
+ foo bar
75
+ baz bang
76
+ DESC
77
+
78
+ task = new_task(:testing, @namespace, :desc => desc)
79
+ assert_equal "Here is a line wrapped with spacing in it.\n\n foo bar\n baz bang", task.description
80
+ end
81
+
82
+ def test_brief_description_should_return_first_sentence_in_description
83
+ desc = "This is the task. It does all kinds of things."
84
+ task = new_task(:testing, @namespace, :desc => desc)
85
+ assert_equal "This is the task.", task.brief_description
86
+ end
87
+
88
+ def test_brief_description_should_truncate_if_length_given
89
+ desc = "This is the task that does all kinds of things. And then some."
90
+ task = new_task(:testing, @namespace, :desc => desc)
91
+ assert_equal "This is the task ...", task.brief_description(20)
92
+ end
93
+
94
+ def test_brief_description_should_not_break_at_period_in_middle_of_sentence
95
+ task = new_task(:testing, @namespace, :desc => "Take file.txt and copy it.")
96
+ assert_equal "Take file.txt and copy it.", task.brief_description
97
+
98
+ task = new_task(:testing, @namespace, :desc => "Take file.txt and copy it. Then do something else.")
99
+ assert_equal "Take file.txt and copy it.", task.brief_description
100
+ end
101
+ end
@@ -0,0 +1,160 @@
1
+ require 'utils'
2
+ require 'capistrano/transfer'
3
+
4
+ class TransferTest < Test::Unit::TestCase
5
+ def test_class_process_should_delegate_to_instance_process
6
+ Capistrano::Transfer.expects(:new).with(:up, "from", "to", %w(a b c), {}).returns(mock('transfer', :process! => nil)).yields
7
+ yielded = false
8
+ Capistrano::Transfer.process(:up, "from", "to", %w(a b c), {}) { yielded = true }
9
+ assert yielded
10
+ end
11
+
12
+ def test_default_transport_is_sftp
13
+ transfer = Capistrano::Transfer.new(:up, "from", "to", [])
14
+ assert_equal :sftp, transfer.transport
15
+ end
16
+
17
+ def test_active_is_true_when_any_sftp_transfers_are_active
18
+ returns = [false, false, true]
19
+ sessions = [session('app1', :sftp), session('app2', :sftp), session('app3', :sftp)].each { |s| s.xsftp.expects(:upload).returns(stub('operation', :active? => returns.shift)) }
20
+ transfer = Capistrano::Transfer.new(:up, "from", "to", sessions, :via => :sftp)
21
+ assert_equal true, transfer.active?
22
+ end
23
+
24
+ def test_active_is_false_when_all_sftp_transfers_are_not_active
25
+ sessions = [session('app1', :sftp), session('app2', :sftp)].each { |s| s.xsftp.expects(:upload).returns(stub('operation', :active? => false)) }
26
+ transfer = Capistrano::Transfer.new(:up, "from", "to", sessions, :via => :sftp)
27
+ assert_equal false, transfer.active?
28
+ end
29
+
30
+ def test_active_is_true_when_any_scp_transfers_are_active
31
+ returns = [false, false, true]
32
+ sessions = [session('app1', :scp), session('app2', :scp), session('app3', :scp)].each do |s|
33
+ channel = stub('channel', :[]= => nil, :active? => returns.shift)
34
+ s.scp.expects(:upload).returns(channel)
35
+ end
36
+ transfer = Capistrano::Transfer.new(:up, "from", "to", sessions, :via => :scp)
37
+ assert_equal true, transfer.active?
38
+ end
39
+
40
+ def test_active_is_false_when_all_scp_transfers_are_not_active
41
+ sessions = [session('app1', :scp), session('app2', :scp), session('app3', :scp)].each do |s|
42
+ channel = stub('channel', :[]= => nil, :active? => false)
43
+ s.scp.expects(:upload).returns(channel)
44
+ end
45
+ transfer = Capistrano::Transfer.new(:up, "from", "to", sessions, :via => :scp)
46
+ assert_equal false, transfer.active?
47
+ end
48
+
49
+ [:up, :down].each do |direction|
50
+ define_method("test_sftp_#{direction}load_from_file_to_file_should_normalize_from_and_to") do
51
+ sessions = [session('app1', :sftp), session('app2', :sftp)]
52
+
53
+ sessions.each do |session|
54
+ session.xsftp.expects("#{direction}load".to_sym).with("from-#{session.xserver.host}", "to-#{session.xserver.host}",
55
+ :properties => { :server => session.xserver, :host => session.xserver.host })
56
+ end
57
+
58
+ transfer = Capistrano::Transfer.new(direction, "from-$CAPISTRANO:HOST$", "to-$CAPISTRANO:HOST$", sessions)
59
+ end
60
+
61
+ define_method("test_scp_#{direction}load_from_file_to_file_should_normalize_from_and_to") do
62
+ sessions = [session('app1', :scp), session('app2', :scp)]
63
+
64
+ sessions.each do |session|
65
+ session.scp.expects("#{direction}load".to_sym).returns({}).with("from-#{session.xserver.host}", "to-#{session.xserver.host}", :via => :scp)
66
+ end
67
+
68
+ transfer = Capistrano::Transfer.new(direction, "from-$CAPISTRANO:HOST$", "to-$CAPISTRANO:HOST$", sessions, :via => :scp)
69
+ end
70
+ end
71
+
72
+ def test_sftp_upload_from_IO_to_file_should_clone_the_IO_for_each_connection
73
+ sessions = [session('app1', :sftp), session('app2', :sftp)]
74
+ io = StringIO.new("from here")
75
+
76
+ sessions.each do |session|
77
+ session.xsftp.expects(:upload).with do |from, to, opts|
78
+ from != io && from.is_a?(StringIO) && from.string == io.string &&
79
+ to == "/to/here-#{session.xserver.host}" &&
80
+ opts[:properties][:server] == session.xserver &&
81
+ opts[:properties][:host] == session.xserver.host
82
+ end
83
+ end
84
+
85
+ transfer = Capistrano::Transfer.new(:up, StringIO.new("from here"), "/to/here-$CAPISTRANO:HOST$", sessions)
86
+ end
87
+
88
+ def test_scp_upload_from_IO_to_file_should_clone_the_IO_for_each_connection
89
+ sessions = [session('app1', :scp), session('app2', :scp)]
90
+ io = StringIO.new("from here")
91
+
92
+ sessions.each do |session|
93
+ channel = mock('channel')
94
+ channel.expects(:[]=).with(:server, session.xserver)
95
+ channel.expects(:[]=).with(:host, session.xserver.host)
96
+ session.scp.expects(:upload).returns(channel).with do |from, to, opts|
97
+ from != io && from.is_a?(StringIO) && from.string == io.string &&
98
+ to == "/to/here-#{session.xserver.host}"
99
+ end
100
+ end
101
+
102
+ transfer = Capistrano::Transfer.new(:up, StringIO.new("from here"), "/to/here-$CAPISTRANO:HOST$", sessions, :via => :scp)
103
+ end
104
+
105
+ def test_process_should_block_until_transfer_is_no_longer_active
106
+ transfer = Capistrano::Transfer.new(:up, "from", "to", [])
107
+ transfer.expects(:process_iteration).times(4).yields.returns(true,true,true,false)
108
+ transfer.expects(:active?).times(4)
109
+ transfer.process!
110
+ end
111
+
112
+ def test_errors_raised_for_a_sftp_session_should_abort_session_and_continue_with_remaining_sessions
113
+ s = session('app1')
114
+ error = ExceptionWithSession.new(s)
115
+ transfer = Capistrano::Transfer.new(:up, "from", "to", [])
116
+ transfer.expects(:process_iteration).raises(error).times(3).returns(true, false)
117
+ txfr = mock('transfer', :abort! => true)
118
+ txfr.expects(:[]=).with(:failed, true)
119
+ txfr.expects(:[]=).with(:error, error)
120
+ transfer.expects(:session_map).returns(s => txfr)
121
+ transfer.process!
122
+ end
123
+
124
+ def test_errors_raised_for_a_scp_session_should_abort_session_and_continue_with_remaining_sessions
125
+ s = session('app1')
126
+ error = ExceptionWithSession.new(s)
127
+ transfer = Capistrano::Transfer.new(:up, "from", "to", [], :via => :scp)
128
+ transfer.expects(:process_iteration).raises(error).times(3).returns(true, false)
129
+ txfr = mock('channel', :close => true)
130
+ txfr.expects(:[]=).with(:failed, true)
131
+ txfr.expects(:[]=).with(:error, error)
132
+ transfer.expects(:session_map).returns(s => txfr)
133
+ transfer.process!
134
+ end
135
+
136
+ private
137
+
138
+ class ExceptionWithSession < ::Exception
139
+ attr_reader :session
140
+
141
+ def initialize(session)
142
+ @session = session
143
+ super()
144
+ end
145
+ end
146
+
147
+ def session(host, mode=nil)
148
+ session = stub('session', :xserver => stub('server', :host => host))
149
+ case mode
150
+ when :sftp
151
+ sftp = stub('sftp')
152
+ session.expects(:sftp).with(false).returns(sftp)
153
+ sftp.expects(:connect).yields(sftp).returns(sftp)
154
+ session.stubs(:xsftp).returns(sftp)
155
+ when :scp
156
+ session.stubs(:scp).returns(stub('scp'))
157
+ end
158
+ session
159
+ end
160
+ end
data/test/utils.rb ADDED
@@ -0,0 +1,38 @@
1
+ begin
2
+ require 'rubygems'
3
+ gem 'mocha'
4
+ rescue LoadError
5
+ end
6
+
7
+ require 'test/unit'
8
+ require 'mocha'
9
+ require 'capistrano/server_definition'
10
+
11
+ module TestExtensions
12
+ def server(host, options={})
13
+ Capistrano::ServerDefinition.new(host, options)
14
+ end
15
+
16
+ def namespace(fqn=nil)
17
+ space = stub(:roles => {}, :fully_qualified_name => fqn, :default_task => nil)
18
+ yield(space) if block_given?
19
+ space
20
+ end
21
+
22
+ def role(space, name, *args)
23
+ opts = args.last.is_a?(Hash) ? args.pop : {}
24
+ space.roles[name] ||= []
25
+ space.roles[name].concat(args.map { |h| Capistrano::ServerDefinition.new(h, opts) })
26
+ end
27
+
28
+ def new_task(name, namespace=@namespace, options={}, &block)
29
+ block ||= Proc.new {}
30
+ task = Capistrano::TaskDefinition.new(name, namespace, options, &block)
31
+ assert_equal block, task.body
32
+ return task
33
+ end
34
+ end
35
+
36
+ class Test::Unit::TestCase
37
+ include TestExtensions
38
+ end