specinfra-backend-docker_lxc 0.1.0

Sign up to get free protection for your applications and to get access to all the features.
checksums.yaml ADDED
@@ -0,0 +1,7 @@
1
+ ---
2
+ SHA1:
3
+ metadata.gz: a1c14192777898a3b3aa32dc8b8f9da1c63b7b94
4
+ data.tar.gz: 77ad983a32d05ab822792931e4dfd46292c2643b
5
+ SHA512:
6
+ metadata.gz: b758cc718f35faee74d97e85fa018dedbf90809232e3ceac19061f292490c78640d362d40021e5e466a9ff691a23ad5299fc2484ce8b8190ec41046955a478f5
7
+ data.tar.gz: 8d6295e0d1aae74f40bff56d3f0e5ef6a63ff502cf6d643d6a0fffb7421db57b2c4dada2e8d4a25ae957070a1e58803f3263bb9f3a9a715000eb45e6b60b1420
data/.yardopts ADDED
@@ -0,0 +1,7 @@
1
+ --markup markdown
2
+ -
3
+ CHANGELOG.md
4
+ CONTRIBUTING.md
5
+ README.md
6
+ TESTING.md
7
+ TODO.md
data/CHANGELOG.md ADDED
@@ -0,0 +1,7 @@
1
+ # CHANGELOG for specinfra-backend-docker_lxc
2
+
3
+ This file is used to list changes made in each version of `specinfra-backend-docker_lxc`.
4
+
5
+ ## 0.1.0 (2015-11-14)
6
+
7
+ * Initial release of `specinfra-backend-docker_lxc`
data/CONTRIBUTING.md ADDED
@@ -0,0 +1,13 @@
1
+ # Contributing
2
+
3
+ 1. [Fork the repository on GitHub](https://help.github.com/articles/fork-a-repo).
4
+ 2. Create a named feature branch (`$ git checkout -b my-new-feature`).
5
+ 3. Write tests for your change (if applicable).
6
+ 4. Write your change.
7
+ 5. Add documentation to your change.
8
+ 6. [Run the tests](https://github.com/zuazo/specinfra-backend-docker_lxc/blob/master/TESTING.md), ensuring they all pass (`$ bundle exec rake`). Try as much as possible **not to reduce coverage**.
9
+ 7. Commit your change (`$ git commit -am 'Add some feature'`).
10
+ 8. Push to the branch (`$ git push origin my-new-feature`).
11
+ 9. [Submit a Pull Request using GitHub](https://help.github.com/articles/creating-a-pull-request).
12
+
13
+ You can see the [TODO.md](https://github.com/zuazo/specinfra-backend-docker_lxc/blob/master/TODO.md) file if you're looking for inspiration.
data/LICENSE ADDED
@@ -0,0 +1,190 @@
1
+ Apache License
2
+ Version 2.0, January 2004
3
+ http://www.apache.org/licenses/
4
+
5
+ TERMS AND CONDITIONS FOR USE, REPRODUCTION, AND DISTRIBUTION
6
+
7
+ 1. Definitions.
8
+
9
+ "License" shall mean the terms and conditions for use, reproduction,
10
+ and distribution as defined by Sections 1 through 9 of this document.
11
+
12
+ "Licensor" shall mean the copyright owner or entity authorized by
13
+ the copyright owner that is granting the License.
14
+
15
+ "Legal Entity" shall mean the union of the acting entity and all
16
+ other entities that control, are controlled by, or are under common
17
+ control with that entity. For the purposes of this definition,
18
+ "control" means (i) the power, direct or indirect, to cause the
19
+ direction or management of such entity, whether by contract or
20
+ otherwise, or (ii) ownership of fifty percent (50%) or more of the
21
+ outstanding shares, or (iii) beneficial ownership of such entity.
22
+
23
+ "You" (or "Your") shall mean an individual or Legal Entity
24
+ exercising permissions granted by this License.
25
+
26
+ "Source" form shall mean the preferred form for making modifications,
27
+ including but not limited to software source code, documentation
28
+ source, and configuration files.
29
+
30
+ "Object" form shall mean any form resulting from mechanical
31
+ transformation or translation of a Source form, including but
32
+ not limited to compiled object code, generated documentation,
33
+ and conversions to other media types.
34
+
35
+ "Work" shall mean the work of authorship, whether in Source or
36
+ Object form, made available under the License, as indicated by a
37
+ copyright notice that is included in or attached to the work
38
+ (an example is provided in the Appendix below).
39
+
40
+ "Derivative Works" shall mean any work, whether in Source or Object
41
+ form, that is based on (or derived from) the Work and for which the
42
+ editorial revisions, annotations, elaborations, or other modifications
43
+ represent, as a whole, an original work of authorship. For the purposes
44
+ of this License, Derivative Works shall not include works that remain
45
+ separable from, or merely link (or bind by name) to the interfaces of,
46
+ the Work and Derivative Works thereof.
47
+
48
+ "Contribution" shall mean any work of authorship, including
49
+ the original version of the Work and any modifications or additions
50
+ to that Work or Derivative Works thereof, that is intentionally
51
+ submitted to Licensor for inclusion in the Work by the copyright owner
52
+ or by an individual or Legal Entity authorized to submit on behalf of
53
+ the copyright owner. For the purposes of this definition, "submitted"
54
+ means any form of electronic, verbal, or written communication sent
55
+ to the Licensor or its representatives, including but not limited to
56
+ communication on electronic mailing lists, source code control systems,
57
+ and issue tracking systems that are managed by, or on behalf of, the
58
+ Licensor for the purpose of discussing and improving the Work, but
59
+ excluding communication that is conspicuously marked or otherwise
60
+ designated in writing by the copyright owner as "Not a Contribution."
61
+
62
+ "Contributor" shall mean Licensor and any individual or Legal Entity
63
+ on behalf of whom a Contribution has been received by Licensor and
64
+ subsequently incorporated within the Work.
65
+
66
+ 2. Grant of Copyright License. Subject to the terms and conditions of
67
+ this License, each Contributor hereby grants to You a perpetual,
68
+ worldwide, non-exclusive, no-charge, royalty-free, irrevocable
69
+ copyright license to reproduce, prepare Derivative Works of,
70
+ publicly display, publicly perform, sublicense, and distribute the
71
+ Work and such Derivative Works in Source or Object form.
72
+
73
+ 3. Grant of Patent License. Subject to the terms and conditions of
74
+ this License, each Contributor hereby grants to You a perpetual,
75
+ worldwide, non-exclusive, no-charge, royalty-free, irrevocable
76
+ (except as stated in this section) patent license to make, have made,
77
+ use, offer to sell, sell, import, and otherwise transfer the Work,
78
+ where such license applies only to those patent claims licensable
79
+ by such Contributor that are necessarily infringed by their
80
+ Contribution(s) alone or by combination of their Contribution(s)
81
+ with the Work to which such Contribution(s) was submitted. If You
82
+ institute patent litigation against any entity (including a
83
+ cross-claim or counterclaim in a lawsuit) alleging that the Work
84
+ or a Contribution incorporated within the Work constitutes direct
85
+ or contributory patent infringement, then any patent licenses
86
+ granted to You under this License for that Work shall terminate
87
+ as of the date such litigation is filed.
88
+
89
+ 4. Redistribution. You may reproduce and distribute copies of the
90
+ Work or Derivative Works thereof in any medium, with or without
91
+ modifications, and in Source or Object form, provided that You
92
+ meet the following conditions:
93
+
94
+ (a) You must give any other recipients of the Work or
95
+ Derivative Works a copy of this License; and
96
+
97
+ (b) You must cause any modified files to carry prominent notices
98
+ stating that You changed the files; and
99
+
100
+ (c) You must retain, in the Source form of any Derivative Works
101
+ that You distribute, all copyright, patent, trademark, and
102
+ attribution notices from the Source form of the Work,
103
+ excluding those notices that do not pertain to any part of
104
+ the Derivative Works; and
105
+
106
+ (d) If the Work includes a "NOTICE" text file as part of its
107
+ distribution, then any Derivative Works that You distribute must
108
+ include a readable copy of the attribution notices contained
109
+ within such NOTICE file, excluding those notices that do not
110
+ pertain to any part of the Derivative Works, in at least one
111
+ of the following places: within a NOTICE text file distributed
112
+ as part of the Derivative Works; within the Source form or
113
+ documentation, if provided along with the Derivative Works; or,
114
+ within a display generated by the Derivative Works, if and
115
+ wherever such third-party notices normally appear. The contents
116
+ of the NOTICE file are for informational purposes only and
117
+ do not modify the License. You may add Your own attribution
118
+ notices within Derivative Works that You distribute, alongside
119
+ or as an addendum to the NOTICE text from the Work, provided
120
+ that such additional attribution notices cannot be construed
121
+ as modifying the License.
122
+
123
+ You may add Your own copyright statement to Your modifications and
124
+ may provide additional or different license terms and conditions
125
+ for use, reproduction, or distribution of Your modifications, or
126
+ for any such Derivative Works as a whole, provided Your use,
127
+ reproduction, and distribution of the Work otherwise complies with
128
+ the conditions stated in this License.
129
+
130
+ 5. Submission of Contributions. Unless You explicitly state otherwise,
131
+ any Contribution intentionally submitted for inclusion in the Work
132
+ by You to the Licensor shall be under the terms and conditions of
133
+ this License, without any additional terms or conditions.
134
+ Notwithstanding the above, nothing herein shall supersede or modify
135
+ the terms of any separate license agreement you may have executed
136
+ with Licensor regarding such Contributions.
137
+
138
+ 6. Trademarks. This License does not grant permission to use the trade
139
+ names, trademarks, service marks, or product names of the Licensor,
140
+ except as required for reasonable and customary use in describing the
141
+ origin of the Work and reproducing the content of the NOTICE file.
142
+
143
+ 7. Disclaimer of Warranty. Unless required by applicable law or
144
+ agreed to in writing, Licensor provides the Work (and each
145
+ Contributor provides its Contributions) on an "AS IS" BASIS,
146
+ WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or
147
+ implied, including, without limitation, any warranties or conditions
148
+ of TITLE, NON-INFRINGEMENT, MERCHANTABILITY, or FITNESS FOR A
149
+ PARTICULAR PURPOSE. You are solely responsible for determining the
150
+ appropriateness of using or redistributing the Work and assume any
151
+ risks associated with Your exercise of permissions under this License.
152
+
153
+ 8. Limitation of Liability. In no event and under no legal theory,
154
+ whether in tort (including negligence), contract, or otherwise,
155
+ unless required by applicable law (such as deliberate and grossly
156
+ negligent acts) or agreed to in writing, shall any Contributor be
157
+ liable to You for damages, including any direct, indirect, special,
158
+ incidental, or consequential damages of any character arising as a
159
+ result of this License or out of the use or inability to use the
160
+ Work (including but not limited to damages for loss of goodwill,
161
+ work stoppage, computer failure or malfunction, or any and all
162
+ other commercial damages or losses), even if such Contributor
163
+ has been advised of the possibility of such damages.
164
+
165
+ 9. Accepting Warranty or Additional Liability. While redistributing
166
+ the Work or Derivative Works thereof, You may choose to offer,
167
+ and charge a fee for, acceptance of support, warranty, indemnity,
168
+ or other liability obligations and/or rights consistent with this
169
+ License. However, in accepting such obligations, You may act only
170
+ on Your own behalf and on Your sole responsibility, not on behalf
171
+ of any other Contributor, and only if You agree to indemnify,
172
+ defend, and hold each Contributor harmless for any liability
173
+ incurred by, or claims asserted against, such Contributor by reason
174
+ of your accepting any such warranty or additional liability.
175
+
176
+ END OF TERMS AND CONDITIONS
177
+
178
+ Copyright 2015 Xabier de Zuazo
179
+
180
+ Licensed under the Apache License, Version 2.0 (the "License");
181
+ you may not use this file except in compliance with the License.
182
+ You may obtain a copy of the License at
183
+
184
+ http://www.apache.org/licenses/LICENSE-2.0
185
+
186
+ Unless required by applicable law or agreed to in writing, software
187
+ distributed under the License is distributed on an "AS IS" BASIS,
188
+ WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
189
+ See the License for the specific language governing permissions and
190
+ limitations under the License.
data/README.md ADDED
@@ -0,0 +1,100 @@
1
+ # Specinfra Docker LXC Backend
2
+ [![Gem Version](http://img.shields.io/gem/v/specinfra-backend-docker_lxc.svg?style=flat)](http://badge.fury.io/rb/specinfra-backend-docker_lxc)
3
+ [![Dependency Status](http://img.shields.io/gemnasium/zuazo/specinfra-backend-docker_lxc.svg?style=flat)](https://gemnasium.com/zuazo/specinfra-backend-docker_lxc)
4
+ [![Code Climate](http://img.shields.io/codeclimate/github/zuazo/specinfra-backend-docker_lxc.svg?style=flat)](https://codeclimate.com/github/zuazo/specinfra-backend-docker_lxc)
5
+ [![Circle CI](https://circleci.com/gh/zuazo/specinfra-backend-docker_lxc/tree/master.svg?style=shield)](https://circleci.com/gh/zuazo/specinfra-backend-docker_lxc/tree/master)
6
+ [![Travis CI](http://img.shields.io/travis/zuazo/specinfra-backend-docker_lxc/0.1.0.svg?style=flat)](https://travis-ci.org/zuazo/specinfra-backend-docker_lxc)
7
+ [![Coverage Status](http://img.shields.io/coveralls/zuazo/specinfra-backend-docker_lxc/0.1.0.svg?style=flat)](https://coveralls.io/r/zuazo/specinfra-backend-docker_lxc?branch=0.1.0)
8
+ [![Inline docs](http://inch-ci.org/github/zuazo/specinfra-backend-docker_lxc.svg?branch=master&style=flat)](http://inch-ci.org/github/zuazo/specinfra-backend-docker_lxc)
9
+
10
+ [Serverspec](http://serverspec.org/)/[Specinfra](https://github.com/mizzy/specinfra) backend for Docker LXC execution driver.
11
+
12
+ ## Requirements
13
+
14
+ * Recommended Docker `1.7.0` or higher.
15
+ * `sudo` installed (not in the container).
16
+ * `lxc-attach` binary installed (included in the `lxc` package).
17
+
18
+ ## Installation
19
+
20
+ Add this line to your application's Gemfile:
21
+
22
+ ```ruby
23
+ # Gemfile
24
+
25
+ gem 'specinfra-backend-docker_lxc', '~> 0.1.0'
26
+ ```
27
+
28
+ And then execute:
29
+
30
+ $ bundle
31
+
32
+ Or install it yourself as:
33
+
34
+ $ gem install specinfra-backend-docker_lxc
35
+
36
+ ## Usage
37
+
38
+ ```ruby
39
+ require 'serverspec'
40
+ require 'specinfra/backend/docker_lxc'
41
+
42
+ set :docker_image, ENV['DOCKER_IMAGE_ID']
43
+ set :backend, :docker_lxc
44
+
45
+ describe 'Dockerfile run' do
46
+ describe service('httpd') do
47
+ it { should be_enabled }
48
+ it { should be_running }
49
+ end
50
+ end
51
+ ```
52
+
53
+ ## Configuration
54
+
55
+ Uses the following `Specinfra` configuration options:
56
+
57
+ - `:sudo_options`: Sudo command argument list as string or as array.
58
+ - `:sudo_path`: Sudo binary directory.
59
+ - `:sudo_password`
60
+ - `:disable_sudo`: whether to disable Sudo (enabled by default).
61
+
62
+ For example:
63
+
64
+ ```ruby
65
+ set :sudo_password, 'mBnriM8SKhRtIww7xgUi'
66
+ ```
67
+
68
+ ## Testing
69
+
70
+ See [TESTING.md](https://github.com/zuazo/specinfra-backend-docker_lxc/blob/master/TESTING.md).
71
+
72
+ ## Contributing
73
+
74
+ Please do not hesitate to [open an issue](https://github.com/zuazo/specinfra-backend-docker_lxc/issues/new) with any questions or problems.
75
+
76
+ See [CONTRIBUTING.md](https://github.com/zuazo/specinfra-backend-docker_lxc/blob/master/CONTRIBUTING.md).
77
+
78
+ ## TODO
79
+
80
+ See [TODO.md](https://github.com/zuazo/specinfra-backend-docker_lxc/blob/master/TODO.md).
81
+
82
+ ## License and Author
83
+
84
+ | | |
85
+ |:---------------------|:-----------------------------------------|
86
+ | **Author:** | [Xabier de Zuazo](https://github.com/zuazo) (<xabier@zuazo.org>)
87
+ | **Copyright:** | Copyright (c) 2015 Xabier de Zuazo
88
+ | **License:** | Apache License, Version 2.0
89
+
90
+ Licensed under the Apache License, Version 2.0 (the "License");
91
+ you may not use this file except in compliance with the License.
92
+ You may obtain a copy of the License at
93
+
94
+ http://www.apache.org/licenses/LICENSE-2.0
95
+
96
+ Unless required by applicable law or agreed to in writing, software
97
+ distributed under the License is distributed on an "AS IS" BASIS,
98
+ WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
99
+ See the License for the specific language governing permissions and
100
+ limitations under the License.
data/Rakefile ADDED
@@ -0,0 +1,57 @@
1
+ # encoding: UTF-8
2
+ # -*- mode: ruby -*-
3
+ # vi: set ft=ruby :
4
+
5
+ # More info at https://github.com/ruby/rake/blob/master/doc/rakefile.rdoc
6
+
7
+ require 'bundler'
8
+ Bundler::GemHelper.install_tasks
9
+
10
+ desc 'Clean some generated files'
11
+ task :clean do
12
+ %w(
13
+ .bundle
14
+ .cache
15
+ coverage
16
+ doc
17
+ *.gem
18
+ Gemfile.lock
19
+ .inch
20
+ vendor
21
+ .yardoc
22
+ ).each { |f| FileUtils.rm_rf(Dir.glob(f)) }
23
+ end
24
+
25
+ desc 'Generate Ruby documentation'
26
+ task :yard do
27
+ require 'yard'
28
+ YARD::Rake::YardocTask.new do |t|
29
+ t.stats_options = %w(--list-undoc)
30
+ end
31
+ end
32
+
33
+ task doc: %w(yard)
34
+
35
+ desc 'Run RuboCop style checks'
36
+ task :rubocop do
37
+ require 'rubocop/rake_task'
38
+ RuboCop::RakeTask.new
39
+ end
40
+
41
+ desc 'Run all style checks'
42
+ task style: %w(rubocop)
43
+
44
+ require 'rspec/core/rake_task'
45
+
46
+ {
47
+ test: '{unit,integration}',
48
+ unit: 'unit',
49
+ integration: 'integration'
50
+ }.each do |test, dir|
51
+ RSpec::Core::RakeTask.new(test) do |t|
52
+ t.pattern = "spec/#{dir}/**{,/*/**}/*_spec.rb"
53
+ t.verbose = true
54
+ end
55
+ end
56
+
57
+ task default: %w(style test)
data/TESTING.md ADDED
@@ -0,0 +1,30 @@
1
+ # Testing
2
+
3
+ ## Installing the Requirements
4
+
5
+ You can install gem dependencies with bundler:
6
+
7
+ $ gem install bundler
8
+ $ bundler install
9
+
10
+ ## Generate Documentation
11
+
12
+ $ bundle exec rake doc
13
+
14
+ This will generate the HTML documentation in the `doc/` directory.
15
+
16
+ ## All the Tests
17
+
18
+ $ bundle exec rake test
19
+
20
+ ## Running the Syntax Style Tests
21
+
22
+ $ bundle exec rake style
23
+
24
+ ## Running the Unit Tests
25
+
26
+ $ bundle exec rake unit
27
+
28
+ ## Running the Integration Tests
29
+
30
+ $ bundle exec rake integration
data/TODO.md ADDED
@@ -0,0 +1,4 @@
1
+ # TODO for specinfra-backend-docker_lxc
2
+
3
+ * [ ] Add `.available?` method to check the availability of `lxc-attach` command.
4
+ * [ ] Add `lxc_attach_path` option.
@@ -0,0 +1,88 @@
1
+ # encoding: UTF-8
2
+ #
3
+ # Author:: Xabier de Zuazo (<xabier@zuazo.org>)
4
+ # Copyright:: Copyright (c) 2015 Xabier de Zuazo
5
+ # License:: Apache License, Version 2.0
6
+ #
7
+ # Licensed under the Apache License, Version 2.0 (the "License");
8
+ # you may not use this file except in compliance with the License.
9
+ # You may obtain a copy of the License at
10
+ #
11
+ # http://www.apache.org/licenses/LICENSE-2.0
12
+ #
13
+ # Unless required by applicable law or agreed to in writing, software
14
+ # distributed under the License is distributed on an "AS IS" BASIS,
15
+ # WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
16
+ # See the License for the specific language governing permissions and
17
+ # limitations under the License.
18
+ #
19
+
20
+ require 'specinfra/backend/docker'
21
+ require 'specinfra/backend/docker_lxc/shell_helpers'
22
+ require 'specinfra/backend/docker_lxc/exceptions'
23
+
24
+ # Command Execution Framework for Serverspec, Itamae and so on.
25
+ module Specinfra
26
+ # Specinfra backend types.
27
+ module Backend
28
+ # Specinfra and Serverspec backend for Docker LXC execution driver.
29
+ class DockerLxc < Docker
30
+ include Specinfra::Backend::DockerLxc::ShellHelpers
31
+
32
+ protected
33
+
34
+ # Generates `lxc-attach` command to run.
35
+ #
36
+ # @param cmd [String] the commands to run inside docker.
37
+ # @return [Array] the command to run as unescaped array.
38
+ def lxc_attach_command(cmd)
39
+ id = @container.id
40
+ ['lxc-attach', '-n', id, '--', 'sh', '-c', cmd]
41
+ end
42
+
43
+ # Parses `lxc-attach` command output and raises an exception if it is an
44
+ # error from the `lxc-attach` program.
45
+ #
46
+ # @param stderr [String] command *stderr* output.
47
+ # @param exit_status [Fixnum] command exit status.
48
+ # @return nil
49
+ def lxc_attach_result_assert(stderr, exit_status)
50
+ return if exit_status == 0
51
+ return if stderr.match(/\A(lxc-attach|lxc_container|sudo): /).nil?
52
+ fail LxcAttachError, stderr
53
+ end
54
+
55
+ # Parses a rescued exception and returns the command result.
56
+ #
57
+ # @param exception [Exception] the exception to parse.
58
+ # @param stdout [String] the *stdout* output.
59
+ # @param stderr [String] the *stderr* output.
60
+ # @param status [Fixnum] the command exit status.
61
+ # @return [Specinfra::CommandResult] the generated result object.
62
+ def erroneous_result(exception, stdout, stderr, status)
63
+ err =
64
+ stderr.nil? ? ([exception.message] + exception.backtrace) : [stderr]
65
+ sta = status.is_a?(Fixnum) && status != 0 ? status : 1
66
+ CommandResult.new(stdout: stdout, stderr: err.join, exit_status: sta)
67
+ end
68
+
69
+ # Runs a command inside a Docker container.
70
+ #
71
+ # @param cmd [String] the command to run.
72
+ # @param opts [Hash] options to pass to {Open3.popen3}.
73
+ # @return [Specinfra::CommandResult] the result.
74
+ def docker_run!(cmd, opts = {})
75
+ stdout, stderr, status = shell_command!(lxc_attach_command(cmd), opts)
76
+ lxc_attach_result_assert(stderr, status)
77
+ CommandResult.new(
78
+ stdout: stdout, stderr: stderr, exit_status: status
79
+ )
80
+ rescue LxcAttachError
81
+ raise
82
+ rescue => e
83
+ @container.kill
84
+ erroneous_result(e, stdout, stderr, status)
85
+ end
86
+ end
87
+ end
88
+ end
@@ -0,0 +1,27 @@
1
+ # encoding: UTF-8
2
+ #
3
+ # Author:: Xabier de Zuazo (<xabier@zuazo.org>)
4
+ # Copyright:: Copyright (c) 2015 Xabier de Zuazo
5
+ # License:: Apache License, Version 2.0
6
+ #
7
+ # Licensed under the Apache License, Version 2.0 (the "License");
8
+ # you may not use this file except in compliance with the License.
9
+ # You may obtain a copy of the License at
10
+ #
11
+ # http://www.apache.org/licenses/LICENSE-2.0
12
+ #
13
+ # Unless required by applicable law or agreed to in writing, software
14
+ # distributed under the License is distributed on an "AS IS" BASIS,
15
+ # WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
16
+ # See the License for the specific language governing permissions and
17
+ # limitations under the License.
18
+ #
19
+
20
+ module Specinfra
21
+ module Backend
22
+ class DockerLxc < Docker
23
+ # Exception raised for `lxc-attach` command errors.
24
+ class LxcAttachError < RuntimeError; end
25
+ end
26
+ end
27
+ end
@@ -0,0 +1,222 @@
1
+ # encoding: UTF-8
2
+ #
3
+ # Author:: Xabier de Zuazo (<xabier@zuazo.org>)
4
+ # Copyright:: Copyright (c) 2015 Xabier de Zuazo
5
+ # License:: Apache License, Version 2.0
6
+ #
7
+ # Licensed under the Apache License, Version 2.0 (the "License");
8
+ # you may not use this file except in compliance with the License.
9
+ # You may obtain a copy of the License at
10
+ #
11
+ # http://www.apache.org/licenses/LICENSE-2.0
12
+ #
13
+ # Unless required by applicable law or agreed to in writing, software
14
+ # distributed under the License is distributed on an "AS IS" BASIS,
15
+ # WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
16
+ # See the License for the specific language governing permissions and
17
+ # limitations under the License.
18
+ #
19
+
20
+ require 'open3'
21
+ require 'shellwords'
22
+ require 'etc'
23
+
24
+ module Specinfra
25
+ module Backend
26
+ class DockerLxc < Docker
27
+ # Helpers to work with the shell and with `sudo`.
28
+ #
29
+ # Uses the following `Specinfra` configuration options:
30
+ #
31
+ # - `:sudo_options`: Sudo command argument list as string or as array.
32
+ # - `:sudo_path`: Sudo binary directory.
33
+ # - `:sudo_password`
34
+ # - `:disable_sudo`: whether to disable Sudo (enabled by default).
35
+ #
36
+ # Based on the official `Specinfra::Backend::Ssh` code.
37
+ #
38
+ # @example
39
+ # class MyBackend < Specinfra::Backend::Base
40
+ # include Specinfra::Backend::DockerLxc::ShellHelpers
41
+ #
42
+ # def my_backend_run!
43
+ # stdout, stderr, status = shell_command!('uname -a')
44
+ # CommandResult.new(
45
+ # stdout: stdout, stderr: stderr, exit_status: status
46
+ # )
47
+ # end
48
+ # end
49
+ module ShellHelpers
50
+ protected
51
+
52
+ # Returns the prompt used by Sudo to ask for the password.
53
+ #
54
+ # @return [String] the prompt.
55
+ # @example
56
+ # sudo_prompt #=> "Password: "
57
+ def sudo_prompt
58
+ 'Password: '
59
+ end
60
+
61
+ # Reads the `:sudo_password` configuration option.
62
+ #
63
+ # @return [String] the password.
64
+ # @example
65
+ # set :sudo_password, 'y0mVT1CYM0uRiHxjLfNV'
66
+ # sudo_password #=> "y0mVT1CYM0uRiHxjLfNV"
67
+ def sudo_password
68
+ Specinfra.configuration.sudo_password
69
+ end
70
+
71
+ # Whether the `:sudo_password` is configured.
72
+ #
73
+ # @return [TrueClass, FalseClass] true if password is configured.
74
+ # @example
75
+ # sudo_password? #=> false
76
+ def sudo_password?
77
+ sudo_password ? true : false
78
+ end
79
+
80
+ # Returns the Sudo program arguments to use by default.
81
+ #
82
+ # @return [String] the arguments properly escaped.
83
+ # @example
84
+ # default_sudo_args #=> ""
85
+ # default_sudo_args #=> "-S -p Password:\\ "
86
+ def default_sudo_args
87
+ args = []
88
+ args += ['-S', '-p', sudo_prompt] if sudo_password?
89
+ args.shelljoin
90
+ end
91
+
92
+ # Returns the Sudo program arguments.
93
+ #
94
+ # Includes both the default and the arguments configured using `set`.
95
+ #
96
+ # @return [String] the arguments properly escaped.
97
+ # @example
98
+ # set :sudo_options, '-a -b -c'
99
+ # sudo_args #=> "-S -p Password:\\ -a -b -c"
100
+ def sudo_args
101
+ sudo_options = Specinfra.configuration.sudo_options
102
+ if sudo_options
103
+ sudo_options = sudo_options.shelljoin if sudo_options.is_a?(Array)
104
+ "#{default_sudo_args} #{sudo_options}"
105
+ else
106
+ default_sudo_args
107
+ end
108
+ end
109
+
110
+ # Gets the Sudo binary path.
111
+ #
112
+ # @return [String] the Sudo binary.
113
+ # @example
114
+ # sudo_bin #=> "sudo"
115
+ # set :sudo_path, '/opt/sudo/bin'
116
+ # sudo_bin #=> "/opt/sudo/bin/sudo"
117
+ def sudo_bin
118
+ sudo_path = Specinfra.configuration.sudo_path
119
+ sudo_bin = sudo_path ? "#{sudo_path}/sudo" : 'sudo'
120
+ sudo_bin.shellescape
121
+ end
122
+
123
+ # Adds Sudo to a command.
124
+ #
125
+ # @param cmd_str [String] the command to run. Must be escaped.
126
+ # @return [String] the command escaped and including `sudo`.
127
+ # @example
128
+ # set :sudo_password, 'y0mVT1CYM0uRiHxjLfNV'
129
+ # sudo_command('uname -a') #=> "sudo -S -p Password:\\ -- uname -a"
130
+ # set :disable_sudo, true
131
+ # sudo_command('uname -a') #=> "uname -a"
132
+ def sudo_command(cmd_str)
133
+ "#{sudo_bin} #{sudo_args} -- #{cmd_str}"
134
+ end
135
+
136
+ # Checks if we need to use Sudo.
137
+ #
138
+ # @return [TrueClass, FalseClass] whether we need to use Sudo.
139
+ # @example
140
+ # sudo? #=> true
141
+ # set :disable_sudo, true
142
+ # sudo? #=> false
143
+ def sudo?
144
+ disable_sudo = Specinfra.configuration.disable_sudo
145
+ Etc.getlogin != 'root' && !disable_sudo
146
+ end
147
+
148
+ # Escapes a shell command.
149
+ #
150
+ # It only escapes it when passed as array.
151
+ #
152
+ # @param cmd [Array<String>, String] the command.
153
+ # @return [String] the command escaped.
154
+ # @example
155
+ # escape_command(['sudo', '-p', 'Password: ')
156
+ # #=> "sudo -p Password:\\ "
157
+ # escape_command('uname -a') #=> "uname -a"
158
+ def escape_command(cmd)
159
+ return cmd if cmd.is_a?(String)
160
+ cmd.shelljoin
161
+ end
162
+
163
+ # Generates the command to run including the Sudo prefix if configured.
164
+ #
165
+ # The command needs to be escaped only if passed as string.
166
+ #
167
+ # @param cmd [Array<String>, String] the command.
168
+ # @return [String] the command escaped.
169
+ # @example
170
+ # generate_escaped_command('uname -a') #=> "sudo uname -a"
171
+ # set :sudo_password, 'y0mVT1CYM0uRiHxjLfNV'
172
+ # generate_escaped_command('uname -a')
173
+ # #=> "sudo -p Password:\\ uname -a"
174
+ # set :disable_sudo, true
175
+ # generate_escaped_command('uname -a') #=> "uname -a"
176
+ def generate_escaped_command(cmd)
177
+ if sudo?
178
+ sudo_command(escape_command(cmd))
179
+ else
180
+ escape_command(cmd)
181
+ end
182
+ end
183
+
184
+ # Writes the password to the *stdin* when asked on *stderr*.
185
+ #
186
+ # @param stdin [IO] stdin file descriptor.
187
+ # @param stderr [IO] stderr file descriptor.
188
+ # @return [String] the string read from stderr without including the
189
+ # password prompt.
190
+ # @example
191
+ # write_sudo_password(stdin, stderr) #=> ""
192
+ def write_sudo_password(stdin, stderr)
193
+ return '' unless sudo_password?
194
+ read = stderr.gets(sudo_prompt.length)
195
+ return read.to_s unless read == sudo_prompt
196
+ stdin.puts "#{sudo_password}\n"
197
+ ''
198
+ end
199
+
200
+ # Runs a command, including Sudo if required.
201
+ #
202
+ # If the command is passed as string, must be properly escaped.
203
+ #
204
+ # @param cmd [Array<String>, String] the command.
205
+ # @return [Array<String, Fixnum>] the array contents: *stdout*, *stderr*
206
+ # and the *exit status*.
207
+ # @example
208
+ # shell_command!('id')
209
+ # #=> ["", "uid=0(root) gid=0(root) groups=0(root)\n", 0]
210
+ # @api public
211
+ def shell_command!(cmd, opts = {})
212
+ cmd_escaped = generate_escaped_command(cmd)
213
+ Open3.popen3(cmd_escaped, opts) do |stdin, stdout, stderr, wait_thr|
214
+ read = write_sudo_password(stdin, stderr)
215
+ stdin.close
216
+ [stdout.read, read + stderr.read, wait_thr.value.exitstatus]
217
+ end
218
+ end
219
+ end
220
+ end
221
+ end
222
+ end
@@ -0,0 +1,56 @@
1
+ # encoding: UTF-8
2
+ #
3
+ # Author:: Xabier de Zuazo (<xabier@zuazo.org>)
4
+ # Copyright:: Copyright (c) 2015 Xabier de Zuazo
5
+ # License:: Apache License, Version 2.0
6
+ #
7
+ # Licensed under the Apache License, Version 2.0 (the "License");
8
+ # you may not use this file except in compliance with the License.
9
+ # You may obtain a copy of the License at
10
+ #
11
+ # http://www.apache.org/licenses/LICENSE-2.0
12
+ #
13
+ # Unless required by applicable law or agreed to in writing, software
14
+ # distributed under the License is distributed on an "AS IS" BASIS,
15
+ # WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
16
+ # See the License for the specific language governing permissions and
17
+ # limitations under the License.
18
+ #
19
+
20
+ require 'simplecov'
21
+ if ENV['TRAVIS']
22
+ require 'coveralls'
23
+ SimpleCov.formatter = Coveralls::SimpleCov::Formatter
24
+ end
25
+ SimpleCov.start do
26
+ add_filter '/spec/'
27
+ end
28
+
29
+ require 'specinfra'
30
+ require 'specinfra/backend/docker_lxc'
31
+ require 'docker'
32
+
33
+ Specinfra.configuration.backend(:base)
34
+
35
+ require 'should_not/rspec'
36
+
37
+ require 'support/docker_logger'
38
+ require 'support/rspec_filters'
39
+ require 'support/disabled_docker_lxc'
40
+ require 'support/docker_lxc_helpers'
41
+
42
+ RSpec.configure do |config|
43
+ # Prohibit using the should syntax
44
+ config.expect_with :rspec do |spec|
45
+ spec.syntax = :expect
46
+ end
47
+
48
+ config.order = 'random'
49
+
50
+ config.color = true
51
+ config.tty = true
52
+
53
+ config.filter_run_excluding lxc_driver: true unless lxc_execution_driver?
54
+
55
+ config.before(:each) { DockerLxcHelpers.configuration_reset }
56
+ end
metadata ADDED
@@ -0,0 +1,225 @@
1
+ --- !ruby/object:Gem::Specification
2
+ name: specinfra-backend-docker_lxc
3
+ version: !ruby/object:Gem::Version
4
+ version: 0.1.0
5
+ platform: ruby
6
+ authors:
7
+ - Xabier de Zuazo
8
+ autorequire:
9
+ bindir: bin
10
+ cert_chain: []
11
+ date: 2015-11-14 00:00:00.000000000 Z
12
+ dependencies:
13
+ - !ruby/object:Gem::Dependency
14
+ name: specinfra
15
+ requirement: !ruby/object:Gem::Requirement
16
+ requirements:
17
+ - - "~>"
18
+ - !ruby/object:Gem::Version
19
+ version: '2.13'
20
+ type: :runtime
21
+ prerelease: false
22
+ version_requirements: !ruby/object:Gem::Requirement
23
+ requirements:
24
+ - - "~>"
25
+ - !ruby/object:Gem::Version
26
+ version: '2.13'
27
+ - !ruby/object:Gem::Dependency
28
+ name: rake
29
+ requirement: !ruby/object:Gem::Requirement
30
+ requirements:
31
+ - - "~>"
32
+ - !ruby/object:Gem::Version
33
+ version: '10.0'
34
+ type: :development
35
+ prerelease: false
36
+ version_requirements: !ruby/object:Gem::Requirement
37
+ requirements:
38
+ - - "~>"
39
+ - !ruby/object:Gem::Version
40
+ version: '10.0'
41
+ - !ruby/object:Gem::Dependency
42
+ name: rspec-core
43
+ requirement: !ruby/object:Gem::Requirement
44
+ requirements:
45
+ - - "~>"
46
+ - !ruby/object:Gem::Version
47
+ version: '3.1'
48
+ type: :development
49
+ prerelease: false
50
+ version_requirements: !ruby/object:Gem::Requirement
51
+ requirements:
52
+ - - "~>"
53
+ - !ruby/object:Gem::Version
54
+ version: '3.1'
55
+ - !ruby/object:Gem::Dependency
56
+ name: rspec-expectations
57
+ requirement: !ruby/object:Gem::Requirement
58
+ requirements:
59
+ - - "~>"
60
+ - !ruby/object:Gem::Version
61
+ version: '3.1'
62
+ type: :development
63
+ prerelease: false
64
+ version_requirements: !ruby/object:Gem::Requirement
65
+ requirements:
66
+ - - "~>"
67
+ - !ruby/object:Gem::Version
68
+ version: '3.1'
69
+ - !ruby/object:Gem::Dependency
70
+ name: rspec-mocks
71
+ requirement: !ruby/object:Gem::Requirement
72
+ requirements:
73
+ - - "~>"
74
+ - !ruby/object:Gem::Version
75
+ version: '3.1'
76
+ type: :development
77
+ prerelease: false
78
+ version_requirements: !ruby/object:Gem::Requirement
79
+ requirements:
80
+ - - "~>"
81
+ - !ruby/object:Gem::Version
82
+ version: '3.1'
83
+ - !ruby/object:Gem::Dependency
84
+ name: coveralls
85
+ requirement: !ruby/object:Gem::Requirement
86
+ requirements:
87
+ - - "~>"
88
+ - !ruby/object:Gem::Version
89
+ version: '0.7'
90
+ type: :development
91
+ prerelease: false
92
+ version_requirements: !ruby/object:Gem::Requirement
93
+ requirements:
94
+ - - "~>"
95
+ - !ruby/object:Gem::Version
96
+ version: '0.7'
97
+ - !ruby/object:Gem::Dependency
98
+ name: simplecov
99
+ requirement: !ruby/object:Gem::Requirement
100
+ requirements:
101
+ - - "~>"
102
+ - !ruby/object:Gem::Version
103
+ version: '0.9'
104
+ type: :development
105
+ prerelease: false
106
+ version_requirements: !ruby/object:Gem::Requirement
107
+ requirements:
108
+ - - "~>"
109
+ - !ruby/object:Gem::Version
110
+ version: '0.9'
111
+ - !ruby/object:Gem::Dependency
112
+ name: should_not
113
+ requirement: !ruby/object:Gem::Requirement
114
+ requirements:
115
+ - - "~>"
116
+ - !ruby/object:Gem::Version
117
+ version: '1.1'
118
+ type: :development
119
+ prerelease: false
120
+ version_requirements: !ruby/object:Gem::Requirement
121
+ requirements:
122
+ - - "~>"
123
+ - !ruby/object:Gem::Version
124
+ version: '1.1'
125
+ - !ruby/object:Gem::Dependency
126
+ name: rubocop
127
+ requirement: !ruby/object:Gem::Requirement
128
+ requirements:
129
+ - - "~>"
130
+ - !ruby/object:Gem::Version
131
+ version: 0.35.0
132
+ type: :development
133
+ prerelease: false
134
+ version_requirements: !ruby/object:Gem::Requirement
135
+ requirements:
136
+ - - "~>"
137
+ - !ruby/object:Gem::Version
138
+ version: 0.35.0
139
+ - !ruby/object:Gem::Dependency
140
+ name: yard
141
+ requirement: !ruby/object:Gem::Requirement
142
+ requirements:
143
+ - - "~>"
144
+ - !ruby/object:Gem::Version
145
+ version: '0.8'
146
+ type: :development
147
+ prerelease: false
148
+ version_requirements: !ruby/object:Gem::Requirement
149
+ requirements:
150
+ - - "~>"
151
+ - !ruby/object:Gem::Version
152
+ version: '0.8'
153
+ - !ruby/object:Gem::Dependency
154
+ name: docker-api
155
+ requirement: !ruby/object:Gem::Requirement
156
+ requirements:
157
+ - - "~>"
158
+ - !ruby/object:Gem::Version
159
+ version: '1.22'
160
+ type: :development
161
+ prerelease: false
162
+ version_requirements: !ruby/object:Gem::Requirement
163
+ requirements:
164
+ - - "~>"
165
+ - !ruby/object:Gem::Version
166
+ version: '1.22'
167
+ - !ruby/object:Gem::Dependency
168
+ name: serverspec
169
+ requirement: !ruby/object:Gem::Requirement
170
+ requirements:
171
+ - - "~>"
172
+ - !ruby/object:Gem::Version
173
+ version: '2.24'
174
+ type: :development
175
+ prerelease: false
176
+ version_requirements: !ruby/object:Gem::Requirement
177
+ requirements:
178
+ - - "~>"
179
+ - !ruby/object:Gem::Version
180
+ version: '2.24'
181
+ description: Serverspec/Specinfra backend for Docker LXC execution driver.
182
+ email: xabier@zuazo.org
183
+ executables: []
184
+ extensions: []
185
+ extra_rdoc_files: []
186
+ files:
187
+ - ".yardopts"
188
+ - CHANGELOG.md
189
+ - CONTRIBUTING.md
190
+ - LICENSE
191
+ - README.md
192
+ - Rakefile
193
+ - TESTING.md
194
+ - TODO.md
195
+ - lib/specinfra/backend/docker_lxc.rb
196
+ - lib/specinfra/backend/docker_lxc/exceptions.rb
197
+ - lib/specinfra/backend/docker_lxc/shell_helpers.rb
198
+ - spec/spec_helper.rb
199
+ homepage: https://github.com/zuazo/specinfra-backend-docker_lxc
200
+ licenses:
201
+ - Apache-2.0
202
+ metadata: {}
203
+ post_install_message:
204
+ rdoc_options: []
205
+ require_paths:
206
+ - lib
207
+ required_ruby_version: !ruby/object:Gem::Requirement
208
+ requirements:
209
+ - - ">="
210
+ - !ruby/object:Gem::Version
211
+ version: 2.0.0
212
+ required_rubygems_version: !ruby/object:Gem::Requirement
213
+ requirements:
214
+ - - ">="
215
+ - !ruby/object:Gem::Version
216
+ version: '0'
217
+ requirements: []
218
+ rubyforge_project:
219
+ rubygems_version: 2.2.2
220
+ signing_key:
221
+ specification_version: 4
222
+ summary: Specinfra Docker LXC Backend
223
+ test_files:
224
+ - spec/spec_helper.rb
225
+ has_rdoc: