beaker-facter 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,15 @@
1
+ ---
2
+ !binary "U0hBMQ==":
3
+ metadata.gz: !binary |-
4
+ OThlZDI4MmUxYTY5MTEyZDhlYzA2NmI4MGU2MTI2N2Y0Zjk1OTljNw==
5
+ data.tar.gz: !binary |-
6
+ ZGVmYTRiNjEzZjNhYWJlYTIxMzI2MmFkNmRkNmNlYTE3MjA0MDlmMQ==
7
+ SHA512:
8
+ metadata.gz: !binary |-
9
+ MTYwM2I1ODI3MjI2ODIyZGFiMWYxZTViZTQ0M2YwZjM0MWM4Yjc2ZWIwNDM0
10
+ M2RhZmVmMmYyYTk4MmM0NGJmNTNmNmZjYWRkOGE5OWMyZjVlN2EyODZhYjhl
11
+ N2ZmMzIzZmM5OGIxNDRmNzVjMTQ5NTRiYTI2MmRjN2Q3ZTFlNzQ=
12
+ data.tar.gz: !binary |-
13
+ MzM5YzVmYmU4MzM2ZmI2MjczZWJjNGViNmQ5ZDJlOWJjNTVjYjE0NGJkZDhk
14
+ MjYyMzA5NjM0OGE2M2QxZWE3MWZjMmM0ODcwMWEzMjViYzRkYTg5OWQwM2Mz
15
+ NGQwMWM2Zjg5OGU4NjA4OWU1NzVlZjJmM2EyN2ZiOWZkOTY0NjU=
data/.gitignore ADDED
@@ -0,0 +1,25 @@
1
+ *.swp
2
+ log/*
3
+ !.gitignore
4
+ junit
5
+ acceptance-tests
6
+ pkg
7
+ Gemfile.lock
8
+ options.rb
9
+ test.cfg
10
+ .yardoc
11
+ coverage
12
+ .bundle
13
+ .vendor
14
+ _vendor
15
+ tmp/
16
+ doc
17
+ # JetBrains IDEA
18
+ *.iml
19
+ .idea/
20
+ # rbenv file
21
+ .ruby-version
22
+ .ruby-gemset
23
+ # Vagrant folder
24
+ .vagrant/
25
+ .vagrant_files/
data/.rspec ADDED
@@ -0,0 +1,3 @@
1
+ --format documentation
2
+ --color
3
+ --tty
data/.simplecov ADDED
@@ -0,0 +1,9 @@
1
+ SimpleCov.configure do
2
+ add_filter 'spec/'
3
+ add_filter 'vendor/'
4
+ add_filter do |file|
5
+ file.lines_of_code < 10
6
+ end
7
+ end
8
+
9
+ SimpleCov.start if ENV['BEAKER_FACTER_COVERAGE']
data/Gemfile ADDED
@@ -0,0 +1,27 @@
1
+ source ENV['GEM_SOURCE'] || "https://rubygems.org"
2
+
3
+ gemspec
4
+
5
+
6
+
7
+ def location_for(place, fake_version = nil)
8
+ if place =~ /^(git:[^#]*)#(.*)/
9
+ [fake_version, { :git => $1, :branch => $2, :require => false }].compact
10
+ elsif place =~ /^file:\/\/(.*)/
11
+ ['>= 0', { :path => File.expand_path($1), :require => false }]
12
+ else
13
+ [place, { :require => false }]
14
+ end
15
+ end
16
+
17
+
18
+ # We don't put beaker in as a test dependency because we
19
+ # don't want to create a transitive dependency
20
+ group :acceptance_testing do
21
+ gem "beaker", *location_for(ENV['BEAKER_VERSION'] || '~> 2.0')
22
+ end
23
+
24
+
25
+ if File.exists? "#{__FILE__}.local"
26
+ eval(File.read("#{__FILE__}.local"), binding)
27
+ end
data/HISTORY.md ADDED
@@ -0,0 +1,8 @@
1
+ # default - History
2
+ ## Tags
3
+ * [LATEST - 5 Nov, 2015 (ece69e29)](#LATEST)
4
+
5
+ ## Details
6
+ ### <a name = "LATEST">LATEST - 5 Nov, 2015 (ece69e29)
7
+
8
+ * Initial release.
data/LICENSE ADDED
@@ -0,0 +1,201 @@
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
+ APPENDIX: How to apply the Apache License to your work.
179
+
180
+ To apply the Apache License to your work, attach the following
181
+ boilerplate notice, with the fields enclosed by brackets "{}"
182
+ replaced with your own identifying information. (Don't include
183
+ the brackets!) The text should be enclosed in the appropriate
184
+ comment syntax for the file format. We also recommend that a
185
+ file or class name and description of purpose be included on the
186
+ same "printed page" as the copyright notice for easier
187
+ identification within third-party archives.
188
+
189
+ Copyright {yyyy} {name of copyright owner}
190
+
191
+ Licensed under the Apache License, Version 2.0 (the "License");
192
+ you may not use this file except in compliance with the License.
193
+ You may obtain a copy of the License at
194
+
195
+ http://www.apache.org/licenses/LICENSE-2.0
196
+
197
+ Unless required by applicable law or agreed to in writing, software
198
+ distributed under the License is distributed on an "AS IS" BASIS,
199
+ WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
200
+ See the License for the specific language governing permissions and
201
+ limitations under the License.
data/README.md ADDED
@@ -0,0 +1,2 @@
1
+ # beaker-facter
2
+ Beaker DSL Extension Facter Helpers
data/Rakefile ADDED
@@ -0,0 +1,185 @@
1
+ require 'rspec/core/rake_task'
2
+
3
+ namespace :test do
4
+
5
+ namespace :spec do
6
+
7
+ desc "Run spec tests"
8
+ RSpec::Core::RakeTask.new(:run) do |t|
9
+ t.rspec_opts = ['--color']
10
+ t.pattern = 'spec/'
11
+ end
12
+
13
+ desc "Run spec tests with coverage"
14
+ RSpec::Core::RakeTask.new(:coverage) do |t|
15
+ ENV['COVERAGE'] = 'y'
16
+ t.rspec_opts = ['--color']
17
+ t.pattern = 'spec/'
18
+ end
19
+
20
+ end
21
+
22
+ namespace :acceptance do
23
+
24
+ def hosts
25
+ ENV['HOSTS'] || ENV['CONFIG'] || 'acceptance/config/nodes/redhat-7-x86_64.yaml'
26
+ end
27
+
28
+ def tests
29
+ ENV['TESTS'] || ENV['TEST'] || 'acceptance/tests'
30
+ end
31
+
32
+ def generate_beaker_cli_flags
33
+ # the options file (including the default options might also have tests
34
+ # they'll get merged with the below by beaker
35
+ tests_opt = "--tests=#{tests}" if tests
36
+
37
+ hosts_opt = "--hosts=#{hosts}" if hosts
38
+
39
+ overriding_options = ENV['OPTIONS'].to_s
40
+
41
+ # compact to remove the nil elements
42
+ [hosts_opt, tests_opt, *overriding_options.split(' ')].compact
43
+ end
44
+
45
+ desc 'Component functional tests for beaker-facter.'
46
+ task :quick do
47
+ sh('beaker',
48
+ '--type', 'foss',
49
+ '--load-path', 'acceptance/lib',
50
+ '--pre-suite', 'acceptance/pre-suite',
51
+ '--keyfile', ENV['KEY'] || "#{ENV['HOME']}/.ssh/id_rsa",
52
+ *generate_beaker_cli_flags,
53
+ )
54
+ end
55
+
56
+ desc 'Component functional tests with coverage for beaker-facter.'
57
+ task :coverage do
58
+ ENV['COVERAGE'] = 'y'
59
+ sh('beaker',
60
+ '--type', 'foss',
61
+ '--load-path', 'acceptance/lib',
62
+ '--pre-suite', 'acceptance/pre-suite',
63
+ '--keyfile', ENV['KEY'] || "#{ENV['HOME']}/.ssh/id_rsa",
64
+ *generate_beaker_cli_flags,
65
+ )
66
+ end
67
+ end
68
+
69
+ end
70
+
71
+ # namespace-named default tasks.
72
+ # these are the default tasks invoked when only the namespace is referenced.
73
+ # they're needed because `task :default` in those blocks doesn't work as expected.
74
+ task 'test:spec' => 'test:spec:run'
75
+ task 'test:acceptance' => 'test:acceptance:quick'
76
+
77
+ # global defaults
78
+ task :test => 'test:spec'
79
+ task :default => :test
80
+
81
+ ###########################################################
82
+ #
83
+ # Documentation Tasks
84
+ #
85
+ ###########################################################
86
+ DOCS_DAEMON = "yard server --reload --daemon --server thin"
87
+ FOREGROUND_SERVER = 'bundle exec yard server --reload --verbose --server thin lib/beaker'
88
+
89
+ def running?( cmdline )
90
+ ps = `ps -ef`
91
+ found = ps.lines.grep( /#{Regexp.quote( cmdline )}/ )
92
+ if found.length > 1
93
+ raise StandardError, "Found multiple YARD Servers. Don't know what to do."
94
+ end
95
+
96
+ yes = found.empty? ? false : true
97
+ return yes, found.first
98
+ end
99
+
100
+ def pid_from( output )
101
+ output.squeeze(' ').strip.split(' ')[1]
102
+ end
103
+
104
+ desc 'Start the documentation server in the foreground'
105
+ task :docs => 'docs:clear' do
106
+ original_dir = Dir.pwd
107
+ Dir.chdir( File.expand_path(File.dirname(__FILE__)) )
108
+ sh FOREGROUND_SERVER
109
+ Dir.chdir( original_dir )
110
+ end
111
+
112
+ namespace :docs do
113
+
114
+ desc 'Clear the generated documentation cache'
115
+ task :clear do
116
+ original_dir = Dir.pwd
117
+ Dir.chdir( File.expand_path(File.dirname(__FILE__)) )
118
+ sh 'rm -rf docs'
119
+ Dir.chdir( original_dir )
120
+ end
121
+
122
+ desc 'Generate static documentation'
123
+ task :gen => 'docs:clear' do
124
+ original_dir = Dir.pwd
125
+ Dir.chdir( File.expand_path(File.dirname(__FILE__)) )
126
+ output = `bundle exec yard doc`
127
+ puts output
128
+ if output =~ /\[warn\]|\[error\]/
129
+ fail "Errors/Warnings during yard documentation generation"
130
+ end
131
+ Dir.chdir( original_dir )
132
+ end
133
+
134
+ desc 'Run the documentation server in the background, alias `bg`'
135
+ task :background => 'docs:clear' do
136
+ yes, output = running?( DOCS_DAEMON )
137
+ if yes
138
+ puts "Not starting a new YARD Server..."
139
+ puts "Found one running with pid #{pid_from( output )}."
140
+ else
141
+ original_dir = Dir.pwd
142
+ Dir.chdir( File.expand_path(File.dirname(__FILE__)) )
143
+ sh "bundle exec #{DOCS_DAEMON}"
144
+ Dir.chdir( original_dir )
145
+ end
146
+ end
147
+
148
+ task(:bg) { Rake::Task['docs:background'].invoke }
149
+
150
+ desc 'Check the status of the documentation server'
151
+ task :status do
152
+ yes, output = running?( DOCS_DAEMON )
153
+ if yes
154
+ pid = pid_from( output )
155
+ puts "Found a YARD Server running with pid #{pid}"
156
+ else
157
+ puts "Could not find a running YARD Server."
158
+ end
159
+ end
160
+
161
+ desc "Stop a running YARD Server"
162
+ task :stop do
163
+ yes, output = running?( DOCS_DAEMON )
164
+ if yes
165
+ pid = pid_from( output )
166
+ puts "Found a YARD Server running with pid #{pid}"
167
+ `kill #{pid}`
168
+ puts "Stopping..."
169
+ yes, output = running?( DOCS_DAEMON )
170
+ if yes
171
+ `kill -9 #{pid}`
172
+ yes, output = running?( DOCS_DAEMON )
173
+ if yes
174
+ puts "Could not Stop Server!"
175
+ else
176
+ puts "Server stopped."
177
+ end
178
+ else
179
+ puts "Server stopped."
180
+ end
181
+ else
182
+ puts "Could not find a running YARD Server"
183
+ end
184
+ end
185
+ end
@@ -0,0 +1,8 @@
1
+ HOSTS:
2
+ ubuntu-server-1404-x64:
3
+ roles:
4
+ - master
5
+ platform: ubuntu-14.04-amd64
6
+ box: puppetlabs/ubuntu-14.04-64-nocm
7
+ box_url: https://vagrantcloud.com/puppetlabs/boxes/ubuntu-14.04-64-nocm
8
+ hypervisor: vagrant
@@ -0,0 +1,108 @@
1
+ module Acceptance
2
+ module Install
3
+ PLATFORM_PATTERNS = {
4
+ :redhat => /fedora|el|centos/,
5
+ :debian => /debian|ubuntu/,
6
+ :debian_ruby18 => /debian|ubuntu-lucid|ubuntu-precise/,
7
+ :solaris => /solaris/,
8
+ :windows => /windows/,
9
+ }.freeze
10
+
11
+ # Installs packages on the hosts.
12
+ #
13
+ # @param hosts [Array<Host>] Array of hosts to install packages to.
14
+ # @param package_hash [Hash{Symbol=>Array<String,Array<String,String>>}]
15
+ # Keys should be a symbol for a platform in PLATFORM_PATTERNS. Values
16
+ # should be an array of package names to install, or of two element
17
+ # arrays where a[0] is the command we expect to find on the platform
18
+ # and a[1] is the package name (when they are different).
19
+ # @param options [Hash{Symbol=>Boolean}]
20
+ # @option options [Boolean] :check_if_exists First check to see if
21
+ # command is present before installing package. (Default false)
22
+ # @return true
23
+ def install_packages_on(hosts, package_hash, options = {})
24
+ check_if_exists = options[:check_if_exists]
25
+ hosts = [hosts] unless hosts.kind_of?(Array)
26
+ hosts.each do |host|
27
+ package_hash.each do |platform_key,package_list|
28
+ if pattern = PLATFORM_PATTERNS[platform_key]
29
+ if pattern.match(host['platform'])
30
+ package_list.each do |cmd_pkg|
31
+ if cmd_pkg.kind_of?(Array)
32
+ command, package = cmd_pkg
33
+ else
34
+ command = package = cmd_pkg
35
+ end
36
+ if !check_if_exists || !host.check_for_package(command)
37
+ host.logger.notify("Installing #{package}")
38
+ additional_switches = '--allow-unauthenticated' if platform_key == :debian
39
+ host.install_package(package, additional_switches)
40
+ end
41
+ end
42
+ end
43
+ else
44
+ raise("Unknown platform '#{platform_key}' in package_hash")
45
+ end
46
+ end
47
+ end
48
+ return true
49
+ end
50
+ module_function :install_packages_on
51
+
52
+ def install_repos_on(host, project, sha, repo_configs_dir)
53
+ platform = host['platform'].with_version_codename
54
+ platform_configs_dir = File.join(repo_configs_dir,platform)
55
+ tld = sha == 'nightly' ? 'nightlies.puppetlabs.com' : 'builds.puppetlabs.lan'
56
+ project = sha == 'nightly' ? project + '-latest' : project
57
+ sha = sha == 'nightly' ? nil : sha
58
+
59
+ case platform
60
+ when /^(fedora|el|centos)-(\d+)-(.+)$/
61
+ variant = (($1 == 'centos') ? 'el' : $1)
62
+ fedora_prefix = ((variant == 'fedora') ? 'f' : '')
63
+ version = $2
64
+ arch = $3
65
+
66
+ repo_filename = "pl-%s%s-%s-%s%s-%s.repo" % [
67
+ project,
68
+ sha ? '-' + sha : '',
69
+ variant,
70
+ fedora_prefix,
71
+ version,
72
+ arch
73
+ ]
74
+ repo_url = "http://%s/%s/%s/repo_configs/rpm/%s" % [tld, project, sha, repo_filename]
75
+
76
+ on host, "curl --silent --show-error --output /etc/yum.repos.d/#{repo_filename} #{repo_url}"
77
+ when /^(debian|ubuntu)-([^-]+)-(.+)$/
78
+ variant = $1
79
+ version = $2
80
+ arch = $3
81
+
82
+ list_filename = "pl-%s%s-%s.list" % [
83
+ project,
84
+ sha ? '-' + sha : '',
85
+ version
86
+ ]
87
+ list_url = "http://%s/%s/%s/repo_configs/deb/%s" % [tld, project, sha, list_filename]
88
+
89
+ on host, "curl --silent --show-error --output /etc/apt/sources.list.d/#{list_filename} #{list_url}"
90
+ on host, "apt-get update"
91
+ else
92
+ if project == 'puppet-agent'
93
+ opts = {
94
+ :puppet_collection => 'PC1',
95
+ :puppet_agent_sha => ENV['SHA'],
96
+ :puppet_agent_version => ENV['SUITE_VERSION'] || ENV['SHA']
97
+ }
98
+ # this installs puppet-agent on windows (msi) and osx (dmg)
99
+ install_puppet_agent_dev_repo_on(agent, opts)
100
+ else
101
+ fail_test("No repository installation step for #{platform} yet...")
102
+ end
103
+ end
104
+ end
105
+ module_function :install_repos_on
106
+
107
+ end
108
+ end
@@ -0,0 +1,21 @@
1
+ require 'acceptance/install'
2
+ extend Acceptance::Install
3
+
4
+ step 'Install facter' do
5
+ sha = ENV['SHA'] || ENV['GIT_SHA'] || 'nightly'
6
+ hosts.each do |host|
7
+ install_repos_on(host, 'puppet-agent', sha, 'repo-configs')
8
+ host.add_env_var('PATH', '/opt/puppetlabs/bin:${PATH}')
9
+ end
10
+
11
+ PACKAGES = {
12
+ :redhat => [
13
+ 'puppet-agent',
14
+ ],
15
+ :debian => [
16
+ 'puppet-agent',
17
+ ],
18
+ }
19
+
20
+ install_packages_on(hosts, PACKAGES)
21
+ end
@@ -0,0 +1,39 @@
1
+ test_name 'fact_on, fact, facter, oh my' do
2
+ facts_expected = [/timezone => \w\w\w/, /facterversion => \d+\.\d+\.\d+/]
3
+ agents.each do |agent|
4
+ on(agent, 'ln -s /opt/puppetlabs/bin/facter /opt/puppetlabs/bin/cfacter')
5
+ step "facter should return all facts on #{agent}" do
6
+ facts = on(agent,facter).stdout
7
+ facts_expected.each do |expected|
8
+ assert_match(expected, facts, 'expected fact not found using facter helper')
9
+ end
10
+ end
11
+
12
+ step "cfacter should return all facts on #{agent}" do
13
+ facts = on(agent,facter).stdout
14
+ facts_expected.each do |expected|
15
+ assert_match(expected, facts, 'expected fact not found using facter helper')
16
+ end
17
+ end
18
+
19
+ step "fact_on should work on a single host" do
20
+ fact = fact_on(agent,'osfamily')
21
+ assert_match(/\w+/, fact, 'osfamily fact not found using fact_on helper')
22
+ end
23
+ end
24
+
25
+
26
+ step "fact_on should work on a multiple hosts and return an array" do
27
+ facts = fact_on(hosts,'hostname')
28
+ assert(facts.kind_of?(Array), 'fact_on did not return an array when given an array')
29
+ facts.each.with_index do |fact,index|
30
+ assert_equal(hosts[index].hostname.split('.').first, fact, 'hostname fact not found using fact helper on multiple hosts')
31
+ end
32
+ end
33
+
34
+ step "fact should work on default host" do
35
+ fact = fact('ipaddress')
36
+ assert_equal(default.ip, fact, 'ipaddress fact not found using fact helper on default host')
37
+ end
38
+
39
+ end
@@ -0,0 +1,35 @@
1
+ # -*- encoding: utf-8 -*-
2
+ $LOAD_PATH.unshift File.expand_path("../lib", __FILE__)
3
+ require 'beaker-facter/version'
4
+
5
+ Gem::Specification.new do |s|
6
+ s.name = "beaker-facter"
7
+ s.version = Beaker::DSL::Helpers::Facter::Version::STRING
8
+ s.authors = ["Puppetlabs"]
9
+ s.email = ["qe-team@puppetlabs.com"]
10
+ s.homepage = "https://github.com/puppetlabs/beaker-facter"
11
+ s.summary = %q{Beaker DSL Extension Facter Helpers}
12
+ s.description = %q{For use with the Beaker acceptance testing tool}
13
+ s.license = 'Apache2'
14
+
15
+ s.files = `git ls-files`.split("\n")
16
+ s.test_files = `git ls-files -- {test,spec,features}/*`.split("\n")
17
+ s.executables = `git ls-files -- bin/*`.split("\n").map{ |f| File.basename(f) }
18
+ s.require_paths = ["lib"]
19
+
20
+ # Testing dependencies
21
+ s.add_development_dependency 'rspec', '~> 3.0'
22
+ s.add_development_dependency 'rspec-its'
23
+ s.add_development_dependency 'fakefs', '~> 0.6'
24
+ s.add_development_dependency 'rake', '~> 10.1'
25
+ s.add_development_dependency 'simplecov'
26
+ s.add_development_dependency 'pry', '~> 0.10'
27
+
28
+ # Documentation dependencies
29
+ s.add_development_dependency 'yard'
30
+ s.add_development_dependency 'markdown'
31
+ s.add_development_dependency 'thin'
32
+
33
+ # Run time dependencies
34
+ s.add_runtime_dependency 'stringify-hash', '~> 0.0.0'
35
+ end
data/bin/beaker-facter ADDED
@@ -0,0 +1,36 @@
1
+ #!/usr/bin/env ruby
2
+
3
+ require 'rubygems' unless defined?(Gem)
4
+ require 'beaker-facter'
5
+
6
+ VERSION_STRING =
7
+ "
8
+ ,'````',
9
+ .`',.' `..'``', ____
10
+ ,' `', .-' '-.
11
+ ; ..' /) '. ,'``', ,.., ,..,
12
+ ', ,., .'`/ (/ \ ,.' '` '.' ;
13
+ '.,.`'.,,,.` `''` ; ; ,` ',
14
+ | | ; ;
15
+ beaker-facter ___| | `., ; ,... ,'
16
+ .-' ; ; `.,.' `,.' `,. ;
17
+ .' /) \ / `.,,'
18
+ / (/ '. .' '.
19
+ ; '-._ _.-' \
20
+ | |)( ;
21
+ | (__) |
22
+ ; ;/ |
23
+ \ // ;
24
+ '. .'/ /
25
+ '-._ _.-' /. .'
26
+ )( / '-._ _.-'
27
+ (__) / )(
28
+ | / (__)
29
+ | / _,'
30
+ | / _,'
31
+ |/._,'
32
+ "
33
+
34
+ puts VERSION_STRING % [Beaker::DSL::Helpers::Hiera::Version::STRING]
35
+
36
+ exit 0
@@ -0,0 +1,88 @@
1
+ module Beaker
2
+ module DSL
3
+ # These are wrappers to equivalent {Beaker::Command} objects
4
+ # so that command line actions are executed within an appropriate and
5
+ # configurable environment.
6
+ #
7
+ # I find most of these adapters of suspicious value and have deprecated
8
+ # many of them.
9
+ module Wrappers
10
+
11
+ # This is hairy and because of legacy code it will take a bit more
12
+ # work to disentangle all of the things that are being passed into
13
+ # this catchall param.
14
+ #
15
+ def facter(*args)
16
+ options = args.last.is_a?(Hash) ? args.pop : {}
17
+ options['ENV'] ||= {}
18
+ options[:cmdexe] = true
19
+ Command.new('facter', args, options )
20
+ end
21
+
22
+ # This is hairy and because of legacy code it will take a bit more
23
+ # work to disentangle all of the things that are being passed into
24
+ # this catchall param.
25
+ #
26
+ def cfacter(*args)
27
+ options = args.last.is_a?(Hash) ? args.pop : {}
28
+ options['ENV'] ||= {}
29
+ options[:cmdexe] = true
30
+ Command.new('cfacter', args, options )
31
+ end
32
+
33
+ end
34
+
35
+
36
+ module Helpers
37
+ # Methods that help you interact with your facter installation, facter must be installed
38
+ # for these methods to execute correctly
39
+ module Facter
40
+
41
+ # @!macro [new] common_opts
42
+ # @param [Hash{Symbol=>String}] opts Options to alter execution.
43
+ # @option opts [Boolean] :silent (false) Do not produce log output
44
+ # @option opts [Array<Fixnum>] :acceptable_exit_codes ([0]) An array
45
+ # (or range) of integer exit codes that should be considered
46
+ # acceptable. An error will be thrown if the exit code does not
47
+ # match one of the values in this list.
48
+ # @option opts [Boolean] :accept_all_exit_codes (false) Consider all
49
+ # exit codes as passing.
50
+ # @option opts [Boolean] :dry_run (false) Do not actually execute any
51
+ # commands on the SUT
52
+ # @option opts [String] :stdin (nil) Input to be provided during command
53
+ # execution on the SUT.
54
+ # @option opts [Boolean] :pty (false) Execute this command in a pseudoterminal.
55
+ # @option opts [Boolean] :expect_connection_failure (false) Expect this command
56
+ # to result in a connection failure, reconnect and continue execution.
57
+ # @option opts [Hash{String=>String}] :environment ({}) These will be
58
+ # treated as extra environment variables that should be set before
59
+ # running the command.
60
+ #
61
+
62
+ # Get a facter fact from a provided host
63
+ #
64
+ # @param [Host, Array<Host>, String, Symbol] host One or more hosts to act upon,
65
+ # or a role (String or Symbol) that identifies one or more hosts.
66
+ # @param [String] name The name of the fact to query for
67
+ # @!macro common_opts
68
+ #
69
+ # @return String The value of the fact 'name' on the provided host
70
+ # @raise [FailTest] Raises an exception if call to facter fails
71
+ def fact_on(host, name, opts = {})
72
+ result = on host, facter(name, opts)
73
+ if result.kind_of?(Array)
74
+ result.map { |res| res.stdout.chomp }
75
+ else
76
+ result.stdout.chomp
77
+ end
78
+ end
79
+
80
+ # Get a facter fact from the default host
81
+ # @see #fact_on
82
+ def fact(name, opts = {})
83
+ fact_on(default, name, opts)
84
+ end
85
+ end
86
+ end
87
+ end
88
+ end
@@ -0,0 +1,11 @@
1
+ module Beaker
2
+ module DSL
3
+ module Helpers
4
+ module Facter
5
+ module Version
6
+ STRING = '0.1.0'
7
+ end
8
+ end
9
+ end
10
+ end
11
+ end
@@ -0,0 +1,11 @@
1
+ module Beaker
2
+ module DSL
3
+ module Helpers
4
+ module Facter
5
+ require 'stringify-hash'
6
+ require 'beaker-facter/helpers'
7
+ require 'beaker-facter/version'
8
+ end
9
+ end
10
+ end
11
+ end
@@ -0,0 +1,79 @@
1
+ require 'spec_helper'
2
+
3
+ class ClassMixedWithDSLHelpers
4
+ include BeakerTestHelpers
5
+ include Beaker::DSL::Helpers::Facter
6
+ include Beaker::DSL::Wrappers
7
+
8
+ def logger
9
+ RSpec::Mocks::Double.new('logger').as_null_object
10
+ end
11
+ end
12
+
13
+ describe ClassMixedWithDSLHelpers do
14
+
15
+ let( :command ){ 'ls' }
16
+ let( :beaker_command ) { class_double("Beaker::DSL::Wrappers::Command").as_stubbed_const(:transfer_nested_constants => true) }
17
+ let( :host ) { double.as_null_object }
18
+ let( :result ) { BeakerTestHelpers::Result.new( host, command ) }
19
+
20
+ let( :master ) { make_host( 'master', :roles => %w( master agent default) ) }
21
+ let( :agent ) { make_host( 'agent', :roles => %w( agent ) ) }
22
+ let( :dash ) { make_host( 'console', :roles => %w( dashboard agent ) ) }
23
+ let( :db ) { make_host( 'db', :roles => %w( database agent ) ) }
24
+ let( :custom ) { make_host( 'custom', :roles => %w( custom agent ) ) }
25
+ let( :hosts ) { [ master, agent, dash, db, custom ] }
26
+ let(:empty_opts) { {'ENV' => {}, :cmdexe => true } }
27
+
28
+ describe '#facter' do
29
+ it 'should split out the options and pass "facter" as first arg to Command' do
30
+ expect( beaker_command ).to receive( :new ).
31
+ with('facter', [ '-p' ], empty_opts)
32
+ subject.facter( '-p' )
33
+ end
34
+ end
35
+
36
+ describe '#cfacter' do
37
+ it 'should split out the options and pass "cfacter" as first arg to Command' do
38
+ expect( beaker_command ).to receive( :new ).
39
+ with('cfacter', [ '-p' ], empty_opts)
40
+ subject.cfacter( '-p' )
41
+ end
42
+ end
43
+
44
+ describe '#fact_on' do
45
+ it 'retrieves a fact on a single host' do
46
+ result.stdout = "family\n"
47
+ expect( subject ).to receive(:facter).with('osfamily',{}).once
48
+ expect( subject ).to receive(:on).and_return(result)
49
+
50
+ expect( subject.fact_on('host','osfamily') ).to be === result.stdout.chomp
51
+ end
52
+
53
+ it 'retrieves an array of facts from multiple hosts' do
54
+ allow( subject ).to receive( :hosts ).and_return( hosts )
55
+ times = hosts.length
56
+ results = []
57
+ hosts.each.with_index do |host, index|
58
+ results[index] = BeakerTestHelpers::Result.new( host, command )
59
+ results[index].stdout = "family\n"
60
+ end
61
+ expected = ["family"] * hosts.length
62
+ expect( subject ).to receive(:facter).with('osfamily',{}).once
63
+ expect( subject ).to receive(:on).and_return(results)
64
+
65
+ expect( subject.fact_on(hosts,'osfamily') ).to be === expected
66
+ end
67
+ end
68
+
69
+ describe '#fact' do
70
+ it 'delegates to #fact_on with the default host' do
71
+ allow( subject ).to receive(:hosts).and_return(hosts)
72
+ allow( subject ).to receive( :default ).and_return( hosts[0] )
73
+ expect( subject ).to receive(:fact_on).with(master,"osfamily",{}).once
74
+
75
+ subject.fact('osfamily')
76
+ end
77
+ end
78
+
79
+ end
@@ -0,0 +1,29 @@
1
+ # These are specifically to mock Beaker methods necessary for testing
2
+ # that will be available during runtime because this is never run separate
3
+ # from Beaker itself.
4
+ #
5
+ # Including Beaker as a dependency would not work as a solution to this issue,
6
+ # since that would make a cycle in the dependency graph, at least until
7
+ # Beaker 3.0 happens and this is no longer a dependency of Beaker's.
8
+ module BeakerTestHelpers
9
+
10
+ class Result
11
+ attr_accessor :host, :cmd, :exit_code, :stdout, :stderr, :output,
12
+ :raw_stdout, :raw_stderr, :raw_output
13
+ def initialize(host, cmd)
14
+ @host = host
15
+ @cmd = cmd
16
+ @stdout = ''
17
+ @stderr = ''
18
+ @output = ''
19
+ @exit_code = nil
20
+ end
21
+ end
22
+
23
+ def block_on hosts
24
+ if block_given?
25
+ yield hosts
26
+ end
27
+ end
28
+
29
+ end
data/spec/helpers.rb ADDED
@@ -0,0 +1,111 @@
1
+ module TestFileHelpers
2
+ def create_files file_array
3
+ file_array.each do |f|
4
+ FileUtils.mkdir_p File.dirname(f)
5
+ FileUtils.touch f
6
+ end
7
+ end
8
+
9
+ def fog_file_contents
10
+ { :default => { :aws_access_key_id => "IMANACCESSKEY",
11
+ :aws_secret_access_key => "supersekritkey",
12
+ :aix_hypervisor_server => "aix_hypervisor.labs.net",
13
+ :aix_hypervisor_username => "aixer",
14
+ :aix_hypervisor_keyfile => "/Users/user/.ssh/id_rsa-acceptance",
15
+ :solaris_hypervisor_server => "solaris_hypervisor.labs.net",
16
+ :solaris_hypervisor_username => "harness",
17
+ :solaris_hypervisor_keyfile => "/Users/user/.ssh/id_rsa-old.private",
18
+ :solaris_hypervisor_vmpath => "rpoooool/zs",
19
+ :solaris_hypervisor_snappaths => ["rpoooool/USER/z0"],
20
+ :vsphere_server => "vsphere.labs.net",
21
+ :vsphere_username => "vsphere@labs.com",
22
+ :vsphere_password => "supersekritpassword",
23
+ } }
24
+ end
25
+
26
+ end
27
+
28
+ module HostHelpers
29
+ HOST_DEFAULTS = { :platform => 'unix',
30
+ :snapshot => 'pe',
31
+ :box => 'box_name',
32
+ :roles => ['agent'],
33
+ :snapshot => 'snap',
34
+ :ip => 'default.ip.address',
35
+ :box => 'default_box_name',
36
+ :box_url => 'http://default.box.url',
37
+ }
38
+
39
+ HOST_NAME = "vm%d"
40
+ HOST_SNAPSHOT = "snapshot%d"
41
+ HOST_IP = "ip.address.for.%s"
42
+ HOST_BOX = "%s_of_my_box"
43
+ HOST_BOX_URL = "http://address.for.my.box.%s"
44
+ HOST_TEMPLATE = "%s_has_a_template"
45
+
46
+ def logger
47
+ double( 'logger' ).as_null_object
48
+ end
49
+
50
+ def make_opts
51
+ opts = StringifyHash.new
52
+ opts.merge( { :logger => logger,
53
+ :host_config => 'sample.config',
54
+ :type => nil,
55
+ :pooling_api => 'http://vcloud.delivery.puppetlabs.net/',
56
+ :datastore => 'instance0',
57
+ :folder => 'Delivery/Quality Assurance/Staging/Dynamic',
58
+ :resourcepool => 'delivery/Quality Assurance/Staging/Dynamic',
59
+ :gce_project => 'beaker-compute',
60
+ :gce_keyfile => '/path/to/keyfile.p12',
61
+ :gce_password => 'notasecret',
62
+ :gce_email => '12345678910@developer.gserviceaccount.com',
63
+ } )
64
+ end
65
+
66
+ def generate_result (name, opts )
67
+ result = double( 'result' )
68
+ stdout = opts.has_key?(:stdout) ? opts[:stdout] : name
69
+ stderr = opts.has_key?(:stderr) ? opts[:stderr] : name
70
+ exit_code = opts.has_key?(:exit_code) ? opts[:exit_code] : 0
71
+ exit_code = [exit_code].flatten
72
+ allow( result ).to receive( :stdout ).and_return( stdout )
73
+ allow( result ).to receive( :stderr ).and_return( stderr )
74
+ allow( result ).to receive( :exit_code ).and_return( *exit_code )
75
+ result
76
+ end
77
+
78
+ def make_host_opts name, opts
79
+ make_opts.merge( { 'HOSTS' => { name => opts } } ).merge( opts )
80
+ end
81
+
82
+ def make_host name, host_hash
83
+ host_hash = StringifyHash.new.merge(HOST_DEFAULTS.merge(host_hash))
84
+
85
+ host = make_opts.merge(host_hash)
86
+
87
+ allow(host).to receive( :name ).and_return( name )
88
+ allow(host).to receive( :to_s ).and_return( name )
89
+ allow(host).to receive( :exec ).and_return( generate_result( name, host_hash ) )
90
+ host
91
+ end
92
+
93
+ def make_hosts preset_opts = {}, amt = 3
94
+ hosts = []
95
+ (1..amt).each do |num|
96
+ name = HOST_NAME % num
97
+ opts = { :snapshot => HOST_SNAPSHOT % num,
98
+ :ip => HOST_IP % name,
99
+ :template => HOST_TEMPLATE % name,
100
+ :box => HOST_BOX % name,
101
+ :box_url => HOST_BOX_URL % name }.merge( preset_opts )
102
+ hosts << make_host(name, opts)
103
+ end
104
+ hosts
105
+ end
106
+
107
+ def make_instance instance_data = {}
108
+ OpenStruct.new instance_data
109
+ end
110
+
111
+ end
@@ -0,0 +1,17 @@
1
+ if ENV["COVERAGE"]
2
+ require 'simplecov'
3
+ SimpleCov.start do
4
+ add_filter '/spec/'
5
+ add_filter '.bundle/gems'
6
+ end
7
+ end
8
+ require 'beaker-facter'
9
+ require 'beaker_test_helpers'
10
+ require 'helpers'
11
+
12
+ require 'rspec/its'
13
+
14
+ RSpec.configure do |config|
15
+ config.include TestFileHelpers
16
+ config.include HostHelpers
17
+ end
metadata ADDED
@@ -0,0 +1,207 @@
1
+ --- !ruby/object:Gem::Specification
2
+ name: beaker-facter
3
+ version: !ruby/object:Gem::Version
4
+ version: 0.1.0
5
+ platform: ruby
6
+ authors:
7
+ - Puppetlabs
8
+ autorequire:
9
+ bindir: bin
10
+ cert_chain: []
11
+ date: 2016-01-29 00:00:00.000000000 Z
12
+ dependencies:
13
+ - !ruby/object:Gem::Dependency
14
+ name: rspec
15
+ requirement: !ruby/object:Gem::Requirement
16
+ requirements:
17
+ - - ~>
18
+ - !ruby/object:Gem::Version
19
+ version: '3.0'
20
+ type: :development
21
+ prerelease: false
22
+ version_requirements: !ruby/object:Gem::Requirement
23
+ requirements:
24
+ - - ~>
25
+ - !ruby/object:Gem::Version
26
+ version: '3.0'
27
+ - !ruby/object:Gem::Dependency
28
+ name: rspec-its
29
+ requirement: !ruby/object:Gem::Requirement
30
+ requirements:
31
+ - - ! '>='
32
+ - !ruby/object:Gem::Version
33
+ version: '0'
34
+ type: :development
35
+ prerelease: false
36
+ version_requirements: !ruby/object:Gem::Requirement
37
+ requirements:
38
+ - - ! '>='
39
+ - !ruby/object:Gem::Version
40
+ version: '0'
41
+ - !ruby/object:Gem::Dependency
42
+ name: fakefs
43
+ requirement: !ruby/object:Gem::Requirement
44
+ requirements:
45
+ - - ~>
46
+ - !ruby/object:Gem::Version
47
+ version: '0.6'
48
+ type: :development
49
+ prerelease: false
50
+ version_requirements: !ruby/object:Gem::Requirement
51
+ requirements:
52
+ - - ~>
53
+ - !ruby/object:Gem::Version
54
+ version: '0.6'
55
+ - !ruby/object:Gem::Dependency
56
+ name: rake
57
+ requirement: !ruby/object:Gem::Requirement
58
+ requirements:
59
+ - - ~>
60
+ - !ruby/object:Gem::Version
61
+ version: '10.1'
62
+ type: :development
63
+ prerelease: false
64
+ version_requirements: !ruby/object:Gem::Requirement
65
+ requirements:
66
+ - - ~>
67
+ - !ruby/object:Gem::Version
68
+ version: '10.1'
69
+ - !ruby/object:Gem::Dependency
70
+ name: simplecov
71
+ requirement: !ruby/object:Gem::Requirement
72
+ requirements:
73
+ - - ! '>='
74
+ - !ruby/object:Gem::Version
75
+ version: '0'
76
+ type: :development
77
+ prerelease: false
78
+ version_requirements: !ruby/object:Gem::Requirement
79
+ requirements:
80
+ - - ! '>='
81
+ - !ruby/object:Gem::Version
82
+ version: '0'
83
+ - !ruby/object:Gem::Dependency
84
+ name: pry
85
+ requirement: !ruby/object:Gem::Requirement
86
+ requirements:
87
+ - - ~>
88
+ - !ruby/object:Gem::Version
89
+ version: '0.10'
90
+ type: :development
91
+ prerelease: false
92
+ version_requirements: !ruby/object:Gem::Requirement
93
+ requirements:
94
+ - - ~>
95
+ - !ruby/object:Gem::Version
96
+ version: '0.10'
97
+ - !ruby/object:Gem::Dependency
98
+ name: yard
99
+ requirement: !ruby/object:Gem::Requirement
100
+ requirements:
101
+ - - ! '>='
102
+ - !ruby/object:Gem::Version
103
+ version: '0'
104
+ type: :development
105
+ prerelease: false
106
+ version_requirements: !ruby/object:Gem::Requirement
107
+ requirements:
108
+ - - ! '>='
109
+ - !ruby/object:Gem::Version
110
+ version: '0'
111
+ - !ruby/object:Gem::Dependency
112
+ name: markdown
113
+ requirement: !ruby/object:Gem::Requirement
114
+ requirements:
115
+ - - ! '>='
116
+ - !ruby/object:Gem::Version
117
+ version: '0'
118
+ type: :development
119
+ prerelease: false
120
+ version_requirements: !ruby/object:Gem::Requirement
121
+ requirements:
122
+ - - ! '>='
123
+ - !ruby/object:Gem::Version
124
+ version: '0'
125
+ - !ruby/object:Gem::Dependency
126
+ name: thin
127
+ requirement: !ruby/object:Gem::Requirement
128
+ requirements:
129
+ - - ! '>='
130
+ - !ruby/object:Gem::Version
131
+ version: '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'
139
+ - !ruby/object:Gem::Dependency
140
+ name: stringify-hash
141
+ requirement: !ruby/object:Gem::Requirement
142
+ requirements:
143
+ - - ~>
144
+ - !ruby/object:Gem::Version
145
+ version: 0.0.0
146
+ type: :runtime
147
+ prerelease: false
148
+ version_requirements: !ruby/object:Gem::Requirement
149
+ requirements:
150
+ - - ~>
151
+ - !ruby/object:Gem::Version
152
+ version: 0.0.0
153
+ description: For use with the Beaker acceptance testing tool
154
+ email:
155
+ - qe-team@puppetlabs.com
156
+ executables:
157
+ - beaker-facter
158
+ extensions: []
159
+ extra_rdoc_files: []
160
+ files:
161
+ - .gitignore
162
+ - .rspec
163
+ - .simplecov
164
+ - Gemfile
165
+ - HISTORY.md
166
+ - LICENSE
167
+ - README.md
168
+ - Rakefile
169
+ - acceptance/config/nodes/vagrant-ubuntu-1404.yml
170
+ - acceptance/lib/acceptance/install.rb
171
+ - acceptance/pre-suite/010_install.rb
172
+ - acceptance/tests/facter.rb
173
+ - beaker-facter.gemspec
174
+ - bin/beaker-facter
175
+ - lib/beaker-facter.rb
176
+ - lib/beaker-facter/helpers.rb
177
+ - lib/beaker-facter/version.rb
178
+ - spec/beaker-facter/helpers_spec.rb
179
+ - spec/beaker_test_helpers.rb
180
+ - spec/helpers.rb
181
+ - spec/spec_helper.rb
182
+ homepage: https://github.com/puppetlabs/beaker-facter
183
+ licenses:
184
+ - Apache2
185
+ metadata: {}
186
+ post_install_message:
187
+ rdoc_options: []
188
+ require_paths:
189
+ - lib
190
+ required_ruby_version: !ruby/object:Gem::Requirement
191
+ requirements:
192
+ - - ! '>='
193
+ - !ruby/object:Gem::Version
194
+ version: '0'
195
+ required_rubygems_version: !ruby/object:Gem::Requirement
196
+ requirements:
197
+ - - ! '>='
198
+ - !ruby/object:Gem::Version
199
+ version: '0'
200
+ requirements: []
201
+ rubyforge_project:
202
+ rubygems_version: 2.4.6
203
+ signing_key:
204
+ specification_version: 4
205
+ summary: Beaker DSL Extension Facter Helpers
206
+ test_files: []
207
+ has_rdoc: