chef-dk 0.8.0 → 0.9.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 (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,58 +1,58 @@
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
- module ChefDK
19
- module Policyfile
20
- module Reports
21
-
22
- # Defines a table with a flexible number of columns and prints rows in
23
- # the table. Columns are defined ahead of time, by calling the #column
24
- # method, individual rows are printed by calling #print_row with the data
25
- # for each cell.
26
- class TablePrinter
27
-
28
- attr_reader :ui
29
-
30
- def initialize(ui)
31
- @ui = ui
32
- @column_widths = []
33
-
34
- yield self
35
- end
36
-
37
- # Defines a column. If a collection is given, it is mapped to an array
38
- # of strings and the longest string is used as the left justify width
39
- # for that column when rows are printed.
40
- def column(collection = [])
41
- @column_widths << (collection.map(&:to_s).map(&:size).max || 0)
42
- end
43
-
44
- # Print a row.
45
- def print_row(*cells)
46
- row = ""
47
- cells.each_with_index do |cell_data, i|
48
- row << cell_data.to_s.ljust(@column_widths[i])
49
- row << " "
50
- end
51
- ui.msg(row.strip)
52
- end
53
-
54
- end
55
- end
56
- end
57
- end
58
-
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
+ module ChefDK
19
+ module Policyfile
20
+ module Reports
21
+
22
+ # Defines a table with a flexible number of columns and prints rows in
23
+ # the table. Columns are defined ahead of time, by calling the #column
24
+ # method, individual rows are printed by calling #print_row with the data
25
+ # for each cell.
26
+ class TablePrinter
27
+
28
+ attr_reader :ui
29
+
30
+ def initialize(ui)
31
+ @ui = ui
32
+ @column_widths = []
33
+
34
+ yield self
35
+ end
36
+
37
+ # Defines a column. If a collection is given, it is mapped to an array
38
+ # of strings and the longest string is used as the left justify width
39
+ # for that column when rows are printed.
40
+ def column(collection = [])
41
+ @column_widths << (collection.map(&:to_s).map(&:size).max || 0)
42
+ end
43
+
44
+ # Print a row.
45
+ def print_row(*cells)
46
+ row = ""
47
+ cells.each_with_index do |cell_data, i|
48
+ row << cell_data.to_s.ljust(@column_widths[i])
49
+ row << " "
50
+ end
51
+ ui.msg(row.strip)
52
+ end
53
+
54
+ end
55
+ end
56
+ end
57
+ end
58
+
@@ -1,70 +1,70 @@
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 'chef-dk/policyfile/reports/table_printer'
19
-
20
- module ChefDK
21
- module Policyfile
22
- module Reports
23
- class Upload
24
-
25
- attr_reader :reused_cbs
26
- attr_reader :uploaded_cbs
27
- attr_reader :ui
28
-
29
- def initialize(reused_cbs: [], uploaded_cbs: [], ui: nil)
30
- @reused_cbs = reused_cbs
31
- @uploaded_cbs = uploaded_cbs
32
- @ui = ui
33
-
34
- @justify_name_width = nil
35
- @justify_version_width = nil
36
- end
37
-
38
- def show
39
- reused_cbs.each do |cb_with_lock|
40
- lock = cb_with_lock.lock
41
- table.print_row("Using", lock.name, lock.version, "(#{lock.identifier[0,8]})")
42
- end
43
-
44
- uploaded_cbs.each do |cb_with_lock|
45
- lock = cb_with_lock.lock
46
- table.print_row("Uploaded", lock.name, lock.version, "(#{lock.identifier[0,8]})")
47
- end
48
- end
49
-
50
- def table
51
- @table ||= TablePrinter.new(ui) do |t|
52
- t.column(%w[ Using Uploaded ])
53
- t.column(cookbook_names)
54
- t.column(cookbook_version_numbers)
55
- t.column
56
- end
57
- end
58
-
59
- def cookbook_names
60
- (reused_cbs + uploaded_cbs).map { |e| e.lock.name }
61
- end
62
-
63
- def cookbook_version_numbers
64
- (reused_cbs + uploaded_cbs).map { |e| e.lock.version }
65
- end
66
-
67
- end
68
- end
69
- end
70
- 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 'chef-dk/policyfile/reports/table_printer'
19
+
20
+ module ChefDK
21
+ module Policyfile
22
+ module Reports
23
+ class Upload
24
+
25
+ attr_reader :reused_cbs
26
+ attr_reader :uploaded_cbs
27
+ attr_reader :ui
28
+
29
+ def initialize(reused_cbs: [], uploaded_cbs: [], ui: nil)
30
+ @reused_cbs = reused_cbs
31
+ @uploaded_cbs = uploaded_cbs
32
+ @ui = ui
33
+
34
+ @justify_name_width = nil
35
+ @justify_version_width = nil
36
+ end
37
+
38
+ def show
39
+ reused_cbs.each do |cb_with_lock|
40
+ lock = cb_with_lock.lock
41
+ table.print_row("Using", lock.name, lock.version, "(#{lock.identifier[0,8]})")
42
+ end
43
+
44
+ uploaded_cbs.each do |cb_with_lock|
45
+ lock = cb_with_lock.lock
46
+ table.print_row("Uploaded", lock.name, lock.version, "(#{lock.identifier[0,8]})")
47
+ end
48
+ end
49
+
50
+ def table
51
+ @table ||= TablePrinter.new(ui) do |t|
52
+ t.column(%w[ Using Uploaded ])
53
+ t.column(cookbook_names)
54
+ t.column(cookbook_version_numbers)
55
+ t.column
56
+ end
57
+ end
58
+
59
+ def cookbook_names
60
+ (reused_cbs + uploaded_cbs).map { |e| e.lock.name }
61
+ end
62
+
63
+ def cookbook_version_numbers
64
+ (reused_cbs + uploaded_cbs).map { |e| e.lock.version }
65
+ end
66
+
67
+ end
68
+ end
69
+ end
70
+ end
@@ -1,298 +1,298 @@
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 'semverse'
19
-
20
- require 'chef-dk/exceptions'
21
-
22
- module ChefDK
23
- module Policyfile
24
-
25
- class SolutionDependencies
26
-
27
- Cookbook = Struct.new(:name, :version)
28
-
29
- class Cookbook
30
-
31
- VALID_STRING_FORMAT = /\A[^\s]+ \([^\s]+\)\Z/
32
-
33
- def self.valid_str?(str)
34
- !!(str =~ VALID_STRING_FORMAT)
35
- end
36
-
37
- def self.parse(str)
38
- name, version_w_parens = str.split(' ')
39
- version = version_w_parens[/\(([^)]+)\)/, 1]
40
- new(name, version)
41
- end
42
-
43
- def to_s
44
- "#{name} (#{version})"
45
- end
46
-
47
- def eql?(other)
48
- other.kind_of?(self.class) and
49
- other.name == name and
50
- other.version == version
51
- end
52
-
53
- def hash
54
- [name, version].hash
55
- end
56
-
57
- end
58
-
59
- def self.from_lock(lock_data)
60
- new.tap {|e| e.consume_lock_data(lock_data) }
61
- end
62
-
63
- attr_reader :policyfile_dependencies
64
-
65
- attr_reader :cookbook_dependencies
66
-
67
- def initialize
68
- @policyfile_dependencies = []
69
- @cookbook_dependencies = {}
70
- end
71
-
72
- def add_policyfile_dep(cookbook, constraint)
73
- @policyfile_dependencies << [ cookbook, Semverse::Constraint.new(constraint) ]
74
- end
75
-
76
- def add_cookbook_dep(cookbook_name, version, dependency_list)
77
- cookbook = Cookbook.new(cookbook_name, version)
78
- add_cookbook_obj_dep(cookbook, dependency_list)
79
- end
80
-
81
- def update_cookbook_dep(cookbook_name, new_version, new_dependency_list)
82
- @cookbook_dependencies.delete_if { |cb, _deps| cb.name == cookbook_name }
83
- add_cookbook_dep(cookbook_name, new_version, new_dependency_list)
84
- end
85
-
86
- def consume_lock_data(lock_data)
87
- unless lock_data.key?("Policyfile") and lock_data.key?("dependencies")
88
- msg = %Q|lockfile solution_dependencies must be a Hash of the form `{"Policyfile": [], "dependencies": {} }' (got: #{lock_data.inspect})|
89
- raise InvalidLockfile, msg
90
- end
91
-
92
- set_policyfile_deps_from_lock_data(lock_data)
93
- set_cookbook_deps_from_lock_data(lock_data)
94
- end
95
-
96
- def test_conflict!(cookbook_name, version)
97
- unless have_cookbook_dep?(cookbook_name, version)
98
- raise CookbookNotInWorkingSet, "Cookbook #{cookbook_name} (#{version}) not in the working set, cannot test for conflicts"
99
- end
100
-
101
- assert_cookbook_version_valid!(cookbook_name, version)
102
- assert_cookbook_deps_valid!(cookbook_name, version)
103
- end
104
-
105
- def to_lock
106
- { "Policyfile" => policyfile_dependencies_for_lock, "dependencies" => cookbook_deps_for_lock }
107
- end
108
-
109
- def policyfile_dependencies_for_lock
110
- policyfile_dependencies.map do |name, constraint|
111
- [ name, constraint.to_s ]
112
- end
113
- end
114
-
115
- def cookbook_deps_for_lock
116
- cookbook_dependencies.inject({}) do |map, (cookbook, deps)|
117
- map[cookbook.to_s] = deps.map do |name, constraint|
118
- [ name, constraint.to_s ]
119
- end
120
- map
121
- end
122
- end
123
-
124
- private
125
-
126
- def add_cookbook_obj_dep(cookbook, dependency_map)
127
- @cookbook_dependencies[cookbook] = dependency_map.map do |dep_name, constraint|
128
- [ dep_name, Semverse::Constraint.new(constraint) ]
129
- end
130
- end
131
-
132
- def assert_cookbook_version_valid!(cookbook_name, version)
133
- policyfile_conflicts = policyfile_conflicts_with(cookbook_name, version)
134
- cookbook_conflicts = cookbook_conflicts_with(cookbook_name, version)
135
- all_conflicts = policyfile_conflicts + cookbook_conflicts
136
-
137
- return false if all_conflicts.empty?
138
-
139
- details = all_conflicts.map { |source, name, constraint| "#{source} depends on #{name} #{constraint}" }
140
- message = "Cookbook #{cookbook_name} (#{version}) conflicts with other dependencies:\n"
141
- full_message = message + details.join("\n")
142
- raise DependencyConflict, full_message
143
- end
144
-
145
- def assert_cookbook_deps_valid!(cookbook_name, version)
146
- dependency_conflicts = cookbook_deps_conflicts_for(cookbook_name, version)
147
- return false if dependency_conflicts.empty?
148
- message = "Cookbook #{cookbook_name} (#{version}) has dependency constraints that cannot be met by the existing cookbook set:\n"
149
- full_message = message + dependency_conflicts.join("\n")
150
- raise DependencyConflict, full_message
151
- end
152
-
153
- def policyfile_conflicts_with(cookbook_name, version)
154
- policyfile_conflicts = []
155
-
156
- @policyfile_dependencies.each do |dep_name, constraint|
157
- if dep_name == cookbook_name and !constraint.satisfies?(version)
158
- policyfile_conflicts << ['Policyfile', dep_name, constraint]
159
- end
160
- end
161
-
162
- policyfile_conflicts
163
- end
164
-
165
- def cookbook_conflicts_with(cookbook_name, version)
166
- cookbook_conflicts = []
167
-
168
- @cookbook_dependencies.each do |top_level_dep_name, dependencies|
169
- dependencies.each do |dep_name, constraint|
170
- if dep_name == cookbook_name and !constraint.satisfies?(version)
171
- cookbook_conflicts << [top_level_dep_name, dep_name, constraint]
172
- end
173
- end
174
- end
175
-
176
- cookbook_conflicts
177
- end
178
-
179
- def cookbook_deps_conflicts_for(cookbook_name, version)
180
- conflicts = []
181
- transitive_deps = find_cookbook_dep_by_name_and_version(cookbook_name, version)
182
- transitive_deps.each do |name, constraint|
183
- existing_cookbook = find_cookbook_dep_by_name(name)
184
- if existing_cookbook.nil?
185
- conflicts << "Cookbook #{name} isn't included in the existing cookbook set."
186
- elsif !constraint.satisfies?(existing_cookbook[0].version)
187
- conflicts << "Dependency on #{name} #{constraint} conflicts with existing version #{existing_cookbook[0]}"
188
- end
189
- end
190
- conflicts
191
- end
192
-
193
- def have_cookbook_dep?(name, version)
194
- @cookbook_dependencies.key?(Cookbook.new(name, version))
195
- end
196
-
197
- def find_cookbook_dep_by_name(name)
198
- @cookbook_dependencies.find { |k,v| k.name == name }
199
- end
200
-
201
- def find_cookbook_dep_by_name_and_version(name, version)
202
- @cookbook_dependencies[Cookbook.new(name, version)]
203
- end
204
-
205
- def set_policyfile_deps_from_lock_data(lock_data)
206
- policyfile_deps_data = lock_data["Policyfile"]
207
-
208
- unless policyfile_deps_data.kind_of?(Array)
209
- msg = "lockfile solution_dependencies Policyfile dependencies must be an array of cookbooks and constraints (got: #{policyfile_deps_data.inspect})"
210
- raise InvalidLockfile, msg
211
- end
212
-
213
- policyfile_deps_data.each do |entry|
214
- add_policyfile_dep_from_lock_data(entry)
215
- end
216
- end
217
-
218
- def add_policyfile_dep_from_lock_data(entry)
219
- unless entry.kind_of?(Array) and entry.size == 2
220
- msg = %Q(lockfile solution_dependencies Policyfile dependencies entry must be like [ "$COOKBOOK_NAME", "$CONSTRAINT" ] (got: #{entry.inspect}))
221
- raise InvalidLockfile, msg
222
- end
223
-
224
- cookbook_name, constraint = entry
225
-
226
- unless cookbook_name.kind_of?(String) and !cookbook_name.empty?
227
- msg = "lockfile solution_dependencies Policyfile dependencies entry. Cookbook name portion must be a string (got: #{entry.inspect})"
228
- raise InvalidLockfile, msg
229
- end
230
-
231
- unless constraint.kind_of?(String) and !constraint.empty?
232
- msg = "malformed lockfile solution_dependencies Policyfile dependencies entry. Version constraint portion must be a string (got: #{entry.inspect})"
233
- raise InvalidLockfile, msg
234
- end
235
- add_policyfile_dep(cookbook_name, constraint)
236
- rescue Semverse::InvalidConstraintFormat
237
- msg = "malformed lockfile solution_dependencies Policyfile dependencies entry. Version constraint portion must be a valid version constraint (got: #{entry.inspect})"
238
- raise InvalidLockfile, msg
239
- end
240
-
241
- def set_cookbook_deps_from_lock_data(lock_data)
242
- cookbook_dependencies_data = lock_data["dependencies"]
243
-
244
- unless cookbook_dependencies_data.kind_of?(Hash)
245
- msg = "lockfile solution_dependencies dependencies entry must be a Hash (JSON object) of dependencies (got: #{cookbook_dependencies_data.inspect})"
246
- raise InvalidLockfile, msg
247
- end
248
-
249
- cookbook_dependencies_data.each do |name_and_version, deps_list|
250
- add_cookbook_dep_from_lock_data(name_and_version, deps_list)
251
- end
252
- end
253
-
254
- def add_cookbook_dep_from_lock_data(name_and_version, deps_list)
255
- unless name_and_version.kind_of?(String)
256
- show = "#{name_and_version.inspect} => #{deps_list.inspect}"
257
- msg = %Q(lockfile cookbook_dependencies entries must be of the form "$COOKBOOK_NAME ($VERSION)" => [ $dependency, ...] (got: #{show}) )
258
- raise InvalidLockfile, msg
259
- end
260
-
261
- unless Cookbook.valid_str?(name_and_version)
262
- msg = %Q(lockfile cookbook_dependencies entry keys must be of the form "$COOKBOOK_NAME ($VERSION)" (got: #{name_and_version.inspect}) )
263
- raise InvalidLockfile, msg
264
- end
265
-
266
- unless deps_list.kind_of?(Array)
267
- msg = %Q(lockfile cookbook_dependencies entry values must be an Array like [ [ "$COOKBOOK_NAME", "$CONSTRAINT" ], ... ] (got: #{deps_list.inspect}) )
268
- raise InvalidLockfile, msg
269
- end
270
-
271
- deps_list.each do |entry|
272
-
273
- unless entry.kind_of?(Array) and entry.size == 2
274
- msg = %Q(lockfile solution_dependencies dependencies entry must be like [ "$COOKBOOK_NAME", "$CONSTRAINT" ] (got: #{entry.inspect}))
275
- raise InvalidLockfile, msg
276
- end
277
-
278
- dep_name, constraint = entry
279
-
280
- unless dep_name.kind_of?(String) and !dep_name.empty?
281
- msg = "malformed lockfile solution_dependencies dependencies entry. Cookbook name portion must be a string (got: #{entry.inspect})"
282
- raise InvalidLockfile, msg
283
- end
284
-
285
- unless constraint.kind_of?(String) and !constraint.empty?
286
- msg = "malformed lockfile solution_dependencies dependencies entry. Version constraint portion must be a string (got: #{entry.inspect})"
287
- raise InvalidLockfile, msg
288
- end
289
- end
290
-
291
- cookbook = Cookbook.parse(name_and_version)
292
- add_cookbook_obj_dep(cookbook, deps_list)
293
- end
294
-
295
- end
296
-
297
- end
298
- 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 'semverse'
19
+
20
+ require 'chef-dk/exceptions'
21
+
22
+ module ChefDK
23
+ module Policyfile
24
+
25
+ class SolutionDependencies
26
+
27
+ Cookbook = Struct.new(:name, :version)
28
+
29
+ class Cookbook
30
+
31
+ VALID_STRING_FORMAT = /\A[^\s]+ \([^\s]+\)\Z/
32
+
33
+ def self.valid_str?(str)
34
+ !!(str =~ VALID_STRING_FORMAT)
35
+ end
36
+
37
+ def self.parse(str)
38
+ name, version_w_parens = str.split(' ')
39
+ version = version_w_parens[/\(([^)]+)\)/, 1]
40
+ new(name, version)
41
+ end
42
+
43
+ def to_s
44
+ "#{name} (#{version})"
45
+ end
46
+
47
+ def eql?(other)
48
+ other.kind_of?(self.class) and
49
+ other.name == name and
50
+ other.version == version
51
+ end
52
+
53
+ def hash
54
+ [name, version].hash
55
+ end
56
+
57
+ end
58
+
59
+ def self.from_lock(lock_data)
60
+ new.tap {|e| e.consume_lock_data(lock_data) }
61
+ end
62
+
63
+ attr_reader :policyfile_dependencies
64
+
65
+ attr_reader :cookbook_dependencies
66
+
67
+ def initialize
68
+ @policyfile_dependencies = []
69
+ @cookbook_dependencies = {}
70
+ end
71
+
72
+ def add_policyfile_dep(cookbook, constraint)
73
+ @policyfile_dependencies << [ cookbook, Semverse::Constraint.new(constraint) ]
74
+ end
75
+
76
+ def add_cookbook_dep(cookbook_name, version, dependency_list)
77
+ cookbook = Cookbook.new(cookbook_name, version)
78
+ add_cookbook_obj_dep(cookbook, dependency_list)
79
+ end
80
+
81
+ def update_cookbook_dep(cookbook_name, new_version, new_dependency_list)
82
+ @cookbook_dependencies.delete_if { |cb, _deps| cb.name == cookbook_name }
83
+ add_cookbook_dep(cookbook_name, new_version, new_dependency_list)
84
+ end
85
+
86
+ def consume_lock_data(lock_data)
87
+ unless lock_data.key?("Policyfile") and lock_data.key?("dependencies")
88
+ msg = %Q|lockfile solution_dependencies must be a Hash of the form `{"Policyfile": [], "dependencies": {} }' (got: #{lock_data.inspect})|
89
+ raise InvalidLockfile, msg
90
+ end
91
+
92
+ set_policyfile_deps_from_lock_data(lock_data)
93
+ set_cookbook_deps_from_lock_data(lock_data)
94
+ end
95
+
96
+ def test_conflict!(cookbook_name, version)
97
+ unless have_cookbook_dep?(cookbook_name, version)
98
+ raise CookbookNotInWorkingSet, "Cookbook #{cookbook_name} (#{version}) not in the working set, cannot test for conflicts"
99
+ end
100
+
101
+ assert_cookbook_version_valid!(cookbook_name, version)
102
+ assert_cookbook_deps_valid!(cookbook_name, version)
103
+ end
104
+
105
+ def to_lock
106
+ { "Policyfile" => policyfile_dependencies_for_lock, "dependencies" => cookbook_deps_for_lock }
107
+ end
108
+
109
+ def policyfile_dependencies_for_lock
110
+ policyfile_dependencies.map do |name, constraint|
111
+ [ name, constraint.to_s ]
112
+ end
113
+ end
114
+
115
+ def cookbook_deps_for_lock
116
+ cookbook_dependencies.inject({}) do |map, (cookbook, deps)|
117
+ map[cookbook.to_s] = deps.map do |name, constraint|
118
+ [ name, constraint.to_s ]
119
+ end
120
+ map
121
+ end
122
+ end
123
+
124
+ private
125
+
126
+ def add_cookbook_obj_dep(cookbook, dependency_map)
127
+ @cookbook_dependencies[cookbook] = dependency_map.map do |dep_name, constraint|
128
+ [ dep_name, Semverse::Constraint.new(constraint) ]
129
+ end
130
+ end
131
+
132
+ def assert_cookbook_version_valid!(cookbook_name, version)
133
+ policyfile_conflicts = policyfile_conflicts_with(cookbook_name, version)
134
+ cookbook_conflicts = cookbook_conflicts_with(cookbook_name, version)
135
+ all_conflicts = policyfile_conflicts + cookbook_conflicts
136
+
137
+ return false if all_conflicts.empty?
138
+
139
+ details = all_conflicts.map { |source, name, constraint| "#{source} depends on #{name} #{constraint}" }
140
+ message = "Cookbook #{cookbook_name} (#{version}) conflicts with other dependencies:\n"
141
+ full_message = message + details.join("\n")
142
+ raise DependencyConflict, full_message
143
+ end
144
+
145
+ def assert_cookbook_deps_valid!(cookbook_name, version)
146
+ dependency_conflicts = cookbook_deps_conflicts_for(cookbook_name, version)
147
+ return false if dependency_conflicts.empty?
148
+ message = "Cookbook #{cookbook_name} (#{version}) has dependency constraints that cannot be met by the existing cookbook set:\n"
149
+ full_message = message + dependency_conflicts.join("\n")
150
+ raise DependencyConflict, full_message
151
+ end
152
+
153
+ def policyfile_conflicts_with(cookbook_name, version)
154
+ policyfile_conflicts = []
155
+
156
+ @policyfile_dependencies.each do |dep_name, constraint|
157
+ if dep_name == cookbook_name and !constraint.satisfies?(version)
158
+ policyfile_conflicts << ['Policyfile', dep_name, constraint]
159
+ end
160
+ end
161
+
162
+ policyfile_conflicts
163
+ end
164
+
165
+ def cookbook_conflicts_with(cookbook_name, version)
166
+ cookbook_conflicts = []
167
+
168
+ @cookbook_dependencies.each do |top_level_dep_name, dependencies|
169
+ dependencies.each do |dep_name, constraint|
170
+ if dep_name == cookbook_name and !constraint.satisfies?(version)
171
+ cookbook_conflicts << [top_level_dep_name, dep_name, constraint]
172
+ end
173
+ end
174
+ end
175
+
176
+ cookbook_conflicts
177
+ end
178
+
179
+ def cookbook_deps_conflicts_for(cookbook_name, version)
180
+ conflicts = []
181
+ transitive_deps = find_cookbook_dep_by_name_and_version(cookbook_name, version)
182
+ transitive_deps.each do |name, constraint|
183
+ existing_cookbook = find_cookbook_dep_by_name(name)
184
+ if existing_cookbook.nil?
185
+ conflicts << "Cookbook #{name} isn't included in the existing cookbook set."
186
+ elsif !constraint.satisfies?(existing_cookbook[0].version)
187
+ conflicts << "Dependency on #{name} #{constraint} conflicts with existing version #{existing_cookbook[0]}"
188
+ end
189
+ end
190
+ conflicts
191
+ end
192
+
193
+ def have_cookbook_dep?(name, version)
194
+ @cookbook_dependencies.key?(Cookbook.new(name, version))
195
+ end
196
+
197
+ def find_cookbook_dep_by_name(name)
198
+ @cookbook_dependencies.find { |k,v| k.name == name }
199
+ end
200
+
201
+ def find_cookbook_dep_by_name_and_version(name, version)
202
+ @cookbook_dependencies[Cookbook.new(name, version)]
203
+ end
204
+
205
+ def set_policyfile_deps_from_lock_data(lock_data)
206
+ policyfile_deps_data = lock_data["Policyfile"]
207
+
208
+ unless policyfile_deps_data.kind_of?(Array)
209
+ msg = "lockfile solution_dependencies Policyfile dependencies must be an array of cookbooks and constraints (got: #{policyfile_deps_data.inspect})"
210
+ raise InvalidLockfile, msg
211
+ end
212
+
213
+ policyfile_deps_data.each do |entry|
214
+ add_policyfile_dep_from_lock_data(entry)
215
+ end
216
+ end
217
+
218
+ def add_policyfile_dep_from_lock_data(entry)
219
+ unless entry.kind_of?(Array) and entry.size == 2
220
+ msg = %Q(lockfile solution_dependencies Policyfile dependencies entry must be like [ "$COOKBOOK_NAME", "$CONSTRAINT" ] (got: #{entry.inspect}))
221
+ raise InvalidLockfile, msg
222
+ end
223
+
224
+ cookbook_name, constraint = entry
225
+
226
+ unless cookbook_name.kind_of?(String) and !cookbook_name.empty?
227
+ msg = "lockfile solution_dependencies Policyfile dependencies entry. Cookbook name portion must be a string (got: #{entry.inspect})"
228
+ raise InvalidLockfile, msg
229
+ end
230
+
231
+ unless constraint.kind_of?(String) and !constraint.empty?
232
+ msg = "malformed lockfile solution_dependencies Policyfile dependencies entry. Version constraint portion must be a string (got: #{entry.inspect})"
233
+ raise InvalidLockfile, msg
234
+ end
235
+ add_policyfile_dep(cookbook_name, constraint)
236
+ rescue Semverse::InvalidConstraintFormat
237
+ msg = "malformed lockfile solution_dependencies Policyfile dependencies entry. Version constraint portion must be a valid version constraint (got: #{entry.inspect})"
238
+ raise InvalidLockfile, msg
239
+ end
240
+
241
+ def set_cookbook_deps_from_lock_data(lock_data)
242
+ cookbook_dependencies_data = lock_data["dependencies"]
243
+
244
+ unless cookbook_dependencies_data.kind_of?(Hash)
245
+ msg = "lockfile solution_dependencies dependencies entry must be a Hash (JSON object) of dependencies (got: #{cookbook_dependencies_data.inspect})"
246
+ raise InvalidLockfile, msg
247
+ end
248
+
249
+ cookbook_dependencies_data.each do |name_and_version, deps_list|
250
+ add_cookbook_dep_from_lock_data(name_and_version, deps_list)
251
+ end
252
+ end
253
+
254
+ def add_cookbook_dep_from_lock_data(name_and_version, deps_list)
255
+ unless name_and_version.kind_of?(String)
256
+ show = "#{name_and_version.inspect} => #{deps_list.inspect}"
257
+ msg = %Q(lockfile cookbook_dependencies entries must be of the form "$COOKBOOK_NAME ($VERSION)" => [ $dependency, ...] (got: #{show}) )
258
+ raise InvalidLockfile, msg
259
+ end
260
+
261
+ unless Cookbook.valid_str?(name_and_version)
262
+ msg = %Q(lockfile cookbook_dependencies entry keys must be of the form "$COOKBOOK_NAME ($VERSION)" (got: #{name_and_version.inspect}) )
263
+ raise InvalidLockfile, msg
264
+ end
265
+
266
+ unless deps_list.kind_of?(Array)
267
+ msg = %Q(lockfile cookbook_dependencies entry values must be an Array like [ [ "$COOKBOOK_NAME", "$CONSTRAINT" ], ... ] (got: #{deps_list.inspect}) )
268
+ raise InvalidLockfile, msg
269
+ end
270
+
271
+ deps_list.each do |entry|
272
+
273
+ unless entry.kind_of?(Array) and entry.size == 2
274
+ msg = %Q(lockfile solution_dependencies dependencies entry must be like [ "$COOKBOOK_NAME", "$CONSTRAINT" ] (got: #{entry.inspect}))
275
+ raise InvalidLockfile, msg
276
+ end
277
+
278
+ dep_name, constraint = entry
279
+
280
+ unless dep_name.kind_of?(String) and !dep_name.empty?
281
+ msg = "malformed lockfile solution_dependencies dependencies entry. Cookbook name portion must be a string (got: #{entry.inspect})"
282
+ raise InvalidLockfile, msg
283
+ end
284
+
285
+ unless constraint.kind_of?(String) and !constraint.empty?
286
+ msg = "malformed lockfile solution_dependencies dependencies entry. Version constraint portion must be a string (got: #{entry.inspect})"
287
+ raise InvalidLockfile, msg
288
+ end
289
+ end
290
+
291
+ cookbook = Cookbook.parse(name_and_version)
292
+ add_cookbook_obj_dep(cookbook, deps_list)
293
+ end
294
+
295
+ end
296
+
297
+ end
298
+ end