chef-dk 0.5.0.rc.1 → 0.5.0

Sign up to get free protection for your applications and to get access to all the features.
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
+