minestrone 0.0.1

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 (96) hide show
  1. checksums.yaml +7 -0
  2. data/.github/workflows/main.yml +32 -0
  3. data/.gitignore +5 -0
  4. data/Gemfile +10 -0
  5. data/LICENSE.txt +22 -0
  6. data/README.md +35 -0
  7. data/Rakefile +10 -0
  8. data/bin/capify +89 -0
  9. data/bin/min +5 -0
  10. data/docs/lib-codebase-map.md +162 -0
  11. data/docs/lib-dependency-graph.svg +129 -0
  12. data/lib/minestrone/callback.rb +45 -0
  13. data/lib/minestrone/cli/help.rb +131 -0
  14. data/lib/minestrone/cli/help.txt +72 -0
  15. data/lib/minestrone/cli/options.rb +232 -0
  16. data/lib/minestrone/cli.rb +159 -0
  17. data/lib/minestrone/command.rb +177 -0
  18. data/lib/minestrone/configuration/actions/file_transfer.rb +53 -0
  19. data/lib/minestrone/configuration/actions/inspect.rb +46 -0
  20. data/lib/minestrone/configuration/actions/invocation.rb +202 -0
  21. data/lib/minestrone/configuration/alias_task.rb +29 -0
  22. data/lib/minestrone/configuration/callbacks.rb +129 -0
  23. data/lib/minestrone/configuration/connections.rb +66 -0
  24. data/lib/minestrone/configuration/execution.rb +139 -0
  25. data/lib/minestrone/configuration/loading.rb +207 -0
  26. data/lib/minestrone/configuration/log_formatters.rb +75 -0
  27. data/lib/minestrone/configuration/namespaces.rb +225 -0
  28. data/lib/minestrone/configuration/servers.rb +70 -0
  29. data/lib/minestrone/configuration/variables.rb +115 -0
  30. data/lib/minestrone/configuration.rb +69 -0
  31. data/lib/minestrone/errors.rb +17 -0
  32. data/lib/minestrone/ext/string.rb +7 -0
  33. data/lib/minestrone/extensions.rb +56 -0
  34. data/lib/minestrone/logger.rb +171 -0
  35. data/lib/minestrone/processable.rb +50 -0
  36. data/lib/minestrone/recipes/deploy/assets.rb +194 -0
  37. data/lib/minestrone/recipes/deploy/bundler.rb +81 -0
  38. data/lib/minestrone/recipes/deploy/dependencies.rb +44 -0
  39. data/lib/minestrone/recipes/deploy/local_dependency.rb +45 -0
  40. data/lib/minestrone/recipes/deploy/remote_dependency.rb +119 -0
  41. data/lib/minestrone/recipes/deploy/scm/base.rb +204 -0
  42. data/lib/minestrone/recipes/deploy/scm/git.rb +284 -0
  43. data/lib/minestrone/recipes/deploy/scm/none.rb +54 -0
  44. data/lib/minestrone/recipes/deploy/scm.rb +22 -0
  45. data/lib/minestrone/recipes/deploy/strategy/base.rb +87 -0
  46. data/lib/minestrone/recipes/deploy/strategy/copy.rb +353 -0
  47. data/lib/minestrone/recipes/deploy/strategy/remote_cache.rb +80 -0
  48. data/lib/minestrone/recipes/deploy/strategy.rb +22 -0
  49. data/lib/minestrone/recipes/deploy.rb +639 -0
  50. data/lib/minestrone/recipes/standard.rb +23 -0
  51. data/lib/minestrone/recipes/templates/maintenance.rhtml +53 -0
  52. data/lib/minestrone/server_definition.rb +56 -0
  53. data/lib/minestrone/ssh.rb +81 -0
  54. data/lib/minestrone/task_definition.rb +82 -0
  55. data/lib/minestrone/transfer.rb +205 -0
  56. data/lib/minestrone/version.rb +11 -0
  57. data/lib/minestrone.rb +3 -0
  58. data/minestrone.gemspec +32 -0
  59. data/test/cli/execute_test.rb +130 -0
  60. data/test/cli/help_test.rb +178 -0
  61. data/test/cli/options_test.rb +315 -0
  62. data/test/cli/ui_test.rb +26 -0
  63. data/test/cli_test.rb +17 -0
  64. data/test/command_test.rb +305 -0
  65. data/test/configuration/actions/file_transfer_test.rb +61 -0
  66. data/test/configuration/actions/inspect_test.rb +76 -0
  67. data/test/configuration/actions/invocation_test.rb +258 -0
  68. data/test/configuration/alias_task_test.rb +110 -0
  69. data/test/configuration/callbacks_test.rb +201 -0
  70. data/test/configuration/connections_test.rb +192 -0
  71. data/test/configuration/execution_test.rb +176 -0
  72. data/test/configuration/loading_test.rb +149 -0
  73. data/test/configuration/namespace_dsl_test.rb +325 -0
  74. data/test/configuration/servers_test.rb +100 -0
  75. data/test/configuration/variables_test.rb +191 -0
  76. data/test/configuration_test.rb +77 -0
  77. data/test/deploy/local_dependency_test.rb +61 -0
  78. data/test/deploy/remote_dependency_test.rb +146 -0
  79. data/test/deploy/scm/base_test.rb +55 -0
  80. data/test/deploy/scm/git_test.rb +260 -0
  81. data/test/deploy/scm/none_test.rb +26 -0
  82. data/test/deploy/strategy/copy_test.rb +360 -0
  83. data/test/extensions_test.rb +69 -0
  84. data/test/fixtures/cli_integration.rb +5 -0
  85. data/test/fixtures/config.rb +4 -0
  86. data/test/fixtures/custom.rb +3 -0
  87. data/test/logger_formatting_test.rb +149 -0
  88. data/test/logger_test.rb +134 -0
  89. data/test/recipes_test.rb +26 -0
  90. data/test/server_definition_test.rb +121 -0
  91. data/test/ssh_test.rb +99 -0
  92. data/test/task_definition_test.rb +117 -0
  93. data/test/transfer_test.rb +172 -0
  94. data/test/utils.rb +28 -0
  95. data/test/version_test.rb +11 -0
  96. metadata +258 -0
@@ -0,0 +1,69 @@
1
+ require "utils"
2
+ require 'minestrone'
3
+
4
+ class ExtensionsTest < Test::Unit::TestCase
5
+ module CustomExtension
6
+ def do_something(command)
7
+ run(command)
8
+ end
9
+ end
10
+
11
+ def setup
12
+ @config = Minestrone::Configuration.new
13
+ end
14
+
15
+ def teardown
16
+ Minestrone::EXTENSIONS.keys.each { |e| Minestrone.remove_plugin(e) }
17
+ end
18
+
19
+ def test_register_plugin_should_add_instance_method_on_configuration_and_return_true
20
+ assert !@config.respond_to?(:custom_stuff)
21
+ assert Minestrone.plugin(:custom_stuff, CustomExtension)
22
+ assert @config.respond_to?(:custom_stuff)
23
+ end
24
+
25
+ def test_register_plugin_that_already_exists_should_return_false
26
+ assert Minestrone.plugin(:custom_stuff, CustomExtension)
27
+ assert !Minestrone.plugin(:custom_stuff, CustomExtension)
28
+ end
29
+
30
+ def test_register_plugin_with_public_method_name_should_fail
31
+ method = Minestrone::Configuration.public_instance_methods.first
32
+ assert_not_nil method, "need a public instance method for testing"
33
+ assert_raises(Minestrone::Error) { Minestrone.plugin(method, CustomExtension) }
34
+ end
35
+
36
+ def test_register_plugin_with_protected_method_name_should_fail
37
+ method = Minestrone::Configuration.protected_instance_methods.first
38
+ assert_not_nil method, "need a protected instance method for testing"
39
+ assert_raises(Minestrone::Error) { Minestrone.plugin(method, CustomExtension) }
40
+ end
41
+
42
+ def test_register_plugin_with_private_method_name_should_fail
43
+ method = Minestrone::Configuration.private_instance_methods.first
44
+ assert_not_nil method, "need a private instance method for testing"
45
+ assert_raises(Minestrone::Error) { Minestrone.plugin(method, CustomExtension) }
46
+ end
47
+
48
+ def test_unregister_plugin_that_does_not_exist_should_return_false
49
+ assert !Minestrone.remove_plugin(:custom_stuff)
50
+ end
51
+
52
+ def test_unregister_plugin_should_remove_method_and_return_true
53
+ assert Minestrone.plugin(:custom_stuff, CustomExtension)
54
+ assert @config.respond_to?(:custom_stuff)
55
+ assert Minestrone.remove_plugin(:custom_stuff)
56
+ assert !@config.respond_to?(:custom_stuff)
57
+ end
58
+
59
+ def test_registered_plugin_proxy_should_return_proxy_object
60
+ Minestrone.plugin(:custom_stuff, CustomExtension)
61
+ assert_instance_of Minestrone::ExtensionProxy, @config.custom_stuff
62
+ end
63
+
64
+ def test_proxy_object_should_delegate_to_configuration
65
+ Minestrone.plugin(:custom_stuff, CustomExtension)
66
+ @config.expects(:run).with("hello")
67
+ @config.custom_stuff.do_something("hello")
68
+ end
69
+ end
@@ -0,0 +1,5 @@
1
+ server "www.minestrone.test"
2
+
3
+ task :testing do
4
+ set :testing_occurred, true
5
+ end
@@ -0,0 +1,4 @@
1
+ set :application, "foo"
2
+ set :repository, "1/2/#{application}"
3
+
4
+ server "www.example.com"
@@ -0,0 +1,3 @@
1
+ ConfigurationLoadingTest::MockConfig.instance(:must_exist).load do
2
+ ping! :custom
3
+ end
@@ -0,0 +1,149 @@
1
+ require File.expand_path("../utils", __FILE__)
2
+ require 'minestrone/logger'
3
+ require 'stringio'
4
+
5
+ Minestrone::Logger.class_eval do
6
+ # Allows formatters to be changed during tests
7
+ def self.formatters=(formatters)
8
+ @formatters = formatters
9
+ @sorted_formatters = nil
10
+ end
11
+ end
12
+
13
+ class LoggerFormattingTest < Test::Unit::TestCase
14
+ def setup
15
+ @io = StringIO.new
16
+ @io.stubs(:tty?).returns(true)
17
+ @logger = Minestrone::Logger.new(:output => @io, :level => 3)
18
+ end
19
+
20
+ def test_matching_with_style_and_color
21
+ Minestrone::Logger.formatters = [{ :match => /^err ::/, :color => :red, :style => :underscore, :level => 0 }]
22
+ @logger.log(0, "err :: Error Occurred")
23
+ assert @io.string.include? "\e[4;31merr :: Error Occurred\e[0m"
24
+ end
25
+
26
+ def test_style_without_color
27
+ Minestrone::Logger.formatters = [{ :match => /.*/, :style => :underscore, :level => 0 }]
28
+ @logger.log(0, "test message")
29
+ # Default color should be blank (0m)
30
+ assert @io.string.include? "\e[4;0mtest message\e[0m"
31
+ end
32
+
33
+ def test_prepending_text
34
+ Minestrone::Logger.formatters = [{ :match => /^executing/, :level => 0, :prepend => '== Currently ' }]
35
+ @logger.log(0, "executing task")
36
+ assert @io.string.include? '== Currently executing task'
37
+ end
38
+
39
+ def test_replacing_matched_text
40
+ Minestrone::Logger.formatters = [{ :match => /^executing/, :level => 0, :replace => 'running' }]
41
+ @logger.log(0, "executing task")
42
+ assert @io.string.include? 'running task'
43
+ end
44
+
45
+ def test_prepending_timestamps
46
+ Minestrone::Logger.formatters = [{ :match => /.*/, :level => 0, :timestamp => true }]
47
+ @logger.log(0, "test message")
48
+ assert @io.string.match(/\d{4}-\d{2}-\d{2} \d{2}:\d{2}:\d{2} test message/)
49
+ end
50
+
51
+ def test_formatter_priorities
52
+ Minestrone::Logger.formatters = [
53
+ { :match => /.*/, :color => :red, :level => 0, :priority => -10 },
54
+ { :match => /.*/, :color => :blue, :level => 0, :priority => -20, :prepend => '###' }
55
+ ]
56
+
57
+ @logger.log(0, "test message")
58
+ # Only the red formatter (color 31) should be applied.
59
+ assert @io.string.include? "\e[31mtest message"
60
+ # The blue formatter should not have prepended $$$
61
+ assert !@io.string.include?('###')
62
+ end
63
+
64
+ def test_no_formatting_if_no_color_or_style
65
+ Minestrone::Logger.formatters = []
66
+ @logger.log(0, "test message")
67
+ assert @io.string.include? "*** test message"
68
+ end
69
+
70
+ def test_formatter_log_levels
71
+ Minestrone::Logger.formatters = [{ :match => /.*/, :color => :blue, :level => 3 }]
72
+ @logger.log(0, "test message")
73
+ # Should not match log level
74
+ assert @io.string.include? "*** test message"
75
+
76
+ clear_logger
77
+ @logger.log(3, "test message")
78
+ # Should match log level and apply blue color
79
+ assert @io.string.include? "\e[34mtest message"
80
+ end
81
+
82
+ private
83
+
84
+ def colorize(message, color, style = nil)
85
+ style = "#{style};" if style
86
+ "\e[#{style}#{color}m" + message + "\e[0m"
87
+ end
88
+
89
+ def clear_logger
90
+ @io = StringIO.new
91
+ @io.stubs(:tty?).returns(true)
92
+ @logger.device = @io
93
+ end
94
+ end
95
+
96
+ class DefaultLoggerFormattersTest < Test::Unit::TestCase
97
+ def setup
98
+ @expected_default_formatter_values = [
99
+ # TRACE
100
+ { :match => /command finished/, :color => :white, :style => :dim, :level => 3, :priority => -10 },
101
+ { :match => /executing locally/, :color => :yellow, :level => 3, :priority => -20 },
102
+
103
+ # DEBUG
104
+ { :match => /executing `.*/, :color => :green, :level => 2, :priority => -10, :timestamp => true },
105
+ { :match => /.*/, :color => :yellow, :level => 2, :priority => -30 },
106
+
107
+ # INFO
108
+ { :match => /.*out\] (fatal:|ERROR:).*/, :color => :red, :level => 1, :priority => -10 },
109
+ { :match => /Permission denied/, :color => :red, :level => 1, :priority => -20 },
110
+ { :match => /sh: .+: command not found/, :color => :magenta, :level => 1, :priority => -30 },
111
+
112
+ # IMPORTANT
113
+ { :match => /^err ::/, :color => :red, :level => 0, :priority => -10 },
114
+ { :match => /.*/, :color => :blue, :level => 0, :priority => -20 }
115
+ ]
116
+
117
+ @custom_default_formatter_values = [
118
+ { :match => /.*/, :color => :white }
119
+ ]
120
+
121
+ end
122
+
123
+ def test_default_formatters_api
124
+ assert Minestrone::Logger.respond_to? :default_formatters
125
+ assert Minestrone::Logger.respond_to? :default_formatters=
126
+ end
127
+
128
+ def test_default_formatters_values
129
+ assert_equal @expected_default_formatter_values, Minestrone::Logger.default_formatters
130
+ assert_equal @expected_default_formatter_values, Minestrone::Logger.instance_variable_get("@formatters")
131
+ assert_equal nil, Minestrone::Logger.instance_variable_get("@sorted_formatters")
132
+ end
133
+
134
+ def test_set_default_formatters_values
135
+ # when given an array
136
+ Minestrone::Logger.default_formatters = @custom_default_formatter_values
137
+
138
+ assert_equal @custom_default_formatter_values, Minestrone::Logger.default_formatters
139
+ Minestrone::Logger.default_formatters = @custom_default_formatter_values
140
+ assert_equal @custom_default_formatter_values, Minestrone::Logger.instance_variable_get("@formatters")
141
+ assert_equal nil, Minestrone::Logger.instance_variable_get("@sorted_formatters")
142
+
143
+ # when given a single formatter values hash
144
+ Minestrone::Logger.default_formatters = @custom_default_formatter_values.first
145
+
146
+ assert_equal @custom_default_formatter_values, Minestrone::Logger.default_formatters
147
+ end
148
+
149
+ end
@@ -0,0 +1,134 @@
1
+ require "utils"
2
+ require 'minestrone/logger'
3
+ require 'stringio'
4
+
5
+ class LoggerTest < Test::Unit::TestCase
6
+ def setup
7
+ @io = StringIO.new
8
+ # Turn off formatting for these tests. Formatting is tested in `logger_formatting_test.rb`.
9
+ @logger = Minestrone::Logger.new(:output => @io, :disable_formatters => true)
10
+ end
11
+
12
+ def test_logger_should_use_STDERR_by_default
13
+ logger = Minestrone::Logger.new
14
+ assert_equal STDERR, logger.device
15
+ end
16
+
17
+ def test_logger_should_have_log_level_0
18
+ logger = Minestrone::Logger.new
19
+ assert_equal 0, logger.level
20
+ end
21
+
22
+ def test_logger_should_use_level_form_options
23
+ logger = Minestrone::Logger.new :level => 4
24
+ assert_equal 4, logger.level
25
+ end
26
+
27
+ def test_logger_should_use_output_option_if_output_responds_to_puts
28
+ logger = Minestrone::Logger.new(:output => STDOUT)
29
+ assert_equal STDOUT, logger.device
30
+ end
31
+
32
+ def test_logger_should_open_file_if_output_does_not_respond_to_puts
33
+ File.expects(:open).with("logs/minestrone.log", "a").returns(:mock)
34
+ logger = Minestrone::Logger.new(:output => "logs/minestrone.log")
35
+ assert_equal :mock, logger.device
36
+ end
37
+
38
+ def test_close_should_not_close_device_if_device_is_default
39
+ logger = Minestrone::Logger.new
40
+ logger.device.expects(:close).never
41
+ logger.close
42
+ end
43
+
44
+ def test_close_should_not_close_device_is_device_is_explicitly_given
45
+ logger = Minestrone::Logger.new(:output => STDOUT)
46
+ STDOUT.expects(:close).never
47
+ logger.close
48
+ end
49
+
50
+ def test_close_should_close_device_when_device_was_implicitly_opened
51
+ f = mock("file", :close => nil)
52
+ File.expects(:open).with("logs/minestrone.log", "a").returns(f)
53
+ logger = Minestrone::Logger.new(:output => "logs/minestrone.log")
54
+ logger.close
55
+ end
56
+
57
+ def test_log_with_level_greater_than_threshold_should_ignore_message
58
+ @logger.level = 3
59
+ @logger.log(4, "message")
60
+ assert @io.string.empty?
61
+ end
62
+
63
+ def test_log_with_level_equal_to_threshold_should_log_message
64
+ @logger.level = 3
65
+ @logger.log(3, "message")
66
+ assert @io.string.include?("message")
67
+ end
68
+
69
+ def test_log_with_level_less_than_threshold_should_log_message
70
+ @logger.level = 3
71
+ @logger.log(2, "message")
72
+ assert @io.string.include?("message")
73
+ end
74
+
75
+ def test_log_with_multiline_message_should_log_each_line_separately
76
+ @logger.log(0, "first line\nsecond line")
77
+ assert @io.string.include?("*** first line")
78
+ assert @io.string.include?("*** second line")
79
+ end
80
+
81
+ def test_log_with_line_prefix_should_insert_line_prefix_before_message
82
+ @logger.log(0, "message", "prefix")
83
+ assert @io.string.include?("*** [prefix] message")
84
+ end
85
+
86
+ def test_log_with_level_0_should_have_strong_indent
87
+ @logger.log(0, "message")
88
+ assert @io.string.match(/^\*\*\* message/)
89
+ end
90
+
91
+ def test_log_with_level_1_should_have_weaker_indent
92
+ @logger.level = 1
93
+ @logger.log(1, "message")
94
+ assert @io.string.match(/^ \*\* message/)
95
+ end
96
+
97
+ def test_log_with_level_2_should_have_weaker_indent
98
+ @logger.level = 2
99
+ @logger.log(2, "message")
100
+ assert @io.string.match(/^ \* message/)
101
+ end
102
+
103
+ def test_log_with_level_3_should_have_weakest_indent
104
+ @logger.level = 3
105
+ @logger.log(3, "message")
106
+ assert @io.string.match(/^ message/)
107
+ end
108
+
109
+ def test_important_should_delegate_to_log_with_level_IMPORTANT
110
+ @logger.expects(:log).with(Minestrone::Logger::IMPORTANT, "message", "prefix")
111
+ @logger.important("message", "prefix")
112
+ end
113
+
114
+ def test_info_should_delegate_to_log_with_level_INFO
115
+ @logger.expects(:log).with(Minestrone::Logger::INFO, "message", "prefix")
116
+ @logger.info("message", "prefix")
117
+ end
118
+
119
+ def test_debug_should_delegate_to_log_with_level_DEBUG
120
+ @logger.expects(:log).with(Minestrone::Logger::DEBUG, "message", "prefix")
121
+ @logger.debug("message", "prefix")
122
+ end
123
+
124
+ def test_trace_should_delegate_to_log_with_level_TRACE
125
+ @logger.expects(:log).with(Minestrone::Logger::TRACE, "message", "prefix")
126
+ @logger.trace("message", "prefix")
127
+ end
128
+
129
+ def test_ordering_of_levels
130
+ assert Minestrone::Logger::IMPORTANT < Minestrone::Logger::INFO
131
+ assert Minestrone::Logger::INFO < Minestrone::Logger::DEBUG
132
+ assert Minestrone::Logger::DEBUG < Minestrone::Logger::TRACE
133
+ end
134
+ end
@@ -0,0 +1,26 @@
1
+ require 'utils'
2
+ require 'minestrone/configuration'
3
+
4
+ class RecipesTest < Test::Unit::TestCase
5
+
6
+ def setup
7
+ @config = Minestrone::Configuration.new
8
+ @config.stubs(:logger).returns(stub_everything)
9
+ end
10
+
11
+ def test_current_releases_does_not_cause_error_on_dry_run
12
+ @config.dry_run = true
13
+ @config.load 'deploy'
14
+ @config.load do
15
+ server "example.com"
16
+ set :application, "foo"
17
+ task :dry_run_test do
18
+ fetch :current_release
19
+ end
20
+ end
21
+
22
+ assert_nothing_raised do
23
+ @config.dry_run_test
24
+ end
25
+ end
26
+ end
@@ -0,0 +1,121 @@
1
+ require "utils"
2
+ require 'minestrone/server_definition'
3
+
4
+ class ServerDefinitionTest < Test::Unit::TestCase
5
+ def test_new_without_credentials_or_port_should_set_values_to_defaults
6
+ server = Minestrone::ServerDefinition.new("www.minestrone.test")
7
+ assert_equal "www.minestrone.test", server.host
8
+ assert_nil server.user
9
+ assert_nil server.port
10
+ end
11
+
12
+ def test_new_with_encoded_user_should_extract_user_and_use_default_port
13
+ server = Minestrone::ServerDefinition.new("jamis@www.minestrone.test")
14
+ assert_equal "www.minestrone.test", server.host
15
+ assert_equal "jamis", server.user
16
+ assert_nil server.port
17
+ end
18
+
19
+ def test_new_with_encoded_port_should_extract_port_and_use_default_user
20
+ server = Minestrone::ServerDefinition.new("www.minestrone.test:8080")
21
+ assert_equal "www.minestrone.test", server.host
22
+ assert_nil server.user
23
+ assert_equal 8080, server.port
24
+ end
25
+
26
+ def test_new_with_encoded_user_and_port_should_extract_user_and_port
27
+ server = Minestrone::ServerDefinition.new("jamis@www.minestrone.test:8080")
28
+ assert_equal "www.minestrone.test", server.host
29
+ assert_equal "jamis", server.user
30
+ assert_equal 8080, server.port
31
+ end
32
+
33
+ def test_new_with_user_as_option_should_use_given_user
34
+ server = Minestrone::ServerDefinition.new("www.minestrone.test", :user => "jamis")
35
+ assert_equal "www.minestrone.test", server.host
36
+ assert_equal "jamis", server.user
37
+ assert_nil server.port
38
+ end
39
+
40
+ def test_new_with_port_as_option_should_use_given_user
41
+ server = Minestrone::ServerDefinition.new("www.minestrone.test", :port => 8080)
42
+ assert_equal "www.minestrone.test", server.host
43
+ assert_nil server.user
44
+ assert_equal 8080, server.port
45
+ end
46
+
47
+ def test_encoded_value_should_override_hash_option
48
+ server = Minestrone::ServerDefinition.new("jamis@www.minestrone.test:8080", :user => "david", :port => 8081)
49
+ assert_equal "www.minestrone.test", server.host
50
+ assert_equal "jamis", server.user
51
+ assert_equal 8080, server.port
52
+ assert server.options.empty?
53
+ end
54
+
55
+ def test_new_with_option_should_dup_option_hash
56
+ options = {}
57
+ server = Minestrone::ServerDefinition.new("www.minestrone.test", options)
58
+ assert_not_equal options.object_id, server.options.object_id
59
+ end
60
+
61
+ def test_new_with_options_should_keep_options
62
+ server = Minestrone::ServerDefinition.new("www.minestrone.test", :ssh_options => { :forward_agent => true })
63
+ assert_equal({ :forward_agent => true }, server.options[:ssh_options])
64
+ end
65
+
66
+ def test_default_user_should_try_to_guess_username
67
+ ENV.stubs(:[]).returns(nil)
68
+ assert_equal "not-specified", Minestrone::ServerDefinition.default_user
69
+
70
+ ENV.stubs(:[]).returns(nil)
71
+ ENV.stubs(:[]).with("USERNAME").returns("ryan")
72
+ assert_equal "ryan", Minestrone::ServerDefinition.default_user
73
+
74
+ ENV.stubs(:[]).returns(nil)
75
+ ENV.stubs(:[]).with("USER").returns("jamis")
76
+ assert_equal "jamis", Minestrone::ServerDefinition.default_user
77
+ end
78
+
79
+ def test_comparison_should_match_when_host_user_port_are_same
80
+ s1 = server("jamis@www.minestrone.test:8080")
81
+ s2 = server("www.minestrone.test", :user => "jamis", :port => 8080)
82
+ assert_equal s1, s2
83
+ assert_equal s1.hash, s2.hash
84
+ assert s1.eql?(s2)
85
+ end
86
+
87
+ def test_servers_should_be_comparable
88
+ s1 = server("jamis@www.minestrone.test:8080")
89
+ s2 = server("www.alphabet.test:1234")
90
+ s3 = server("jamis@www.minestrone.test:8075")
91
+ s4 = server("billy@www.minestrone.test:8080")
92
+
93
+ assert s2 < s1
94
+ assert s3 < s1
95
+ assert s4 < s1
96
+ assert s2 < s3
97
+ assert s2 < s4
98
+ assert s3 < s4
99
+ end
100
+
101
+ def test_comparison_should_not_match_when_any_of_host_user_port_differ
102
+ s1 = server("jamis@www.minestrone.test:8080")
103
+ s2 = server("bob@www.minestrone.test:8080")
104
+ s3 = server("jamis@www.minestrone.test:8081")
105
+ s4 = server("jamis@app.minestrone.test:8080")
106
+ assert_not_equal s1, s2
107
+ assert_not_equal s1, s3
108
+ assert_not_equal s1, s4
109
+ assert_not_equal s2, s3
110
+ assert_not_equal s2, s4
111
+ assert_not_equal s3, s4
112
+ end
113
+
114
+ def test_to_s
115
+ assert_equal "www.minestrone.test", server("www.minestrone.test").to_s
116
+ assert_equal "www.minestrone.test", server("www.minestrone.test:22").to_s
117
+ assert_equal "www.minestrone.test:1234", server("www.minestrone.test:1234").to_s
118
+ assert_equal "jamis@www.minestrone.test", server("jamis@www.minestrone.test").to_s
119
+ assert_equal "jamis@www.minestrone.test:1234", server("jamis@www.minestrone.test:1234").to_s
120
+ end
121
+ end
data/test/ssh_test.rb ADDED
@@ -0,0 +1,99 @@
1
+ require "utils"
2
+ require 'minestrone/ssh'
3
+
4
+ class SSHTest < Test::Unit::TestCase
5
+ def setup
6
+ Minestrone::ServerDefinition.stubs(:default_user).returns("default-user")
7
+ @options = { :auth_methods => %w(publickey),
8
+ :config => false }
9
+ @server = server("minestrone")
10
+ Net::SSH.stubs(:configuration_for).returns({})
11
+ end
12
+
13
+ def test_connect_with_bare_server_without_options_or_config_with_public_key_succeeding_should_only_loop_once
14
+ Net::SSH.expects(:start).with(@server.host, "default-user", @options).returns(success = Object.new)
15
+ assert_equal success, Minestrone::SSH.connect(@server)
16
+ end
17
+
18
+ def test_connect_with_bare_server_without_options_with_public_key_failing_should_raise_error
19
+ Net::SSH.expects(:start).with(@server.host, "default-user", @options).raises(Net::SSH::AuthenticationFailed)
20
+ assert_raises(Net::SSH::AuthenticationFailed) do
21
+ Minestrone::SSH.connect(@server, :password => "f4b13n")
22
+ end
23
+ end
24
+
25
+ def test_connect_with_bare_server_and_user_via_public_key_should_pass_user_to_net_ssh
26
+ Net::SSH.expects(:start).with(@server.host, "jamis", @options).returns(success = Object.new)
27
+ assert_equal success, Minestrone::SSH.connect(@server, :user => "jamis")
28
+ end
29
+
30
+ def test_connect_with_bare_server_with_explicit_port_should_pass_port_to_net_ssh
31
+ Net::SSH.expects(:start).with(@server.host, "default-user", @options.merge(:port => 1234)).returns(success = Object.new)
32
+ assert_equal success, Minestrone::SSH.connect(@server, :port => 1234)
33
+ end
34
+
35
+ def test_connect_with_server_with_user_should_pass_user_to_net_ssh
36
+ server = server("jamis@minestrone")
37
+ Net::SSH.expects(:start).with(server.host, "jamis", @options).returns(success = Object.new)
38
+ assert_equal success, Minestrone::SSH.connect(server)
39
+ end
40
+
41
+ def test_connect_with_server_with_port_should_pass_port_to_net_ssh
42
+ server = server("minestrone:1235")
43
+ Net::SSH.expects(:start).with(server.host, "default-user", @options.merge(:port => 1235)).returns(success = Object.new)
44
+ assert_equal success, Minestrone::SSH.connect(server)
45
+ end
46
+
47
+ def test_connect_with_server_with_user_and_port_should_pass_user_and_port_to_net_ssh
48
+ server = server("jamis@minestrone:1235")
49
+ Net::SSH.expects(:start).with(server.host, "jamis", @options.merge(:port => 1235)).returns(success = Object.new)
50
+ assert_equal success, Minestrone::SSH.connect(server)
51
+ end
52
+
53
+ def test_connect_with_server_with_other_ssh_options_should_pass_ssh_options_to_net_ssh
54
+ server = server("jamis@minestrone:1235", :ssh_options => { :keys => %w(some_valid_key), :hmac => 'none' })
55
+ Net::SSH.expects(:start).with(server.host, "jamis", @options.merge(:port => 1235, :keys => %w(some_valid_key), :hmac => 'none' )).returns(success = Object.new)
56
+ assert_equal success, Minestrone::SSH.connect(server)
57
+ end
58
+
59
+ def test_connect_with_ssh_options_should_use_ssh_options
60
+ ssh_options = { :username => "JamisMan", :port => 8125, :config => false }
61
+ Net::SSH.expects(:start).with(@server.host, "JamisMan", @options.merge(:port => 8125, :config => false)).returns(success = Object.new)
62
+ assert_equal success, Minestrone::SSH.connect(@server, {:ssh_options => ssh_options})
63
+ end
64
+
65
+ def test_connect_with_options_and_ssh_options_should_see_options_override_ssh_options
66
+ ssh_options = { :username => "JamisMan", :port => 8125, :forward_agent => true }
67
+ Net::SSH.expects(:start).with(@server.host, "jamis", @options.merge(:port => 1235, :forward_agent => true)).returns(success = Object.new)
68
+ assert_equal success, Minestrone::SSH.connect(@server, :ssh_options => ssh_options, :user => "jamis", :port => 1235)
69
+ end
70
+
71
+ def test_connect_with_verbose_option_should_set_verbose_option_on_ssh
72
+ Net::SSH.expects(:start).with(@server.host, "default-user", @options).returns(success = Object.new)
73
+ assert_equal success, Minestrone::SSH.connect(@server, :verbose => 0)
74
+ Net::SSH.expects(:start).with(@server.host, "default-user", @options.merge(:verbose => :debug)).returns(success = Object.new)
75
+ assert_equal success, Minestrone::SSH.connect(@server, :verbose => 1)
76
+ Net::SSH.expects(:start).with(@server.host, "default-user", @options.merge(:verbose => :debug)).returns(success = Object.new)
77
+ assert_equal success, Minestrone::SSH.connect(@server, :verbose => 2)
78
+ end
79
+
80
+ def test_connect_with_ssh_options_should_see_server_options_override_ssh_options
81
+ ssh_options = { :username => "JamisMan", :port => 8125, :forward_agent => true }
82
+ server = server("jamis@minestrone:1235")
83
+ Net::SSH.expects(:start).with(server.host, "jamis", @options.merge(:port => 1235, :forward_agent => true, :config => false)).returns(success = Object.new)
84
+ assert_equal success, Minestrone::SSH.connect(server, {:ssh_options => ssh_options})
85
+ end
86
+
87
+ def test_connect_should_add_xserver_accessor_to_connection
88
+ Net::SSH.expects(:start).with(@server.host, "default-user", @options).returns(success = Object.new)
89
+ assert_equal success, Minestrone::SSH.connect(@server)
90
+ assert success.respond_to?(:xserver)
91
+ assert success.respond_to?(:xserver)
92
+ assert_equal success.xserver, @server
93
+ end
94
+
95
+ def test_connect_should_ignore_custom_auth_methods
96
+ Net::SSH.expects(:start).with(@server.host, "default-user", @options).returns(success = Object.new)
97
+ assert_equal success, Minestrone::SSH.connect(@server, :ssh_options => { :auth_methods => %w(password keyboard-interactive) })
98
+ end
99
+ end