rory-deploy 1.8.4.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 (44) hide show
  1. checksums.yaml +7 -0
  2. data/.gitignore +10 -0
  3. data/CONTRIBUTORS.md +77 -0
  4. data/Gemfile +5 -0
  5. data/LICENSE +19 -0
  6. data/README.md +574 -0
  7. data/Rakefile +15 -0
  8. data/bin/rory +80 -0
  9. data/bin/rory-gen-config +79 -0
  10. data/lib/capistrano_dsl.rb +91 -0
  11. data/lib/centurion.rb +9 -0
  12. data/lib/centurion/deploy.rb +139 -0
  13. data/lib/centurion/deploy_dsl.rb +180 -0
  14. data/lib/centurion/docker_registry.rb +89 -0
  15. data/lib/centurion/docker_server.rb +79 -0
  16. data/lib/centurion/docker_server_group.rb +33 -0
  17. data/lib/centurion/docker_via_api.rb +166 -0
  18. data/lib/centurion/docker_via_cli.rb +81 -0
  19. data/lib/centurion/dogestry.rb +92 -0
  20. data/lib/centurion/logging.rb +28 -0
  21. data/lib/centurion/service.rb +218 -0
  22. data/lib/centurion/shell.rb +46 -0
  23. data/lib/centurion/version.rb +3 -0
  24. data/lib/core_ext/numeric_bytes.rb +94 -0
  25. data/lib/tasks/centurion.rake +15 -0
  26. data/lib/tasks/deploy.rake +250 -0
  27. data/lib/tasks/info.rake +24 -0
  28. data/lib/tasks/list.rake +56 -0
  29. data/rory-deploy.gemspec +33 -0
  30. data/spec/capistrano_dsl_spec.rb +67 -0
  31. data/spec/deploy_dsl_spec.rb +184 -0
  32. data/spec/deploy_spec.rb +212 -0
  33. data/spec/docker_registry_spec.rb +105 -0
  34. data/spec/docker_server_group_spec.rb +31 -0
  35. data/spec/docker_server_spec.rb +92 -0
  36. data/spec/docker_via_api_spec.rb +246 -0
  37. data/spec/docker_via_cli_spec.rb +91 -0
  38. data/spec/dogestry_spec.rb +73 -0
  39. data/spec/logging_spec.rb +41 -0
  40. data/spec/service_spec.rb +288 -0
  41. data/spec/spec_helper.rb +7 -0
  42. data/spec/support/matchers/capistrano_dsl_matchers.rb +13 -0
  43. data/spec/support/matchers/exit_code_matches.rb +38 -0
  44. metadata +214 -0
@@ -0,0 +1,73 @@
1
+ require 'spec_helper'
2
+ require 'centurion/dogestry'
3
+
4
+ describe Centurion::Dogestry do
5
+ let(:dogestry_options) {
6
+ {
7
+ aws_access_key_id: "abc",
8
+ aws_secret_key: "xyz",
9
+ s3_bucket: "s3-registry-test"
10
+ }
11
+ }
12
+ let(:registry) { Centurion::Dogestry.new(dogestry_options) }
13
+ let(:repo) { 'google/golang' }
14
+ let(:pull_hosts) {
15
+ [
16
+ 'tcp://host-1:2375',
17
+ 'tcp://host-2:2375'
18
+ ]
19
+ }
20
+ let(:flags) { "-pullhosts #{pull_hosts.join(',')}"}
21
+
22
+ describe '#aws_access_key_id' do
23
+ it 'returns correct value' do
24
+ expect(registry.aws_access_key_id).to eq(dogestry_options[:aws_access_key_id])
25
+ end
26
+ end
27
+
28
+ describe '#aws_secret_key' do
29
+ it 'returns correct value' do
30
+ expect(registry.aws_secret_key).to eq(dogestry_options[:aws_secret_key])
31
+ end
32
+ end
33
+
34
+ describe '#s3_bucket' do
35
+ it 'returns correct value' do
36
+ expect(registry.s3_bucket).to eq(dogestry_options[:s3_bucket])
37
+ end
38
+ end
39
+
40
+ describe '#s3_region' do
41
+ it 'returns correct default value' do
42
+ expect(registry.s3_region).to eq("us-east-1")
43
+ end
44
+ end
45
+
46
+ describe '#s3_url' do
47
+ it 'returns correct value' do
48
+ expect(registry.s3_url).to eq("s3://#{registry.s3_bucket}/?region=#{registry.s3_region}")
49
+ end
50
+ end
51
+
52
+ describe '#exec_command' do
53
+ it 'returns correct value' do
54
+ expect(registry.exec_command('pull', repo)).to start_with('dogestry')
55
+ end
56
+ end
57
+
58
+ describe '#pull' do
59
+ it 'returns correct value' do
60
+ if registry.which('dogestry')
61
+ expect(registry).to receive(:echo).with("dogestry #{flags} pull #{registry.s3_url} #{repo}")
62
+ registry.pull(repo, pull_hosts)
63
+ end
64
+ end
65
+ end
66
+
67
+ describe '#which' do
68
+ it 'finds dogestry command line' do
69
+ allow(File).to receive(:executable?).and_return(true)
70
+ expect(registry.which('dogestry')).to_not be_nil
71
+ end
72
+ end
73
+ end
@@ -0,0 +1,41 @@
1
+ require 'spec_helper'
2
+ require 'centurion/logging'
3
+
4
+ class TestLogging
5
+ extend Centurion::Logging
6
+ def self.logger
7
+ log
8
+ end
9
+ end
10
+
11
+ describe Centurion::Logging do
12
+ let(:message) { %w{ something something_else } }
13
+
14
+ context '#info' do
15
+ it 'passes through to Logger' do
16
+ expect(TestLogging.logger).to receive(:info).with(message.join(' '))
17
+ TestLogging.info(*message)
18
+ end
19
+ end
20
+
21
+ context '#warn' do
22
+ it 'passes through to Logger' do
23
+ expect(TestLogging.logger).to receive(:warn).with(message.join(' '))
24
+ TestLogging.warn(*message)
25
+ end
26
+ end
27
+
28
+ context '#debug' do
29
+ it 'passes through to Logger' do
30
+ expect(TestLogging.logger).to receive(:debug).with(message.join(' '))
31
+ TestLogging.debug(*message)
32
+ end
33
+ end
34
+
35
+ context '#error' do
36
+ it 'passes through to Logger' do
37
+ expect(TestLogging.logger).to receive(:error).with(message.join(' '))
38
+ TestLogging.error(*message)
39
+ end
40
+ end
41
+ end
@@ -0,0 +1,288 @@
1
+ require 'spec_helper'
2
+ require 'centurion/service'
3
+
4
+ describe Centurion::Service do
5
+
6
+ let(:service) { Centurion::Service.new(:redis) }
7
+ let(:hostname) { 'shakespeare' }
8
+ let(:image) { 'redis' }
9
+
10
+ it 'creates a service from the environment' do
11
+ extend Capistrano::DSL
12
+ set_current_environment(:test)
13
+ set(:name, 'mycontainer')
14
+ set(:image, image)
15
+ set(:hostname, hostname)
16
+ set(:binds, [ Centurion::Service::Volume.new('/foo', '/foo/bar') ])
17
+ set(:port_bindings, [ Centurion::Service::PortBinding.new(12340, 80, 'tcp') ])
18
+
19
+ svc = Centurion::Service.from_env
20
+ expect(svc.name).to eq('mycontainer')
21
+ expect(svc.image).to eq(image)
22
+ expect(svc.dns).to be_nil
23
+ expect(svc.volumes.size).to eq(1)
24
+ expect(svc.volumes.first.host_volume).to eq('/foo')
25
+ expect(svc.port_bindings.size).to eq(1)
26
+ expect(svc.port_bindings.first.container_port).to eq(80)
27
+ end
28
+
29
+ it 'starts with a command' do
30
+ service.command = ['redis-server']
31
+ expect(service.command).to eq(['redis-server'])
32
+ end
33
+
34
+ it 'has memory bounds' do
35
+ service.memory = 1024
36
+ expect(service.memory).to eq(1024)
37
+ end
38
+
39
+ it 'rejects non-numeric memory bounds' do
40
+ expect(-> { service.memory = 'all' }).to raise_error
41
+ end
42
+
43
+ it 'has cpu shares bounds' do
44
+ service.cpu_shares = 512
45
+ expect(service.cpu_shares).to eq(512)
46
+ end
47
+
48
+ it 'rejects non-numeric cpu shares' do
49
+ expect(-> { service.cpu_shares = 'all' }).to raise_error
50
+ end
51
+
52
+ it 'has a custom dns association' do
53
+ service.dns = 'redis.example.com'
54
+ expect(service.dns).to eq('redis.example.com')
55
+ end
56
+
57
+ it 'boots from a docker image' do
58
+ service.image = 'registry.hub.docker.com/library/redis'
59
+ expect(service.image).to eq('registry.hub.docker.com/library/redis')
60
+ end
61
+
62
+ it 'has env vars' do
63
+ service.add_env_vars(SLAVE_OF: '127.0.0.1')
64
+ service.add_env_vars(USE_AOF: '1')
65
+ expect(service.env_vars).to eq(SLAVE_OF: '127.0.0.1', USE_AOF: '1')
66
+ end
67
+
68
+ it 'has volume bindings' do
69
+ service.add_volume('/volumes/redis/data', '/data')
70
+ service.add_volume('/volumes/redis/config', '/config')
71
+ expect(service.volumes).to eq([Centurion::Service::Volume.new('/volumes/redis/data', '/data'),
72
+ Centurion::Service::Volume.new('/volumes/redis/config', '/config')])
73
+ end
74
+
75
+ it 'has port mappings' do
76
+ service.add_port_bindings(8000, 6379, 'tcp', '127.0.0.1')
77
+ service.add_port_bindings(18000, 16379, 'tcp', '127.0.0.1')
78
+ expect(service.port_bindings).to eq([Centurion::Service::PortBinding.new(8000, 6379, 'tcp', '127.0.0.1'),
79
+ Centurion::Service::PortBinding.new(18000, 16379, 'tcp', '127.0.0.1')])
80
+ end
81
+
82
+ it 'builds a list of public ports for the service' do
83
+ service.add_port_bindings(8000, 6379, 'tcp', '127.0.0.1')
84
+ service.add_port_bindings(18000, 16379, 'tcp', '127.0.0.1')
85
+ expect(service.public_ports).to eq([8000, 18000])
86
+ end
87
+
88
+ context 'building a container configuration' do
89
+ service = Centurion::Service.new(:redis)
90
+ service.image = 'http://registry.hub.docker.com/library/redis'
91
+ service.command = ['redis-server', '--appendonly', 'yes']
92
+ service.memory = 1024
93
+ service.cpu_shares = 512
94
+ service.add_env_vars(SLAVE_OF: '127.0.0.2')
95
+ service.add_port_bindings(8000, 6379, 'tcp', '10.0.0.1')
96
+ service.network_mode = 'host'
97
+ service.add_volume('/volumes/redis.8000', '/data')
98
+
99
+ it 'builds a valid docker container configuration' do
100
+ expect(service.build_config('example.com')).to eq({
101
+ 'Image' => 'http://registry.hub.docker.com/library/redis',
102
+ 'Cmd' => ['redis-server', '--appendonly', 'yes'],
103
+ 'Memory' => 1024,
104
+ 'CpuShares' => 512,
105
+ 'ExposedPorts' => {'6379/tcp' => {}},
106
+ 'Env' => ['SLAVE_OF=127.0.0.2'],
107
+ 'Volumes' => {'/data' => {}},
108
+
109
+ # TODO: Ignoring this for now because Docker 1.6
110
+ # https://github.com/newrelic/centurion/issues/117
111
+ # 'VolumesFrom' => 'parent'
112
+ })
113
+ end
114
+
115
+ it 'overrides the default hostname when passed a block' do
116
+ expect(service.build_config('example.com') { |s| "host.#{s}" }).to eq({
117
+ 'Image' => 'http://registry.hub.docker.com/library/redis',
118
+ 'Hostname' => 'host.example.com',
119
+ 'Cmd' => ['redis-server', '--appendonly', 'yes'],
120
+ 'Memory' => 1024,
121
+ 'CpuShares' => 512,
122
+ 'ExposedPorts' => {'6379/tcp' => {}},
123
+ 'Env' => ['SLAVE_OF=127.0.0.2'],
124
+ 'Volumes' => {'/data' => {}},
125
+
126
+ # TODO: Ignoring this for now because Docker 1.6
127
+ # https://github.com/newrelic/centurion/issues/117
128
+ # 'VolumesFrom' => 'parent'
129
+ })
130
+ end
131
+ end
132
+
133
+ it 'interpolates hostname into env variables' do
134
+ allow(Socket).to receive(:getaddrinfo).and_return([["AF_INET", 0, "93.184.216.34", "93.184.216.34", 2, 1, 6]])
135
+ service = Centurion::Service.new(:redis)
136
+ service.add_env_vars(HOST: '%DOCKER_HOSTNAME%')
137
+
138
+ expect(service.build_config('example.com')['Env']).to eq(['HOST=example.com'])
139
+ end
140
+
141
+ it 'interpolates host ip into env variables' do
142
+ allow(Socket).to receive(:getaddrinfo).and_return([["AF_INET", 0, "93.184.216.34", "93.184.216.34", 2, 1, 6]])
143
+ service = Centurion::Service.new(:redis)
144
+ service.add_env_vars(HOST: '%DOCKER_HOST_IP%')
145
+
146
+ expect(service.build_config('example.com')['Env']).to eq(['HOST=93.184.216.34'])
147
+ end
148
+
149
+ it 'does not blow up on non-string values' do
150
+ expect { service.add_env_vars(SOMETHING: true) }.not_to raise_error
151
+ end
152
+
153
+ it 'builds a valid docker host configuration' do
154
+ service = Centurion::Service.new(:redis)
155
+ service.dns = 'example.com'
156
+ service.add_port_bindings(8000, 6379)
157
+ service.cap_adds = ['IPC_BIND', 'NET_RAW']
158
+ service.cap_drops = ['DAC_OVERRIDE']
159
+ service.add_volume('/volumes/redis.8000', '/data')
160
+
161
+ expect(service.build_host_config(Centurion::Service::RestartPolicy.new('on-failure', 10))).to eq({
162
+ 'Binds' => ['/volumes/redis.8000:/data'],
163
+ 'CapAdd' => ['IPC_BIND', 'NET_RAW'],
164
+ 'CapDrop' => ['DAC_OVERRIDE'],
165
+ 'PortBindings' => {
166
+ '6379/tcp' => [{'HostPort' => '8000'}]
167
+ },
168
+ 'NetworkMode' => 'bridge',
169
+ 'Dns' => 'example.com',
170
+ 'RestartPolicy' => {
171
+ 'Name' => 'on-failure',
172
+ 'MaximumRetryCount' => 10
173
+ }
174
+ })
175
+ end
176
+
177
+ it 'ignores garbage restart policy' do
178
+ service = Centurion::Service.new(:redis)
179
+
180
+ expect(service.build_host_config(Centurion::Service::RestartPolicy.new('garbage'))).to eq({
181
+ 'Binds' => [],
182
+ 'CapAdd' => [],
183
+ 'CapDrop' => [],
184
+ 'PortBindings' => {},
185
+ 'NetworkMode' => 'bridge',
186
+ 'RestartPolicy' => {
187
+ 'Name' => 'on-failure',
188
+ 'MaximumRetryCount' => 10
189
+ }
190
+ })
191
+ end
192
+
193
+ it 'accepts "no" restart policy' do
194
+ service = Centurion::Service.new(:redis)
195
+
196
+ expect(service.build_host_config(Centurion::Service::RestartPolicy.new('no'))).to eq({
197
+ 'Binds' => [],
198
+ 'CapAdd' => [],
199
+ 'CapDrop' => [],
200
+ 'PortBindings' => {},
201
+ 'NetworkMode' => 'bridge',
202
+ 'RestartPolicy' => {
203
+ 'Name' => 'no',
204
+ }
205
+ })
206
+ end
207
+
208
+ it 'accepts "always" restart policy' do
209
+ service = Centurion::Service.new(:redis)
210
+
211
+ expect(service.build_host_config(Centurion::Service::RestartPolicy.new('always'))).to eq({
212
+ 'Binds' => [],
213
+ 'CapAdd' => [],
214
+ 'CapDrop' => [],
215
+ 'PortBindings' => {},
216
+ 'NetworkMode' => 'bridge',
217
+ 'RestartPolicy' => {
218
+ 'Name' => 'always',
219
+ }
220
+ })
221
+ end
222
+
223
+ it 'accepts "on-failure" restart policy with retry count' do
224
+ service = Centurion::Service.new(:redis)
225
+
226
+ expect(service.build_host_config(Centurion::Service::RestartPolicy.new('on-failure', 50))).to eq({
227
+ 'Binds' => [],
228
+ 'CapAdd' => [],
229
+ 'CapDrop' => [],
230
+ 'NetworkMode' => 'bridge',
231
+ 'PortBindings' => {},
232
+ 'RestartPolicy' => {
233
+ 'Name' => 'on-failure',
234
+ 'MaximumRetryCount' => 50
235
+ }
236
+ })
237
+ end
238
+
239
+ it 'builds docker configuration for volume binds' do
240
+ service.add_volume('/volumes/redis/data', '/data')
241
+ expect(service.volume_binds_config).to eq(['/volumes/redis/data:/data'])
242
+ end
243
+
244
+ it 'builds docker configuration for port bindings' do
245
+ service.add_port_bindings(8000, 6379, 'tcp', '127.0.0.1')
246
+ expect(service.port_bindings_config).to eq({
247
+ '6379/tcp' => [{'HostPort' => '8000', 'HostIp' => '127.0.0.1'}]
248
+ })
249
+ end
250
+
251
+ it 'builds docker configuration for port bindings without host ip' do
252
+ service.add_port_bindings(8000, 6379, 'tcp')
253
+ expect(service.port_bindings_config).to eq({
254
+ '6379/tcp' => [{'HostPort' => '8000'}]
255
+ })
256
+ end
257
+
258
+ it 'builds docker configuration for container-linked networking' do
259
+ service.network_mode = 'container:a2e8937b'
260
+ expect(service.build_host_config(Centurion::Service::RestartPolicy.new('on-failure', 50))).to eq({
261
+ 'Binds' => [],
262
+ 'CapAdd' => [],
263
+ 'CapDrop' => [],
264
+ 'NetworkMode' => 'container:a2e8937b',
265
+ 'PortBindings' => {},
266
+ 'RestartPolicy' => {
267
+ 'Name' => 'on-failure',
268
+ 'MaximumRetryCount' => 50
269
+ }
270
+ })
271
+ end
272
+
273
+ it 'builds docker configuration for host networking' do
274
+ service.network_mode = 'host'
275
+ expect(service.build_host_config(Centurion::Service::RestartPolicy.new('on-failure', 50))).to eq({
276
+ 'Binds' => [],
277
+ 'CapAdd' => [],
278
+ 'CapDrop' => [],
279
+ 'NetworkMode' => 'host',
280
+ 'PortBindings' => {},
281
+ 'RestartPolicy' => {
282
+ 'Name' => 'on-failure',
283
+ 'MaximumRetryCount' => 50
284
+ }
285
+ })
286
+ end
287
+
288
+ end
@@ -0,0 +1,7 @@
1
+ require 'simplecov'
2
+ SimpleCov.start do
3
+ add_filter '/spec'
4
+ end
5
+
6
+ $: << File.expand_path(File.join(File.dirname(__FILE__), '..', 'lib'))
7
+ Dir[File.dirname(__FILE__) + '/support/**/*.rb'].each {|f| require f}
@@ -0,0 +1,13 @@
1
+ RSpec::Matchers.define :have_key_and_value do |expected_key, expected_value|
2
+ match do |actual|
3
+ actual.env[actual.current_environment].has_key?(expected_key.to_sym) && (actual.fetch(expected_key.to_sym) == expected_value)
4
+ end
5
+
6
+ failure_message do |actual|
7
+ "expected that #{actual.env[actual.current_environment].keys.inspect} would include #{expected_key.inspect} with value #{expected_value.inspect}"
8
+ end
9
+
10
+ failure_message_when_negated do |actual|
11
+ "expected that #{actual.env[actual.current_environment].keys.join(', ')} would not include #{expected_key.inspect} with value #{expected_value.inspect}"
12
+ end
13
+ end
@@ -0,0 +1,38 @@
1
+ # https://gist.github.com/mmasashi/58bd7e2668836a387856
2
+ RSpec::Matchers.define :terminate do |code|
3
+ actual = nil
4
+
5
+ def supports_block_expectations?
6
+ true
7
+ end
8
+
9
+ match do |block|
10
+ begin
11
+ block.call
12
+ rescue SystemExit => e
13
+ actual = e.status
14
+ end
15
+ actual and actual == status_code
16
+ end
17
+
18
+ chain :with_code do |status_code|
19
+ @status_code = status_code
20
+ end
21
+
22
+ failure_message do |block|
23
+ "expected block to call exit(#{status_code}) but exit" +
24
+ (actual.nil? ? " not called" : "(#{actual}) was called")
25
+ end
26
+
27
+ failure_message_when_negated do |block|
28
+ "expected block not to call exit(#{status_code})"
29
+ end
30
+
31
+ description do
32
+ "expect block to call exit(#{status_code})"
33
+ end
34
+
35
+ def status_code
36
+ @status_code ||= 0
37
+ end
38
+ end