chef-dk 0.8.0 → 0.9.0

Sign up to get free protection for your applications and to get access to all the features.
Files changed (299) hide show
  1. checksums.yaml +4 -4
  2. data/CONTRIBUTING.md +190 -190
  3. data/Gemfile +26 -0
  4. data/LICENSE +201 -201
  5. data/README.md +276 -276
  6. data/Rakefile +18 -18
  7. data/bin/chef +25 -25
  8. data/lib/chef-dk.rb +19 -19
  9. data/lib/chef-dk/authenticated_http.rb +40 -40
  10. data/lib/chef-dk/builtin_commands.rb +60 -60
  11. data/lib/chef-dk/chef_runner.rb +98 -98
  12. data/lib/chef-dk/cli.rb +200 -200
  13. data/lib/chef-dk/command/base.rb +79 -75
  14. data/lib/chef-dk/command/clean_policy_cookbooks.rb +116 -116
  15. data/lib/chef-dk/command/clean_policy_revisions.rb +113 -113
  16. data/lib/chef-dk/command/delete_policy.rb +122 -122
  17. data/lib/chef-dk/command/delete_policy_group.rb +122 -122
  18. data/lib/chef-dk/command/diff.rb +316 -316
  19. data/lib/chef-dk/command/env.rb +90 -90
  20. data/lib/chef-dk/command/exec.rb +45 -45
  21. data/lib/chef-dk/command/export.rb +151 -153
  22. data/lib/chef-dk/command/gem.rb +47 -47
  23. data/lib/chef-dk/command/generate.rb +120 -118
  24. data/lib/chef-dk/command/generator_commands.rb +80 -80
  25. data/lib/chef-dk/command/generator_commands/app.rb +107 -107
  26. data/lib/chef-dk/command/generator_commands/attribute.rb +37 -37
  27. data/lib/chef-dk/command/generator_commands/base.rb +121 -121
  28. data/lib/chef-dk/command/generator_commands/cookbook.rb +119 -108
  29. data/lib/chef-dk/command/generator_commands/cookbook_code_file.rb +100 -100
  30. data/lib/chef-dk/command/generator_commands/cookbook_file.rb +45 -45
  31. data/lib/chef-dk/command/generator_commands/generator_generator.rb +177 -0
  32. data/lib/chef-dk/command/generator_commands/lwrp.rb +36 -36
  33. data/lib/chef-dk/command/generator_commands/policyfile.rb +86 -83
  34. data/lib/chef-dk/command/generator_commands/recipe.rb +36 -36
  35. data/lib/chef-dk/command/generator_commands/repo.rb +96 -96
  36. data/lib/chef-dk/command/generator_commands/template.rb +46 -46
  37. data/lib/chef-dk/command/install.rb +121 -121
  38. data/lib/chef-dk/command/provision.rb +438 -438
  39. data/lib/chef-dk/command/push.rb +118 -118
  40. data/lib/chef-dk/command/push_archive.rb +126 -126
  41. data/lib/chef-dk/command/shell_init.rb +180 -180
  42. data/lib/chef-dk/command/show_policy.rb +165 -165
  43. data/lib/chef-dk/command/undelete.rb +155 -155
  44. data/lib/chef-dk/command/update.rb +129 -129
  45. data/lib/chef-dk/command/verify.rb +490 -453
  46. data/lib/chef-dk/commands_map.rb +115 -115
  47. data/lib/chef-dk/completions/bash.sh.erb +5 -5
  48. data/lib/chef-dk/completions/chef.fish.erb +10 -10
  49. data/lib/chef-dk/completions/zsh.zsh.erb +21 -21
  50. data/lib/chef-dk/component_test.rb +171 -171
  51. data/lib/chef-dk/configurable.rb +57 -52
  52. data/lib/chef-dk/cookbook_metadata.rb +45 -45
  53. data/lib/chef-dk/cookbook_omnifetch.rb +32 -32
  54. data/lib/chef-dk/cookbook_profiler/git.rb +151 -151
  55. data/lib/chef-dk/cookbook_profiler/identifiers.rb +72 -72
  56. data/lib/chef-dk/cookbook_profiler/null_scm.rb +32 -32
  57. data/lib/chef-dk/exceptions.rb +113 -113
  58. data/lib/chef-dk/generator.rb +163 -162
  59. data/lib/chef-dk/helpers.rb +159 -159
  60. data/lib/chef-dk/pager.rb +106 -106
  61. data/lib/chef-dk/policyfile/chef_repo_cookbook_source.rb +122 -122
  62. data/lib/chef-dk/policyfile/chef_server_cookbook_source.rb +54 -54
  63. data/lib/chef-dk/policyfile/community_cookbook_source.rb +82 -82
  64. data/lib/chef-dk/policyfile/comparison_base.rb +124 -124
  65. data/lib/chef-dk/policyfile/cookbook_location_specification.rb +133 -133
  66. data/lib/chef-dk/policyfile/cookbook_locks.rb +466 -466
  67. data/lib/chef-dk/policyfile/cookbook_sources.rb +21 -21
  68. data/lib/chef-dk/policyfile/differ.rb +266 -266
  69. data/lib/chef-dk/policyfile/dsl.rb +197 -197
  70. data/lib/chef-dk/policyfile/lister.rb +232 -232
  71. data/lib/chef-dk/policyfile/null_cookbook_source.rb +45 -45
  72. data/lib/chef-dk/policyfile/read_cookbook_for_compat_mode_upload.rb +124 -124
  73. data/lib/chef-dk/policyfile/reports/install.rb +70 -70
  74. data/lib/chef-dk/policyfile/reports/table_printer.rb +58 -58
  75. data/lib/chef-dk/policyfile/reports/upload.rb +70 -70
  76. data/lib/chef-dk/policyfile/solution_dependencies.rb +298 -298
  77. data/lib/chef-dk/policyfile/storage_config.rb +100 -100
  78. data/lib/chef-dk/policyfile/undo_record.rb +142 -142
  79. data/lib/chef-dk/policyfile/undo_stack.rb +130 -130
  80. data/lib/chef-dk/policyfile/uploader.rb +213 -213
  81. data/lib/chef-dk/policyfile_compiler.rb +322 -322
  82. data/lib/chef-dk/policyfile_lock.rb +552 -552
  83. data/lib/chef-dk/policyfile_services/clean_policies.rb +95 -95
  84. data/lib/chef-dk/policyfile_services/clean_policy_cookbooks.rb +125 -125
  85. data/lib/chef-dk/policyfile_services/export_repo.rb +309 -281
  86. data/lib/chef-dk/policyfile_services/install.rb +125 -125
  87. data/lib/chef-dk/policyfile_services/push.rb +114 -114
  88. data/lib/chef-dk/policyfile_services/push_archive.rb +173 -173
  89. data/lib/chef-dk/policyfile_services/rm_policy.rb +142 -142
  90. data/lib/chef-dk/policyfile_services/rm_policy_group.rb +86 -86
  91. data/lib/chef-dk/policyfile_services/show_policy.rb +237 -237
  92. data/lib/chef-dk/policyfile_services/undelete.rb +108 -108
  93. data/lib/chef-dk/policyfile_services/update_attributes.rb +104 -104
  94. data/lib/chef-dk/service_exception_inspectors.rb +25 -25
  95. data/lib/chef-dk/service_exception_inspectors/base.rb +40 -40
  96. data/lib/chef-dk/service_exception_inspectors/http.rb +121 -121
  97. data/lib/chef-dk/service_exceptions.rb +143 -143
  98. data/lib/chef-dk/shell_out.rb +36 -36
  99. data/lib/chef-dk/skeletons/code_generator/files/default/Berksfile +3 -3
  100. data/lib/chef-dk/skeletons/code_generator/files/default/chefignore +100 -100
  101. data/lib/chef-dk/skeletons/code_generator/files/default/cookbook_readmes/README-policy.md +9 -9
  102. data/lib/chef-dk/skeletons/code_generator/files/default/cookbook_readmes/README.md +54 -54
  103. data/lib/chef-dk/skeletons/code_generator/files/default/gitignore +16 -16
  104. data/lib/chef-dk/skeletons/code_generator/files/default/repo/README.md +28 -28
  105. data/lib/chef-dk/skeletons/code_generator/files/default/repo/cookbooks/example/README.md +27 -0
  106. data/lib/chef-dk/skeletons/code_generator/files/default/repo/cookbooks/example/attributes/default.rb +7 -7
  107. data/lib/chef-dk/skeletons/code_generator/files/default/repo/cookbooks/example/metadata.rb +3 -3
  108. data/lib/chef-dk/skeletons/code_generator/files/default/repo/cookbooks/example/recipes/default.rb +8 -8
  109. data/lib/chef-dk/skeletons/code_generator/files/default/repo/data_bags/README.md +57 -57
  110. data/lib/chef-dk/skeletons/code_generator/files/default/repo/data_bags/example/example_item.json +3 -3
  111. data/lib/chef-dk/skeletons/code_generator/files/default/repo/environments/README.md +9 -9
  112. data/lib/chef-dk/skeletons/code_generator/files/default/repo/environments/example.json +12 -12
  113. data/lib/chef-dk/skeletons/code_generator/files/default/repo/roles/README.md +8 -8
  114. data/lib/chef-dk/skeletons/code_generator/files/default/repo/roles/example.json +12 -12
  115. data/lib/chef-dk/skeletons/code_generator/files/default/serverspec_spec_helper.rb +8 -3
  116. data/lib/chef-dk/skeletons/code_generator/files/default/spec_helper.rb +2 -2
  117. data/lib/chef-dk/skeletons/code_generator/metadata.rb +8 -8
  118. data/lib/chef-dk/skeletons/code_generator/recipes/app.rb +97 -97
  119. data/lib/chef-dk/skeletons/code_generator/recipes/attribute.rb +12 -12
  120. data/lib/chef-dk/skeletons/code_generator/recipes/cookbook.rb +104 -92
  121. data/lib/chef-dk/skeletons/code_generator/recipes/cookbook_file.rb +24 -24
  122. data/lib/chef-dk/skeletons/code_generator/recipes/lwrp.rb +23 -23
  123. data/lib/chef-dk/skeletons/code_generator/recipes/policyfile.rb +8 -8
  124. data/lib/chef-dk/skeletons/code_generator/recipes/recipe.rb +27 -27
  125. data/lib/chef-dk/skeletons/code_generator/recipes/repo.rb +48 -47
  126. data/lib/chef-dk/skeletons/code_generator/recipes/template.rb +32 -32
  127. data/lib/chef-dk/skeletons/code_generator/templates/default/LICENSE.all_rights.erb +3 -3
  128. data/lib/chef-dk/skeletons/code_generator/templates/default/LICENSE.apache2.erb +201 -201
  129. data/lib/chef-dk/skeletons/code_generator/templates/default/LICENSE.gplv2.erb +339 -339
  130. data/lib/chef-dk/skeletons/code_generator/templates/default/LICENSE.gplv3.erb +674 -674
  131. data/lib/chef-dk/skeletons/code_generator/templates/default/LICENSE.mit.erb +21 -21
  132. data/lib/chef-dk/skeletons/code_generator/templates/default/Policyfile.rb.erb +20 -16
  133. data/lib/chef-dk/skeletons/code_generator/templates/default/README.md.erb +4 -4
  134. data/lib/chef-dk/skeletons/code_generator/templates/default/kitchen.yml.erb +16 -16
  135. data/lib/chef-dk/skeletons/code_generator/templates/default/kitchen_policyfile.yml.erb +27 -0
  136. data/lib/chef-dk/skeletons/code_generator/templates/default/metadata.rb.erb +7 -7
  137. data/lib/chef-dk/skeletons/code_generator/templates/default/recipe.rb.erb +5 -5
  138. data/lib/chef-dk/skeletons/code_generator/templates/default/recipe_spec.rb.erb +20 -20
  139. data/lib/chef-dk/skeletons/code_generator/templates/default/repo/gitignore.erb +11 -11
  140. data/lib/chef-dk/skeletons/code_generator/templates/default/serverspec_default_spec.rb.erb +9 -9
  141. data/lib/chef-dk/ui.rb +58 -58
  142. data/lib/chef-dk/version.rb +20 -20
  143. data/lib/kitchen/provisioner/policyfile_zero.rb +193 -164
  144. data/spec/shared/a_file_generator.rb +125 -125
  145. data/spec/shared/a_generated_file.rb +12 -12
  146. data/spec/shared/command_with_ui_object.rb +11 -11
  147. data/spec/shared/custom_generator_cookbook.rb +117 -117
  148. data/spec/shared/fixture_cookbook_checksums.rb +47 -47
  149. data/spec/shared/setup_git_cookbooks.rb +53 -53
  150. data/spec/spec_helper.rb +49 -48
  151. data/spec/test_helpers.rb +84 -84
  152. data/spec/unit/chef_runner_spec.rb +111 -110
  153. data/spec/unit/cli_spec.rb +357 -357
  154. data/spec/unit/command/base_spec.rb +169 -136
  155. data/spec/unit/command/clean_policy_cookbooks_spec.rb +181 -181
  156. data/spec/unit/command/clean_policy_revisions_spec.rb +181 -181
  157. data/spec/unit/command/delete_policy_group_spec.rb +207 -207
  158. data/spec/unit/command/delete_policy_spec.rb +207 -207
  159. data/spec/unit/command/diff_spec.rb +312 -312
  160. data/spec/unit/command/env_spec.rb +52 -52
  161. data/spec/unit/command/exec_spec.rb +179 -179
  162. data/spec/unit/command/export_spec.rb +189 -189
  163. data/spec/unit/command/generate_spec.rb +142 -142
  164. data/spec/unit/command/generator_commands/app_spec.rb +169 -169
  165. data/spec/unit/command/generator_commands/attribute_spec.rb +32 -32
  166. data/spec/unit/command/generator_commands/cookbook_file_spec.rb +32 -32
  167. data/spec/unit/command/generator_commands/cookbook_spec.rb +320 -240
  168. data/spec/unit/command/generator_commands/generator_generator_spec.rb +229 -0
  169. data/spec/unit/command/generator_commands/lwrp_spec.rb +32 -32
  170. data/spec/unit/command/generator_commands/policyfile_spec.rb +125 -125
  171. data/spec/unit/command/generator_commands/recipe_spec.rb +34 -34
  172. data/spec/unit/command/generator_commands/repo_spec.rb +283 -283
  173. data/spec/unit/command/generator_commands/template_spec.rb +32 -32
  174. data/spec/unit/command/install_spec.rb +179 -179
  175. data/spec/unit/command/provision_spec.rb +592 -592
  176. data/spec/unit/command/push_archive_spec.rb +153 -153
  177. data/spec/unit/command/push_spec.rb +199 -199
  178. data/spec/unit/command/shell_init_spec.rb +329 -329
  179. data/spec/unit/command/show_policy_spec.rb +235 -235
  180. data/spec/unit/command/undelete_spec.rb +246 -246
  181. data/spec/unit/command/update_spec.rb +251 -251
  182. data/spec/unit/command/verify_spec.rb +323 -322
  183. data/spec/unit/commands_map_spec.rb +57 -57
  184. data/spec/unit/component_test_spec.rb +126 -126
  185. data/spec/unit/cookbook_metadata_spec.rb +98 -98
  186. data/spec/unit/cookbook_profiler/git_spec.rb +176 -176
  187. data/spec/unit/cookbook_profiler/identifiers_spec.rb +83 -83
  188. data/spec/unit/fixtures/chef-runner-cookbooks/test_cookbook/recipes/recipe_one.rb +9 -9
  189. data/spec/unit/fixtures/chef-runner-cookbooks/test_cookbook/recipes/recipe_two.rb +9 -9
  190. data/spec/unit/fixtures/command/cli_test_command.rb +26 -26
  191. data/spec/unit/fixtures/command/explicit_path_example.rb +7 -7
  192. data/spec/unit/fixtures/cookbook_cache/baz-f59ee7a5bca6a4e606b67f7f856b768d847c39bb/.kitchen.yml +16 -16
  193. data/spec/unit/fixtures/cookbook_cache/baz-f59ee7a5bca6a4e606b67f7f856b768d847c39bb/Berksfile +3 -3
  194. data/spec/unit/fixtures/cookbook_cache/baz-f59ee7a5bca6a4e606b67f7f856b768d847c39bb/README.md +4 -4
  195. data/spec/unit/fixtures/cookbook_cache/baz-f59ee7a5bca6a4e606b67f7f856b768d847c39bb/chefignore +96 -96
  196. data/spec/unit/fixtures/cookbook_cache/baz-f59ee7a5bca6a4e606b67f7f856b768d847c39bb/metadata.rb +8 -8
  197. data/spec/unit/fixtures/cookbook_cache/baz-f59ee7a5bca6a4e606b67f7f856b768d847c39bb/recipes/default.rb +8 -8
  198. data/spec/unit/fixtures/cookbook_cache/dep_of_bar-1.2.3/.kitchen.yml +16 -16
  199. data/spec/unit/fixtures/cookbook_cache/dep_of_bar-1.2.3/Berksfile +3 -3
  200. data/spec/unit/fixtures/cookbook_cache/dep_of_bar-1.2.3/README.md +4 -4
  201. data/spec/unit/fixtures/cookbook_cache/dep_of_bar-1.2.3/chefignore +96 -96
  202. data/spec/unit/fixtures/cookbook_cache/dep_of_bar-1.2.3/metadata.rb +8 -8
  203. data/spec/unit/fixtures/cookbook_cache/dep_of_bar-1.2.3/recipes/default.rb +8 -8
  204. data/spec/unit/fixtures/cookbook_cache/foo-1.0.0/.kitchen.yml +16 -16
  205. data/spec/unit/fixtures/cookbook_cache/foo-1.0.0/Berksfile +3 -3
  206. data/spec/unit/fixtures/cookbook_cache/foo-1.0.0/README.md +4 -4
  207. data/spec/unit/fixtures/cookbook_cache/foo-1.0.0/chefignore +96 -96
  208. data/spec/unit/fixtures/cookbook_cache/foo-1.0.0/metadata.rb +8 -8
  209. data/spec/unit/fixtures/cookbook_cache/foo-1.0.0/recipes/default.rb +8 -8
  210. data/spec/unit/fixtures/cookbooks_api/pruned_small_universe.json +1321 -1321
  211. data/spec/unit/fixtures/cookbooks_api/small_universe.json +2987 -2987
  212. data/spec/unit/fixtures/cookbooks_api/universe.json +1 -1
  213. data/spec/unit/fixtures/cookbooks_api/update_fixtures.rb +36 -36
  214. data/spec/unit/fixtures/dev_cookbooks/README.md +16 -16
  215. data/spec/unit/fixtures/eg_omnibus_dir/valid/embedded/apps/berkshelf/integration_test +2 -2
  216. data/spec/unit/fixtures/eg_omnibus_dir/valid/embedded/apps/berkshelf/verify_me +5 -5
  217. data/spec/unit/fixtures/eg_omnibus_dir/valid/embedded/apps/chef/verify_me +3 -3
  218. data/spec/unit/fixtures/eg_omnibus_dir/valid/embedded/apps/test-kitchen/verify_me +2 -2
  219. data/spec/unit/fixtures/example_cookbook/.gitignore +17 -17
  220. data/spec/unit/fixtures/example_cookbook/.kitchen.yml +16 -16
  221. data/spec/unit/fixtures/example_cookbook/Berksfile +3 -3
  222. data/spec/unit/fixtures/example_cookbook/README.md +4 -4
  223. data/spec/unit/fixtures/example_cookbook/chefignore +96 -96
  224. data/spec/unit/fixtures/example_cookbook/metadata.rb +8 -8
  225. data/spec/unit/fixtures/example_cookbook/recipes/default.rb +8 -8
  226. data/spec/unit/fixtures/example_cookbook_metadata_json_only/.gitignore +17 -17
  227. data/spec/unit/fixtures/example_cookbook_metadata_json_only/.kitchen.yml +16 -16
  228. data/spec/unit/fixtures/example_cookbook_metadata_json_only/Berksfile +3 -3
  229. data/spec/unit/fixtures/example_cookbook_metadata_json_only/README.md +4 -4
  230. data/spec/unit/fixtures/example_cookbook_metadata_json_only/chefignore +96 -96
  231. data/spec/unit/fixtures/example_cookbook_metadata_json_only/metadata.json +5 -5
  232. data/spec/unit/fixtures/example_cookbook_metadata_json_only/recipes/default.rb +8 -8
  233. data/spec/unit/fixtures/example_cookbook_no_metadata/.gitignore +17 -17
  234. data/spec/unit/fixtures/example_cookbook_no_metadata/.kitchen.yml +16 -16
  235. data/spec/unit/fixtures/example_cookbook_no_metadata/Berksfile +3 -3
  236. data/spec/unit/fixtures/example_cookbook_no_metadata/README.md +4 -4
  237. data/spec/unit/fixtures/example_cookbook_no_metadata/chefignore +96 -96
  238. data/spec/unit/fixtures/example_cookbook_no_metadata/recipes/default.rb +8 -8
  239. data/spec/unit/fixtures/local_path_cookbooks/another-local-cookbook/README.md +4 -4
  240. data/spec/unit/fixtures/local_path_cookbooks/another-local-cookbook/chefignore +96 -96
  241. data/spec/unit/fixtures/local_path_cookbooks/another-local-cookbook/metadata.rb +8 -8
  242. data/spec/unit/fixtures/local_path_cookbooks/another-local-cookbook/recipes/default.rb +8 -8
  243. data/spec/unit/fixtures/local_path_cookbooks/cookbook-with-a-dep/Berksfile +3 -3
  244. data/spec/unit/fixtures/local_path_cookbooks/cookbook-with-a-dep/README.md +4 -4
  245. data/spec/unit/fixtures/local_path_cookbooks/cookbook-with-a-dep/chefignore +96 -96
  246. data/spec/unit/fixtures/local_path_cookbooks/cookbook-with-a-dep/metadata.rb +9 -9
  247. data/spec/unit/fixtures/local_path_cookbooks/cookbook-with-a-dep/recipes/default.rb +8 -8
  248. data/spec/unit/fixtures/local_path_cookbooks/local-cookbook/.kitchen.yml +16 -16
  249. data/spec/unit/fixtures/local_path_cookbooks/local-cookbook/Berksfile +3 -3
  250. data/spec/unit/fixtures/local_path_cookbooks/local-cookbook/README.md +4 -4
  251. data/spec/unit/fixtures/local_path_cookbooks/local-cookbook/chefignore +96 -96
  252. data/spec/unit/fixtures/local_path_cookbooks/local-cookbook/metadata.rb +8 -8
  253. data/spec/unit/fixtures/local_path_cookbooks/local-cookbook/recipes/default.rb +8 -8
  254. data/spec/unit/fixtures/local_path_cookbooks/metadata-missing/README.md +2 -2
  255. data/spec/unit/fixtures/local_path_cookbooks/noignore-f59ee7a5bca6a4e606b67f7f856b768d847c39bb/.kitchen.yml +16 -16
  256. data/spec/unit/fixtures/local_path_cookbooks/noignore-f59ee7a5bca6a4e606b67f7f856b768d847c39bb/README.md +4 -4
  257. data/spec/unit/fixtures/local_path_cookbooks/noignore-f59ee7a5bca6a4e606b67f7f856b768d847c39bb/metadata.rb +8 -8
  258. data/spec/unit/fixtures/local_path_cookbooks/noignore-f59ee7a5bca6a4e606b67f7f856b768d847c39bb/recipes/default.rb +8 -8
  259. data/spec/unit/generator_spec.rb +120 -120
  260. data/spec/unit/helpers_spec.rb +92 -92
  261. data/spec/unit/pager_spec.rb +119 -119
  262. data/spec/unit/policyfile/chef_repo_cookbook_source_spec.rb +66 -66
  263. data/spec/unit/policyfile/chef_server_cookbook_source_spec.rb +34 -34
  264. data/spec/unit/policyfile/community_cookbook_source_spec.rb +51 -51
  265. data/spec/unit/policyfile/comparison_base_spec.rb +343 -343
  266. data/spec/unit/policyfile/cookbook_location_specification_spec.rb +252 -252
  267. data/spec/unit/policyfile/cookbook_locks_spec.rb +529 -529
  268. data/spec/unit/policyfile/differ_spec.rb +687 -687
  269. data/spec/unit/policyfile/lister_spec.rb +272 -272
  270. data/spec/unit/policyfile/null_cookbook_source_spec.rb +35 -35
  271. data/spec/unit/policyfile/read_cookbook_for_compat_mode_upload_spec.rb +92 -92
  272. data/spec/unit/policyfile/reports/install_spec.rb +115 -115
  273. data/spec/unit/policyfile/reports/upload_spec.rb +96 -96
  274. data/spec/unit/policyfile/solution_dependencies_spec.rb +145 -145
  275. data/spec/unit/policyfile/storage_config_spec.rb +172 -172
  276. data/spec/unit/policyfile/undo_record_spec.rb +260 -260
  277. data/spec/unit/policyfile/undo_stack_spec.rb +266 -266
  278. data/spec/unit/policyfile/uploader_spec.rb +410 -410
  279. data/spec/unit/policyfile_demands_spec.rb +876 -876
  280. data/spec/unit/policyfile_evaluation_spec.rb +441 -441
  281. data/spec/unit/policyfile_lock_build_spec.rb +1056 -1056
  282. data/spec/unit/policyfile_lock_install_spec.rb +138 -138
  283. data/spec/unit/policyfile_lock_serialization_spec.rb +425 -425
  284. data/spec/unit/policyfile_lock_validation_spec.rb +611 -611
  285. data/spec/unit/policyfile_services/clean_policies_spec.rb +236 -236
  286. data/spec/unit/policyfile_services/clean_policy_cookbooks_spec.rb +275 -275
  287. data/spec/unit/policyfile_services/export_repo_spec.rb +439 -416
  288. data/spec/unit/policyfile_services/install_spec.rb +191 -191
  289. data/spec/unit/policyfile_services/push_archive_spec.rb +345 -345
  290. data/spec/unit/policyfile_services/push_spec.rb +233 -233
  291. data/spec/unit/policyfile_services/rm_policy_group_spec.rb +241 -241
  292. data/spec/unit/policyfile_services/rm_policy_spec.rb +266 -266
  293. data/spec/unit/policyfile_services/show_policy_spec.rb +889 -889
  294. data/spec/unit/policyfile_services/undelete_spec.rb +304 -304
  295. data/spec/unit/policyfile_services/update_attributes_spec.rb +217 -217
  296. data/spec/unit/service_exception_inspectors/base_spec.rb +43 -43
  297. data/spec/unit/service_exception_inspectors/http_spec.rb +140 -140
  298. data/spec/unit/shell_out_spec.rb +34 -34
  299. metadata +9 -3
@@ -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