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.
Files changed (45) hide show
  1. checksums.yaml +4 -4
  2. data/README.md +63 -24
  3. data/lib/chef-dk/builtin_commands.rb +2 -0
  4. data/lib/chef-dk/command/diff.rb +312 -0
  5. data/lib/chef-dk/command/push.rb +1 -1
  6. data/lib/chef-dk/command/shell_init.rb +21 -3
  7. data/lib/chef-dk/command/update.rb +28 -5
  8. data/lib/chef-dk/configurable.rb +1 -1
  9. data/lib/chef-dk/exceptions.rb +3 -0
  10. data/lib/chef-dk/pager.rb +106 -0
  11. data/lib/chef-dk/policyfile/chef_repo_cookbook_source.rb +114 -0
  12. data/lib/chef-dk/policyfile/comparison_base.rb +124 -0
  13. data/lib/chef-dk/policyfile/cookbook_sources.rb +1 -0
  14. data/lib/chef-dk/policyfile/differ.rb +266 -0
  15. data/lib/chef-dk/policyfile/dsl.rb +26 -3
  16. data/lib/chef-dk/policyfile/uploader.rb +4 -5
  17. data/lib/chef-dk/policyfile_compiler.rb +8 -0
  18. data/lib/chef-dk/policyfile_lock.rb +135 -3
  19. data/lib/chef-dk/policyfile_services/install.rb +1 -0
  20. data/lib/chef-dk/policyfile_services/update_attributes.rb +104 -0
  21. data/lib/chef-dk/service_exceptions.rb +12 -0
  22. data/lib/chef-dk/ui.rb +8 -0
  23. data/lib/chef-dk/version.rb +1 -1
  24. data/spec/spec_helper.rb +6 -0
  25. data/spec/test_helpers.rb +4 -0
  26. data/spec/unit/command/diff_spec.rb +283 -0
  27. data/spec/unit/command/shell_init_spec.rb +19 -2
  28. data/spec/unit/command/update_spec.rb +96 -0
  29. data/spec/unit/command/verify_spec.rb +0 -6
  30. data/spec/unit/fixtures/local_path_cookbooks/cookbook-with-a-dep/Berksfile +3 -0
  31. data/spec/unit/fixtures/local_path_cookbooks/cookbook-with-a-dep/README.md +4 -0
  32. data/spec/unit/fixtures/local_path_cookbooks/cookbook-with-a-dep/chefignore +96 -0
  33. data/spec/unit/fixtures/local_path_cookbooks/cookbook-with-a-dep/metadata.rb +9 -0
  34. data/spec/unit/fixtures/local_path_cookbooks/cookbook-with-a-dep/recipes/default.rb +8 -0
  35. data/spec/unit/pager_spec.rb +119 -0
  36. data/spec/unit/policyfile/chef_repo_cookbook_source_spec.rb +66 -0
  37. data/spec/unit/policyfile/comparison_base_spec.rb +343 -0
  38. data/spec/unit/policyfile/differ_spec.rb +687 -0
  39. data/spec/unit/policyfile_evaluation_spec.rb +87 -0
  40. data/spec/unit/policyfile_lock_build_spec.rb +247 -8
  41. data/spec/unit/policyfile_lock_serialization_spec.rb +47 -0
  42. data/spec/unit/policyfile_services/export_repo_spec.rb +2 -0
  43. data/spec/unit/policyfile_services/push_spec.rb +2 -0
  44. data/spec/unit/policyfile_services/update_attributes_spec.rb +217 -0
  45. 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
+