chef-dk 3.9.0 → 3.10.1

Sign up to get free protection for your applications and to get access to all the features.
Files changed (377) hide show
  1. checksums.yaml +4 -4
  2. data/Gemfile +139 -139
  3. data/Gemfile.lock +917 -919
  4. data/LICENSE +201 -201
  5. data/Rakefile +77 -77
  6. data/bin/chef +25 -25
  7. data/chef-dk.gemspec +60 -60
  8. data/lib/chef-dk.rb +19 -19
  9. data/lib/chef-dk/authenticated_http.rb +22 -22
  10. data/lib/chef-dk/builtin_commands.rb +62 -62
  11. data/lib/chef-dk/chef_runner.rb +114 -114
  12. data/lib/chef-dk/chef_server_api_multi.rb +73 -73
  13. data/lib/chef-dk/cli.rb +201 -201
  14. data/lib/chef-dk/command/base.rb +79 -79
  15. data/lib/chef-dk/command/clean_policy_cookbooks.rb +114 -114
  16. data/lib/chef-dk/command/clean_policy_revisions.rb +111 -111
  17. data/lib/chef-dk/command/delete_policy.rb +120 -120
  18. data/lib/chef-dk/command/delete_policy_group.rb +120 -120
  19. data/lib/chef-dk/command/describe_cookbook.rb +95 -95
  20. data/lib/chef-dk/command/diff.rb +315 -315
  21. data/lib/chef-dk/command/env.rb +89 -89
  22. data/lib/chef-dk/command/exec.rb +44 -44
  23. data/lib/chef-dk/command/export.rb +155 -155
  24. data/lib/chef-dk/command/gem.rb +47 -47
  25. data/lib/chef-dk/command/generate.rb +126 -126
  26. data/lib/chef-dk/command/generator_commands.rb +83 -83
  27. data/lib/chef-dk/command/generator_commands/app.rb +106 -106
  28. data/lib/chef-dk/command/generator_commands/attribute.rb +36 -36
  29. data/lib/chef-dk/command/generator_commands/base.rb +157 -157
  30. data/lib/chef-dk/command/generator_commands/build_cookbook.rb +125 -125
  31. data/lib/chef-dk/command/generator_commands/chef_exts/generator_desc_resource.rb +85 -85
  32. data/lib/chef-dk/command/generator_commands/chef_exts/quieter_doc_formatter.rb +38 -38
  33. data/lib/chef-dk/command/generator_commands/chef_exts/recipe_dsl_ext.rb +39 -39
  34. data/lib/chef-dk/command/generator_commands/cookbook.rb +251 -251
  35. data/lib/chef-dk/command/generator_commands/cookbook_code_file.rb +100 -100
  36. data/lib/chef-dk/command/generator_commands/cookbook_file.rb +45 -45
  37. data/lib/chef-dk/command/generator_commands/generator_generator.rb +174 -174
  38. data/lib/chef-dk/command/generator_commands/helpers.rb +36 -36
  39. data/lib/chef-dk/command/generator_commands/policyfile.rb +124 -124
  40. data/lib/chef-dk/command/generator_commands/recipe.rb +36 -36
  41. data/lib/chef-dk/command/generator_commands/repo.rb +123 -123
  42. data/lib/chef-dk/command/generator_commands/resource.rb +36 -36
  43. data/lib/chef-dk/command/generator_commands/template.rb +46 -46
  44. data/lib/chef-dk/command/install.rb +120 -120
  45. data/lib/chef-dk/command/provision.rb +439 -439
  46. data/lib/chef-dk/command/push.rb +117 -117
  47. data/lib/chef-dk/command/push_archive.rb +125 -125
  48. data/lib/chef-dk/command/shell_init.rb +179 -179
  49. data/lib/chef-dk/command/show_policy.rb +163 -163
  50. data/lib/chef-dk/command/undelete.rb +154 -154
  51. data/lib/chef-dk/command/update.rb +139 -139
  52. data/lib/chef-dk/command/verify.rb +638 -638
  53. data/lib/chef-dk/commands_map.rb +113 -113
  54. data/lib/chef-dk/completions/bash.sh.erb +5 -5
  55. data/lib/chef-dk/completions/chef.fish.erb +10 -10
  56. data/lib/chef-dk/completions/zsh.zsh.erb +21 -21
  57. data/lib/chef-dk/component_test.rb +227 -227
  58. data/lib/chef-dk/configurable.rb +88 -88
  59. data/lib/chef-dk/cookbook_metadata.rb +45 -45
  60. data/lib/chef-dk/cookbook_omnifetch.rb +32 -32
  61. data/lib/chef-dk/cookbook_profiler/git.rb +152 -152
  62. data/lib/chef-dk/cookbook_profiler/identifiers.rb +72 -72
  63. data/lib/chef-dk/cookbook_profiler/null_scm.rb +31 -31
  64. data/lib/chef-dk/exceptions.rb +151 -151
  65. data/lib/chef-dk/generator.rb +165 -165
  66. data/lib/chef-dk/helpers.rb +176 -176
  67. data/lib/chef-dk/pager.rb +104 -104
  68. data/lib/chef-dk/policyfile/artifactory_cookbook_source.rb +102 -102
  69. data/lib/chef-dk/policyfile/attribute_merge_checker.rb +110 -110
  70. data/lib/chef-dk/policyfile/chef_repo_cookbook_source.rb +138 -138
  71. data/lib/chef-dk/policyfile/chef_server_cookbook_source.rb +99 -99
  72. data/lib/chef-dk/policyfile/chef_server_lock_fetcher.rb +167 -167
  73. data/lib/chef-dk/policyfile/community_cookbook_source.rb +95 -95
  74. data/lib/chef-dk/policyfile/comparison_base.rb +123 -123
  75. data/lib/chef-dk/policyfile/cookbook_location_specification.rb +154 -154
  76. data/lib/chef-dk/policyfile/cookbook_locks.rb +466 -466
  77. data/lib/chef-dk/policyfile/cookbook_sources.rb +23 -23
  78. data/lib/chef-dk/policyfile/delivery_supermarket_source.rb +89 -89
  79. data/lib/chef-dk/policyfile/differ.rb +263 -263
  80. data/lib/chef-dk/policyfile/dsl.rb +288 -288
  81. data/lib/chef-dk/policyfile/git_lock_fetcher.rb +265 -265
  82. data/lib/chef-dk/policyfile/included_policies_cookbook_source.rb +156 -156
  83. data/lib/chef-dk/policyfile/lister.rb +229 -229
  84. data/lib/chef-dk/policyfile/local_lock_fetcher.rb +132 -129
  85. data/lib/chef-dk/policyfile/lock_applier.rb +80 -80
  86. data/lib/chef-dk/policyfile/lock_fetcher_mixin.rb +37 -0
  87. data/lib/chef-dk/policyfile/null_cookbook_source.rb +49 -49
  88. data/lib/chef-dk/policyfile/policyfile_location_specification.rb +128 -125
  89. data/lib/chef-dk/policyfile/read_cookbook_for_compat_mode_upload.rb +124 -124
  90. data/lib/chef-dk/policyfile/remote_lock_fetcher.rb +108 -0
  91. data/lib/chef-dk/policyfile/reports/install.rb +69 -69
  92. data/lib/chef-dk/policyfile/reports/table_printer.rb +57 -57
  93. data/lib/chef-dk/policyfile/reports/upload.rb +70 -70
  94. data/lib/chef-dk/policyfile/solution_dependencies.rb +311 -311
  95. data/lib/chef-dk/policyfile/source_uri.rb +57 -57
  96. data/lib/chef-dk/policyfile/storage_config.rb +112 -112
  97. data/lib/chef-dk/policyfile/undo_record.rb +139 -139
  98. data/lib/chef-dk/policyfile/undo_stack.rb +128 -128
  99. data/lib/chef-dk/policyfile/uploader.rb +222 -222
  100. data/lib/chef-dk/policyfile_compiler.rb +528 -528
  101. data/lib/chef-dk/policyfile_lock.rb +581 -581
  102. data/lib/chef-dk/policyfile_services/clean_policies.rb +95 -95
  103. data/lib/chef-dk/policyfile_services/clean_policy_cookbooks.rb +123 -123
  104. data/lib/chef-dk/policyfile_services/export_repo.rb +419 -419
  105. data/lib/chef-dk/policyfile_services/install.rb +167 -167
  106. data/lib/chef-dk/policyfile_services/push.rb +112 -112
  107. data/lib/chef-dk/policyfile_services/push_archive.rb +164 -164
  108. data/lib/chef-dk/policyfile_services/rm_policy.rb +141 -141
  109. data/lib/chef-dk/policyfile_services/rm_policy_group.rb +85 -85
  110. data/lib/chef-dk/policyfile_services/show_policy.rb +234 -234
  111. data/lib/chef-dk/policyfile_services/undelete.rb +108 -108
  112. data/lib/chef-dk/policyfile_services/update_attributes.rb +110 -110
  113. data/lib/chef-dk/service_exception_inspectors.rb +24 -24
  114. data/lib/chef-dk/service_exception_inspectors/base.rb +39 -39
  115. data/lib/chef-dk/service_exception_inspectors/http.rb +119 -119
  116. data/lib/chef-dk/service_exceptions.rb +142 -142
  117. data/lib/chef-dk/shell_out.rb +36 -36
  118. data/lib/chef-dk/skeletons/code_generator/files/default/Berksfile +4 -4
  119. data/lib/chef-dk/skeletons/code_generator/files/default/build_cookbook/.kitchen.yml +21 -21
  120. data/lib/chef-dk/skeletons/code_generator/files/default/build_cookbook/README.md +146 -146
  121. data/lib/chef-dk/skeletons/code_generator/files/default/build_cookbook/test-fixture-recipe.rb +9 -9
  122. data/lib/chef-dk/skeletons/code_generator/files/default/chefignore +104 -104
  123. data/lib/chef-dk/skeletons/code_generator/files/default/cookbook_readmes/README-policy.md +9 -9
  124. data/lib/chef-dk/skeletons/code_generator/files/default/cookbook_readmes/README.md +66 -66
  125. data/lib/chef-dk/skeletons/code_generator/files/default/delivery-config.json +17 -17
  126. data/lib/chef-dk/skeletons/code_generator/files/default/delivery-project.toml +36 -36
  127. data/lib/chef-dk/skeletons/code_generator/files/default/gitignore +22 -22
  128. data/lib/chef-dk/skeletons/code_generator/files/default/repo/README.md +24 -24
  129. data/lib/chef-dk/skeletons/code_generator/files/default/repo/cookbooks/example/README.md +27 -27
  130. data/lib/chef-dk/skeletons/code_generator/files/default/repo/cookbooks/example/attributes/default.rb +8 -8
  131. data/lib/chef-dk/skeletons/code_generator/files/default/repo/cookbooks/example/metadata.rb +7 -7
  132. data/lib/chef-dk/skeletons/code_generator/files/default/repo/cookbooks/example/recipes/default.rb +9 -9
  133. data/lib/chef-dk/skeletons/code_generator/files/default/repo/data_bags/README.md +56 -56
  134. data/lib/chef-dk/skeletons/code_generator/files/default/repo/data_bags/example/example_item.json +3 -3
  135. data/lib/chef-dk/skeletons/code_generator/files/default/repo/dot-chef-repo.txt +6 -6
  136. data/lib/chef-dk/skeletons/code_generator/files/default/repo/environments/README.md +9 -9
  137. data/lib/chef-dk/skeletons/code_generator/files/default/repo/environments/example.json +12 -12
  138. data/lib/chef-dk/skeletons/code_generator/files/default/repo/policyfiles/README.md +24 -24
  139. data/lib/chef-dk/skeletons/code_generator/files/default/repo/roles/README.md +9 -9
  140. data/lib/chef-dk/skeletons/code_generator/files/default/repo/roles/example.json +12 -12
  141. data/lib/chef-dk/skeletons/code_generator/files/default/spec_helper.rb +3 -3
  142. data/lib/chef-dk/skeletons/code_generator/files/default/spec_helper_policyfile.rb +3 -3
  143. data/lib/chef-dk/skeletons/code_generator/metadata.rb +8 -8
  144. data/lib/chef-dk/skeletons/code_generator/recipes/app.rb +89 -89
  145. data/lib/chef-dk/skeletons/code_generator/recipes/attribute.rb +13 -13
  146. data/lib/chef-dk/skeletons/code_generator/recipes/build_cookbook.rb +177 -177
  147. data/lib/chef-dk/skeletons/code_generator/recipes/cookbook.rb +161 -161
  148. data/lib/chef-dk/skeletons/code_generator/recipes/cookbook_file.rb +25 -25
  149. data/lib/chef-dk/skeletons/code_generator/recipes/helpers.rb +21 -21
  150. data/lib/chef-dk/skeletons/code_generator/recipes/policyfile.rb +9 -9
  151. data/lib/chef-dk/skeletons/code_generator/recipes/recipe.rb +52 -52
  152. data/lib/chef-dk/skeletons/code_generator/recipes/repo.rb +68 -68
  153. data/lib/chef-dk/skeletons/code_generator/recipes/resource.rb +13 -13
  154. data/lib/chef-dk/skeletons/code_generator/recipes/template.rb +32 -32
  155. data/lib/chef-dk/skeletons/code_generator/templates/default/CHANGELOG.md.erb +11 -11
  156. data/lib/chef-dk/skeletons/code_generator/templates/default/LICENSE.all_rights.erb +3 -3
  157. data/lib/chef-dk/skeletons/code_generator/templates/default/LICENSE.apachev2.erb +201 -201
  158. data/lib/chef-dk/skeletons/code_generator/templates/default/LICENSE.gplv2.erb +339 -339
  159. data/lib/chef-dk/skeletons/code_generator/templates/default/LICENSE.gplv3.erb +674 -674
  160. data/lib/chef-dk/skeletons/code_generator/templates/default/LICENSE.mit.erb +21 -21
  161. data/lib/chef-dk/skeletons/code_generator/templates/default/Policyfile.rb.erb +25 -25
  162. data/lib/chef-dk/skeletons/code_generator/templates/default/README.md.erb +4 -4
  163. data/lib/chef-dk/skeletons/code_generator/templates/default/attribute.rb.erb +0 -0
  164. data/lib/chef-dk/skeletons/code_generator/templates/default/build_cookbook/Berksfile.erb +7 -7
  165. data/lib/chef-dk/skeletons/code_generator/templates/default/build_cookbook/metadata.rb.erb +10 -10
  166. data/lib/chef-dk/skeletons/code_generator/templates/default/build_cookbook/recipe.rb.erb +8 -8
  167. data/lib/chef-dk/skeletons/code_generator/templates/default/cookbook_file.erb +0 -0
  168. data/lib/chef-dk/skeletons/code_generator/templates/default/helpers.rb.erb +39 -39
  169. data/lib/chef-dk/skeletons/code_generator/templates/default/inspec_default_test.rb.erb +16 -16
  170. data/lib/chef-dk/skeletons/code_generator/templates/default/kitchen.yml.erb +26 -26
  171. data/lib/chef-dk/skeletons/code_generator/templates/default/kitchen_dokken.yml.erb +31 -31
  172. data/lib/chef-dk/skeletons/code_generator/templates/default/kitchen_policyfile.yml.erb +33 -33
  173. data/lib/chef-dk/skeletons/code_generator/templates/default/metadata.rb.erb +20 -20
  174. data/lib/chef-dk/skeletons/code_generator/templates/default/recipe.rb.erb +5 -5
  175. data/lib/chef-dk/skeletons/code_generator/templates/default/recipe_spec.rb.erb +35 -35
  176. data/lib/chef-dk/skeletons/code_generator/templates/default/repo/gitignore.erb +128 -128
  177. data/lib/chef-dk/skeletons/code_generator/templates/default/resource.rb.erb +1 -1
  178. data/lib/chef-dk/skeletons/code_generator/templates/default/template.erb +0 -0
  179. data/lib/chef-dk/ui.rb +57 -57
  180. data/lib/chef-dk/version.rb +20 -20
  181. data/lib/kitchen/provisioner/policyfile_zero.rb +195 -195
  182. data/spec/shared/a_file_generator.rb +125 -125
  183. data/spec/shared/a_generated_file.rb +12 -12
  184. data/spec/shared/command_with_ui_object.rb +11 -11
  185. data/spec/shared/custom_generator_cookbook.rb +136 -136
  186. data/spec/shared/fixture_cookbook_checksums.rb +46 -46
  187. data/spec/shared/setup_git_committer_config.rb +54 -54
  188. data/spec/shared/setup_git_cookbooks.rb +53 -53
  189. data/spec/spec_helper.rb +51 -51
  190. data/spec/test_helpers.rb +84 -84
  191. data/spec/unit/chef_runner_spec.rb +139 -139
  192. data/spec/unit/chef_server_api_multi_spec.rb +120 -120
  193. data/spec/unit/cli_spec.rb +377 -377
  194. data/spec/unit/command/base_spec.rb +172 -172
  195. data/spec/unit/command/clean_policy_cookbooks_spec.rb +180 -180
  196. data/spec/unit/command/clean_policy_revisions_spec.rb +180 -180
  197. data/spec/unit/command/delete_policy_group_spec.rb +206 -206
  198. data/spec/unit/command/delete_policy_spec.rb +206 -206
  199. data/spec/unit/command/diff_spec.rb +311 -311
  200. data/spec/unit/command/env_spec.rb +52 -52
  201. data/spec/unit/command/exec_spec.rb +178 -178
  202. data/spec/unit/command/export_spec.rb +199 -199
  203. data/spec/unit/command/generate_spec.rb +142 -142
  204. data/spec/unit/command/generator_commands/app_spec.rb +166 -166
  205. data/spec/unit/command/generator_commands/attribute_spec.rb +31 -31
  206. data/spec/unit/command/generator_commands/base_spec.rb +181 -181
  207. data/spec/unit/command/generator_commands/build_cookbook_spec.rb +377 -377
  208. data/spec/unit/command/generator_commands/chef_exts/generator_desc_resource_spec.rb +97 -97
  209. data/spec/unit/command/generator_commands/chef_exts/recipe_dsl_ext_spec.rb +111 -111
  210. data/spec/unit/command/generator_commands/cookbook_file_spec.rb +31 -31
  211. data/spec/unit/command/generator_commands/cookbook_spec.rb +765 -765
  212. data/spec/unit/command/generator_commands/generator_generator_spec.rb +227 -227
  213. data/spec/unit/command/generator_commands/helpers_spec.rb +31 -31
  214. data/spec/unit/command/generator_commands/policyfile_spec.rb +223 -223
  215. data/spec/unit/command/generator_commands/recipe_spec.rb +37 -37
  216. data/spec/unit/command/generator_commands/repo_spec.rb +374 -374
  217. data/spec/unit/command/generator_commands/resource_spec.rb +31 -31
  218. data/spec/unit/command/generator_commands/template_spec.rb +31 -31
  219. data/spec/unit/command/install_spec.rb +179 -179
  220. data/spec/unit/command/provision_spec.rb +589 -589
  221. data/spec/unit/command/push_archive_spec.rb +153 -153
  222. data/spec/unit/command/push_spec.rb +198 -198
  223. data/spec/unit/command/shell_init_spec.rb +339 -339
  224. data/spec/unit/command/show_policy_spec.rb +234 -234
  225. data/spec/unit/command/undelete_spec.rb +244 -244
  226. data/spec/unit/command/update_spec.rb +283 -283
  227. data/spec/unit/command/verify_spec.rb +342 -342
  228. data/spec/unit/commands_map_spec.rb +57 -57
  229. data/spec/unit/component_test_spec.rb +128 -128
  230. data/spec/unit/configurable_spec.rb +68 -68
  231. data/spec/unit/cookbook_metadata_spec.rb +96 -96
  232. data/spec/unit/cookbook_profiler/git_spec.rb +176 -176
  233. data/spec/unit/cookbook_profiler/identifiers_spec.rb +81 -81
  234. data/spec/unit/fixtures/chef-runner-cookbooks/test_cookbook/recipes/recipe_one.rb +9 -9
  235. data/spec/unit/fixtures/chef-runner-cookbooks/test_cookbook/recipes/recipe_two.rb +9 -9
  236. data/spec/unit/fixtures/command/cli_test_command.rb +26 -26
  237. data/spec/unit/fixtures/command/explicit_path_example.rb +7 -7
  238. data/spec/unit/fixtures/configurable/test_config_loader.rb +5 -5
  239. data/spec/unit/fixtures/configurable/test_configurable.rb +10 -10
  240. data/spec/unit/fixtures/cookbook_cache/baz-f59ee7a5bca6a4e606b67f7f856b768d847c39bb/.kitchen.yml +16 -16
  241. data/spec/unit/fixtures/cookbook_cache/baz-f59ee7a5bca6a4e606b67f7f856b768d847c39bb/Berksfile +3 -3
  242. data/spec/unit/fixtures/cookbook_cache/baz-f59ee7a5bca6a4e606b67f7f856b768d847c39bb/README.md +4 -4
  243. data/spec/unit/fixtures/cookbook_cache/baz-f59ee7a5bca6a4e606b67f7f856b768d847c39bb/chefignore +96 -96
  244. data/spec/unit/fixtures/cookbook_cache/baz-f59ee7a5bca6a4e606b67f7f856b768d847c39bb/metadata.rb +8 -8
  245. data/spec/unit/fixtures/cookbook_cache/baz-f59ee7a5bca6a4e606b67f7f856b768d847c39bb/recipes/default.rb +8 -8
  246. data/spec/unit/fixtures/cookbook_cache/dep_of_bar-1.2.3/.kitchen.yml +16 -16
  247. data/spec/unit/fixtures/cookbook_cache/dep_of_bar-1.2.3/Berksfile +3 -3
  248. data/spec/unit/fixtures/cookbook_cache/dep_of_bar-1.2.3/README.md +4 -4
  249. data/spec/unit/fixtures/cookbook_cache/dep_of_bar-1.2.3/chefignore +96 -96
  250. data/spec/unit/fixtures/cookbook_cache/dep_of_bar-1.2.3/metadata.rb +8 -8
  251. data/spec/unit/fixtures/cookbook_cache/dep_of_bar-1.2.3/recipes/default.rb +8 -8
  252. data/spec/unit/fixtures/cookbook_cache/foo-1.0.0/.kitchen.yml +16 -16
  253. data/spec/unit/fixtures/cookbook_cache/foo-1.0.0/Berksfile +3 -3
  254. data/spec/unit/fixtures/cookbook_cache/foo-1.0.0/README.md +4 -4
  255. data/spec/unit/fixtures/cookbook_cache/foo-1.0.0/chefignore +96 -96
  256. data/spec/unit/fixtures/cookbook_cache/foo-1.0.0/metadata.rb +8 -8
  257. data/spec/unit/fixtures/cookbook_cache/foo-1.0.0/recipes/default.rb +8 -8
  258. data/spec/unit/fixtures/cookbooks_api/chef_server_universe.json +56 -56
  259. data/spec/unit/fixtures/cookbooks_api/pruned_chef_server_universe.json +30 -30
  260. data/spec/unit/fixtures/cookbooks_api/pruned_small_universe.json +1321 -1321
  261. data/spec/unit/fixtures/cookbooks_api/small_universe.json +2987 -2987
  262. data/spec/unit/fixtures/cookbooks_api/universe.json +1 -1
  263. data/spec/unit/fixtures/cookbooks_api/update_fixtures.rb +33 -33
  264. data/spec/unit/fixtures/dev_cookbooks/README.md +16 -16
  265. data/spec/unit/fixtures/dev_cookbooks/bar-cookbook.gitbundle +0 -0
  266. data/spec/unit/fixtures/eg_omnibus_dir/missing_apps/bin/.keep +0 -0
  267. data/spec/unit/fixtures/eg_omnibus_dir/missing_apps/embedded/.keep +0 -0
  268. data/spec/unit/fixtures/eg_omnibus_dir/missing_apps/embedded/bin/.keep +0 -0
  269. data/spec/unit/fixtures/eg_omnibus_dir/missing_component/bin/.keep +0 -0
  270. data/spec/unit/fixtures/eg_omnibus_dir/missing_component/embedded/apps/berkshelf/.keep +0 -0
  271. data/spec/unit/fixtures/eg_omnibus_dir/missing_component/embedded/apps/test-kitchen/.keep +0 -0
  272. data/spec/unit/fixtures/eg_omnibus_dir/missing_component/embedded/bin/.keep +0 -0
  273. data/spec/unit/fixtures/eg_omnibus_dir/valid/bin/.keep +0 -0
  274. data/spec/unit/fixtures/eg_omnibus_dir/valid/embedded/apps/berkshelf/integration_test +2 -2
  275. data/spec/unit/fixtures/eg_omnibus_dir/valid/embedded/apps/berkshelf/verify_me +5 -5
  276. data/spec/unit/fixtures/eg_omnibus_dir/valid/embedded/apps/chef-dk/.keep +0 -0
  277. data/spec/unit/fixtures/eg_omnibus_dir/valid/embedded/apps/chef/verify_me +3 -3
  278. data/spec/unit/fixtures/eg_omnibus_dir/valid/embedded/apps/test-kitchen/verify_me +2 -2
  279. data/spec/unit/fixtures/eg_omnibus_dir/valid/embedded/bin/.keep +0 -0
  280. data/spec/unit/fixtures/example_app/Policyfile.rb +0 -0
  281. data/spec/unit/fixtures/example_cookbook/.gitignore +17 -17
  282. data/spec/unit/fixtures/example_cookbook/.kitchen.yml +16 -16
  283. data/spec/unit/fixtures/example_cookbook/Berksfile +3 -3
  284. data/spec/unit/fixtures/example_cookbook/README.md +4 -4
  285. data/spec/unit/fixtures/example_cookbook/chefignore +96 -96
  286. data/spec/unit/fixtures/example_cookbook/metadata.rb +8 -8
  287. data/spec/unit/fixtures/example_cookbook/recipes/default.rb +8 -8
  288. data/spec/unit/fixtures/example_cookbook_metadata_json_only/.gitignore +17 -17
  289. data/spec/unit/fixtures/example_cookbook_metadata_json_only/.kitchen.yml +16 -16
  290. data/spec/unit/fixtures/example_cookbook_metadata_json_only/Berksfile +3 -3
  291. data/spec/unit/fixtures/example_cookbook_metadata_json_only/README.md +4 -4
  292. data/spec/unit/fixtures/example_cookbook_metadata_json_only/chefignore +96 -96
  293. data/spec/unit/fixtures/example_cookbook_metadata_json_only/metadata.json +5 -5
  294. data/spec/unit/fixtures/example_cookbook_metadata_json_only/recipes/default.rb +8 -8
  295. data/spec/unit/fixtures/example_cookbook_no_metadata/.gitignore +17 -17
  296. data/spec/unit/fixtures/example_cookbook_no_metadata/.kitchen.yml +16 -16
  297. data/spec/unit/fixtures/example_cookbook_no_metadata/Berksfile +3 -3
  298. data/spec/unit/fixtures/example_cookbook_no_metadata/README.md +4 -4
  299. data/spec/unit/fixtures/example_cookbook_no_metadata/chefignore +96 -96
  300. data/spec/unit/fixtures/example_cookbook_no_metadata/recipes/default.rb +8 -8
  301. data/spec/unit/fixtures/local_path_cookbooks/another-local-cookbook/README.md +4 -4
  302. data/spec/unit/fixtures/local_path_cookbooks/another-local-cookbook/chefignore +96 -96
  303. data/spec/unit/fixtures/local_path_cookbooks/another-local-cookbook/metadata.rb +8 -8
  304. data/spec/unit/fixtures/local_path_cookbooks/another-local-cookbook/recipes/default.rb +8 -8
  305. data/spec/unit/fixtures/local_path_cookbooks/cookbook-with-a-dep/Berksfile +3 -3
  306. data/spec/unit/fixtures/local_path_cookbooks/cookbook-with-a-dep/README.md +4 -4
  307. data/spec/unit/fixtures/local_path_cookbooks/cookbook-with-a-dep/chefignore +96 -96
  308. data/spec/unit/fixtures/local_path_cookbooks/cookbook-with-a-dep/metadata.rb +9 -9
  309. data/spec/unit/fixtures/local_path_cookbooks/cookbook-with-a-dep/recipes/default.rb +8 -8
  310. data/spec/unit/fixtures/local_path_cookbooks/local-cookbook/.kitchen.yml +16 -16
  311. data/spec/unit/fixtures/local_path_cookbooks/local-cookbook/Berksfile +3 -3
  312. data/spec/unit/fixtures/local_path_cookbooks/local-cookbook/README.md +4 -4
  313. data/spec/unit/fixtures/local_path_cookbooks/local-cookbook/chefignore +96 -96
  314. data/spec/unit/fixtures/local_path_cookbooks/local-cookbook/extra/extra_file.txt +0 -0
  315. data/spec/unit/fixtures/local_path_cookbooks/local-cookbook/metadata.rb +8 -8
  316. data/spec/unit/fixtures/local_path_cookbooks/local-cookbook/recipes/default.rb +8 -8
  317. data/spec/unit/fixtures/local_path_cookbooks/metadata-missing/README.md +2 -2
  318. data/spec/unit/fixtures/local_path_cookbooks/noignore-f59ee7a5bca6a4e606b67f7f856b768d847c39bb/.kitchen.yml +16 -16
  319. data/spec/unit/fixtures/local_path_cookbooks/noignore-f59ee7a5bca6a4e606b67f7f856b768d847c39bb/README.md +4 -4
  320. data/spec/unit/fixtures/local_path_cookbooks/noignore-f59ee7a5bca6a4e606b67f7f856b768d847c39bb/metadata.rb +8 -8
  321. data/spec/unit/fixtures/local_path_cookbooks/noignore-f59ee7a5bca6a4e606b67f7f856b768d847c39bb/recipes/default.rb +8 -8
  322. data/spec/unit/generator_spec.rb +119 -119
  323. data/spec/unit/pager_spec.rb +117 -117
  324. data/spec/unit/policyfile/artifactory_cookbook_source_spec.rb +59 -59
  325. data/spec/unit/policyfile/attribute_merge_checker_spec.rb +80 -80
  326. data/spec/unit/policyfile/chef_repo_cookbook_source_spec.rb +93 -93
  327. data/spec/unit/policyfile/chef_server_cookbook_source_spec.rb +55 -55
  328. data/spec/unit/policyfile/chef_server_lock_fetcher_spec.rb +161 -161
  329. data/spec/unit/policyfile/community_cookbook_source_spec.rb +83 -83
  330. data/spec/unit/policyfile/comparison_base_spec.rb +340 -340
  331. data/spec/unit/policyfile/cookbook_location_specification_spec.rb +347 -347
  332. data/spec/unit/policyfile/cookbook_locks_spec.rb +527 -527
  333. data/spec/unit/policyfile/delivery_supermarket_source_spec.rb +129 -129
  334. data/spec/unit/policyfile/differ_spec.rb +686 -686
  335. data/spec/unit/policyfile/git_lock_fetcher_spec.rb +155 -155
  336. data/spec/unit/policyfile/included_policies_cookbook_source_spec.rb +242 -242
  337. data/spec/unit/policyfile/lister_spec.rb +268 -268
  338. data/spec/unit/policyfile/local_lock_fetcher_spec.rb +199 -173
  339. data/spec/unit/policyfile/lock_applier_spec.rb +100 -100
  340. data/spec/unit/policyfile/lock_fetcher_mixin_spec.rb +60 -0
  341. data/spec/unit/policyfile/null_cookbook_source_spec.rb +34 -34
  342. data/spec/unit/policyfile/read_cookbook_for_compat_mode_upload_spec.rb +92 -92
  343. data/spec/unit/policyfile/remote_lock_fetcher_spec.rb +129 -0
  344. data/spec/unit/policyfile/reports/install_spec.rb +114 -114
  345. data/spec/unit/policyfile/reports/upload_spec.rb +94 -94
  346. data/spec/unit/policyfile/solution_dependencies_spec.rb +170 -170
  347. data/spec/unit/policyfile/source_uri_spec.rb +36 -36
  348. data/spec/unit/policyfile/storage_config_spec.rb +180 -180
  349. data/spec/unit/policyfile/undo_record_spec.rb +258 -258
  350. data/spec/unit/policyfile/undo_stack_spec.rb +265 -265
  351. data/spec/unit/policyfile/uploader_spec.rb +410 -410
  352. data/spec/unit/policyfile_demands_spec.rb +1197 -1197
  353. data/spec/unit/policyfile_evaluation_spec.rb +628 -628
  354. data/spec/unit/policyfile_includes_dsl_spec.rb +220 -159
  355. data/spec/unit/policyfile_includes_spec.rb +720 -720
  356. data/spec/unit/policyfile_install_with_includes_spec.rb +232 -232
  357. data/spec/unit/policyfile_lock_build_spec.rb +1065 -1065
  358. data/spec/unit/policyfile_lock_install_spec.rb +137 -137
  359. data/spec/unit/policyfile_lock_serialization_spec.rb +424 -424
  360. data/spec/unit/policyfile_lock_validation_spec.rb +608 -608
  361. data/spec/unit/policyfile_services/clean_policies_spec.rb +236 -236
  362. data/spec/unit/policyfile_services/clean_policy_cookbooks_spec.rb +272 -272
  363. data/spec/unit/policyfile_services/export_repo_spec.rb +473 -473
  364. data/spec/unit/policyfile_services/install_spec.rb +209 -209
  365. data/spec/unit/policyfile_services/push_archive_spec.rb +359 -359
  366. data/spec/unit/policyfile_services/push_spec.rb +249 -249
  367. data/spec/unit/policyfile_services/rm_policy_group_spec.rb +237 -237
  368. data/spec/unit/policyfile_services/rm_policy_spec.rb +263 -263
  369. data/spec/unit/policyfile_services/show_policy_spec.rb +887 -887
  370. data/spec/unit/policyfile_services/undelete_spec.rb +302 -302
  371. data/spec/unit/policyfile_services/update_attributes_spec.rb +229 -229
  372. data/spec/unit/policyfile_services/update_spec.rb +162 -162
  373. data/spec/unit/service_exception_inspectors/base_spec.rb +41 -41
  374. data/spec/unit/service_exception_inspectors/http_spec.rb +138 -138
  375. data/spec/unit/shell_out_spec.rb +34 -34
  376. data/warning.txt +9 -9
  377. metadata +8 -2
@@ -1,311 +1,311 @@
1
- #
2
- # Copyright:: Copyright (c) 2014-2018, 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
- require "set"
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) &&
49
- other.name == name &&
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") && 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.sort
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.sort.to_h
122
- end
123
-
124
- def transitive_deps(names)
125
- deps = Set.new
126
- to_explore = names.dup
127
- until to_explore.empty?
128
- ck_name = to_explore.shift
129
- next unless deps.add?(ck_name) # explore each ck only once
130
- my_deps = find_cookbook_dep_by_name(ck_name)
131
- dep_names = my_deps[1].map(&:first)
132
- to_explore += dep_names
133
- end
134
- deps.to_a.sort
135
- end
136
-
137
- private
138
-
139
- def add_cookbook_obj_dep(cookbook, dependency_map)
140
- @cookbook_dependencies[cookbook] = dependency_map.map do |dep_name, constraint|
141
- [ dep_name, Semverse::Constraint.new(constraint) ]
142
- end
143
- end
144
-
145
- def assert_cookbook_version_valid!(cookbook_name, version)
146
- policyfile_conflicts = policyfile_conflicts_with(cookbook_name, version)
147
- cookbook_conflicts = cookbook_conflicts_with(cookbook_name, version)
148
- all_conflicts = policyfile_conflicts + cookbook_conflicts
149
-
150
- return false if all_conflicts.empty?
151
-
152
- details = all_conflicts.map { |source, name, constraint| "#{source} depends on #{name} #{constraint}" }
153
- message = "Cookbook #{cookbook_name} (#{version}) conflicts with other dependencies:\n"
154
- full_message = message + details.join("\n")
155
- raise DependencyConflict, full_message
156
- end
157
-
158
- def assert_cookbook_deps_valid!(cookbook_name, version)
159
- dependency_conflicts = cookbook_deps_conflicts_for(cookbook_name, version)
160
- return false if dependency_conflicts.empty?
161
- message = "Cookbook #{cookbook_name} (#{version}) has dependency constraints that cannot be met by the existing cookbook set:\n"
162
- full_message = message + dependency_conflicts.join("\n")
163
- raise DependencyConflict, full_message
164
- end
165
-
166
- def policyfile_conflicts_with(cookbook_name, version)
167
- policyfile_conflicts = []
168
-
169
- @policyfile_dependencies.each do |dep_name, constraint|
170
- if dep_name == cookbook_name && !constraint.satisfies?(version)
171
- policyfile_conflicts << ["Policyfile", dep_name, constraint]
172
- end
173
- end
174
-
175
- policyfile_conflicts
176
- end
177
-
178
- def cookbook_conflicts_with(cookbook_name, version)
179
- cookbook_conflicts = []
180
-
181
- @cookbook_dependencies.each do |top_level_dep_name, dependencies|
182
- dependencies.each do |dep_name, constraint|
183
- if dep_name == cookbook_name && !constraint.satisfies?(version)
184
- cookbook_conflicts << [top_level_dep_name, dep_name, constraint]
185
- end
186
- end
187
- end
188
-
189
- cookbook_conflicts
190
- end
191
-
192
- def cookbook_deps_conflicts_for(cookbook_name, version)
193
- conflicts = []
194
- transitive_deps = find_cookbook_dep_by_name_and_version(cookbook_name, version)
195
- transitive_deps.each do |name, constraint|
196
- existing_cookbook = find_cookbook_dep_by_name(name)
197
- if existing_cookbook.nil?
198
- conflicts << "Cookbook #{name} isn't included in the existing cookbook set."
199
- elsif !constraint.satisfies?(existing_cookbook[0].version)
200
- conflicts << "Dependency on #{name} #{constraint} conflicts with existing version #{existing_cookbook[0]}"
201
- end
202
- end
203
- conflicts
204
- end
205
-
206
- def have_cookbook_dep?(name, version)
207
- @cookbook_dependencies.key?(Cookbook.new(name, version))
208
- end
209
-
210
- def find_cookbook_dep_by_name(name)
211
- @cookbook_dependencies.find { |k, v| k.name == name }
212
- end
213
-
214
- def find_cookbook_dep_by_name_and_version(name, version)
215
- @cookbook_dependencies[Cookbook.new(name, version)]
216
- end
217
-
218
- def set_policyfile_deps_from_lock_data(lock_data)
219
- policyfile_deps_data = lock_data["Policyfile"]
220
-
221
- unless policyfile_deps_data.kind_of?(Array)
222
- msg = "lockfile solution_dependencies Policyfile dependencies must be an array of cookbooks and constraints (got: #{policyfile_deps_data.inspect})"
223
- raise InvalidLockfile, msg
224
- end
225
-
226
- policyfile_deps_data.each do |entry|
227
- add_policyfile_dep_from_lock_data(entry)
228
- end
229
- end
230
-
231
- def add_policyfile_dep_from_lock_data(entry)
232
- unless entry.kind_of?(Array) && entry.size == 2
233
- msg = %Q{lockfile solution_dependencies Policyfile dependencies entry must be like [ "$COOKBOOK_NAME", "$CONSTRAINT" ] (got: #{entry.inspect})}
234
- raise InvalidLockfile, msg
235
- end
236
-
237
- cookbook_name, constraint = entry
238
-
239
- unless cookbook_name.kind_of?(String) && !cookbook_name.empty?
240
- msg = "lockfile solution_dependencies Policyfile dependencies entry. Cookbook name portion must be a string (got: #{entry.inspect})"
241
- raise InvalidLockfile, msg
242
- end
243
-
244
- unless constraint.kind_of?(String) && !constraint.empty?
245
- msg = "malformed lockfile solution_dependencies Policyfile dependencies entry. Version constraint portion must be a string (got: #{entry.inspect})"
246
- raise InvalidLockfile, msg
247
- end
248
- add_policyfile_dep(cookbook_name, constraint)
249
- rescue Semverse::InvalidConstraintFormat
250
- msg = "malformed lockfile solution_dependencies Policyfile dependencies entry. Version constraint portion must be a valid version constraint (got: #{entry.inspect})"
251
- raise InvalidLockfile, msg
252
- end
253
-
254
- def set_cookbook_deps_from_lock_data(lock_data)
255
- cookbook_dependencies_data = lock_data["dependencies"]
256
-
257
- unless cookbook_dependencies_data.kind_of?(Hash)
258
- msg = "lockfile solution_dependencies dependencies entry must be a Hash (JSON object) of dependencies (got: #{cookbook_dependencies_data.inspect})"
259
- raise InvalidLockfile, msg
260
- end
261
-
262
- cookbook_dependencies_data.each do |name_and_version, deps_list|
263
- add_cookbook_dep_from_lock_data(name_and_version, deps_list)
264
- end
265
- end
266
-
267
- def add_cookbook_dep_from_lock_data(name_and_version, deps_list)
268
- unless name_and_version.kind_of?(String)
269
- show = "#{name_and_version.inspect} => #{deps_list.inspect}"
270
- msg = %Q{lockfile cookbook_dependencies entries must be of the form "$COOKBOOK_NAME ($VERSION)" => [ $dependency, ...] (got: #{show}) }
271
- raise InvalidLockfile, msg
272
- end
273
-
274
- unless Cookbook.valid_str?(name_and_version)
275
- msg = %Q{lockfile cookbook_dependencies entry keys must be of the form "$COOKBOOK_NAME ($VERSION)" (got: #{name_and_version.inspect}) }
276
- raise InvalidLockfile, msg
277
- end
278
-
279
- unless deps_list.kind_of?(Array)
280
- msg = %Q{lockfile cookbook_dependencies entry values must be an Array like [ [ "$COOKBOOK_NAME", "$CONSTRAINT" ], ... ] (got: #{deps_list.inspect}) }
281
- raise InvalidLockfile, msg
282
- end
283
-
284
- deps_list.each do |entry|
285
-
286
- unless entry.kind_of?(Array) && entry.size == 2
287
- msg = %Q{lockfile solution_dependencies dependencies entry must be like [ "$COOKBOOK_NAME", "$CONSTRAINT" ] (got: #{entry.inspect})}
288
- raise InvalidLockfile, msg
289
- end
290
-
291
- dep_name, constraint = entry
292
-
293
- unless dep_name.kind_of?(String) && !dep_name.empty?
294
- msg = "malformed lockfile solution_dependencies dependencies entry. Cookbook name portion must be a string (got: #{entry.inspect})"
295
- raise InvalidLockfile, msg
296
- end
297
-
298
- unless constraint.kind_of?(String) && !constraint.empty?
299
- msg = "malformed lockfile solution_dependencies dependencies entry. Version constraint portion must be a string (got: #{entry.inspect})"
300
- raise InvalidLockfile, msg
301
- end
302
- end
303
-
304
- cookbook = Cookbook.parse(name_and_version)
305
- add_cookbook_obj_dep(cookbook, deps_list)
306
- end
307
-
308
- end
309
-
310
- end
311
- end
1
+ #
2
+ # Copyright:: Copyright (c) 2014-2018, 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
+ require "set"
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) &&
49
+ other.name == name &&
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") && 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.sort
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.sort.to_h
122
+ end
123
+
124
+ def transitive_deps(names)
125
+ deps = Set.new
126
+ to_explore = names.dup
127
+ until to_explore.empty?
128
+ ck_name = to_explore.shift
129
+ next unless deps.add?(ck_name) # explore each ck only once
130
+ my_deps = find_cookbook_dep_by_name(ck_name)
131
+ dep_names = my_deps[1].map(&:first)
132
+ to_explore += dep_names
133
+ end
134
+ deps.to_a.sort
135
+ end
136
+
137
+ private
138
+
139
+ def add_cookbook_obj_dep(cookbook, dependency_map)
140
+ @cookbook_dependencies[cookbook] = dependency_map.map do |dep_name, constraint|
141
+ [ dep_name, Semverse::Constraint.new(constraint) ]
142
+ end
143
+ end
144
+
145
+ def assert_cookbook_version_valid!(cookbook_name, version)
146
+ policyfile_conflicts = policyfile_conflicts_with(cookbook_name, version)
147
+ cookbook_conflicts = cookbook_conflicts_with(cookbook_name, version)
148
+ all_conflicts = policyfile_conflicts + cookbook_conflicts
149
+
150
+ return false if all_conflicts.empty?
151
+
152
+ details = all_conflicts.map { |source, name, constraint| "#{source} depends on #{name} #{constraint}" }
153
+ message = "Cookbook #{cookbook_name} (#{version}) conflicts with other dependencies:\n"
154
+ full_message = message + details.join("\n")
155
+ raise DependencyConflict, full_message
156
+ end
157
+
158
+ def assert_cookbook_deps_valid!(cookbook_name, version)
159
+ dependency_conflicts = cookbook_deps_conflicts_for(cookbook_name, version)
160
+ return false if dependency_conflicts.empty?
161
+ message = "Cookbook #{cookbook_name} (#{version}) has dependency constraints that cannot be met by the existing cookbook set:\n"
162
+ full_message = message + dependency_conflicts.join("\n")
163
+ raise DependencyConflict, full_message
164
+ end
165
+
166
+ def policyfile_conflicts_with(cookbook_name, version)
167
+ policyfile_conflicts = []
168
+
169
+ @policyfile_dependencies.each do |dep_name, constraint|
170
+ if dep_name == cookbook_name && !constraint.satisfies?(version)
171
+ policyfile_conflicts << ["Policyfile", dep_name, constraint]
172
+ end
173
+ end
174
+
175
+ policyfile_conflicts
176
+ end
177
+
178
+ def cookbook_conflicts_with(cookbook_name, version)
179
+ cookbook_conflicts = []
180
+
181
+ @cookbook_dependencies.each do |top_level_dep_name, dependencies|
182
+ dependencies.each do |dep_name, constraint|
183
+ if dep_name == cookbook_name && !constraint.satisfies?(version)
184
+ cookbook_conflicts << [top_level_dep_name, dep_name, constraint]
185
+ end
186
+ end
187
+ end
188
+
189
+ cookbook_conflicts
190
+ end
191
+
192
+ def cookbook_deps_conflicts_for(cookbook_name, version)
193
+ conflicts = []
194
+ transitive_deps = find_cookbook_dep_by_name_and_version(cookbook_name, version)
195
+ transitive_deps.each do |name, constraint|
196
+ existing_cookbook = find_cookbook_dep_by_name(name)
197
+ if existing_cookbook.nil?
198
+ conflicts << "Cookbook #{name} isn't included in the existing cookbook set."
199
+ elsif !constraint.satisfies?(existing_cookbook[0].version)
200
+ conflicts << "Dependency on #{name} #{constraint} conflicts with existing version #{existing_cookbook[0]}"
201
+ end
202
+ end
203
+ conflicts
204
+ end
205
+
206
+ def have_cookbook_dep?(name, version)
207
+ @cookbook_dependencies.key?(Cookbook.new(name, version))
208
+ end
209
+
210
+ def find_cookbook_dep_by_name(name)
211
+ @cookbook_dependencies.find { |k, v| k.name == name }
212
+ end
213
+
214
+ def find_cookbook_dep_by_name_and_version(name, version)
215
+ @cookbook_dependencies[Cookbook.new(name, version)]
216
+ end
217
+
218
+ def set_policyfile_deps_from_lock_data(lock_data)
219
+ policyfile_deps_data = lock_data["Policyfile"]
220
+
221
+ unless policyfile_deps_data.kind_of?(Array)
222
+ msg = "lockfile solution_dependencies Policyfile dependencies must be an array of cookbooks and constraints (got: #{policyfile_deps_data.inspect})"
223
+ raise InvalidLockfile, msg
224
+ end
225
+
226
+ policyfile_deps_data.each do |entry|
227
+ add_policyfile_dep_from_lock_data(entry)
228
+ end
229
+ end
230
+
231
+ def add_policyfile_dep_from_lock_data(entry)
232
+ unless entry.kind_of?(Array) && entry.size == 2
233
+ msg = %Q{lockfile solution_dependencies Policyfile dependencies entry must be like [ "$COOKBOOK_NAME", "$CONSTRAINT" ] (got: #{entry.inspect})}
234
+ raise InvalidLockfile, msg
235
+ end
236
+
237
+ cookbook_name, constraint = entry
238
+
239
+ unless cookbook_name.kind_of?(String) && !cookbook_name.empty?
240
+ msg = "lockfile solution_dependencies Policyfile dependencies entry. Cookbook name portion must be a string (got: #{entry.inspect})"
241
+ raise InvalidLockfile, msg
242
+ end
243
+
244
+ unless constraint.kind_of?(String) && !constraint.empty?
245
+ msg = "malformed lockfile solution_dependencies Policyfile dependencies entry. Version constraint portion must be a string (got: #{entry.inspect})"
246
+ raise InvalidLockfile, msg
247
+ end
248
+ add_policyfile_dep(cookbook_name, constraint)
249
+ rescue Semverse::InvalidConstraintFormat
250
+ msg = "malformed lockfile solution_dependencies Policyfile dependencies entry. Version constraint portion must be a valid version constraint (got: #{entry.inspect})"
251
+ raise InvalidLockfile, msg
252
+ end
253
+
254
+ def set_cookbook_deps_from_lock_data(lock_data)
255
+ cookbook_dependencies_data = lock_data["dependencies"]
256
+
257
+ unless cookbook_dependencies_data.kind_of?(Hash)
258
+ msg = "lockfile solution_dependencies dependencies entry must be a Hash (JSON object) of dependencies (got: #{cookbook_dependencies_data.inspect})"
259
+ raise InvalidLockfile, msg
260
+ end
261
+
262
+ cookbook_dependencies_data.each do |name_and_version, deps_list|
263
+ add_cookbook_dep_from_lock_data(name_and_version, deps_list)
264
+ end
265
+ end
266
+
267
+ def add_cookbook_dep_from_lock_data(name_and_version, deps_list)
268
+ unless name_and_version.kind_of?(String)
269
+ show = "#{name_and_version.inspect} => #{deps_list.inspect}"
270
+ msg = %Q{lockfile cookbook_dependencies entries must be of the form "$COOKBOOK_NAME ($VERSION)" => [ $dependency, ...] (got: #{show}) }
271
+ raise InvalidLockfile, msg
272
+ end
273
+
274
+ unless Cookbook.valid_str?(name_and_version)
275
+ msg = %Q{lockfile cookbook_dependencies entry keys must be of the form "$COOKBOOK_NAME ($VERSION)" (got: #{name_and_version.inspect}) }
276
+ raise InvalidLockfile, msg
277
+ end
278
+
279
+ unless deps_list.kind_of?(Array)
280
+ msg = %Q{lockfile cookbook_dependencies entry values must be an Array like [ [ "$COOKBOOK_NAME", "$CONSTRAINT" ], ... ] (got: #{deps_list.inspect}) }
281
+ raise InvalidLockfile, msg
282
+ end
283
+
284
+ deps_list.each do |entry|
285
+
286
+ unless entry.kind_of?(Array) && entry.size == 2
287
+ msg = %Q{lockfile solution_dependencies dependencies entry must be like [ "$COOKBOOK_NAME", "$CONSTRAINT" ] (got: #{entry.inspect})}
288
+ raise InvalidLockfile, msg
289
+ end
290
+
291
+ dep_name, constraint = entry
292
+
293
+ unless dep_name.kind_of?(String) && !dep_name.empty?
294
+ msg = "malformed lockfile solution_dependencies dependencies entry. Cookbook name portion must be a string (got: #{entry.inspect})"
295
+ raise InvalidLockfile, msg
296
+ end
297
+
298
+ unless constraint.kind_of?(String) && !constraint.empty?
299
+ msg = "malformed lockfile solution_dependencies dependencies entry. Version constraint portion must be a string (got: #{entry.inspect})"
300
+ raise InvalidLockfile, msg
301
+ end
302
+ end
303
+
304
+ cookbook = Cookbook.parse(name_and_version)
305
+ add_cookbook_obj_dep(cookbook, deps_list)
306
+ end
307
+
308
+ end
309
+
310
+ end
311
+ end