chef-dk 0.9.0 → 0.10.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 (306) hide show
  1. checksums.yaml +4 -4
  2. data/CONTRIBUTING.md +190 -190
  3. data/Gemfile +26 -26
  4. data/LICENSE +201 -201
  5. data/README.md +292 -276
  6. data/Rakefile +18 -18
  7. data/bin/chef +25 -25
  8. data/chef-dk.gemspec +62 -0
  9. data/lib/chef-dk.rb +19 -19
  10. data/lib/chef-dk/authenticated_http.rb +40 -40
  11. data/lib/chef-dk/builtin_commands.rb +60 -60
  12. data/lib/chef-dk/chef_runner.rb +98 -98
  13. data/lib/chef-dk/cli.rb +200 -200
  14. data/lib/chef-dk/command/base.rb +79 -79
  15. data/lib/chef-dk/command/clean_policy_cookbooks.rb +116 -116
  16. data/lib/chef-dk/command/clean_policy_revisions.rb +113 -113
  17. data/lib/chef-dk/command/delete_policy.rb +122 -122
  18. data/lib/chef-dk/command/delete_policy_group.rb +122 -122
  19. data/lib/chef-dk/command/diff.rb +316 -316
  20. data/lib/chef-dk/command/env.rb +90 -90
  21. data/lib/chef-dk/command/exec.rb +45 -45
  22. data/lib/chef-dk/command/export.rb +151 -151
  23. data/lib/chef-dk/command/gem.rb +47 -47
  24. data/lib/chef-dk/command/generate.rb +120 -120
  25. data/lib/chef-dk/command/generator_commands.rb +80 -80
  26. data/lib/chef-dk/command/generator_commands/app.rb +107 -107
  27. data/lib/chef-dk/command/generator_commands/attribute.rb +37 -37
  28. data/lib/chef-dk/command/generator_commands/base.rb +121 -121
  29. data/lib/chef-dk/command/generator_commands/cookbook.rb +153 -119
  30. data/lib/chef-dk/command/generator_commands/cookbook_code_file.rb +100 -100
  31. data/lib/chef-dk/command/generator_commands/cookbook_file.rb +45 -45
  32. data/lib/chef-dk/command/generator_commands/generator_generator.rb +177 -177
  33. data/lib/chef-dk/command/generator_commands/lwrp.rb +36 -36
  34. data/lib/chef-dk/command/generator_commands/policyfile.rb +127 -86
  35. data/lib/chef-dk/command/generator_commands/recipe.rb +36 -36
  36. data/lib/chef-dk/command/generator_commands/repo.rb +125 -96
  37. data/lib/chef-dk/command/generator_commands/template.rb +46 -46
  38. data/lib/chef-dk/command/install.rb +121 -121
  39. data/lib/chef-dk/command/provision.rb +438 -438
  40. data/lib/chef-dk/command/push.rb +118 -118
  41. data/lib/chef-dk/command/push_archive.rb +126 -126
  42. data/lib/chef-dk/command/shell_init.rb +180 -180
  43. data/lib/chef-dk/command/show_policy.rb +165 -165
  44. data/lib/chef-dk/command/undelete.rb +155 -155
  45. data/lib/chef-dk/command/update.rb +129 -129
  46. data/lib/chef-dk/command/verify.rb +534 -490
  47. data/lib/chef-dk/commands_map.rb +115 -115
  48. data/lib/chef-dk/completions/bash.sh.erb +5 -5
  49. data/lib/chef-dk/completions/chef.fish.erb +10 -10
  50. data/lib/chef-dk/completions/zsh.zsh.erb +21 -21
  51. data/lib/chef-dk/component_test.rb +173 -171
  52. data/lib/chef-dk/configurable.rb +57 -57
  53. data/lib/chef-dk/cookbook_metadata.rb +45 -45
  54. data/lib/chef-dk/cookbook_omnifetch.rb +32 -32
  55. data/lib/chef-dk/cookbook_profiler/git.rb +151 -151
  56. data/lib/chef-dk/cookbook_profiler/identifiers.rb +72 -72
  57. data/lib/chef-dk/cookbook_profiler/null_scm.rb +32 -32
  58. data/lib/chef-dk/exceptions.rb +126 -113
  59. data/lib/chef-dk/generator.rb +163 -163
  60. data/lib/chef-dk/helpers.rb +159 -159
  61. data/lib/chef-dk/pager.rb +106 -106
  62. data/lib/chef-dk/policyfile/chef_repo_cookbook_source.rb +138 -122
  63. data/lib/chef-dk/policyfile/chef_server_cookbook_source.rb +54 -54
  64. data/lib/chef-dk/policyfile/community_cookbook_source.rb +97 -82
  65. data/lib/chef-dk/policyfile/comparison_base.rb +124 -124
  66. data/lib/chef-dk/policyfile/cookbook_location_specification.rb +141 -133
  67. data/lib/chef-dk/policyfile/cookbook_locks.rb +466 -466
  68. data/lib/chef-dk/policyfile/cookbook_sources.rb +22 -21
  69. data/lib/chef-dk/policyfile/delivery_supermarket_source.rb +90 -0
  70. data/lib/chef-dk/policyfile/differ.rb +266 -266
  71. data/lib/chef-dk/policyfile/dsl.rb +223 -197
  72. data/lib/chef-dk/policyfile/lister.rb +232 -232
  73. data/lib/chef-dk/policyfile/null_cookbook_source.rb +45 -45
  74. data/lib/chef-dk/policyfile/read_cookbook_for_compat_mode_upload.rb +124 -124
  75. data/lib/chef-dk/policyfile/reports/install.rb +70 -70
  76. data/lib/chef-dk/policyfile/reports/table_printer.rb +58 -58
  77. data/lib/chef-dk/policyfile/reports/upload.rb +70 -70
  78. data/lib/chef-dk/policyfile/solution_dependencies.rb +298 -298
  79. data/lib/chef-dk/policyfile/storage_config.rb +100 -100
  80. data/lib/chef-dk/policyfile/undo_record.rb +142 -142
  81. data/lib/chef-dk/policyfile/undo_stack.rb +130 -130
  82. data/lib/chef-dk/policyfile/uploader.rb +213 -213
  83. data/lib/chef-dk/policyfile_compiler.rb +380 -322
  84. data/lib/chef-dk/policyfile_lock.rb +552 -552
  85. data/lib/chef-dk/policyfile_services/clean_policies.rb +95 -95
  86. data/lib/chef-dk/policyfile_services/clean_policy_cookbooks.rb +125 -125
  87. data/lib/chef-dk/policyfile_services/export_repo.rb +309 -309
  88. data/lib/chef-dk/policyfile_services/install.rb +125 -125
  89. data/lib/chef-dk/policyfile_services/push.rb +114 -114
  90. data/lib/chef-dk/policyfile_services/push_archive.rb +173 -173
  91. data/lib/chef-dk/policyfile_services/rm_policy.rb +142 -142
  92. data/lib/chef-dk/policyfile_services/rm_policy_group.rb +86 -86
  93. data/lib/chef-dk/policyfile_services/show_policy.rb +237 -237
  94. data/lib/chef-dk/policyfile_services/undelete.rb +108 -108
  95. data/lib/chef-dk/policyfile_services/update_attributes.rb +104 -104
  96. data/lib/chef-dk/service_exception_inspectors.rb +25 -25
  97. data/lib/chef-dk/service_exception_inspectors/base.rb +40 -40
  98. data/lib/chef-dk/service_exception_inspectors/http.rb +121 -121
  99. data/lib/chef-dk/service_exceptions.rb +143 -143
  100. data/lib/chef-dk/shell_out.rb +36 -36
  101. data/lib/chef-dk/skeletons/code_generator/files/default/Berksfile +3 -3
  102. data/lib/chef-dk/skeletons/code_generator/files/default/chefignore +100 -100
  103. data/lib/chef-dk/skeletons/code_generator/files/default/cookbook_readmes/README-policy.md +9 -9
  104. data/lib/chef-dk/skeletons/code_generator/files/default/cookbook_readmes/README.md +54 -54
  105. data/lib/chef-dk/skeletons/code_generator/files/default/gitignore +16 -16
  106. data/lib/chef-dk/skeletons/code_generator/files/default/repo/README.md +28 -28
  107. data/lib/chef-dk/skeletons/code_generator/files/default/repo/cookbooks/example/README.md +27 -27
  108. data/lib/chef-dk/skeletons/code_generator/files/default/repo/cookbooks/example/attributes/default.rb +7 -7
  109. data/lib/chef-dk/skeletons/code_generator/files/default/repo/cookbooks/example/metadata.rb +3 -3
  110. data/lib/chef-dk/skeletons/code_generator/files/default/repo/cookbooks/example/recipes/default.rb +8 -8
  111. data/lib/chef-dk/skeletons/code_generator/files/default/repo/data_bags/README.md +58 -58
  112. data/lib/chef-dk/skeletons/code_generator/files/default/repo/data_bags/example/example_item.json +3 -3
  113. data/lib/chef-dk/skeletons/code_generator/files/default/repo/dot-chef-repo.txt +6 -0
  114. data/lib/chef-dk/skeletons/code_generator/files/default/repo/environments/README.md +9 -9
  115. data/lib/chef-dk/skeletons/code_generator/files/default/repo/environments/example.json +12 -12
  116. data/lib/chef-dk/skeletons/code_generator/files/default/repo/policies/README.md +24 -0
  117. data/lib/chef-dk/skeletons/code_generator/files/default/repo/roles/README.md +9 -9
  118. data/lib/chef-dk/skeletons/code_generator/files/default/repo/roles/example.json +12 -12
  119. data/lib/chef-dk/skeletons/code_generator/files/default/serverspec_spec_helper.rb +8 -8
  120. data/lib/chef-dk/skeletons/code_generator/files/default/spec_helper.rb +2 -2
  121. data/lib/chef-dk/skeletons/code_generator/files/default/spec_helper_policyfile.rb +2 -0
  122. data/lib/chef-dk/skeletons/code_generator/metadata.rb +8 -8
  123. data/lib/chef-dk/skeletons/code_generator/recipes/app.rb +97 -97
  124. data/lib/chef-dk/skeletons/code_generator/recipes/attribute.rb +12 -12
  125. data/lib/chef-dk/skeletons/code_generator/recipes/cookbook.rb +117 -104
  126. data/lib/chef-dk/skeletons/code_generator/recipes/cookbook_file.rb +24 -24
  127. data/lib/chef-dk/skeletons/code_generator/recipes/lwrp.rb +23 -23
  128. data/lib/chef-dk/skeletons/code_generator/recipes/policyfile.rb +8 -8
  129. data/lib/chef-dk/skeletons/code_generator/recipes/recipe.rb +27 -27
  130. data/lib/chef-dk/skeletons/code_generator/recipes/repo.rb +67 -48
  131. data/lib/chef-dk/skeletons/code_generator/recipes/template.rb +32 -32
  132. data/lib/chef-dk/skeletons/code_generator/templates/default/LICENSE.all_rights.erb +3 -3
  133. data/lib/chef-dk/skeletons/code_generator/templates/default/LICENSE.apache2.erb +201 -201
  134. data/lib/chef-dk/skeletons/code_generator/templates/default/LICENSE.gplv2.erb +339 -339
  135. data/lib/chef-dk/skeletons/code_generator/templates/default/LICENSE.gplv3.erb +674 -674
  136. data/lib/chef-dk/skeletons/code_generator/templates/default/LICENSE.mit.erb +21 -21
  137. data/lib/chef-dk/skeletons/code_generator/templates/default/Policyfile.rb.erb +25 -20
  138. data/lib/chef-dk/skeletons/code_generator/templates/default/README.md.erb +4 -4
  139. data/lib/chef-dk/skeletons/code_generator/templates/default/kitchen.yml.erb +21 -16
  140. data/lib/chef-dk/skeletons/code_generator/templates/default/kitchen_policyfile.yml.erb +32 -27
  141. data/lib/chef-dk/skeletons/code_generator/templates/default/metadata.rb.erb +7 -7
  142. data/lib/chef-dk/skeletons/code_generator/templates/default/recipe.rb.erb +5 -5
  143. data/lib/chef-dk/skeletons/code_generator/templates/default/recipe_spec.rb.erb +20 -20
  144. data/lib/chef-dk/skeletons/code_generator/templates/default/repo/gitignore.erb +11 -11
  145. data/lib/chef-dk/skeletons/code_generator/templates/default/serverspec_default_spec.rb.erb +9 -9
  146. data/lib/chef-dk/ui.rb +58 -58
  147. data/lib/chef-dk/version.rb +20 -20
  148. data/lib/kitchen/provisioner/policyfile_zero.rb +189 -193
  149. data/spec/shared/a_file_generator.rb +125 -125
  150. data/spec/shared/a_generated_file.rb +12 -12
  151. data/spec/shared/command_with_ui_object.rb +11 -11
  152. data/spec/shared/custom_generator_cookbook.rb +117 -117
  153. data/spec/shared/fixture_cookbook_checksums.rb +47 -47
  154. data/spec/shared/setup_git_cookbooks.rb +53 -53
  155. data/spec/spec_helper.rb +50 -49
  156. data/spec/test_helpers.rb +84 -84
  157. data/spec/unit/chef_runner_spec.rb +111 -111
  158. data/spec/unit/cli_spec.rb +357 -357
  159. data/spec/unit/command/base_spec.rb +169 -169
  160. data/spec/unit/command/clean_policy_cookbooks_spec.rb +181 -181
  161. data/spec/unit/command/clean_policy_revisions_spec.rb +181 -181
  162. data/spec/unit/command/delete_policy_group_spec.rb +207 -207
  163. data/spec/unit/command/delete_policy_spec.rb +207 -207
  164. data/spec/unit/command/diff_spec.rb +312 -312
  165. data/spec/unit/command/env_spec.rb +52 -52
  166. data/spec/unit/command/exec_spec.rb +179 -179
  167. data/spec/unit/command/export_spec.rb +189 -189
  168. data/spec/unit/command/generate_spec.rb +142 -142
  169. data/spec/unit/command/generator_commands/app_spec.rb +169 -169
  170. data/spec/unit/command/generator_commands/attribute_spec.rb +32 -32
  171. data/spec/unit/command/generator_commands/cookbook_file_spec.rb +32 -32
  172. data/spec/unit/command/generator_commands/cookbook_spec.rb +450 -320
  173. data/spec/unit/command/generator_commands/generator_generator_spec.rb +229 -229
  174. data/spec/unit/command/generator_commands/lwrp_spec.rb +32 -32
  175. data/spec/unit/command/generator_commands/policyfile_spec.rb +225 -125
  176. data/spec/unit/command/generator_commands/recipe_spec.rb +34 -34
  177. data/spec/unit/command/generator_commands/repo_spec.rb +367 -283
  178. data/spec/unit/command/generator_commands/template_spec.rb +32 -32
  179. data/spec/unit/command/install_spec.rb +179 -179
  180. data/spec/unit/command/provision_spec.rb +592 -592
  181. data/spec/unit/command/push_archive_spec.rb +153 -153
  182. data/spec/unit/command/push_spec.rb +199 -199
  183. data/spec/unit/command/shell_init_spec.rb +329 -329
  184. data/spec/unit/command/show_policy_spec.rb +235 -235
  185. data/spec/unit/command/undelete_spec.rb +246 -246
  186. data/spec/unit/command/update_spec.rb +251 -251
  187. data/spec/unit/command/verify_spec.rb +325 -323
  188. data/spec/unit/commands_map_spec.rb +57 -57
  189. data/spec/unit/component_test_spec.rb +126 -126
  190. data/spec/unit/cookbook_metadata_spec.rb +98 -98
  191. data/spec/unit/cookbook_profiler/git_spec.rb +176 -176
  192. data/spec/unit/cookbook_profiler/identifiers_spec.rb +83 -83
  193. data/spec/unit/fixtures/chef-runner-cookbooks/test_cookbook/recipes/recipe_one.rb +9 -9
  194. data/spec/unit/fixtures/chef-runner-cookbooks/test_cookbook/recipes/recipe_two.rb +9 -9
  195. data/spec/unit/fixtures/command/cli_test_command.rb +26 -26
  196. data/spec/unit/fixtures/command/explicit_path_example.rb +7 -7
  197. data/spec/unit/fixtures/cookbook_cache/baz-f59ee7a5bca6a4e606b67f7f856b768d847c39bb/.kitchen.yml +16 -16
  198. data/spec/unit/fixtures/cookbook_cache/baz-f59ee7a5bca6a4e606b67f7f856b768d847c39bb/Berksfile +3 -3
  199. data/spec/unit/fixtures/cookbook_cache/baz-f59ee7a5bca6a4e606b67f7f856b768d847c39bb/README.md +4 -4
  200. data/spec/unit/fixtures/cookbook_cache/baz-f59ee7a5bca6a4e606b67f7f856b768d847c39bb/chefignore +96 -96
  201. data/spec/unit/fixtures/cookbook_cache/baz-f59ee7a5bca6a4e606b67f7f856b768d847c39bb/metadata.rb +8 -8
  202. data/spec/unit/fixtures/cookbook_cache/baz-f59ee7a5bca6a4e606b67f7f856b768d847c39bb/recipes/default.rb +8 -8
  203. data/spec/unit/fixtures/cookbook_cache/dep_of_bar-1.2.3/.kitchen.yml +16 -16
  204. data/spec/unit/fixtures/cookbook_cache/dep_of_bar-1.2.3/Berksfile +3 -3
  205. data/spec/unit/fixtures/cookbook_cache/dep_of_bar-1.2.3/README.md +4 -4
  206. data/spec/unit/fixtures/cookbook_cache/dep_of_bar-1.2.3/chefignore +96 -96
  207. data/spec/unit/fixtures/cookbook_cache/dep_of_bar-1.2.3/metadata.rb +8 -8
  208. data/spec/unit/fixtures/cookbook_cache/dep_of_bar-1.2.3/recipes/default.rb +8 -8
  209. data/spec/unit/fixtures/cookbook_cache/foo-1.0.0/.kitchen.yml +16 -16
  210. data/spec/unit/fixtures/cookbook_cache/foo-1.0.0/Berksfile +3 -3
  211. data/spec/unit/fixtures/cookbook_cache/foo-1.0.0/README.md +4 -4
  212. data/spec/unit/fixtures/cookbook_cache/foo-1.0.0/chefignore +96 -96
  213. data/spec/unit/fixtures/cookbook_cache/foo-1.0.0/metadata.rb +8 -8
  214. data/spec/unit/fixtures/cookbook_cache/foo-1.0.0/recipes/default.rb +8 -8
  215. data/spec/unit/fixtures/cookbooks_api/pruned_small_universe.json +1321 -1321
  216. data/spec/unit/fixtures/cookbooks_api/small_universe.json +2987 -2987
  217. data/spec/unit/fixtures/cookbooks_api/universe.json +1 -1
  218. data/spec/unit/fixtures/cookbooks_api/update_fixtures.rb +36 -36
  219. data/spec/unit/fixtures/dev_cookbooks/README.md +16 -16
  220. data/spec/unit/fixtures/eg_omnibus_dir/valid/embedded/apps/berkshelf/integration_test +2 -2
  221. data/spec/unit/fixtures/eg_omnibus_dir/valid/embedded/apps/berkshelf/verify_me +5 -5
  222. data/spec/unit/fixtures/eg_omnibus_dir/valid/embedded/apps/chef/verify_me +3 -3
  223. data/spec/unit/fixtures/eg_omnibus_dir/valid/embedded/apps/test-kitchen/verify_me +2 -2
  224. data/spec/unit/fixtures/example_cookbook/.gitignore +17 -17
  225. data/spec/unit/fixtures/example_cookbook/.kitchen.yml +16 -16
  226. data/spec/unit/fixtures/example_cookbook/Berksfile +3 -3
  227. data/spec/unit/fixtures/example_cookbook/README.md +4 -4
  228. data/spec/unit/fixtures/example_cookbook/chefignore +96 -96
  229. data/spec/unit/fixtures/example_cookbook/metadata.rb +8 -8
  230. data/spec/unit/fixtures/example_cookbook/recipes/default.rb +8 -8
  231. data/spec/unit/fixtures/example_cookbook_metadata_json_only/.gitignore +17 -17
  232. data/spec/unit/fixtures/example_cookbook_metadata_json_only/.kitchen.yml +16 -16
  233. data/spec/unit/fixtures/example_cookbook_metadata_json_only/Berksfile +3 -3
  234. data/spec/unit/fixtures/example_cookbook_metadata_json_only/README.md +4 -4
  235. data/spec/unit/fixtures/example_cookbook_metadata_json_only/chefignore +96 -96
  236. data/spec/unit/fixtures/example_cookbook_metadata_json_only/metadata.json +5 -5
  237. data/spec/unit/fixtures/example_cookbook_metadata_json_only/recipes/default.rb +8 -8
  238. data/spec/unit/fixtures/example_cookbook_no_metadata/.gitignore +17 -17
  239. data/spec/unit/fixtures/example_cookbook_no_metadata/.kitchen.yml +16 -16
  240. data/spec/unit/fixtures/example_cookbook_no_metadata/Berksfile +3 -3
  241. data/spec/unit/fixtures/example_cookbook_no_metadata/README.md +4 -4
  242. data/spec/unit/fixtures/example_cookbook_no_metadata/chefignore +96 -96
  243. data/spec/unit/fixtures/example_cookbook_no_metadata/recipes/default.rb +8 -8
  244. data/spec/unit/fixtures/local_path_cookbooks/another-local-cookbook/README.md +4 -4
  245. data/spec/unit/fixtures/local_path_cookbooks/another-local-cookbook/chefignore +96 -96
  246. data/spec/unit/fixtures/local_path_cookbooks/another-local-cookbook/metadata.rb +8 -8
  247. data/spec/unit/fixtures/local_path_cookbooks/another-local-cookbook/recipes/default.rb +8 -8
  248. data/spec/unit/fixtures/local_path_cookbooks/cookbook-with-a-dep/Berksfile +3 -3
  249. data/spec/unit/fixtures/local_path_cookbooks/cookbook-with-a-dep/README.md +4 -4
  250. data/spec/unit/fixtures/local_path_cookbooks/cookbook-with-a-dep/chefignore +96 -96
  251. data/spec/unit/fixtures/local_path_cookbooks/cookbook-with-a-dep/metadata.rb +9 -9
  252. data/spec/unit/fixtures/local_path_cookbooks/cookbook-with-a-dep/recipes/default.rb +8 -8
  253. data/spec/unit/fixtures/local_path_cookbooks/local-cookbook/.kitchen.yml +16 -16
  254. data/spec/unit/fixtures/local_path_cookbooks/local-cookbook/Berksfile +3 -3
  255. data/spec/unit/fixtures/local_path_cookbooks/local-cookbook/README.md +4 -4
  256. data/spec/unit/fixtures/local_path_cookbooks/local-cookbook/chefignore +96 -96
  257. data/spec/unit/fixtures/local_path_cookbooks/local-cookbook/metadata.rb +8 -8
  258. data/spec/unit/fixtures/local_path_cookbooks/local-cookbook/recipes/default.rb +8 -8
  259. data/spec/unit/fixtures/local_path_cookbooks/metadata-missing/README.md +2 -2
  260. data/spec/unit/fixtures/local_path_cookbooks/noignore-f59ee7a5bca6a4e606b67f7f856b768d847c39bb/.kitchen.yml +16 -16
  261. data/spec/unit/fixtures/local_path_cookbooks/noignore-f59ee7a5bca6a4e606b67f7f856b768d847c39bb/README.md +4 -4
  262. data/spec/unit/fixtures/local_path_cookbooks/noignore-f59ee7a5bca6a4e606b67f7f856b768d847c39bb/metadata.rb +8 -8
  263. data/spec/unit/fixtures/local_path_cookbooks/noignore-f59ee7a5bca6a4e606b67f7f856b768d847c39bb/recipes/default.rb +8 -8
  264. data/spec/unit/generator_spec.rb +120 -120
  265. data/spec/unit/helpers_spec.rb +92 -92
  266. data/spec/unit/pager_spec.rb +119 -119
  267. data/spec/unit/policyfile/chef_repo_cookbook_source_spec.rb +93 -66
  268. data/spec/unit/policyfile/chef_server_cookbook_source_spec.rb +34 -34
  269. data/spec/unit/policyfile/community_cookbook_source_spec.rb +84 -51
  270. data/spec/unit/policyfile/comparison_base_spec.rb +343 -343
  271. data/spec/unit/policyfile/cookbook_location_specification_spec.rb +257 -252
  272. data/spec/unit/policyfile/cookbook_locks_spec.rb +529 -529
  273. data/spec/unit/policyfile/delivery_supermarket_source_spec.rb +130 -0
  274. data/spec/unit/policyfile/differ_spec.rb +687 -687
  275. data/spec/unit/policyfile/lister_spec.rb +272 -272
  276. data/spec/unit/policyfile/null_cookbook_source_spec.rb +35 -35
  277. data/spec/unit/policyfile/read_cookbook_for_compat_mode_upload_spec.rb +92 -92
  278. data/spec/unit/policyfile/reports/install_spec.rb +115 -115
  279. data/spec/unit/policyfile/reports/upload_spec.rb +96 -96
  280. data/spec/unit/policyfile/solution_dependencies_spec.rb +145 -145
  281. data/spec/unit/policyfile/storage_config_spec.rb +172 -172
  282. data/spec/unit/policyfile/undo_record_spec.rb +260 -260
  283. data/spec/unit/policyfile/undo_stack_spec.rb +266 -266
  284. data/spec/unit/policyfile/uploader_spec.rb +410 -410
  285. data/spec/unit/policyfile_demands_spec.rb +997 -876
  286. data/spec/unit/policyfile_evaluation_spec.rb +557 -441
  287. data/spec/unit/policyfile_lock_build_spec.rb +1056 -1056
  288. data/spec/unit/policyfile_lock_install_spec.rb +138 -138
  289. data/spec/unit/policyfile_lock_serialization_spec.rb +425 -425
  290. data/spec/unit/policyfile_lock_validation_spec.rb +611 -611
  291. data/spec/unit/policyfile_services/clean_policies_spec.rb +236 -236
  292. data/spec/unit/policyfile_services/clean_policy_cookbooks_spec.rb +275 -275
  293. data/spec/unit/policyfile_services/export_repo_spec.rb +439 -439
  294. data/spec/unit/policyfile_services/install_spec.rb +191 -191
  295. data/spec/unit/policyfile_services/push_archive_spec.rb +345 -345
  296. data/spec/unit/policyfile_services/push_spec.rb +233 -233
  297. data/spec/unit/policyfile_services/rm_policy_group_spec.rb +241 -241
  298. data/spec/unit/policyfile_services/rm_policy_spec.rb +266 -266
  299. data/spec/unit/policyfile_services/show_policy_spec.rb +889 -889
  300. data/spec/unit/policyfile_services/undelete_spec.rb +304 -304
  301. data/spec/unit/policyfile_services/update_attributes_spec.rb +217 -217
  302. data/spec/unit/service_exception_inspectors/base_spec.rb +43 -43
  303. data/spec/unit/service_exception_inspectors/http_spec.rb +140 -140
  304. data/spec/unit/shell_out_spec.rb +34 -34
  305. data/warning.txt +9 -0
  306. metadata +18 -5
@@ -1,529 +1,529 @@
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/cookbook_locks'
20
-
21
- shared_examples_for "Cookbook Lock" do
22
-
23
- let(:cookbook_lock_data) { cookbook_lock.to_lock }
24
-
25
- it "has a cookbook name" do
26
- expect(cookbook_lock.name).to eq(cookbook_name)
27
- end
28
-
29
- it "has a source_options attribute" do
30
- cookbook_lock.source_options = { artifactserver: "https://artifacts.example.com/nginx/1.0.0/download" }
31
- expect(cookbook_lock.source_options).to eq({ artifactserver: "https://artifacts.example.com/nginx/1.0.0/download" })
32
- end
33
-
34
- it "has an identifier attribute" do
35
- cookbook_lock.identifier = "my-opaque-id"
36
- expect(cookbook_lock.identifier).to eq("my-opaque-id")
37
- end
38
-
39
- it "has a dotted_decimal_identifier attribute" do
40
- cookbook_lock.dotted_decimal_identifier = "123.456.789"
41
- expect(cookbook_lock.dotted_decimal_identifier).to eq("123.456.789")
42
- end
43
-
44
- it "has a version attribute" do
45
- cookbook_lock.version = "1.2.3"
46
- expect(cookbook_lock.version).to eq("1.2.3")
47
- end
48
-
49
- it "has a storage config" do
50
- expect(cookbook_lock.storage_config).to eq(storage_config)
51
- end
52
-
53
- context "when the underlying cookbook has not been mutated, or #refresh! has not been called" do
54
-
55
- it "is not updated" do
56
- expect(cookbook_lock).to_not be_updated
57
- end
58
-
59
- it "does not have an updated identifier" do
60
- expect(cookbook_lock.identifier_updated?).to be false
61
- end
62
-
63
- it "does not have an updated version" do
64
- expect(cookbook_lock.version_updated?).to be false
65
- end
66
-
67
- end
68
-
69
- context "when version and identifier attributes are populated" do
70
-
71
- before do
72
- allow(cookbook_lock).to receive(:validate!)
73
-
74
- cookbook_lock.identifier = "my-opaque-id"
75
- cookbook_lock.dotted_decimal_identifier = "123.456.789"
76
- cookbook_lock.version = "1.2.3"
77
- cookbook_lock.source_options = { :sourcekey => "location info" }
78
- end
79
-
80
- it "includes the identifier in the lock data" do
81
- expect(cookbook_lock_data["identifier"]).to eq("my-opaque-id")
82
- end
83
-
84
- it "includes the dotted decimal identifier in the lock data" do
85
- expect(cookbook_lock_data["dotted_decimal_identifier"]).to eq("123.456.789")
86
- end
87
-
88
- it "includes the version in lock data" do
89
- expect(cookbook_lock_data["version"]).to eq("1.2.3")
90
- end
91
-
92
- it "includes the source_options in lock data" do
93
- expect(cookbook_lock_data["source_options"]).to eq({ :sourcekey => "location info" })
94
- end
95
-
96
- it "creates a CookbookLocationSpecification with the source and version data" do
97
- location_spec = cookbook_lock.cookbook_location_spec
98
- expect(location_spec.name).to eq(cookbook_name)
99
- expect(location_spec.version_constraint).to eq(Semverse::Constraint.new("= 1.2.3"))
100
- expect(location_spec.source_options).to eq({ sourcekey: "location info" })
101
- end
102
-
103
- it "delegates #dependencies to cookbook_location_spec" do
104
- deps = [ [ "foo", ">= 0.0.0"], [ "bar", "~> 2.1" ] ]
105
- expect(cookbook_lock.cookbook_location_spec).to receive(:dependencies).and_return(deps)
106
- expect(cookbook_lock.dependencies).to eq(deps)
107
- end
108
-
109
- it "delegates #installed? to the CookbookLocationSpecification" do
110
- location_spec = cookbook_lock.cookbook_location_spec
111
- expect(location_spec).to receive(:installed?).and_return(true)
112
- expect(cookbook_lock).to be_installed
113
- expect(location_spec).to receive(:installed?).and_return(false)
114
- expect(cookbook_lock).to_not be_installed
115
- end
116
-
117
- end
118
-
119
- context "when created from lock data" do
120
-
121
- let(:lock_data) do
122
- {
123
- "identifier" => "my-opaque-id",
124
- "dotted_decimal_identifier" => "123.456.789",
125
- "version" => "1.2.3",
126
- "source_options" => { "sourcekey" => "location info" },
127
- "cache_key" => nil,
128
- "source" => "cookbooks_local_path"
129
- }
130
- end
131
-
132
- before do
133
- cookbook_lock.build_from_lock_data(lock_data)
134
- end
135
-
136
- it "sets the identifier attribute" do
137
- expect(cookbook_lock.identifier).to eq("my-opaque-id")
138
- end
139
-
140
- it "sets the dotted_decimal_identifier attribute" do
141
- expect(cookbook_lock.dotted_decimal_identifier).to eq("123.456.789")
142
- end
143
-
144
- it "sets the version attribute" do
145
- expect(cookbook_lock.version).to eq("1.2.3")
146
- end
147
-
148
- it "sets the source options" do
149
- expect(cookbook_lock.source_options).to eq({ sourcekey: "location info" })
150
- end
151
- end
152
-
153
- end
154
-
155
-
156
- describe ChefDK::Policyfile::CachedCookbook do
157
-
158
- let(:cookbook_name) { "nginx" }
159
-
160
- let(:storage_config) { ChefDK::Policyfile::StorageConfig.new }
161
-
162
- let(:cookbook_lock) do
163
- described_class.new(cookbook_name, storage_config)
164
- end
165
-
166
- include_examples "Cookbook Lock"
167
-
168
- it "has a cache_key attribute" do
169
- cookbook_lock.cache_key = "nginx-1.0.0-example.com"
170
- expect(cookbook_lock.cache_key).to eq("nginx-1.0.0-example.com")
171
- end
172
-
173
- it "has an origin attribute" do
174
- cookbook_lock.origin = "https://artifacts.example.com/nginx/1.0.0/download"
175
- expect(cookbook_lock.origin).to eq("https://artifacts.example.com/nginx/1.0.0/download")
176
- end
177
-
178
- it "errors locating the cookbook when the cache key is not set" do
179
- expect { cookbook_lock.cookbook_path }.to raise_error(ChefDK::MissingCookbookLockData)
180
- end
181
-
182
- it "ignores calls to #refresh!" do
183
- expect { cookbook_lock.refresh! }.to_not raise_error
184
- end
185
-
186
- context "when populated with valid data" do
187
-
188
- let(:cookbook_name) { "foo" }
189
-
190
- let(:cache_path) { File.join(fixtures_path, "cached_cookbooks") }
191
-
192
- before do
193
- cookbook_lock.cache_key = "foo-1.0.0"
194
-
195
- storage_config.cache_path = cache_path
196
- end
197
-
198
- it "gives the path to the cookbook in the cache" do
199
- expect(cookbook_lock.cookbook_path).to eq(File.join(cache_path, "foo-1.0.0"))
200
- end
201
-
202
- end
203
-
204
- end
205
-
206
- describe ChefDK::Policyfile::LocalCookbook do
207
-
208
- let(:cookbook_name) { "nginx" }
209
-
210
- let(:storage_config) { ChefDK::Policyfile::StorageConfig.new }
211
-
212
- let(:scm_profiler) { instance_double("ChefDK::CookbookProfiler::Git", profile_data: {}) }
213
-
214
- let(:cookbook_lock) do
215
- lock = described_class.new(cookbook_name, storage_config)
216
- allow(lock).to receive(:scm_profiler).and_return(scm_profiler)
217
- lock
218
- end
219
-
220
- include_examples "Cookbook Lock"
221
-
222
- describe "gathering identifier info" do
223
- let(:identifiers) do
224
- instance_double("ChefDK::CookbookProfiler::Identifiers",
225
- content_identifier: "abc123",
226
- dotted_decimal_identifier: "111.222.333",
227
- semver_version: "1.2.3")
228
- end
229
-
230
- before do
231
- allow(cookbook_lock).to receive(:identifiers).and_return(identifiers)
232
- cookbook_lock.gather_profile_data
233
- end
234
-
235
- it "sets the content identifier" do
236
- expect(cookbook_lock.identifier).to eq("abc123")
237
- end
238
-
239
- it "sets the backwards compatible dotted decimal identifer equivalent" do
240
- expect(cookbook_lock.dotted_decimal_identifier).to eq("111.222.333")
241
- end
242
-
243
- it "collects the 'real' SemVer version of the cookbook" do
244
- expect(cookbook_lock.version).to eq("1.2.3")
245
- end
246
-
247
- end
248
-
249
- describe "selecting an SCM profiler" do
250
-
251
- let(:cookbook_source_relpath) { "nginx" }
252
-
253
- let(:cookbook_source_path) do
254
- path = File.join(tempdir, cookbook_source_relpath)
255
- FileUtils.mkdir_p(path)
256
- path
257
- end
258
-
259
- # everywhere else, #scm_profiler is stubbed, we need the unstubbed version
260
- let(:cookbook_lock) do
261
- described_class.new(cookbook_name, storage_config)
262
- end
263
-
264
- before do
265
- cookbook_lock.source = cookbook_source_path
266
- end
267
-
268
- after do
269
- clear_tempdir
270
- end
271
-
272
- context "when the cookbook is in a git-repo" do
273
-
274
- before do
275
- FileUtils.mkdir_p(git_dir_path)
276
- end
277
-
278
- context "when the cookbook is a self-contained git repo" do
279
-
280
- let(:git_dir_path) { File.join(cookbook_source_path, ".git") }
281
-
282
- it "selects the git profiler" do
283
- expect(cookbook_lock.scm_profiler).to be_an_instance_of(ChefDK::CookbookProfiler::Git)
284
- end
285
-
286
- end
287
-
288
- context "when the cookbook is a subdirectory of a git repo" do
289
-
290
- let(:cookbook_source_relpath) { "cookbook_repo/nginx" }
291
-
292
- let(:git_dir_path) { File.join(tempdir, "cookbook_repo/.git") }
293
-
294
- it "selects the git profiler" do
295
- expect(cookbook_lock.scm_profiler).to be_an_instance_of(ChefDK::CookbookProfiler::Git)
296
- end
297
-
298
- end
299
-
300
- end
301
-
302
- context "when the cookbook is not in a git repo" do
303
-
304
- it "selects the null profiler" do
305
- expect(cookbook_lock.scm_profiler).to be_an_instance_of(ChefDK::CookbookProfiler::NullSCM)
306
- end
307
-
308
- end
309
-
310
- end
311
-
312
- context "when loading data from a serialized form" do
313
-
314
- let(:previous_lock_data) do
315
- {
316
- "identifier" => "abc123",
317
- "dotted_decimal_identifier" => "111.222.333",
318
- "version" => "1.2.3",
319
- "source" => "../my_repo/nginx",
320
- "source_options" => {
321
- "path" => "../my_repo/nginx"
322
- },
323
- "cache_key" => nil
324
- }
325
- end
326
-
327
- before do
328
- cookbook_lock.build_from_lock_data(previous_lock_data)
329
- end
330
-
331
- it "sets the identifier" do
332
- expect(cookbook_lock.identifier).to eq("abc123")
333
- end
334
-
335
- it "sets the dotted_decimal_identifier" do
336
- expect(cookbook_lock.dotted_decimal_identifier).to eq("111.222.333")
337
- end
338
-
339
- it "sets the version" do
340
- expect(cookbook_lock.version).to eq("1.2.3")
341
- end
342
-
343
- it "sets the source attribute" do
344
- expect(cookbook_lock.source).to eq("../my_repo/nginx")
345
- end
346
-
347
- it "sets the source options, symbolizing keys so the data is compatible with CookbookLocationSpecification" do
348
- expected = { path: "../my_repo/nginx" }
349
- expect(cookbook_lock.source_options).to eq(expected)
350
- end
351
-
352
- it "doesn't refresh scm_data when #lock_data is called" do
353
- allow(scm_profiler).to receive(:profile_data).and_raise("This shouldn't get called")
354
- cookbook_lock.lock_data
355
- end
356
-
357
- context "after the data has been refreshed" do
358
-
359
- before do
360
- allow(cookbook_lock).to receive(:identifiers).and_return(identifiers)
361
- cookbook_lock.refresh!
362
- end
363
-
364
- context "and the underlying hasn't been mutated" do
365
-
366
- let(:identifiers) do
367
- instance_double("ChefDK::CookbookProfiler::Identifiers",
368
- content_identifier: "abc123",
369
- dotted_decimal_identifier: "111.222.333",
370
- semver_version: "1.2.3")
371
- end
372
-
373
- it "has the correct identifier" do
374
- expect(cookbook_lock.identifier).to eq("abc123")
375
- end
376
-
377
- it "has the correct dotted_decimal_identifier" do
378
- expect(cookbook_lock.dotted_decimal_identifier).to eq("111.222.333")
379
- end
380
-
381
- it "has the correct version" do
382
- expect(cookbook_lock.version).to eq("1.2.3")
383
- end
384
-
385
- it "sets the updated flag to false" do
386
- expect(cookbook_lock).to_not be_updated
387
- end
388
-
389
- it "sets the version_updated flag to false" do
390
- expect(cookbook_lock.version_updated?).to be(false)
391
- end
392
-
393
- it "sets the identifier_updated flag to false" do
394
- expect(cookbook_lock.identifier_updated?).to be(false)
395
- end
396
-
397
- end
398
-
399
- context "and the underlying data has been mutated" do
400
- # represents the updated state of the cookbook
401
- let(:identifiers) do
402
- instance_double("ChefDK::CookbookProfiler::Identifiers",
403
- content_identifier: "def456",
404
- dotted_decimal_identifier: "777.888.999",
405
- semver_version: "7.8.9")
406
- end
407
-
408
- it "sets the content identifier to the new identifier" do
409
- expect(cookbook_lock.identifier).to eq("def456")
410
- end
411
-
412
- it "sets the dotted_decimal_identifier to the new identifier" do
413
- expect(cookbook_lock.dotted_decimal_identifier).to eq("777.888.999")
414
- end
415
-
416
- it "sets the SemVer version to the new version" do
417
- expect(cookbook_lock.version).to eq("7.8.9")
418
- end
419
-
420
- it "sets the updated flag to true" do
421
- expect(cookbook_lock).to be_updated
422
- end
423
-
424
- it "sets the version_updated flag to true" do
425
- expect(cookbook_lock.version_updated?).to be(true)
426
- end
427
-
428
- it "sets the identifier_updated flag to true" do
429
- expect(cookbook_lock.identifier_updated?).to be(true)
430
- end
431
- end
432
- end
433
- end
434
-
435
- end
436
-
437
- describe ChefDK::Policyfile::ArchivedCookbook do
438
-
439
- let(:cookbook_name) { "nginx" }
440
-
441
- let(:storage_config) { ChefDK::Policyfile::StorageConfig.new }
442
-
443
- let(:wrapped_cookbook_lock_data) do
444
- {
445
- "identifier" => "abc123",
446
- "dotted_decimal_identifier" => "111.222.333",
447
- "version" => "1.2.3",
448
- "source" => "../my_repo/nginx",
449
- "source_options" => {
450
- # when getting the cookbook location spec, source options needs to have
451
- # symbolic keys, so a round trip via LocalCookbook#build_from_lock_data
452
- # will result in this being a symbol
453
- :path => "../my_repo/nginx"
454
- },
455
- "cache_key" => nil,
456
- "scm_info" => {}
457
- }
458
- end
459
-
460
- let(:wrapped_cookbook_lock) do
461
- lock = ChefDK::Policyfile::LocalCookbook.new(cookbook_name, storage_config)
462
- allow(lock).to receive(:scm_info).and_return({})
463
- lock.build_from_lock_data(wrapped_cookbook_lock_data)
464
- lock
465
- end
466
-
467
- let(:cookbook_lock) do
468
- described_class.new(wrapped_cookbook_lock, storage_config)
469
- end
470
-
471
- let(:archived_cookbook_path) { File.join(storage_config.relative_paths_root, "cookbooks", "nginx-111.222.333") }
472
-
473
- it "sets cookbook_path to the path within the archive" do
474
- expect(cookbook_lock.cookbook_path).to eq(archived_cookbook_path)
475
- end
476
-
477
- it "implements build_from_lock_data" do
478
- msg = "ArchivedCookbook cannot be built from lock data, it can only wrap an existing lock object"
479
- expect { cookbook_lock.build_from_lock_data({}) }.to raise_error(NotImplementedError, msg)
480
- end
481
-
482
- it "implements validate!" do
483
- expect(cookbook_lock.validate!).to be(true)
484
- end
485
-
486
- it "implements refresh!" do
487
- expect(cookbook_lock.refresh!).to be(true)
488
- end
489
-
490
- it "implements installed?" do
491
- allow(File).to receive(:exist?).with(archived_cookbook_path).and_return(false)
492
- allow(File).to receive(:directory?).with(archived_cookbook_path).and_return(false)
493
- expect(cookbook_lock.installed?).to be(false)
494
- allow(File).to receive(:exist?).with(archived_cookbook_path).and_return(true)
495
- allow(File).to receive(:directory?).with(archived_cookbook_path).and_return(true)
496
- expect(cookbook_lock.installed?).to be(true)
497
- end
498
-
499
- describe "delegated behavior" do
500
-
501
- it "sets the identifier" do
502
- expect(cookbook_lock.identifier).to eq("abc123")
503
- end
504
-
505
- it "sets the dotted_decimal_identifier" do
506
- expect(cookbook_lock.dotted_decimal_identifier).to eq("111.222.333")
507
- end
508
-
509
- it "sets the version" do
510
- expect(cookbook_lock.version).to eq("1.2.3")
511
- end
512
-
513
- it "sets the source attribute" do
514
- expect(cookbook_lock.source).to eq("../my_repo/nginx")
515
- end
516
-
517
- it "sets the source options, symbolizing keys so the data is compatible with CookbookLocationSpecification" do
518
- expected = { path: "../my_repo/nginx" }
519
- expect(cookbook_lock.source_options).to eq(expected)
520
- end
521
-
522
- it "returns unchanged data when calling to_lock" do
523
- expect(cookbook_lock.to_lock).to eq(wrapped_cookbook_lock_data)
524
- end
525
-
526
-
527
- end
528
-
529
- end
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/cookbook_locks'
20
+
21
+ shared_examples_for "Cookbook Lock" do
22
+
23
+ let(:cookbook_lock_data) { cookbook_lock.to_lock }
24
+
25
+ it "has a cookbook name" do
26
+ expect(cookbook_lock.name).to eq(cookbook_name)
27
+ end
28
+
29
+ it "has a source_options attribute" do
30
+ cookbook_lock.source_options = { artifactserver: "https://artifacts.example.com/nginx/1.0.0/download" }
31
+ expect(cookbook_lock.source_options).to eq({ artifactserver: "https://artifacts.example.com/nginx/1.0.0/download" })
32
+ end
33
+
34
+ it "has an identifier attribute" do
35
+ cookbook_lock.identifier = "my-opaque-id"
36
+ expect(cookbook_lock.identifier).to eq("my-opaque-id")
37
+ end
38
+
39
+ it "has a dotted_decimal_identifier attribute" do
40
+ cookbook_lock.dotted_decimal_identifier = "123.456.789"
41
+ expect(cookbook_lock.dotted_decimal_identifier).to eq("123.456.789")
42
+ end
43
+
44
+ it "has a version attribute" do
45
+ cookbook_lock.version = "1.2.3"
46
+ expect(cookbook_lock.version).to eq("1.2.3")
47
+ end
48
+
49
+ it "has a storage config" do
50
+ expect(cookbook_lock.storage_config).to eq(storage_config)
51
+ end
52
+
53
+ context "when the underlying cookbook has not been mutated, or #refresh! has not been called" do
54
+
55
+ it "is not updated" do
56
+ expect(cookbook_lock).to_not be_updated
57
+ end
58
+
59
+ it "does not have an updated identifier" do
60
+ expect(cookbook_lock.identifier_updated?).to be false
61
+ end
62
+
63
+ it "does not have an updated version" do
64
+ expect(cookbook_lock.version_updated?).to be false
65
+ end
66
+
67
+ end
68
+
69
+ context "when version and identifier attributes are populated" do
70
+
71
+ before do
72
+ allow(cookbook_lock).to receive(:validate!)
73
+
74
+ cookbook_lock.identifier = "my-opaque-id"
75
+ cookbook_lock.dotted_decimal_identifier = "123.456.789"
76
+ cookbook_lock.version = "1.2.3"
77
+ cookbook_lock.source_options = { :sourcekey => "location info" }
78
+ end
79
+
80
+ it "includes the identifier in the lock data" do
81
+ expect(cookbook_lock_data["identifier"]).to eq("my-opaque-id")
82
+ end
83
+
84
+ it "includes the dotted decimal identifier in the lock data" do
85
+ expect(cookbook_lock_data["dotted_decimal_identifier"]).to eq("123.456.789")
86
+ end
87
+
88
+ it "includes the version in lock data" do
89
+ expect(cookbook_lock_data["version"]).to eq("1.2.3")
90
+ end
91
+
92
+ it "includes the source_options in lock data" do
93
+ expect(cookbook_lock_data["source_options"]).to eq({ :sourcekey => "location info" })
94
+ end
95
+
96
+ it "creates a CookbookLocationSpecification with the source and version data" do
97
+ location_spec = cookbook_lock.cookbook_location_spec
98
+ expect(location_spec.name).to eq(cookbook_name)
99
+ expect(location_spec.version_constraint).to eq(Semverse::Constraint.new("= 1.2.3"))
100
+ expect(location_spec.source_options).to eq({ sourcekey: "location info" })
101
+ end
102
+
103
+ it "delegates #dependencies to cookbook_location_spec" do
104
+ deps = [ [ "foo", ">= 0.0.0"], [ "bar", "~> 2.1" ] ]
105
+ expect(cookbook_lock.cookbook_location_spec).to receive(:dependencies).and_return(deps)
106
+ expect(cookbook_lock.dependencies).to eq(deps)
107
+ end
108
+
109
+ it "delegates #installed? to the CookbookLocationSpecification" do
110
+ location_spec = cookbook_lock.cookbook_location_spec
111
+ expect(location_spec).to receive(:installed?).and_return(true)
112
+ expect(cookbook_lock).to be_installed
113
+ expect(location_spec).to receive(:installed?).and_return(false)
114
+ expect(cookbook_lock).to_not be_installed
115
+ end
116
+
117
+ end
118
+
119
+ context "when created from lock data" do
120
+
121
+ let(:lock_data) do
122
+ {
123
+ "identifier" => "my-opaque-id",
124
+ "dotted_decimal_identifier" => "123.456.789",
125
+ "version" => "1.2.3",
126
+ "source_options" => { "sourcekey" => "location info" },
127
+ "cache_key" => nil,
128
+ "source" => "cookbooks_local_path"
129
+ }
130
+ end
131
+
132
+ before do
133
+ cookbook_lock.build_from_lock_data(lock_data)
134
+ end
135
+
136
+ it "sets the identifier attribute" do
137
+ expect(cookbook_lock.identifier).to eq("my-opaque-id")
138
+ end
139
+
140
+ it "sets the dotted_decimal_identifier attribute" do
141
+ expect(cookbook_lock.dotted_decimal_identifier).to eq("123.456.789")
142
+ end
143
+
144
+ it "sets the version attribute" do
145
+ expect(cookbook_lock.version).to eq("1.2.3")
146
+ end
147
+
148
+ it "sets the source options" do
149
+ expect(cookbook_lock.source_options).to eq({ sourcekey: "location info" })
150
+ end
151
+ end
152
+
153
+ end
154
+
155
+
156
+ describe ChefDK::Policyfile::CachedCookbook do
157
+
158
+ let(:cookbook_name) { "nginx" }
159
+
160
+ let(:storage_config) { ChefDK::Policyfile::StorageConfig.new }
161
+
162
+ let(:cookbook_lock) do
163
+ described_class.new(cookbook_name, storage_config)
164
+ end
165
+
166
+ include_examples "Cookbook Lock"
167
+
168
+ it "has a cache_key attribute" do
169
+ cookbook_lock.cache_key = "nginx-1.0.0-example.com"
170
+ expect(cookbook_lock.cache_key).to eq("nginx-1.0.0-example.com")
171
+ end
172
+
173
+ it "has an origin attribute" do
174
+ cookbook_lock.origin = "https://artifacts.example.com/nginx/1.0.0/download"
175
+ expect(cookbook_lock.origin).to eq("https://artifacts.example.com/nginx/1.0.0/download")
176
+ end
177
+
178
+ it "errors locating the cookbook when the cache key is not set" do
179
+ expect { cookbook_lock.cookbook_path }.to raise_error(ChefDK::MissingCookbookLockData)
180
+ end
181
+
182
+ it "ignores calls to #refresh!" do
183
+ expect { cookbook_lock.refresh! }.to_not raise_error
184
+ end
185
+
186
+ context "when populated with valid data" do
187
+
188
+ let(:cookbook_name) { "foo" }
189
+
190
+ let(:cache_path) { File.join(fixtures_path, "cached_cookbooks") }
191
+
192
+ before do
193
+ cookbook_lock.cache_key = "foo-1.0.0"
194
+
195
+ storage_config.cache_path = cache_path
196
+ end
197
+
198
+ it "gives the path to the cookbook in the cache" do
199
+ expect(cookbook_lock.cookbook_path).to eq(File.join(cache_path, "foo-1.0.0"))
200
+ end
201
+
202
+ end
203
+
204
+ end
205
+
206
+ describe ChefDK::Policyfile::LocalCookbook do
207
+
208
+ let(:cookbook_name) { "nginx" }
209
+
210
+ let(:storage_config) { ChefDK::Policyfile::StorageConfig.new }
211
+
212
+ let(:scm_profiler) { instance_double("ChefDK::CookbookProfiler::Git", profile_data: {}) }
213
+
214
+ let(:cookbook_lock) do
215
+ lock = described_class.new(cookbook_name, storage_config)
216
+ allow(lock).to receive(:scm_profiler).and_return(scm_profiler)
217
+ lock
218
+ end
219
+
220
+ include_examples "Cookbook Lock"
221
+
222
+ describe "gathering identifier info" do
223
+ let(:identifiers) do
224
+ instance_double("ChefDK::CookbookProfiler::Identifiers",
225
+ content_identifier: "abc123",
226
+ dotted_decimal_identifier: "111.222.333",
227
+ semver_version: "1.2.3")
228
+ end
229
+
230
+ before do
231
+ allow(cookbook_lock).to receive(:identifiers).and_return(identifiers)
232
+ cookbook_lock.gather_profile_data
233
+ end
234
+
235
+ it "sets the content identifier" do
236
+ expect(cookbook_lock.identifier).to eq("abc123")
237
+ end
238
+
239
+ it "sets the backwards compatible dotted decimal identifer equivalent" do
240
+ expect(cookbook_lock.dotted_decimal_identifier).to eq("111.222.333")
241
+ end
242
+
243
+ it "collects the 'real' SemVer version of the cookbook" do
244
+ expect(cookbook_lock.version).to eq("1.2.3")
245
+ end
246
+
247
+ end
248
+
249
+ describe "selecting an SCM profiler" do
250
+
251
+ let(:cookbook_source_relpath) { "nginx" }
252
+
253
+ let(:cookbook_source_path) do
254
+ path = File.join(tempdir, cookbook_source_relpath)
255
+ FileUtils.mkdir_p(path)
256
+ path
257
+ end
258
+
259
+ # everywhere else, #scm_profiler is stubbed, we need the unstubbed version
260
+ let(:cookbook_lock) do
261
+ described_class.new(cookbook_name, storage_config)
262
+ end
263
+
264
+ before do
265
+ cookbook_lock.source = cookbook_source_path
266
+ end
267
+
268
+ after do
269
+ clear_tempdir
270
+ end
271
+
272
+ context "when the cookbook is in a git-repo" do
273
+
274
+ before do
275
+ FileUtils.mkdir_p(git_dir_path)
276
+ end
277
+
278
+ context "when the cookbook is a self-contained git repo" do
279
+
280
+ let(:git_dir_path) { File.join(cookbook_source_path, ".git") }
281
+
282
+ it "selects the git profiler" do
283
+ expect(cookbook_lock.scm_profiler).to be_an_instance_of(ChefDK::CookbookProfiler::Git)
284
+ end
285
+
286
+ end
287
+
288
+ context "when the cookbook is a subdirectory of a git repo" do
289
+
290
+ let(:cookbook_source_relpath) { "cookbook_repo/nginx" }
291
+
292
+ let(:git_dir_path) { File.join(tempdir, "cookbook_repo/.git") }
293
+
294
+ it "selects the git profiler" do
295
+ expect(cookbook_lock.scm_profiler).to be_an_instance_of(ChefDK::CookbookProfiler::Git)
296
+ end
297
+
298
+ end
299
+
300
+ end
301
+
302
+ context "when the cookbook is not in a git repo" do
303
+
304
+ it "selects the null profiler" do
305
+ expect(cookbook_lock.scm_profiler).to be_an_instance_of(ChefDK::CookbookProfiler::NullSCM)
306
+ end
307
+
308
+ end
309
+
310
+ end
311
+
312
+ context "when loading data from a serialized form" do
313
+
314
+ let(:previous_lock_data) do
315
+ {
316
+ "identifier" => "abc123",
317
+ "dotted_decimal_identifier" => "111.222.333",
318
+ "version" => "1.2.3",
319
+ "source" => "../my_repo/nginx",
320
+ "source_options" => {
321
+ "path" => "../my_repo/nginx"
322
+ },
323
+ "cache_key" => nil
324
+ }
325
+ end
326
+
327
+ before do
328
+ cookbook_lock.build_from_lock_data(previous_lock_data)
329
+ end
330
+
331
+ it "sets the identifier" do
332
+ expect(cookbook_lock.identifier).to eq("abc123")
333
+ end
334
+
335
+ it "sets the dotted_decimal_identifier" do
336
+ expect(cookbook_lock.dotted_decimal_identifier).to eq("111.222.333")
337
+ end
338
+
339
+ it "sets the version" do
340
+ expect(cookbook_lock.version).to eq("1.2.3")
341
+ end
342
+
343
+ it "sets the source attribute" do
344
+ expect(cookbook_lock.source).to eq("../my_repo/nginx")
345
+ end
346
+
347
+ it "sets the source options, symbolizing keys so the data is compatible with CookbookLocationSpecification" do
348
+ expected = { path: "../my_repo/nginx" }
349
+ expect(cookbook_lock.source_options).to eq(expected)
350
+ end
351
+
352
+ it "doesn't refresh scm_data when #lock_data is called" do
353
+ allow(scm_profiler).to receive(:profile_data).and_raise("This shouldn't get called")
354
+ cookbook_lock.lock_data
355
+ end
356
+
357
+ context "after the data has been refreshed" do
358
+
359
+ before do
360
+ allow(cookbook_lock).to receive(:identifiers).and_return(identifiers)
361
+ cookbook_lock.refresh!
362
+ end
363
+
364
+ context "and the underlying hasn't been mutated" do
365
+
366
+ let(:identifiers) do
367
+ instance_double("ChefDK::CookbookProfiler::Identifiers",
368
+ content_identifier: "abc123",
369
+ dotted_decimal_identifier: "111.222.333",
370
+ semver_version: "1.2.3")
371
+ end
372
+
373
+ it "has the correct identifier" do
374
+ expect(cookbook_lock.identifier).to eq("abc123")
375
+ end
376
+
377
+ it "has the correct dotted_decimal_identifier" do
378
+ expect(cookbook_lock.dotted_decimal_identifier).to eq("111.222.333")
379
+ end
380
+
381
+ it "has the correct version" do
382
+ expect(cookbook_lock.version).to eq("1.2.3")
383
+ end
384
+
385
+ it "sets the updated flag to false" do
386
+ expect(cookbook_lock).to_not be_updated
387
+ end
388
+
389
+ it "sets the version_updated flag to false" do
390
+ expect(cookbook_lock.version_updated?).to be(false)
391
+ end
392
+
393
+ it "sets the identifier_updated flag to false" do
394
+ expect(cookbook_lock.identifier_updated?).to be(false)
395
+ end
396
+
397
+ end
398
+
399
+ context "and the underlying data has been mutated" do
400
+ # represents the updated state of the cookbook
401
+ let(:identifiers) do
402
+ instance_double("ChefDK::CookbookProfiler::Identifiers",
403
+ content_identifier: "def456",
404
+ dotted_decimal_identifier: "777.888.999",
405
+ semver_version: "7.8.9")
406
+ end
407
+
408
+ it "sets the content identifier to the new identifier" do
409
+ expect(cookbook_lock.identifier).to eq("def456")
410
+ end
411
+
412
+ it "sets the dotted_decimal_identifier to the new identifier" do
413
+ expect(cookbook_lock.dotted_decimal_identifier).to eq("777.888.999")
414
+ end
415
+
416
+ it "sets the SemVer version to the new version" do
417
+ expect(cookbook_lock.version).to eq("7.8.9")
418
+ end
419
+
420
+ it "sets the updated flag to true" do
421
+ expect(cookbook_lock).to be_updated
422
+ end
423
+
424
+ it "sets the version_updated flag to true" do
425
+ expect(cookbook_lock.version_updated?).to be(true)
426
+ end
427
+
428
+ it "sets the identifier_updated flag to true" do
429
+ expect(cookbook_lock.identifier_updated?).to be(true)
430
+ end
431
+ end
432
+ end
433
+ end
434
+
435
+ end
436
+
437
+ describe ChefDK::Policyfile::ArchivedCookbook do
438
+
439
+ let(:cookbook_name) { "nginx" }
440
+
441
+ let(:storage_config) { ChefDK::Policyfile::StorageConfig.new }
442
+
443
+ let(:wrapped_cookbook_lock_data) do
444
+ {
445
+ "identifier" => "abc123",
446
+ "dotted_decimal_identifier" => "111.222.333",
447
+ "version" => "1.2.3",
448
+ "source" => "../my_repo/nginx",
449
+ "source_options" => {
450
+ # when getting the cookbook location spec, source options needs to have
451
+ # symbolic keys, so a round trip via LocalCookbook#build_from_lock_data
452
+ # will result in this being a symbol
453
+ :path => "../my_repo/nginx"
454
+ },
455
+ "cache_key" => nil,
456
+ "scm_info" => {}
457
+ }
458
+ end
459
+
460
+ let(:wrapped_cookbook_lock) do
461
+ lock = ChefDK::Policyfile::LocalCookbook.new(cookbook_name, storage_config)
462
+ allow(lock).to receive(:scm_info).and_return({})
463
+ lock.build_from_lock_data(wrapped_cookbook_lock_data)
464
+ lock
465
+ end
466
+
467
+ let(:cookbook_lock) do
468
+ described_class.new(wrapped_cookbook_lock, storage_config)
469
+ end
470
+
471
+ let(:archived_cookbook_path) { File.join(storage_config.relative_paths_root, "cookbooks", "nginx-111.222.333") }
472
+
473
+ it "sets cookbook_path to the path within the archive" do
474
+ expect(cookbook_lock.cookbook_path).to eq(archived_cookbook_path)
475
+ end
476
+
477
+ it "implements build_from_lock_data" do
478
+ msg = "ArchivedCookbook cannot be built from lock data, it can only wrap an existing lock object"
479
+ expect { cookbook_lock.build_from_lock_data({}) }.to raise_error(NotImplementedError, msg)
480
+ end
481
+
482
+ it "implements validate!" do
483
+ expect(cookbook_lock.validate!).to be(true)
484
+ end
485
+
486
+ it "implements refresh!" do
487
+ expect(cookbook_lock.refresh!).to be(true)
488
+ end
489
+
490
+ it "implements installed?" do
491
+ allow(File).to receive(:exist?).with(archived_cookbook_path).and_return(false)
492
+ allow(File).to receive(:directory?).with(archived_cookbook_path).and_return(false)
493
+ expect(cookbook_lock.installed?).to be(false)
494
+ allow(File).to receive(:exist?).with(archived_cookbook_path).and_return(true)
495
+ allow(File).to receive(:directory?).with(archived_cookbook_path).and_return(true)
496
+ expect(cookbook_lock.installed?).to be(true)
497
+ end
498
+
499
+ describe "delegated behavior" do
500
+
501
+ it "sets the identifier" do
502
+ expect(cookbook_lock.identifier).to eq("abc123")
503
+ end
504
+
505
+ it "sets the dotted_decimal_identifier" do
506
+ expect(cookbook_lock.dotted_decimal_identifier).to eq("111.222.333")
507
+ end
508
+
509
+ it "sets the version" do
510
+ expect(cookbook_lock.version).to eq("1.2.3")
511
+ end
512
+
513
+ it "sets the source attribute" do
514
+ expect(cookbook_lock.source).to eq("../my_repo/nginx")
515
+ end
516
+
517
+ it "sets the source options, symbolizing keys so the data is compatible with CookbookLocationSpecification" do
518
+ expected = { path: "../my_repo/nginx" }
519
+ expect(cookbook_lock.source_options).to eq(expected)
520
+ end
521
+
522
+ it "returns unchanged data when calling to_lock" do
523
+ expect(cookbook_lock.to_lock).to eq(wrapped_cookbook_lock_data)
524
+ end
525
+
526
+
527
+ end
528
+
529
+ end