r10k 3.4.0 → 3.6.0
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.
- checksums.yaml +4 -4
- data/.github/pull_request_template.md +4 -0
- data/.github/workflows/docker.yml +25 -1
- data/.github/workflows/release.yml +36 -0
- data/.travis.yml +21 -8
- data/CHANGELOG.mkd +64 -4
- data/CODEOWNERS +1 -1
- data/Gemfile +1 -1
- data/README.mkd +13 -4
- data/azure-pipelines.yml +4 -2
- data/doc/dynamic-environments/configuration.mkd +41 -4
- data/doc/dynamic-environments/git-environments.mkd +1 -1
- data/doc/dynamic-environments/master-configuration.mkd +28 -58
- data/doc/faq.mkd +6 -1
- data/doc/puppetfile.mkd +2 -0
- data/docker/Makefile +19 -3
- data/docker/r10k/Dockerfile +22 -8
- data/docker/r10k/release.Dockerfile +23 -4
- data/integration/Rakefile +2 -2
- data/integration/tests/git_source/git_source_repeated_remote.rb +68 -0
- data/lib/r10k/action/deploy/environment.rb +5 -1
- data/lib/r10k/action/deploy/module.rb +5 -1
- data/lib/r10k/action/runner.rb +4 -4
- data/lib/r10k/cli/deploy.rb +1 -1
- data/lib/r10k/forge/module_release.rb +2 -2
- data/lib/r10k/git/cache.rb +1 -3
- data/lib/r10k/git/stateful_repository.rb +4 -0
- data/lib/r10k/module/base.rb +8 -0
- data/lib/r10k/module/git.rb +4 -0
- data/lib/r10k/puppetfile.rb +26 -6
- data/lib/r10k/settings.rb +1 -1
- data/lib/r10k/source.rb +1 -0
- data/lib/r10k/source/exec.rb +51 -0
- data/lib/r10k/source/git.rb +22 -2
- data/lib/r10k/source/hash.rb +32 -8
- data/lib/r10k/version.rb +4 -1
- data/locales/r10k.pot +33 -10
- data/r10k.gemspec +5 -1
- data/spec/unit/action/deploy/module_spec.rb +15 -2
- data/spec/unit/action/puppetfile/install_spec.rb +4 -1
- data/spec/unit/action/runner_spec.rb +2 -2
- data/spec/unit/forge/module_release_spec.rb +14 -10
- data/spec/unit/puppetfile_spec.rb +67 -2
- data/spec/unit/source/exec_spec.rb +81 -0
- data/spec/unit/source/git_spec.rb +37 -1
- data/spec/unit/source/hash_spec.rb +54 -0
- data/spec/unit/source/yaml_spec.rb +42 -0
- metadata +54 -16
- data/integration/scripts/README.mkd +0 -86
- data/integration/scripts/setup_r10k_env_centos5.sh +0 -23
- data/integration/scripts/setup_r10k_env_centos6.sh +0 -23
- data/integration/scripts/setup_r10k_env_rhel7.sh +0 -23
- data/integration/scripts/setup_r10k_env_sles11.sh +0 -23
- data/integration/scripts/setup_r10k_env_sles12.sh +0 -23
- data/integration/scripts/setup_r10k_env_ubuntu1004.sh +0 -23
- data/integration/scripts/setup_r10k_env_ubuntu1204.sh +0 -23
- data/integration/scripts/setup_r10k_env_ubuntu1404.sh +0 -23
data/lib/r10k/settings.rb
CHANGED
@@ -160,7 +160,7 @@ module R10K
|
|
160
160
|
|
161
161
|
Definition.new(:pool_size, {
|
162
162
|
:desc => "The amount of threads used to concurrently install modules. The default value is 1: install one module at a time.",
|
163
|
-
:default =>
|
163
|
+
:default => 4,
|
164
164
|
:validate => lambda do |value|
|
165
165
|
if !value.is_a?(Integer)
|
166
166
|
raise ArgumentError, "The pool_size setting should be an integer, not a #{value.class}"
|
data/lib/r10k/source.rb
CHANGED
@@ -0,0 +1,51 @@
|
|
1
|
+
require 'r10k/util/subprocess'
|
2
|
+
require 'json'
|
3
|
+
require 'yaml'
|
4
|
+
|
5
|
+
class R10K::Source::Exec < R10K::Source::Hash
|
6
|
+
R10K::Source.register(:exec, self)
|
7
|
+
|
8
|
+
def initialize(name, basedir, options = {})
|
9
|
+
unless @command = options[:command]
|
10
|
+
raise ConfigError, _('Environment source %{name} missing required parameter: command') % {name: name}
|
11
|
+
end
|
12
|
+
|
13
|
+
# We haven't set the environments option yet. We will do that by
|
14
|
+
# overloading the #environments method.
|
15
|
+
super(name, basedir, options)
|
16
|
+
end
|
17
|
+
|
18
|
+
def environments_hash
|
19
|
+
@environments_hash ||= set_environments_hash(run_environments_command)
|
20
|
+
end
|
21
|
+
|
22
|
+
private
|
23
|
+
|
24
|
+
def run_environments_command
|
25
|
+
subproc = R10K::Util::Subprocess.new([@command])
|
26
|
+
subproc.raise_on_fail = true
|
27
|
+
subproc.logger = self.logger
|
28
|
+
procresult = subproc.execute
|
29
|
+
|
30
|
+
begin
|
31
|
+
environments = JSON.parse(procresult.stdout)
|
32
|
+
rescue JSON::ParserError => json_err
|
33
|
+
begin
|
34
|
+
environments = YAML.safe_load(procresult.stdout)
|
35
|
+
rescue Psych::SyntaxError => yaml_err
|
36
|
+
raise R10K::Error, _("Error parsing command output for exec source %{name}:\n" \
|
37
|
+
"Not valid JSON: %{j_msg}\n" \
|
38
|
+
"Not valid YAML: %{y_msg}\n" \
|
39
|
+
"Stdout:\n%{out}") % {name: name, j_msg: json_err.message, y_msg: yaml_err.message, out: procresult.stdout}
|
40
|
+
end
|
41
|
+
end
|
42
|
+
|
43
|
+
unless R10K::Source::Hash.valid_environments_hash?(environments)
|
44
|
+
raise R10K::Error, _("Environment source %{name} command %{cmd} did not return valid environment data.\n" \
|
45
|
+
'Returned: %{data}') % {name: name, cmd: @command, data: environments}
|
46
|
+
end
|
47
|
+
|
48
|
+
# Return the resulting environments hash
|
49
|
+
environments
|
50
|
+
end
|
51
|
+
end
|
data/lib/r10k/source/git.rb
CHANGED
@@ -41,6 +41,10 @@ class R10K::Source::Git < R10K::Source::Base
|
|
41
41
|
# that will be deployed as environments.
|
42
42
|
attr_reader :ignore_branch_prefixes
|
43
43
|
|
44
|
+
# @!attribute [r] filter_command
|
45
|
+
# @return [String] Command to run to filter branches
|
46
|
+
attr_reader :filter_command
|
47
|
+
|
44
48
|
# Initialize the given source.
|
45
49
|
#
|
46
50
|
# @param name [String] The identifier for this source.
|
@@ -61,6 +65,7 @@ class R10K::Source::Git < R10K::Source::Base
|
|
61
65
|
@remote = options[:remote]
|
62
66
|
@invalid_branches = (options[:invalid_branches] || 'correct_and_warn')
|
63
67
|
@ignore_branch_prefixes = options[:ignore_branch_prefixes]
|
68
|
+
@filter_command = options[:filter_command]
|
64
69
|
|
65
70
|
@cache = R10K::Git.cache.generate(@remote)
|
66
71
|
end
|
@@ -115,7 +120,7 @@ class R10K::Source::Git < R10K::Source::Base
|
|
115
120
|
environments.map {|env| env.dirname }
|
116
121
|
end
|
117
122
|
|
118
|
-
def
|
123
|
+
def filter_branches_by_regexp(branches, ignore_prefixes)
|
119
124
|
filter = Regexp.new("^#{Regexp.union(ignore_prefixes)}")
|
120
125
|
branches = branches.reject do |branch|
|
121
126
|
result = filter.match(branch)
|
@@ -127,14 +132,29 @@ class R10K::Source::Git < R10K::Source::Base
|
|
127
132
|
branches
|
128
133
|
end
|
129
134
|
|
135
|
+
def filter_branches_by_command(branches, command)
|
136
|
+
branches.select do |branch|
|
137
|
+
result = system({'GIT_DIR' => @cache.git_dir.to_s, 'R10K_BRANCH' => branch, 'R10K_NAME' => @name.to_s}, command)
|
138
|
+
unless result
|
139
|
+
logger.warn _("Branch `%{name}:%{branch}` filtered out by filter_command %{cmd}") % {name: @name, branch: branch, cmd: command}
|
140
|
+
end
|
141
|
+
result
|
142
|
+
end
|
143
|
+
end
|
144
|
+
|
130
145
|
private
|
131
146
|
|
132
147
|
def branch_names
|
133
148
|
opts = {:prefix => @prefix, :invalid => @invalid_branches, :source => @name}
|
134
149
|
branches = @cache.branches
|
135
150
|
if @ignore_branch_prefixes && !@ignore_branch_prefixes.empty?
|
136
|
-
branches =
|
151
|
+
branches = filter_branches_by_regexp(branches, @ignore_branch_prefixes)
|
152
|
+
end
|
153
|
+
|
154
|
+
if @filter_command && !@filter_command.empty?
|
155
|
+
branches = filter_branches_by_command(branches, @filter_command)
|
137
156
|
end
|
157
|
+
|
138
158
|
branches.map do |branch|
|
139
159
|
R10K::Environment::Name.new(branch, opts)
|
140
160
|
end
|
data/lib/r10k/source/hash.rb
CHANGED
@@ -120,6 +120,16 @@
|
|
120
120
|
#
|
121
121
|
class R10K::Source::Hash < R10K::Source::Base
|
122
122
|
|
123
|
+
include R10K::Logging
|
124
|
+
|
125
|
+
# @param hash [Hash] A hash to validate.
|
126
|
+
# @return [Boolean] False if the hash is obviously invalid. A true return
|
127
|
+
# means _maybe_ it's valid.
|
128
|
+
def self.valid_environments_hash?(hash)
|
129
|
+
# TODO: more robust schema valiation
|
130
|
+
hash.is_a?(Hash)
|
131
|
+
end
|
132
|
+
|
123
133
|
# @param name [String] The identifier for this source.
|
124
134
|
# @param basedir [String] The base directory where the generated environments will be created.
|
125
135
|
# @param options [Hash] An additional set of options for this source. The
|
@@ -131,19 +141,33 @@ class R10K::Source::Hash < R10K::Source::Base
|
|
131
141
|
# @option options [Hash] :environments The hash definition of environments
|
132
142
|
def initialize(name, basedir, options = {})
|
133
143
|
super(name, basedir, options)
|
144
|
+
end
|
134
145
|
|
135
|
-
|
136
|
-
|
137
|
-
|
138
|
-
|
139
|
-
|
140
|
-
|
141
|
-
|
146
|
+
# Set the environment hash for the source. The environment hash is what the
|
147
|
+
# source uses to generate enviroments.
|
148
|
+
# @param hash [Hash] The hash to sanitize and use as the source's environments.
|
149
|
+
# Should be formatted for use with R10K::Environment#from_hash.
|
150
|
+
def set_environments_hash(hash)
|
151
|
+
@environments_hash = hash.reduce({}) do |memo,(name,opts)|
|
152
|
+
R10K::Util::SymbolizeKeys.symbolize_keys!(opts)
|
153
|
+
memo.merge({
|
154
|
+
name => opts.merge({
|
155
|
+
:basedir => @basedir,
|
156
|
+
:dirname => R10K::Environment::Name.new(name, {prefix: @prefix, source: @name}).dirname
|
157
|
+
})
|
158
|
+
})
|
142
159
|
end
|
143
160
|
end
|
144
161
|
|
162
|
+
# Return the sanitized environments hash for this source. The environments
|
163
|
+
# hash should contain objects formatted for use with R10K::Environment#from_hash.
|
164
|
+
# If the hash does not exist it will be built based on @options.
|
165
|
+
def environments_hash
|
166
|
+
@environments_hash ||= set_environments_hash(@options.fetch(:environments, {}))
|
167
|
+
end
|
168
|
+
|
145
169
|
def environments
|
146
|
-
@environments ||=
|
170
|
+
@environments ||= environments_hash.map do |name, hash|
|
147
171
|
R10K::Environment.from_hash(name, hash)
|
148
172
|
end
|
149
173
|
end
|
data/lib/r10k/version.rb
CHANGED
@@ -1,3 +1,6 @@
|
|
1
1
|
module R10K
|
2
|
-
|
2
|
+
# When updating to a new major (X) or minor (Y) version, include `#major` or
|
3
|
+
# `#minor` (respectively) in your commit message to trigger the appropriate
|
4
|
+
# release. Otherwise, a new patch (Z) version will be released.
|
5
|
+
VERSION = '3.6.0'
|
3
6
|
end
|
data/locales/r10k.pot
CHANGED
@@ -1,16 +1,16 @@
|
|
1
1
|
# SOME DESCRIPTIVE TITLE.
|
2
|
-
# Copyright (C)
|
2
|
+
# Copyright (C) 2020 Puppet, Inc.
|
3
3
|
# This file is distributed under the same license as the r10k package.
|
4
|
-
# FIRST AUTHOR <EMAIL@ADDRESS>,
|
4
|
+
# FIRST AUTHOR <EMAIL@ADDRESS>, 2020.
|
5
5
|
#
|
6
6
|
#, fuzzy
|
7
7
|
msgid ""
|
8
8
|
msgstr ""
|
9
|
-
"Project-Id-Version: r10k 3.
|
9
|
+
"Project-Id-Version: r10k 3.4.1-57-g2eb088a\n"
|
10
10
|
"\n"
|
11
11
|
"Report-Msgid-Bugs-To: docs@puppetlabs.com\n"
|
12
|
-
"POT-Creation-Date:
|
13
|
-
"PO-Revision-Date:
|
12
|
+
"POT-Creation-Date: 2020-07-22 16:41+0000\n"
|
13
|
+
"PO-Revision-Date: 2020-07-22 16:41+0000\n"
|
14
14
|
"Last-Translator: FULL NAME <EMAIL@ADDRESS>\n"
|
15
15
|
"Language-Team: LANGUAGE <LL@li.org>\n"
|
16
16
|
"Language: \n"
|
@@ -423,26 +423,49 @@ msgstr ""
|
|
423
423
|
msgid "Setting %{name} requires a URL but '%{value}' could not be parsed as a URL"
|
424
424
|
msgstr ""
|
425
425
|
|
426
|
-
#: ../lib/r10k/source/
|
426
|
+
#: ../lib/r10k/source/exec.rb:10
|
427
|
+
msgid "Environment source %{name} missing required parameter: command"
|
428
|
+
msgstr ""
|
429
|
+
|
430
|
+
#: ../lib/r10k/source/exec.rb:36
|
431
|
+
msgid ""
|
432
|
+
"Error parsing command output for exec source %{name}:\n"
|
433
|
+
"Not valid JSON: %{j_msg}\n"
|
434
|
+
"Not valid YAML: %{y_msg}\n"
|
435
|
+
"Stdout:\n"
|
436
|
+
"%{out}"
|
437
|
+
msgstr ""
|
438
|
+
|
439
|
+
#: ../lib/r10k/source/exec.rb:44
|
440
|
+
msgid ""
|
441
|
+
"Environment source %{name} command %{cmd} did not return valid environment data.\n"
|
442
|
+
"Returned: %{data}"
|
443
|
+
msgstr ""
|
444
|
+
|
445
|
+
#: ../lib/r10k/source/git.rb:77
|
427
446
|
msgid "Fetching '%{remote}' to determine current branches."
|
428
447
|
msgstr ""
|
429
448
|
|
430
|
-
#: ../lib/r10k/source/git.rb:
|
449
|
+
#: ../lib/r10k/source/git.rb:80
|
431
450
|
msgid "Unable to determine current branches for Git source '%{name}' (%{basedir})"
|
432
451
|
msgstr ""
|
433
452
|
|
434
|
-
#: ../lib/r10k/source/git.rb:
|
453
|
+
#: ../lib/r10k/source/git.rb:105
|
435
454
|
msgid "Environment %{env_name} contained non-word characters, correcting name to %{corrected_env_name}"
|
436
455
|
msgstr ""
|
437
456
|
|
438
|
-
#: ../lib/r10k/source/git.rb:
|
457
|
+
#: ../lib/r10k/source/git.rb:109
|
439
458
|
msgid "Environment %{env_name} contained non-word characters, ignoring it."
|
440
459
|
msgstr ""
|
441
460
|
|
442
|
-
#: ../lib/r10k/source/git.rb:
|
461
|
+
#: ../lib/r10k/source/git.rb:128 ../lib/r10k/source/svn.rb:113
|
443
462
|
msgid "Branch %{branch} filtered out by ignore_branch_prefixes %{ibp}"
|
444
463
|
msgstr ""
|
445
464
|
|
465
|
+
#: ../lib/r10k/source/git.rb:139
|
466
|
+
msgid "Branch `%{name}:%{branch}` filtered out by filter_command %{cmd}"
|
467
|
+
msgstr ""
|
468
|
+
|
446
469
|
#: ../lib/r10k/source/yaml.rb:10
|
447
470
|
msgid "Couldn't open environments file %{file}: %{err}"
|
448
471
|
msgstr ""
|
data/r10k.gemspec
CHANGED
@@ -23,7 +23,7 @@ Gem::Specification.new do |s|
|
|
23
23
|
s.license = 'Apache-2.0'
|
24
24
|
|
25
25
|
s.add_dependency 'colored2', '3.1.2'
|
26
|
-
s.add_dependency 'cri',
|
26
|
+
s.add_dependency 'cri', ['>= 2.15.10', '< 3.0.0']
|
27
27
|
|
28
28
|
s.add_dependency 'log4r', '1.1.10'
|
29
29
|
s.add_dependency 'multi_json', '~> 1.10'
|
@@ -31,6 +31,10 @@ Gem::Specification.new do |s|
|
|
31
31
|
s.add_dependency 'puppet_forge', '~> 2.3.0'
|
32
32
|
|
33
33
|
s.add_dependency 'gettext-setup', '~>0.24'
|
34
|
+
# These two pins narrow what is allowed by gettext-setup,
|
35
|
+
# to preserver compatability with Ruby 2.4
|
36
|
+
s.add_dependency 'fast_gettext', '~> 1.1.0'
|
37
|
+
s.add_dependency 'gettext', ['>= 3.0.2', '< 3.3.0']
|
34
38
|
|
35
39
|
s.add_development_dependency 'rspec', '~> 3.1'
|
36
40
|
|
@@ -25,6 +25,10 @@ describe R10K::Action::Deploy::Module do
|
|
25
25
|
it 'can accept a puppet-path option' do
|
26
26
|
described_class.new({ 'puppet-path': '/nonexistent' }, [])
|
27
27
|
end
|
28
|
+
|
29
|
+
it 'can accept a cachedir option' do
|
30
|
+
described_class.new({ cachedir: '/nonexistent' }, [])
|
31
|
+
end
|
28
32
|
end
|
29
33
|
|
30
34
|
describe "with no-force" do
|
@@ -66,8 +70,8 @@ describe R10K::Action::Deploy::Module do
|
|
66
70
|
|
67
71
|
before do
|
68
72
|
allow(subject).to receive(:visit_environment).and_wrap_original do |original, environment, &block|
|
69
|
-
expect(environment.puppetfile).to receive(:
|
70
|
-
[R10K::Module::Local.new(environment.name, '/fakedir', [], environment)]
|
73
|
+
expect(environment.puppetfile).to receive(:modules_by_vcs_cachedir).and_return(
|
74
|
+
{none: [R10K::Module::Local.new(environment.name, '/fakedir', [], environment)]}
|
71
75
|
)
|
72
76
|
original.call(environment, &block)
|
73
77
|
end
|
@@ -123,4 +127,13 @@ describe R10K::Action::Deploy::Module do
|
|
123
127
|
expect(subject.instance_variable_get(:@puppet_path)).to eq('/nonexistent')
|
124
128
|
end
|
125
129
|
end
|
130
|
+
|
131
|
+
describe 'with cachedir' do
|
132
|
+
|
133
|
+
subject { described_class.new({ config: '/some/nonexistent/path', cachedir: '/nonexistent' }, []) }
|
134
|
+
|
135
|
+
it 'sets puppet_path' do
|
136
|
+
expect(subject.instance_variable_get(:@cachedir)).to eq('/nonexistent')
|
137
|
+
end
|
138
|
+
end
|
126
139
|
end
|
@@ -19,12 +19,15 @@ describe R10K::Action::Puppetfile::Install do
|
|
19
19
|
|
20
20
|
describe "installing modules" do
|
21
21
|
let(:modules) do
|
22
|
-
|
22
|
+
(1..4).map do |idx|
|
23
|
+
R10K::Module::Base.new("author/modname#{idx}", "/some/nonexistent/path/modname#{idx}", nil)
|
24
|
+
end
|
23
25
|
end
|
24
26
|
|
25
27
|
before do
|
26
28
|
allow(puppetfile).to receive(:purge!)
|
27
29
|
allow(puppetfile).to receive(:modules).and_return(modules)
|
30
|
+
allow(puppetfile).to receive(:modules_by_vcs_cachedir).and_return({none: modules})
|
28
31
|
end
|
29
32
|
|
30
33
|
it "syncs each module in the Puppetfile" do
|
@@ -94,7 +94,7 @@ describe R10K::Action::Runner do
|
|
94
94
|
else
|
95
95
|
{ "#{conf_path}": override }
|
96
96
|
end
|
97
|
-
expect(global_settings).to receive(:evaluate).with(overrides).and_call_original
|
97
|
+
expect(global_settings).to receive(:evaluate).with(hash_including(overrides)).and_call_original
|
98
98
|
runner.call
|
99
99
|
end
|
100
100
|
end
|
@@ -109,7 +109,7 @@ describe R10K::Action::Runner do
|
|
109
109
|
else
|
110
110
|
{ "#{conf_path}": override }
|
111
111
|
end
|
112
|
-
expect(global_settings).to receive(:evaluate).with(overrides).and_call_original
|
112
|
+
expect(global_settings).to receive(:evaluate).with(hash_including(overrides)).and_call_original
|
113
113
|
runner.call
|
114
114
|
end
|
115
115
|
end
|
@@ -166,33 +166,37 @@ describe R10K::Forge::ModuleRelease do
|
|
166
166
|
end
|
167
167
|
|
168
168
|
describe "#cleanup_unpack_path" do
|
169
|
-
it "ignores the unpack_path if
|
170
|
-
|
171
|
-
expect(
|
169
|
+
it "ignores the unpack_path if the parent doesn't exist" do
|
170
|
+
parent = instance_double('Pathname')
|
171
|
+
expect(parent).to receive(:exist?).and_return false
|
172
|
+
expect(parent).to_not receive(:rmtree)
|
173
|
+
expect(unpack_path).to receive(:parent).and_return(parent)
|
172
174
|
subject.cleanup_unpack_path
|
173
175
|
end
|
174
176
|
|
175
177
|
it "removes the containing directory of unpack_path if it exists" do
|
176
178
|
parent = instance_double('Pathname')
|
177
179
|
expect(parent).to receive(:rmtree)
|
178
|
-
expect(
|
179
|
-
expect(unpack_path).to receive(:parent).and_return(parent)
|
180
|
+
expect(parent).to receive(:exist?).and_return true
|
181
|
+
expect(unpack_path).to receive(:parent).and_return(parent).exactly(2).times
|
180
182
|
subject.cleanup_unpack_path
|
181
183
|
end
|
182
184
|
end
|
183
185
|
|
184
186
|
describe "#cleanup_download_path" do
|
185
|
-
it "ignores the download_path if
|
186
|
-
|
187
|
-
expect(
|
187
|
+
it "ignores the download_path if the parent doesn't exist" do
|
188
|
+
parent = instance_double('Pathname')
|
189
|
+
expect(parent).to receive(:exist?).and_return false
|
190
|
+
expect(parent).to_not receive(:rmtree)
|
191
|
+
expect(download_path).to receive(:parent).and_return(parent)
|
188
192
|
subject.cleanup_download_path
|
189
193
|
end
|
190
194
|
|
191
195
|
it "removes the containing directory of download_path if it exists" do
|
192
196
|
parent = instance_double('Pathname')
|
193
197
|
expect(parent).to receive(:rmtree)
|
194
|
-
expect(
|
195
|
-
expect(download_path).to receive(:parent).and_return(parent)
|
198
|
+
expect(parent).to receive(:exist?).and_return true
|
199
|
+
expect(download_path).to receive(:parent).and_return(parent).exactly(2).times
|
196
200
|
subject.cleanup_download_path
|
197
201
|
end
|
198
202
|
end
|
@@ -127,6 +127,25 @@ describe R10K::Puppetfile do
|
|
127
127
|
|
128
128
|
expect { subject.add_module('puppet/test_module', module_opts) }.to raise_error(R10K::Error, /cannot manage content.*is not within/i).and not_change { subject.modules }
|
129
129
|
end
|
130
|
+
|
131
|
+
it "groups modules by vcs cache location" do
|
132
|
+
module_opts = { install_path: File.join(subject.basedir, 'vendor') }
|
133
|
+
opts1 = module_opts.merge(git: 'git@example.com:puppet/test_module.git')
|
134
|
+
opts2 = module_opts.merge(git: 'git@example.com:puppet/test_module_c.git')
|
135
|
+
sanitized_name1 = "git@example.com-puppet-test_module.git"
|
136
|
+
sanitized_name2 = "git@example.com-puppet-test_module_c.git"
|
137
|
+
|
138
|
+
subject.add_module('puppet/test_module_a', opts1)
|
139
|
+
subject.add_module('puppet/test_module_b', opts1)
|
140
|
+
subject.add_module('puppet/test_module_c', opts2)
|
141
|
+
subject.add_module('puppet/test_module_d', '1.2.3')
|
142
|
+
|
143
|
+
mods_by_cachedir = subject.modules_by_vcs_cachedir
|
144
|
+
|
145
|
+
expect(mods_by_cachedir[:none].length).to be 1
|
146
|
+
expect(mods_by_cachedir[sanitized_name1].length).to be 2
|
147
|
+
expect(mods_by_cachedir[sanitized_name2].length).to be 1
|
148
|
+
end
|
130
149
|
end
|
131
150
|
|
132
151
|
describe "#purge_exclusions" do
|
@@ -154,6 +173,26 @@ describe R10K::Puppetfile do
|
|
154
173
|
end
|
155
174
|
end
|
156
175
|
|
176
|
+
describe '#managed_directories' do
|
177
|
+
it 'returns an array of paths that can be purged' do
|
178
|
+
allow(R10K::Module).to receive(:new).with('puppet/test_module', subject.moduledir, '1.2.3', anything).and_call_original
|
179
|
+
|
180
|
+
subject.add_module('puppet/test_module', '1.2.3')
|
181
|
+
expect(subject.managed_directories).to match_array(["/some/nonexistent/basedir/modules"])
|
182
|
+
end
|
183
|
+
|
184
|
+
context 'with a module with install_path == \'\'' do
|
185
|
+
it 'basedir isn\'t in the list of paths to purge' do
|
186
|
+
module_opts = { install_path: '', git: 'git@example.com:puppet/test_module.git' }
|
187
|
+
|
188
|
+
allow(R10K::Module).to receive(:new).with('puppet/test_module', subject.basedir, module_opts, anything).and_call_original
|
189
|
+
|
190
|
+
subject.add_module('puppet/test_module', module_opts)
|
191
|
+
expect(subject.managed_directories).to be_empty
|
192
|
+
end
|
193
|
+
end
|
194
|
+
end
|
195
|
+
|
157
196
|
describe "evaluating a Puppetfile" do
|
158
197
|
def expect_wrapped_error(orig, pf_path, wrapped_error)
|
159
198
|
expect(orig).to be_a_kind_of(R10K::Error)
|
@@ -268,7 +307,7 @@ describe R10K::Puppetfile do
|
|
268
307
|
mod2 = spy('module')
|
269
308
|
expect(mod2).to receive(:accept).with(visitor)
|
270
309
|
|
271
|
-
expect(subject).to receive(:
|
310
|
+
expect(subject).to receive(:modules_by_vcs_cachedir).and_return({none: [mod1, mod2]})
|
272
311
|
subject.accept(visitor)
|
273
312
|
end
|
274
313
|
|
@@ -289,12 +328,38 @@ describe R10K::Puppetfile do
|
|
289
328
|
mod2 = spy('module')
|
290
329
|
expect(mod2).to receive(:accept).with(visitor)
|
291
330
|
|
292
|
-
expect(subject).to receive(:
|
331
|
+
expect(subject).to receive(:modules_by_vcs_cachedir).and_return({none: [mod1, mod2]})
|
293
332
|
|
294
333
|
expect(Thread).to receive(:new).exactly(pool_size).and_call_original
|
295
334
|
expect(Queue).to receive(:new).and_call_original
|
296
335
|
|
297
336
|
subject.accept(visitor)
|
298
337
|
end
|
338
|
+
|
339
|
+
it "Creates queues of modules grouped by cachedir" do
|
340
|
+
visitor = spy('visitor')
|
341
|
+
expect(visitor).to receive(:visit) do |type, other, &block|
|
342
|
+
expect(type).to eq :puppetfile
|
343
|
+
expect(other).to eq subject
|
344
|
+
block.call
|
345
|
+
end
|
346
|
+
|
347
|
+
mod1 = spy('module1')
|
348
|
+
mod2 = spy('module2')
|
349
|
+
mod3 = spy('module3')
|
350
|
+
mod4 = spy('module4')
|
351
|
+
mod5 = spy('module5')
|
352
|
+
mod6 = spy('module6')
|
353
|
+
|
354
|
+
expect(subject).to receive(:modules_by_vcs_cachedir)
|
355
|
+
.and_return({:none => [mod1, mod2],
|
356
|
+
"foo-cachedir" => [mod3, mod4],
|
357
|
+
"bar-cachedir" => [mod5, mod6]})
|
358
|
+
|
359
|
+
queue = subject.modules_queue(visitor)
|
360
|
+
expect(queue.length).to be 4
|
361
|
+
queue_array = 4.times.map { queue.pop }
|
362
|
+
expect(queue_array).to match_array([[mod1], [mod2], [mod3, mod4], [mod5, mod6]])
|
363
|
+
end
|
299
364
|
end
|
300
365
|
end
|