chef-dk 0.5.0.rc.1 → 0.5.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/README.md +63 -24
- data/lib/chef-dk/builtin_commands.rb +2 -0
- data/lib/chef-dk/command/diff.rb +312 -0
- data/lib/chef-dk/command/push.rb +1 -1
- data/lib/chef-dk/command/shell_init.rb +21 -3
- data/lib/chef-dk/command/update.rb +28 -5
- data/lib/chef-dk/configurable.rb +1 -1
- data/lib/chef-dk/exceptions.rb +3 -0
- data/lib/chef-dk/pager.rb +106 -0
- data/lib/chef-dk/policyfile/chef_repo_cookbook_source.rb +114 -0
- data/lib/chef-dk/policyfile/comparison_base.rb +124 -0
- data/lib/chef-dk/policyfile/cookbook_sources.rb +1 -0
- data/lib/chef-dk/policyfile/differ.rb +266 -0
- data/lib/chef-dk/policyfile/dsl.rb +26 -3
- data/lib/chef-dk/policyfile/uploader.rb +4 -5
- data/lib/chef-dk/policyfile_compiler.rb +8 -0
- data/lib/chef-dk/policyfile_lock.rb +135 -3
- data/lib/chef-dk/policyfile_services/install.rb +1 -0
- data/lib/chef-dk/policyfile_services/update_attributes.rb +104 -0
- data/lib/chef-dk/service_exceptions.rb +12 -0
- data/lib/chef-dk/ui.rb +8 -0
- data/lib/chef-dk/version.rb +1 -1
- data/spec/spec_helper.rb +6 -0
- data/spec/test_helpers.rb +4 -0
- data/spec/unit/command/diff_spec.rb +283 -0
- data/spec/unit/command/shell_init_spec.rb +19 -2
- data/spec/unit/command/update_spec.rb +96 -0
- data/spec/unit/command/verify_spec.rb +0 -6
- data/spec/unit/fixtures/local_path_cookbooks/cookbook-with-a-dep/Berksfile +3 -0
- data/spec/unit/fixtures/local_path_cookbooks/cookbook-with-a-dep/README.md +4 -0
- data/spec/unit/fixtures/local_path_cookbooks/cookbook-with-a-dep/chefignore +96 -0
- data/spec/unit/fixtures/local_path_cookbooks/cookbook-with-a-dep/metadata.rb +9 -0
- data/spec/unit/fixtures/local_path_cookbooks/cookbook-with-a-dep/recipes/default.rb +8 -0
- data/spec/unit/pager_spec.rb +119 -0
- data/spec/unit/policyfile/chef_repo_cookbook_source_spec.rb +66 -0
- data/spec/unit/policyfile/comparison_base_spec.rb +343 -0
- data/spec/unit/policyfile/differ_spec.rb +687 -0
- data/spec/unit/policyfile_evaluation_spec.rb +87 -0
- data/spec/unit/policyfile_lock_build_spec.rb +247 -8
- data/spec/unit/policyfile_lock_serialization_spec.rb +47 -0
- data/spec/unit/policyfile_services/export_repo_spec.rb +2 -0
- data/spec/unit/policyfile_services/push_spec.rb +2 -0
- data/spec/unit/policyfile_services/update_attributes_spec.rb +217 -0
- metadata +62 -6
@@ -0,0 +1,343 @@
|
|
1
|
+
#
|
2
|
+
# Copyright:: Copyright (c) 2015 Chef Software Inc.
|
3
|
+
# License:: Apache License, Version 2.0
|
4
|
+
#
|
5
|
+
# Licensed under the Apache License, Version 2.0 (the "License");
|
6
|
+
# you may not use this file except in compliance with the License.
|
7
|
+
# You may obtain a copy of the License at
|
8
|
+
#
|
9
|
+
# http://www.apache.org/licenses/LICENSE-2.0
|
10
|
+
#
|
11
|
+
# Unless required by applicable law or agreed to in writing, software
|
12
|
+
# distributed under the License is distributed on an "AS IS" BASIS,
|
13
|
+
# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
14
|
+
# See the License for the specific language governing permissions and
|
15
|
+
# limitations under the License.
|
16
|
+
#
|
17
|
+
|
18
|
+
require 'spec_helper'
|
19
|
+
require 'chef-dk/policyfile/comparison_base'
|
20
|
+
|
21
|
+
describe "Policyfile Comparison Bases" do
|
22
|
+
|
23
|
+
let(:minimal_lockfile_json) do
|
24
|
+
<<-E
|
25
|
+
{
|
26
|
+
"revision_id": "6fe753184c8946052d3231bb4212116df28d89a3a5f7ae52832ad408419dd5eb",
|
27
|
+
"name": "install-example",
|
28
|
+
"run_list": [
|
29
|
+
"recipe[local-cookbook::default]"
|
30
|
+
],
|
31
|
+
"cookbook_locks": {
|
32
|
+
"local-cookbook": {
|
33
|
+
"version": "2.3.4",
|
34
|
+
"identifier": "fab501cfaf747901bd82c1bc706beae7dc3a350c",
|
35
|
+
"dotted_decimal_identifier": "70567763561641081.489844270461035.258281553147148",
|
36
|
+
"source": "cookbooks/local-cookbook",
|
37
|
+
"cache_key": null,
|
38
|
+
"scm_info": null,
|
39
|
+
"source_options": {
|
40
|
+
"path": "cookbooks/local-cookbook"
|
41
|
+
}
|
42
|
+
}
|
43
|
+
},
|
44
|
+
"default_attributes": {},
|
45
|
+
"override_attributes": {},
|
46
|
+
"solution_dependencies": {
|
47
|
+
"Policyfile": [
|
48
|
+
[
|
49
|
+
"local-cookbook",
|
50
|
+
">= 0.0.0"
|
51
|
+
]
|
52
|
+
],
|
53
|
+
"dependencies": {
|
54
|
+
"local-cookbook (2.3.4)": [
|
55
|
+
|
56
|
+
]
|
57
|
+
}
|
58
|
+
}
|
59
|
+
}
|
60
|
+
E
|
61
|
+
end
|
62
|
+
|
63
|
+
let(:minimal_lockfile) { FFI_Yajl::Parser.parse(minimal_lockfile_json) }
|
64
|
+
|
65
|
+
describe ChefDK::Policyfile::ComparisonBase::Local do
|
66
|
+
|
67
|
+
let(:policyfile_lock_relpath) { "Policyfile.lock.json" }
|
68
|
+
|
69
|
+
subject(:comparison_base) { described_class.new(policyfile_lock_relpath) }
|
70
|
+
|
71
|
+
before do
|
72
|
+
reset_tempdir
|
73
|
+
end
|
74
|
+
|
75
|
+
after do
|
76
|
+
reset_tempdir
|
77
|
+
end
|
78
|
+
|
79
|
+
it "has the lockfile relative path" do
|
80
|
+
expect(comparison_base.policyfile_lock_relpath).to eq(policyfile_lock_relpath)
|
81
|
+
end
|
82
|
+
|
83
|
+
it "is named local:RELATIVE_PATH" do
|
84
|
+
expect(comparison_base.name).to eq("local:#{policyfile_lock_relpath}")
|
85
|
+
end
|
86
|
+
|
87
|
+
context "when the local lock doesn't exist" do
|
88
|
+
|
89
|
+
it "raises an exception when reading the lockfile" do
|
90
|
+
Dir.chdir(tempdir) do
|
91
|
+
expect { comparison_base.lock }.to raise_error(ChefDK::LockfileNotFound)
|
92
|
+
end
|
93
|
+
end
|
94
|
+
|
95
|
+
end
|
96
|
+
|
97
|
+
context "when the local policyfile lock is not readable", :skip_on_windows do
|
98
|
+
|
99
|
+
before do
|
100
|
+
Dir.chdir(tempdir) do
|
101
|
+
FileUtils.touch(policyfile_lock_relpath)
|
102
|
+
allow(File).to receive(:readable?).with(policyfile_lock_relpath).and_return(false)
|
103
|
+
end
|
104
|
+
end
|
105
|
+
|
106
|
+
it "raises an exception" do
|
107
|
+
Dir.chdir(tempdir) do
|
108
|
+
expect { comparison_base.lock }.to raise_error(ChefDK::LockfileNotFound)
|
109
|
+
end
|
110
|
+
end
|
111
|
+
|
112
|
+
end
|
113
|
+
|
114
|
+
context "when the local policyfile lock is malformed" do
|
115
|
+
|
116
|
+
before do
|
117
|
+
Dir.chdir(tempdir) do
|
118
|
+
File.open(policyfile_lock_relpath, "w+") { |f| f.print("}}}}}}") }
|
119
|
+
end
|
120
|
+
end
|
121
|
+
|
122
|
+
it "raises an exception" do
|
123
|
+
Dir.chdir(tempdir) do
|
124
|
+
expect { comparison_base.lock }.to raise_error(ChefDK::MalformedLockfile)
|
125
|
+
end
|
126
|
+
end
|
127
|
+
|
128
|
+
end
|
129
|
+
|
130
|
+
context "when the local lock exists and is valid JSON" do
|
131
|
+
|
132
|
+
before do
|
133
|
+
Dir.chdir(tempdir) do
|
134
|
+
File.open(policyfile_lock_relpath, "w+") { |f| f.print(minimal_lockfile_json) }
|
135
|
+
end
|
136
|
+
end
|
137
|
+
|
138
|
+
it "reads the local lock and parses the JSON" do
|
139
|
+
Dir.chdir(tempdir) do
|
140
|
+
expect(comparison_base.lock).to eq(minimal_lockfile)
|
141
|
+
end
|
142
|
+
end
|
143
|
+
|
144
|
+
end
|
145
|
+
|
146
|
+
end
|
147
|
+
|
148
|
+
describe ChefDK::Policyfile::ComparisonBase::Git do
|
149
|
+
|
150
|
+
let(:ref) { "master" }
|
151
|
+
|
152
|
+
let(:policyfile_lock_relpath) { "policies/MyPolicy.lock.json" }
|
153
|
+
|
154
|
+
subject(:comparison_base) { described_class.new(ref, policyfile_lock_relpath) }
|
155
|
+
|
156
|
+
it "has the policyfile lock relative path it was created with" do
|
157
|
+
expect(comparison_base.policyfile_lock_relpath).to eq(policyfile_lock_relpath)
|
158
|
+
end
|
159
|
+
|
160
|
+
it "has the ref it was created with" do
|
161
|
+
expect(comparison_base.ref).to eq(ref)
|
162
|
+
end
|
163
|
+
|
164
|
+
it "is named git:REF" do
|
165
|
+
expect(comparison_base.name).to eq("git:master")
|
166
|
+
end
|
167
|
+
|
168
|
+
it "creates a `git show` command for the policyfile lock and ref" do
|
169
|
+
expect(comparison_base.git_cmd_string).to eq("git show master:./policies/MyPolicy.lock.json")
|
170
|
+
expect(comparison_base.git_cmd.command).to eq("git show master:./policies/MyPolicy.lock.json")
|
171
|
+
end
|
172
|
+
|
173
|
+
context "when the git command fails" do
|
174
|
+
|
175
|
+
before do
|
176
|
+
allow(comparison_base.git_cmd).to receive(:run_command)
|
177
|
+
allow(comparison_base.git_cmd).to receive(:error!).and_raise(Mixlib::ShellOut::ShellCommandFailed)
|
178
|
+
allow(comparison_base.git_cmd).to receive(:stderr).and_return("fatal: Not a git repository (or any of the parent directories): .git\n")
|
179
|
+
end
|
180
|
+
|
181
|
+
it "raises an exception when reading the lockfile" do
|
182
|
+
expect { comparison_base.lock }.to raise_error(ChefDK::GitError)
|
183
|
+
end
|
184
|
+
|
185
|
+
end
|
186
|
+
|
187
|
+
context "when the git command succeeds" do
|
188
|
+
|
189
|
+
before do
|
190
|
+
allow(comparison_base.git_cmd).to receive(:run_command)
|
191
|
+
allow(comparison_base.git_cmd).to receive(:error!).and_return(nil)
|
192
|
+
end
|
193
|
+
|
194
|
+
context "and the JSON is malformed" do
|
195
|
+
|
196
|
+
before do
|
197
|
+
allow(comparison_base.git_cmd).to receive(:stdout).and_return("}}}}}")
|
198
|
+
end
|
199
|
+
|
200
|
+
it "raises an exception" do
|
201
|
+
expect { comparison_base.lock }.to raise_error(ChefDK::MalformedLockfile)
|
202
|
+
end
|
203
|
+
|
204
|
+
end
|
205
|
+
|
206
|
+
context "and the JSON is well-formed" do
|
207
|
+
|
208
|
+
before do
|
209
|
+
allow(comparison_base.git_cmd).to receive(:stdout).and_return(minimal_lockfile_json)
|
210
|
+
end
|
211
|
+
|
212
|
+
it "reads the lockfile and parses the JSON" do
|
213
|
+
expect(comparison_base.lock).to eq(minimal_lockfile)
|
214
|
+
end
|
215
|
+
end
|
216
|
+
|
217
|
+
end
|
218
|
+
|
219
|
+
end
|
220
|
+
|
221
|
+
describe ChefDK::Policyfile::ComparisonBase::PolicyGroup do
|
222
|
+
|
223
|
+
let(:group) { "acceptance" }
|
224
|
+
let(:policy_name) { "chatserver" }
|
225
|
+
let(:http_client) { instance_double("ChefDK::AuthenticatedHTTP", url: "https://chef.example/organizations/monkeynews") }
|
226
|
+
|
227
|
+
subject(:comparison_base) { described_class.new(group, policy_name, http_client) }
|
228
|
+
|
229
|
+
it "has the group it was created with" do
|
230
|
+
expect(comparison_base.group).to eq(group)
|
231
|
+
end
|
232
|
+
|
233
|
+
it "has the policy_name it was created with" do
|
234
|
+
expect(comparison_base.policy_name).to eq(policy_name)
|
235
|
+
end
|
236
|
+
|
237
|
+
it "has the HTTP client it was created with" do
|
238
|
+
expect(comparison_base.http_client).to eq(http_client)
|
239
|
+
end
|
240
|
+
|
241
|
+
it "is named policy_group:GROUP" do
|
242
|
+
expect(comparison_base.name).to eq("policy_group:#{group}")
|
243
|
+
end
|
244
|
+
|
245
|
+
context "when there is a non-404 HTTP error fetching the policyfile lock" do
|
246
|
+
|
247
|
+
let(:response) do
|
248
|
+
Net::HTTPResponse.send(:response_class, "500").new("1.0", "500", "Internal Server Error").tap do |r|
|
249
|
+
r.instance_variable_set(:@body, "oops")
|
250
|
+
end
|
251
|
+
end
|
252
|
+
|
253
|
+
let(:http_exception) do
|
254
|
+
begin
|
255
|
+
response.error!
|
256
|
+
rescue => e
|
257
|
+
e
|
258
|
+
end
|
259
|
+
end
|
260
|
+
|
261
|
+
before do
|
262
|
+
allow(http_client).to receive(:get).and_raise(http_exception)
|
263
|
+
end
|
264
|
+
|
265
|
+
it "raises an exception" do
|
266
|
+
exception = nil
|
267
|
+
begin
|
268
|
+
comparison_base.lock
|
269
|
+
rescue => exception
|
270
|
+
expect(exception).to be_a_kind_of(ChefDK::PolicyfileDownloadError)
|
271
|
+
expect(exception.message).to eq("HTTP error attempting to fetch policyfile lock from https://chef.example/organizations/monkeynews")
|
272
|
+
expect(exception.cause).to eq(http_exception)
|
273
|
+
end
|
274
|
+
expect(exception).to_not be_nil
|
275
|
+
end
|
276
|
+
|
277
|
+
end
|
278
|
+
|
279
|
+
context "when the server returns 404 fetching the policyfile lock" do
|
280
|
+
|
281
|
+
let(:response) do
|
282
|
+
Net::HTTPResponse.send(:response_class, "404").new("1.0", "404", "Not Found").tap do |r|
|
283
|
+
r.instance_variable_set(:@body, "nothin' here, chief")
|
284
|
+
end
|
285
|
+
end
|
286
|
+
|
287
|
+
let(:http_exception) do
|
288
|
+
begin
|
289
|
+
response.error!
|
290
|
+
rescue => e
|
291
|
+
e
|
292
|
+
end
|
293
|
+
end
|
294
|
+
|
295
|
+
before do
|
296
|
+
allow(http_client).to receive(:get).and_raise(http_exception)
|
297
|
+
end
|
298
|
+
|
299
|
+
it "raises an exception" do
|
300
|
+
exception = nil
|
301
|
+
begin
|
302
|
+
comparison_base.lock
|
303
|
+
rescue => exception
|
304
|
+
expect(exception).to be_a_kind_of(ChefDK::PolicyfileDownloadError)
|
305
|
+
expect(exception.message).to eq("No policyfile lock named 'chatserver' found in policy_group 'acceptance' at https://chef.example/organizations/monkeynews")
|
306
|
+
expect(exception.cause).to eq(http_exception)
|
307
|
+
end
|
308
|
+
expect(exception).to_not be_nil
|
309
|
+
end
|
310
|
+
|
311
|
+
|
312
|
+
end
|
313
|
+
|
314
|
+
|
315
|
+
context "when a non-HTTP error occurs fetching the policyfile lock" do
|
316
|
+
|
317
|
+
before do
|
318
|
+
allow(http_client).to receive(:get).and_raise(Errno::ECONNREFUSED)
|
319
|
+
end
|
320
|
+
|
321
|
+
it "raises an exception" do
|
322
|
+
expect { comparison_base.lock }.to raise_error(ChefDK::PolicyfileDownloadError)
|
323
|
+
end
|
324
|
+
|
325
|
+
end
|
326
|
+
|
327
|
+
context "when the policyfile lock is fetched from the server" do
|
328
|
+
|
329
|
+
before do
|
330
|
+
expect(http_client).to receive(:get).
|
331
|
+
with("policy_groups/acceptance/policies/chatserver").
|
332
|
+
and_return(minimal_lockfile)
|
333
|
+
end
|
334
|
+
|
335
|
+
it "returns the policyfile lock data" do
|
336
|
+
expect(comparison_base.lock).to eq(minimal_lockfile)
|
337
|
+
end
|
338
|
+
|
339
|
+
end
|
340
|
+
|
341
|
+
end
|
342
|
+
end
|
343
|
+
|
@@ -0,0 +1,687 @@
|
|
1
|
+
#
|
2
|
+
# Copyright:: Copyright (c) 2014 Chef Software Inc.
|
3
|
+
# License:: Apache License, Version 2.0
|
4
|
+
#
|
5
|
+
# Licensed under the Apache License, Version 2.0 (the "License");
|
6
|
+
# you may not use this file except in compliance with the License.
|
7
|
+
# You may obtain a copy of the License at
|
8
|
+
#
|
9
|
+
# http://www.apache.org/licenses/LICENSE-2.0
|
10
|
+
#
|
11
|
+
# Unless required by applicable law or agreed to in writing, software
|
12
|
+
# distributed under the License is distributed on an "AS IS" BASIS,
|
13
|
+
# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
14
|
+
# See the License for the specific language governing permissions and
|
15
|
+
# limitations under the License.
|
16
|
+
#
|
17
|
+
|
18
|
+
require 'spec_helper'
|
19
|
+
require 'chef-dk/policyfile/differ'
|
20
|
+
|
21
|
+
describe ChefDK::Policyfile::Differ do
|
22
|
+
|
23
|
+
let(:old_lock_json) do
|
24
|
+
<<-E
|
25
|
+
{
|
26
|
+
"revision_id": "cf4b8a020bdc1ba6914093a8a07a5514cce8a3a2979a967b1f32ea704a61785b",
|
27
|
+
"name": "jenkins",
|
28
|
+
"run_list": [
|
29
|
+
"recipe[java::default]",
|
30
|
+
"recipe[jenkins::master]",
|
31
|
+
"recipe[policyfile_demo::whatever]",
|
32
|
+
"recipe[policyfile_demo::default]"
|
33
|
+
],
|
34
|
+
"named_run_lists": {
|
35
|
+
"update_jenkins": [
|
36
|
+
"recipe[jenkins::master]",
|
37
|
+
"recipe[policyfile_demo::default]"
|
38
|
+
]
|
39
|
+
},
|
40
|
+
"cookbook_locks": {
|
41
|
+
"policyfile_demo": {
|
42
|
+
"version": "0.1.0",
|
43
|
+
"identifier": "ea96c99da079db9ff3cb22601638fabd5df49599",
|
44
|
+
"dotted_decimal_identifier": "66030937227426267.45022575077627448.275691232073113",
|
45
|
+
"source": "cookbooks/policyfile_demo",
|
46
|
+
"cache_key": null,
|
47
|
+
"scm_info": {
|
48
|
+
"scm": "git",
|
49
|
+
"remote": "git@github.com:danielsdeleo/policyfile-jenkins-demo.git",
|
50
|
+
"revision": "6f92fe8f24fd953a1c40ebb1d7cdb2a4fbbf4d4d",
|
51
|
+
"working_tree_clean": false,
|
52
|
+
"published": true,
|
53
|
+
"synchronized_remote_branches": [
|
54
|
+
"mine/master"
|
55
|
+
]
|
56
|
+
},
|
57
|
+
"source_options": {
|
58
|
+
"path": "cookbooks/policyfile_demo"
|
59
|
+
}
|
60
|
+
},
|
61
|
+
"apt": {
|
62
|
+
"version": "2.7.0",
|
63
|
+
"identifier": "16c57abbd056543f7d5a15dabbb03261024a9c5e",
|
64
|
+
"dotted_decimal_identifier": "6409580415309396.17870749399956400.55392231660638",
|
65
|
+
"cache_key": "apt-2.7.0-supermarket.chef.io",
|
66
|
+
"origin": "https://supermarket.chef.io/api/v1/cookbooks/apt/versions/2.7.0/download",
|
67
|
+
"source_options": {
|
68
|
+
"artifactserver": "https://supermarket.chef.io/api/v1/cookbooks/apt/versions/2.7.0/download",
|
69
|
+
"version": "2.7.0"
|
70
|
+
}
|
71
|
+
},
|
72
|
+
"java": {
|
73
|
+
"version": "1.31.0",
|
74
|
+
"identifier": "9178a38ad3e3baa55b49c1b8d9f4bf6a43dbc358",
|
75
|
+
"dotted_decimal_identifier": "40946515427189690.46543743498115572.210463125914456",
|
76
|
+
"cache_key": "java-1.31.0-supermarket.chef.io",
|
77
|
+
"origin": "https://supermarket.chef.io/api/v1/cookbooks/java/versions/1.31.0/download",
|
78
|
+
"source_options": {
|
79
|
+
"artifactserver": "https://supermarket.chef.io/api/v1/cookbooks/java/versions/1.31.0/download",
|
80
|
+
"version": "1.31.0"
|
81
|
+
}
|
82
|
+
},
|
83
|
+
"jenkins": {
|
84
|
+
"version": "2.2.2",
|
85
|
+
"identifier": "0be380429add00d189b4431059ac967a60052323",
|
86
|
+
"dotted_decimal_identifier": "3346364756581632.58979677444790700.165452341125923",
|
87
|
+
"cache_key": "jenkins-2.2.2-supermarket.chef.io",
|
88
|
+
"origin": "https://supermarket.chef.io/api/v1/cookbooks/jenkins/versions/2.2.2/download",
|
89
|
+
"source_options": {
|
90
|
+
"artifactserver": "https://supermarket.chef.io/api/v1/cookbooks/jenkins/versions/2.2.2/download",
|
91
|
+
"version": "2.2.2"
|
92
|
+
}
|
93
|
+
},
|
94
|
+
"runit": {
|
95
|
+
"version": "1.5.18",
|
96
|
+
"identifier": "1a0aeb2c167a24e0c5120ca7b06ba8c4cff4610c",
|
97
|
+
"dotted_decimal_identifier": "7330354567739940.63267076095586411.185563255955724",
|
98
|
+
"cache_key": "runit-1.5.18-supermarket.chef.io",
|
99
|
+
"origin": "https://supermarket.chef.io/api/v1/cookbooks/runit/versions/1.5.18/download",
|
100
|
+
"source_options": {
|
101
|
+
"artifactserver": "https://supermarket.chef.io/api/v1/cookbooks/runit/versions/1.5.18/download",
|
102
|
+
"version": "1.5.18"
|
103
|
+
}
|
104
|
+
},
|
105
|
+
"build-essential": {
|
106
|
+
"version": "2.2.2",
|
107
|
+
"identifier": "d8ce58401d154378599b0fead81d2c390615602b",
|
108
|
+
"dotted_decimal_identifier": "61025473397593411.33875519727130653.48623426822187",
|
109
|
+
"cache_key": "build-essential-2.2.2-supermarket.chef.io",
|
110
|
+
"origin": "https://supermarket.chef.io/api/v1/cookbooks/build-essential/versions/2.2.2/download",
|
111
|
+
"source_options": {
|
112
|
+
"artifactserver": "https://supermarket.chef.io/api/v1/cookbooks/build-essential/versions/2.2.2/download",
|
113
|
+
"version": "2.2.2"
|
114
|
+
}
|
115
|
+
},
|
116
|
+
"yum": {
|
117
|
+
"version": "3.5.4",
|
118
|
+
"identifier": "f9c778c3cd3908071e0c55722682f96e653b5642",
|
119
|
+
"dotted_decimal_identifier": "70306590695962888.2003363158959746.274252540106306",
|
120
|
+
"cache_key": "yum-3.5.4-supermarket.chef.io",
|
121
|
+
"origin": "https://supermarket.chef.io/api/v1/cookbooks/yum/versions/3.5.4/download",
|
122
|
+
"source_options": {
|
123
|
+
"artifactserver": "https://supermarket.chef.io/api/v1/cookbooks/yum/versions/3.5.4/download",
|
124
|
+
"version": "3.5.4"
|
125
|
+
}
|
126
|
+
},
|
127
|
+
"yum-epel": {
|
128
|
+
"version": "0.6.0",
|
129
|
+
"identifier": "cd74f541ba0341abcc168c74471c349ca68f77b7",
|
130
|
+
"dotted_decimal_identifier": "57830966944203585.48356618235299612.57847413962679",
|
131
|
+
"cache_key": "yum-epel-0.6.0-supermarket.chef.io",
|
132
|
+
"origin": "https://supermarket.chef.io/api/v1/cookbooks/yum-epel/versions/0.6.0/download",
|
133
|
+
"source_options": {
|
134
|
+
"artifactserver": "https://supermarket.chef.io/api/v1/cookbooks/yum-epel/versions/0.6.0/download",
|
135
|
+
"version": "0.6.0"
|
136
|
+
}
|
137
|
+
}
|
138
|
+
},
|
139
|
+
"default_attributes": {
|
140
|
+
"greeting": "Attributes, f*** yeah"
|
141
|
+
},
|
142
|
+
"override_attributes": {
|
143
|
+
"attr_only_updating": "use -a"
|
144
|
+
},
|
145
|
+
"solution_dependencies": {
|
146
|
+
"Policyfile": [
|
147
|
+
[
|
148
|
+
"policyfile_demo",
|
149
|
+
">= 0.0.0"
|
150
|
+
],
|
151
|
+
[
|
152
|
+
"apt",
|
153
|
+
"= 2.7.0"
|
154
|
+
],
|
155
|
+
[
|
156
|
+
"java",
|
157
|
+
"= 1.31.0"
|
158
|
+
],
|
159
|
+
[
|
160
|
+
"jenkins",
|
161
|
+
"= 2.2.2"
|
162
|
+
],
|
163
|
+
[
|
164
|
+
"runit",
|
165
|
+
"= 1.5.18"
|
166
|
+
],
|
167
|
+
[
|
168
|
+
"build-essential",
|
169
|
+
"= 2.2.2"
|
170
|
+
],
|
171
|
+
[
|
172
|
+
"yum",
|
173
|
+
"= 3.5.4"
|
174
|
+
],
|
175
|
+
[
|
176
|
+
"yum-epel",
|
177
|
+
"= 0.6.0"
|
178
|
+
]
|
179
|
+
],
|
180
|
+
"dependencies": {
|
181
|
+
"apt (2.7.0)": [
|
182
|
+
|
183
|
+
],
|
184
|
+
"java (1.31.0)": [
|
185
|
+
|
186
|
+
],
|
187
|
+
"jenkins (2.2.2)": [
|
188
|
+
[
|
189
|
+
"apt",
|
190
|
+
"~> 2.0"
|
191
|
+
],
|
192
|
+
[
|
193
|
+
"runit",
|
194
|
+
"~> 1.5"
|
195
|
+
],
|
196
|
+
[
|
197
|
+
"yum",
|
198
|
+
"~> 3.0"
|
199
|
+
]
|
200
|
+
],
|
201
|
+
"runit (1.5.18)": [
|
202
|
+
[
|
203
|
+
"build-essential",
|
204
|
+
">= 0.0.0"
|
205
|
+
],
|
206
|
+
[
|
207
|
+
"yum",
|
208
|
+
"~> 3.0"
|
209
|
+
],
|
210
|
+
[
|
211
|
+
"yum-epel",
|
212
|
+
">= 0.0.0"
|
213
|
+
]
|
214
|
+
],
|
215
|
+
"build-essential (2.2.2)": [
|
216
|
+
|
217
|
+
],
|
218
|
+
"yum (3.5.4)": [
|
219
|
+
|
220
|
+
],
|
221
|
+
"yum-epel (0.6.0)": [
|
222
|
+
[
|
223
|
+
"yum",
|
224
|
+
"~> 3.0"
|
225
|
+
]
|
226
|
+
],
|
227
|
+
"policyfile_demo (0.1.0)": [
|
228
|
+
|
229
|
+
]
|
230
|
+
}
|
231
|
+
}
|
232
|
+
}
|
233
|
+
E
|
234
|
+
end
|
235
|
+
|
236
|
+
let(:old_lock) { FFI_Yajl::Parser.parse(old_lock_json) }
|
237
|
+
|
238
|
+
let(:new_lock) { old_lock }
|
239
|
+
|
240
|
+
let(:new_rev_id) { "304566f86a620aae85797a3c491a51fb8c6ecf996407e77b8063aa3ee59672c5" }
|
241
|
+
|
242
|
+
let(:ui) { TestHelpers::TestUI.new }
|
243
|
+
|
244
|
+
def output
|
245
|
+
# ANSI codes make the tests harder to read
|
246
|
+
Paint.unpaint(ui.output)
|
247
|
+
end
|
248
|
+
|
249
|
+
subject(:differ) do
|
250
|
+
described_class.new(old_name: "git: HEAD", old_lock: old_lock, new_name: "local disk", new_lock: new_lock, ui: ui)
|
251
|
+
end
|
252
|
+
|
253
|
+
it "has a UI object" do
|
254
|
+
expect(differ.ui).to eq(ui)
|
255
|
+
end
|
256
|
+
|
257
|
+
it "has the old lock data" do
|
258
|
+
expect(differ.old_lock).to eq(old_lock)
|
259
|
+
end
|
260
|
+
|
261
|
+
it "has the old lock `name'" do
|
262
|
+
expect(differ.old_name).to eq("git: HEAD")
|
263
|
+
end
|
264
|
+
|
265
|
+
it "has the new lock data" do
|
266
|
+
expect(differ.new_lock).to eq(new_lock)
|
267
|
+
end
|
268
|
+
|
269
|
+
it "has the new lock `name'" do
|
270
|
+
expect(differ.new_name).to eq("local disk")
|
271
|
+
end
|
272
|
+
|
273
|
+
context "when old and new lock data are the same" do
|
274
|
+
|
275
|
+
let(:new_lock) { old_lock }
|
276
|
+
|
277
|
+
it "has no updates" do
|
278
|
+
expect(differ.different?).to be(false)
|
279
|
+
end
|
280
|
+
|
281
|
+
it "has no updated sections" do
|
282
|
+
expect(differ.updated_sections).to be_empty
|
283
|
+
end
|
284
|
+
|
285
|
+
it "reports that there are no updates" do
|
286
|
+
expected_message = <<-E
|
287
|
+
No changes for policy lock 'jenkins' between 'git: HEAD' and 'local disk'
|
288
|
+
E
|
289
|
+
differ.run_report
|
290
|
+
expect(output).to include(expected_message)
|
291
|
+
end
|
292
|
+
|
293
|
+
end
|
294
|
+
|
295
|
+
context "when the run list is updated" do
|
296
|
+
|
297
|
+
let(:new_lock) do
|
298
|
+
n = old_lock.dup
|
299
|
+
n["revision_id"] = new_rev_id
|
300
|
+
n["run_list"] = old_lock["run_list"].dup
|
301
|
+
n["run_list"].delete_at(2)
|
302
|
+
n["run_list"] += %w{ recipe[one::one] recipe[two::two] recipe[three::three]}
|
303
|
+
n
|
304
|
+
end
|
305
|
+
|
306
|
+
it "has updates" do
|
307
|
+
expect(differ.different?).to be(true)
|
308
|
+
end
|
309
|
+
|
310
|
+
it "has an updated revision_id and run_list" do
|
311
|
+
expect(differ.updated_sections).to match_array(%w{revision_id run_list})
|
312
|
+
end
|
313
|
+
|
314
|
+
it "reports the updated rev_id and run_list" do
|
315
|
+
expected_message = <<-E
|
316
|
+
Policy lock 'jenkins' differs between 'git: HEAD' and 'local disk':
|
317
|
+
|
318
|
+
REVISION ID CHANGED
|
319
|
+
===================
|
320
|
+
|
321
|
+
@@ -1,2 +1,2 @@
|
322
|
+
-cf4b8a020bdc1ba6914093a8a07a5514cce8a3a2979a967b1f32ea704a61785b
|
323
|
+
+304566f86a620aae85797a3c491a51fb8c6ecf996407e77b8063aa3ee59672c5
|
324
|
+
|
325
|
+
RUN LIST CHANGED
|
326
|
+
================
|
327
|
+
|
328
|
+
@@ -1,5 +1,7 @@
|
329
|
+
recipe[java::default]
|
330
|
+
recipe[jenkins::master]
|
331
|
+
-recipe[policyfile_demo::whatever]
|
332
|
+
recipe[policyfile_demo::default]
|
333
|
+
+recipe[one::one]
|
334
|
+
+recipe[two::two]
|
335
|
+
+recipe[three::three]
|
336
|
+
|
337
|
+
E
|
338
|
+
differ.run_report
|
339
|
+
expect(output).to eq(expected_message)
|
340
|
+
end
|
341
|
+
|
342
|
+
# primary goal here is to make sure the differ behaves correctly when diff
|
343
|
+
# "hunks" don't overlap
|
344
|
+
context "when the run_list has non-contiguous changes" do
|
345
|
+
|
346
|
+
let(:old_lock) do
|
347
|
+
FFI_Yajl::Parser.parse(old_lock_json).tap do |l|
|
348
|
+
l["run_list"] = %w{ a b c d e f g h i j k l m n }.map do |letter|
|
349
|
+
"recipe[#{letter}::default]"
|
350
|
+
end
|
351
|
+
end
|
352
|
+
end
|
353
|
+
|
354
|
+
let(:new_lock) do
|
355
|
+
old_lock.dup.tap do |n|
|
356
|
+
n["revision_id"] = new_rev_id
|
357
|
+
n["run_list"] = n["run_list"].dup
|
358
|
+
n["run_list"].delete_at(1)
|
359
|
+
n["run_list"] += %w{ o p q r }.map do |letter|
|
360
|
+
"recipe[#{letter}::new]"
|
361
|
+
end
|
362
|
+
end
|
363
|
+
end
|
364
|
+
|
365
|
+
it "prints the correct changes with context for the run list" do
|
366
|
+
expected_message = <<-E
|
367
|
+
Policy lock 'jenkins' differs between 'git: HEAD' and 'local disk':
|
368
|
+
|
369
|
+
REVISION ID CHANGED
|
370
|
+
===================
|
371
|
+
|
372
|
+
@@ -1,2 +1,2 @@
|
373
|
+
-cf4b8a020bdc1ba6914093a8a07a5514cce8a3a2979a967b1f32ea704a61785b
|
374
|
+
+304566f86a620aae85797a3c491a51fb8c6ecf996407e77b8063aa3ee59672c5
|
375
|
+
|
376
|
+
RUN LIST CHANGED
|
377
|
+
================
|
378
|
+
|
379
|
+
@@ -1,5 +1,4 @@
|
380
|
+
recipe[a::default]
|
381
|
+
-recipe[b::default]
|
382
|
+
recipe[c::default]
|
383
|
+
recipe[d::default]
|
384
|
+
recipe[e::default]
|
385
|
+
@@ -12,4 +11,8 @@
|
386
|
+
recipe[l::default]
|
387
|
+
recipe[m::default]
|
388
|
+
recipe[n::default]
|
389
|
+
+recipe[o::new]
|
390
|
+
+recipe[p::new]
|
391
|
+
+recipe[q::new]
|
392
|
+
+recipe[r::new]
|
393
|
+
|
394
|
+
E
|
395
|
+
differ.run_report
|
396
|
+
expect(output).to eq(expected_message)
|
397
|
+
end
|
398
|
+
|
399
|
+
end
|
400
|
+
end
|
401
|
+
|
402
|
+
context "with a removed cookbook" do
|
403
|
+
|
404
|
+
let(:new_lock) do
|
405
|
+
old_lock.dup.tap do |n|
|
406
|
+
n["revision_id"] = new_rev_id
|
407
|
+
n["cookbook_locks"] = n["cookbook_locks"].dup
|
408
|
+
n["cookbook_locks"].delete("apt")
|
409
|
+
end
|
410
|
+
end
|
411
|
+
|
412
|
+
it "has updates" do
|
413
|
+
expect(differ.different?).to be(true)
|
414
|
+
end
|
415
|
+
|
416
|
+
it "has an updated revision_id and cookbook_locks" do
|
417
|
+
expect(differ.updated_sections).to match_array(%w{revision_id cookbook_locks})
|
418
|
+
end
|
419
|
+
|
420
|
+
it "has removed the 'apt' cookbook" do
|
421
|
+
expect(differ.removed_cookbooks).to eq(%w{apt})
|
422
|
+
end
|
423
|
+
|
424
|
+
it "reports the updated revision_id and removed cookbooks" do
|
425
|
+
expected_message = <<-E
|
426
|
+
Policy lock 'jenkins' differs between 'git: HEAD' and 'local disk':
|
427
|
+
|
428
|
+
REVISION ID CHANGED
|
429
|
+
===================
|
430
|
+
|
431
|
+
@@ -1,2 +1,2 @@
|
432
|
+
-cf4b8a020bdc1ba6914093a8a07a5514cce8a3a2979a967b1f32ea704a61785b
|
433
|
+
+304566f86a620aae85797a3c491a51fb8c6ecf996407e77b8063aa3ee59672c5
|
434
|
+
|
435
|
+
REMOVED COOKBOOKS
|
436
|
+
=================
|
437
|
+
|
438
|
+
apt
|
439
|
+
---
|
440
|
+
|
441
|
+
@@ -1,12 +1 @@
|
442
|
+
-{
|
443
|
+
- "version": "2.7.0",
|
444
|
+
- "identifier": "16c57abbd056543f7d5a15dabbb03261024a9c5e",
|
445
|
+
- "dotted_decimal_identifier": "6409580415309396.17870749399956400.55392231660638",
|
446
|
+
- "cache_key": "apt-2.7.0-supermarket.chef.io",
|
447
|
+
- "origin": "https://supermarket.chef.io/api/v1/cookbooks/apt/versions/2.7.0/download",
|
448
|
+
- "source_options": {
|
449
|
+
- "artifactserver": "https://supermarket.chef.io/api/v1/cookbooks/apt/versions/2.7.0/download",
|
450
|
+
- "version": "2.7.0"
|
451
|
+
- }
|
452
|
+
-}
|
453
|
+
|
454
|
+
E
|
455
|
+
differ.run_report
|
456
|
+
expect(output).to eq(expected_message)
|
457
|
+
end
|
458
|
+
|
459
|
+
end
|
460
|
+
|
461
|
+
context "with an added cookbook" do
|
462
|
+
|
463
|
+
let(:new_cookbook) do
|
464
|
+
{
|
465
|
+
"version" => "2.3.2",
|
466
|
+
"identifier" => "9c6990944d9a347dec8bd375e707ba0aecdc17cd",
|
467
|
+
"dotted_decimal_identifier" => "69437059924760478.24393100994078142.115593340606828",
|
468
|
+
"cache_key" => "bluepill-2.3.2-supermarket.chef.io",
|
469
|
+
"origin" => "https://supermarket.chef.io/api/v1/cookbooks/bluepill/versions/2.3.2/download",
|
470
|
+
"source_options" => {
|
471
|
+
"artifactserver" => "https://supermarket.chef.io/api/v1/cookbooks/bluepill/versions/2.3.2/download",
|
472
|
+
"version" => "2.3.2"
|
473
|
+
}
|
474
|
+
}
|
475
|
+
end
|
476
|
+
|
477
|
+
let(:new_lock) do
|
478
|
+
old_lock.dup.tap do |n|
|
479
|
+
n["revision_id"] = new_rev_id
|
480
|
+
n["cookbook_locks"] = n["cookbook_locks"].dup
|
481
|
+
n["cookbook_locks"]["bluepill"] = new_cookbook
|
482
|
+
end
|
483
|
+
end
|
484
|
+
|
485
|
+
it "has updates" do
|
486
|
+
expect(differ.different?).to be(true)
|
487
|
+
end
|
488
|
+
|
489
|
+
it "has an updated revision_id and cookbook_locks" do
|
490
|
+
expect(differ.updated_sections).to match_array(%w{revision_id cookbook_locks})
|
491
|
+
end
|
492
|
+
|
493
|
+
it "has added the bluepill cookbook" do
|
494
|
+
expect(differ.added_cookbooks).to eq(%w{ bluepill })
|
495
|
+
end
|
496
|
+
|
497
|
+
it "reports the updated revision_id and added cookbook" do
|
498
|
+
expected_message = <<-E
|
499
|
+
Policy lock 'jenkins' differs between 'git: HEAD' and 'local disk':
|
500
|
+
|
501
|
+
REVISION ID CHANGED
|
502
|
+
===================
|
503
|
+
|
504
|
+
@@ -1,2 +1,2 @@
|
505
|
+
-cf4b8a020bdc1ba6914093a8a07a5514cce8a3a2979a967b1f32ea704a61785b
|
506
|
+
+304566f86a620aae85797a3c491a51fb8c6ecf996407e77b8063aa3ee59672c5
|
507
|
+
|
508
|
+
ADDED COOKBOOKS
|
509
|
+
===============
|
510
|
+
|
511
|
+
bluepill
|
512
|
+
--------
|
513
|
+
|
514
|
+
@@ -1 +1,12 @@
|
515
|
+
+{
|
516
|
+
+ "version": "2.3.2",
|
517
|
+
+ "identifier": "9c6990944d9a347dec8bd375e707ba0aecdc17cd",
|
518
|
+
+ "dotted_decimal_identifier": "69437059924760478.24393100994078142.115593340606828",
|
519
|
+
+ "cache_key": "bluepill-2.3.2-supermarket.chef.io",
|
520
|
+
+ "origin": "https://supermarket.chef.io/api/v1/cookbooks/bluepill/versions/2.3.2/download",
|
521
|
+
+ "source_options": {
|
522
|
+
+ "artifactserver": "https://supermarket.chef.io/api/v1/cookbooks/bluepill/versions/2.3.2/download",
|
523
|
+
+ "version": "2.3.2"
|
524
|
+
+ }
|
525
|
+
+}
|
526
|
+
|
527
|
+
E
|
528
|
+
differ.run_report
|
529
|
+
expect(output).to eq(expected_message)
|
530
|
+
end
|
531
|
+
|
532
|
+
end
|
533
|
+
|
534
|
+
context "with a modified cookbook" do
|
535
|
+
|
536
|
+
let(:new_lock) do
|
537
|
+
old_lock.dup.tap do |n|
|
538
|
+
n["revision_id"] = new_rev_id
|
539
|
+
n["cookbook_locks"] = n["cookbook_locks"].dup
|
540
|
+
policyfile_demo = n["cookbook_locks"]["policyfile_demo"].dup
|
541
|
+
policyfile_demo["identifier"] = "f04cc40faf628253fe7d9566d66a1733fb1afbe9"
|
542
|
+
policyfile_demo["dotted_decimal_identifier"] = "67638399371010690.23642238397896298.25512023620585"
|
543
|
+
n["cookbook_locks"]["policyfile_demo"] = policyfile_demo
|
544
|
+
end
|
545
|
+
end
|
546
|
+
|
547
|
+
it "has updates" do
|
548
|
+
expect(differ.different?).to be(true)
|
549
|
+
end
|
550
|
+
|
551
|
+
it "has an updated revision_id and cookbook_locks" do
|
552
|
+
expect(differ.updated_sections).to match_array(%w{revision_id cookbook_locks})
|
553
|
+
end
|
554
|
+
|
555
|
+
it "has modified the 'policyfile_demo' cookbook" do
|
556
|
+
expect(differ.modified_cookbooks).to eq(%w{policyfile_demo})
|
557
|
+
end
|
558
|
+
|
559
|
+
it "reports the updated revision_id and modified policyfile_demo cookbook" do
|
560
|
+
expected_message = <<-E
|
561
|
+
Policy lock 'jenkins' differs between 'git: HEAD' and 'local disk':
|
562
|
+
|
563
|
+
REVISION ID CHANGED
|
564
|
+
===================
|
565
|
+
|
566
|
+
@@ -1,2 +1,2 @@
|
567
|
+
-cf4b8a020bdc1ba6914093a8a07a5514cce8a3a2979a967b1f32ea704a61785b
|
568
|
+
+304566f86a620aae85797a3c491a51fb8c6ecf996407e77b8063aa3ee59672c5
|
569
|
+
|
570
|
+
MODIFIED COOKBOOKS
|
571
|
+
==================
|
572
|
+
|
573
|
+
policyfile_demo
|
574
|
+
---------------
|
575
|
+
|
576
|
+
@@ -1,7 +1,7 @@
|
577
|
+
{
|
578
|
+
"version": "0.1.0",
|
579
|
+
- "identifier": "ea96c99da079db9ff3cb22601638fabd5df49599",
|
580
|
+
- "dotted_decimal_identifier": "66030937227426267.45022575077627448.275691232073113",
|
581
|
+
+ "identifier": "f04cc40faf628253fe7d9566d66a1733fb1afbe9",
|
582
|
+
+ "dotted_decimal_identifier": "67638399371010690.23642238397896298.25512023620585",
|
583
|
+
"source": "cookbooks/policyfile_demo",
|
584
|
+
"cache_key": null,
|
585
|
+
"scm_info": {
|
586
|
+
|
587
|
+
E
|
588
|
+
differ.run_report
|
589
|
+
expect(output).to eq(expected_message)
|
590
|
+
end
|
591
|
+
|
592
|
+
end
|
593
|
+
|
594
|
+
context "with updated default attributes" do
|
595
|
+
|
596
|
+
let(:new_lock) do
|
597
|
+
old_lock.dup.tap do |n|
|
598
|
+
n["revision_id"] = new_rev_id
|
599
|
+
n["default_attributes"] = n["default_attributes"].dup
|
600
|
+
n["default_attributes"]["new_attr"] = "hello"
|
601
|
+
end
|
602
|
+
end
|
603
|
+
|
604
|
+
it "has updates" do
|
605
|
+
expect(differ.different?).to be(true)
|
606
|
+
end
|
607
|
+
|
608
|
+
it "has an updated revision_id and default_attributes" do
|
609
|
+
expect(differ.updated_sections).to match_array(%w{revision_id default_attributes})
|
610
|
+
end
|
611
|
+
|
612
|
+
it "reports the updated revision_id and modified attributes" do
|
613
|
+
expected_output = <<-E
|
614
|
+
Policy lock 'jenkins' differs between 'git: HEAD' and 'local disk':
|
615
|
+
|
616
|
+
REVISION ID CHANGED
|
617
|
+
===================
|
618
|
+
|
619
|
+
@@ -1,2 +1,2 @@
|
620
|
+
-cf4b8a020bdc1ba6914093a8a07a5514cce8a3a2979a967b1f32ea704a61785b
|
621
|
+
+304566f86a620aae85797a3c491a51fb8c6ecf996407e77b8063aa3ee59672c5
|
622
|
+
|
623
|
+
DEFAULT ATTRIBUTES CHANGED
|
624
|
+
==========================
|
625
|
+
|
626
|
+
@@ -1,4 +1,5 @@
|
627
|
+
{
|
628
|
+
- "greeting": "Attributes, f*** yeah"
|
629
|
+
+ "greeting": "Attributes, f*** yeah",
|
630
|
+
+ "new_attr": "hello"
|
631
|
+
}
|
632
|
+
|
633
|
+
E
|
634
|
+
differ.run_report
|
635
|
+
expect(output).to eq(expected_output)
|
636
|
+
end
|
637
|
+
|
638
|
+
end
|
639
|
+
|
640
|
+
context "with updated override_attributes" do
|
641
|
+
|
642
|
+
let(:new_lock) do
|
643
|
+
old_lock.dup.tap do |n|
|
644
|
+
n["revision_id"] = new_rev_id
|
645
|
+
n["override_attributes"] = n["override_attributes"].dup
|
646
|
+
n["override_attributes"]["new_attr"] = "ALL THE DIFF"
|
647
|
+
end
|
648
|
+
end
|
649
|
+
|
650
|
+
it "has updates" do
|
651
|
+
expect(differ.different?).to be(true)
|
652
|
+
end
|
653
|
+
|
654
|
+
it "has an updated revision_id and override_attributes" do
|
655
|
+
expect(differ.updated_sections).to match_array(%w{revision_id override_attributes})
|
656
|
+
end
|
657
|
+
|
658
|
+
it "reports the updated revision_id and override_attributes" do
|
659
|
+
expected_output = <<-E
|
660
|
+
Policy lock 'jenkins' differs between 'git: HEAD' and 'local disk':
|
661
|
+
|
662
|
+
REVISION ID CHANGED
|
663
|
+
===================
|
664
|
+
|
665
|
+
@@ -1,2 +1,2 @@
|
666
|
+
-cf4b8a020bdc1ba6914093a8a07a5514cce8a3a2979a967b1f32ea704a61785b
|
667
|
+
+304566f86a620aae85797a3c491a51fb8c6ecf996407e77b8063aa3ee59672c5
|
668
|
+
|
669
|
+
OVERRIDE ATTRIBUTES CHANGED
|
670
|
+
===========================
|
671
|
+
|
672
|
+
@@ -1,4 +1,5 @@
|
673
|
+
{
|
674
|
+
- "attr_only_updating": "use -a"
|
675
|
+
+ "attr_only_updating": "use -a",
|
676
|
+
+ "new_attr": "ALL THE DIFF"
|
677
|
+
}
|
678
|
+
|
679
|
+
E
|
680
|
+
|
681
|
+
differ.run_report
|
682
|
+
expect(output).to eq(expected_output)
|
683
|
+
end
|
684
|
+
end
|
685
|
+
|
686
|
+
end
|
687
|
+
|