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,473 +1,473 @@
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 "spec_helper"
19
- require "chef-dk/policyfile_services/export_repo"
20
-
21
- describe ChefDK::PolicyfileServices::ExportRepo do
22
-
23
- let(:working_dir) do
24
- path = File.join(tempdir, "policyfile_services_test_working_dir")
25
- Dir.mkdir(path)
26
- path
27
- end
28
-
29
- let(:export_dir) { File.join(tempdir, "export_repo_export_dir") }
30
-
31
- let(:policyfile_rb_explicit_name) { nil }
32
-
33
- let(:policyfile_rb_name) { policyfile_rb_explicit_name || "Policyfile.rb" }
34
-
35
- let(:expanded_policyfile_path) { File.join(working_dir, policyfile_rb_name) }
36
-
37
- let(:policyfile_lock_name) { "Policyfile.lock.json" }
38
-
39
- let(:policyfile_lock_path) { File.join(working_dir, policyfile_lock_name) }
40
-
41
- let(:force_export) { false }
42
-
43
- let(:archive) { false }
44
-
45
- subject(:export_service) do
46
- described_class.new(policyfile: policyfile_rb_explicit_name,
47
- root_dir: working_dir,
48
- export_dir: export_dir,
49
- archive: archive,
50
- force: force_export)
51
- end
52
-
53
- it "uses Policyfile.rb as the default Policyfile name" do
54
- expect(export_service.policyfile_filename).to eq(expanded_policyfile_path)
55
- end
56
-
57
- context "when given an explicit Policyfile name" do
58
-
59
- let(:policyfile_rb_explicit_name) { "MyPolicy.rb" }
60
-
61
- it "uses the given Policyfile name" do
62
- expect(export_service.policyfile_filename).to eq(expanded_policyfile_path)
63
- end
64
-
65
- end
66
-
67
- it "has a destination directory for the export" do
68
- expect(export_service.export_dir).to eq(export_dir)
69
- end
70
-
71
- context "when the policyfile lock is missing" do
72
-
73
- it "raises an error that suggests you run `chef install'" do
74
- expect { export_service.run }.to raise_error(ChefDK::LockfileNotFound)
75
- end
76
-
77
- end
78
-
79
- context "when a lockfile is present" do
80
-
81
- before do
82
- File.open(policyfile_lock_path, "w+") { |f| f.print(lockfile_content) }
83
- end
84
-
85
- context "and the lockfile has invalid JSON" do
86
-
87
- let(:lockfile_content) { ":::" }
88
-
89
- it "errors out" do
90
- expect { export_service.run }.to raise_error(ChefDK::PolicyfileExportRepoError, /Error reading lockfile/)
91
- end
92
-
93
- end
94
-
95
- context "and the lockfile is semantically invalid" do
96
-
97
- let(:lockfile_content) { "{ }" }
98
-
99
- it "errors out" do
100
- expect { export_service.run }.to raise_error(ChefDK::PolicyfileExportRepoError, /Invalid lockfile data/)
101
- end
102
-
103
- end
104
-
105
- context "and the lockfile is valid" do
106
-
107
- let(:local_cookbook_path) { File.join(fixtures_path, "local_path_cookbooks/local-cookbook") }
108
-
109
- let(:revision_id) { "7da81d2c7bb97f904637f97e7f8b487fa4bb1ed682edea7087743dec84c254ec" }
110
-
111
- let(:lockfile_content) do
112
- <<~E
113
- {
114
- "revision_id": "#{revision_id}",
115
- "name": "install-example",
116
- "run_list": [
117
- "recipe[local-cookbook::default]"
118
- ],
119
- "cookbook_locks": {
120
- "local-cookbook": {
121
- "version": "2.3.4",
122
- "identifier": "1e9dfd1134735385b425c056cb5decef9081b92c",
123
- "dotted_decimal_identifier": "42704157235437826.6970356709321892.63549625984142",
124
- "source": "#{local_cookbook_path}",
125
- "cache_key": null,
126
- "scm_info": null,
127
- "source_options": {
128
- "path": "#{local_cookbook_path}"
129
- }
130
- }
131
- },
132
- "default_attributes": {},
133
- "override_attributes": {},
134
- "solution_dependencies": {
135
- "Policyfile": [
136
- [
137
- "local-cookbook",
138
- ">= 0.0.0"
139
- ]
140
- ],
141
- "dependencies": {
142
- "local-cookbook (2.3.4)": [
143
-
144
- ]
145
- }
146
- }
147
- }
148
- E
149
- end
150
-
151
- it "reads the lockfile data" do
152
- lock = export_service.policyfile_lock
153
- expect(lock).to be_an_instance_of(ChefDK::PolicyfileLock)
154
- expect(lock.name).to eq("install-example")
155
- expect(lock.cookbook_locks.size).to eq(1)
156
- expect(lock.cookbook_locks).to have_key("local-cookbook")
157
- end
158
-
159
- it "delegates #policy_name to the lockfile" do
160
- expect(export_service.policy_name).to eq("install-example")
161
- end
162
-
163
- context "when using archive mode" do
164
-
165
- let(:archive) { true }
166
-
167
- # TODO: also support a full file name
168
- context "when the given 'export_dir' is a directory" do
169
-
170
- it "sets the archive file location to $policy_name-$revision.tgz" do
171
- expected = File.join(export_dir, "install-example-7da81d2c7bb97f904637f97e7f8b487fa4bb1ed682edea7087743dec84c254ec.tgz")
172
- expect(export_service.archive_file_location).to eq(expected)
173
- end
174
-
175
- end
176
-
177
- end
178
-
179
- describe "writing updates to the policyfile lock" do
180
-
181
- let(:updated_lockfile_io) { StringIO.new }
182
-
183
- it "validates the lockfile and writes updates to disk" do
184
- allow(File).to receive(:open).and_call_original
185
- expect(File).to receive(:open).with(policyfile_lock_path, "wb+").and_yield(updated_lockfile_io)
186
-
187
- export_service.run
188
- end
189
-
190
- end
191
-
192
- context "copying the cookbooks to the export dir" do
193
-
194
- shared_examples_for "successful_export" do
195
- before do
196
- allow(export_service.policyfile_lock).to receive(:validate_cookbooks!).and_return(true)
197
- export_service.run
198
- end
199
-
200
- let(:cookbook_files) do
201
- base_pathname = Pathname.new(local_cookbook_path)
202
- Dir.glob("#{local_cookbook_path}/**/*").map do |full_path|
203
- Pathname.new(full_path).relative_path_from(base_pathname)
204
- end
205
- end
206
-
207
- let(:expected_files_relative) do
208
- metadata_rb = Pathname.new("metadata.rb")
209
- expected = cookbook_files.delete_if { |p| p == metadata_rb }
210
-
211
- # Berksfile is chefignored
212
- berksfile = Pathname.new("Berksfile")
213
- expected = expected.delete_if { |p| p == berksfile }
214
-
215
- expected << Pathname.new("metadata.json")
216
- end
217
-
218
- let(:cookbook_with_version) { "local-cookbook-1e9dfd1134735385b425c056cb5decef9081b92c" }
219
-
220
- let(:exported_cookbook_root) { Pathname.new(File.join(export_dir, "cookbook_artifacts", cookbook_with_version)) }
221
-
222
- let(:expected_files) do
223
- expected_files_relative.map do |file_rel_path|
224
- exported_cookbook_root + file_rel_path
225
- end
226
- end
227
-
228
- it "copies cookbooks to the target dir in versioned_cookbooks format" do
229
- expected_files.each do |expected_file|
230
- expect(expected_file).to exist
231
- end
232
- end
233
-
234
- # Using JSON form of metadata ensures that we don't rely on anything
235
- # in the ruby code in metadata.rb; commonly folks will do things like
236
- # shell out to git for the version number, etc.
237
- it "writes metadata.json in the exported cookbook, removing metadata.rb" do
238
- metadata_json_path = File.join(exported_cookbook_root, "metadata.json")
239
- metadata_json = FFI_Yajl::Parser.parse(IO.read(metadata_json_path))
240
- expect(metadata_json["version"]).to eq("2.3.4")
241
- end
242
-
243
- it "copies the policyfile lock to policies/POLICY_NAME.json" do
244
- exported_policy_path = File.join(export_dir, "policies", "install-example-#{revision_id}.json")
245
- exported_policy_json = IO.read(exported_policy_path)
246
- expect(exported_policy_json).to eq(FFI_Yajl::Encoder.encode(export_service.policyfile_lock.to_lock, pretty: true))
247
- end
248
-
249
- it "creates a policy_group file for the local policy group with the revision id of the exported policy" do
250
- exported_policy_group_path = File.join(export_dir, "policy_groups", "local.json")
251
- exported_policy_group_data = FFI_Yajl::Parser.parse(IO.read(exported_policy_group_path))
252
-
253
- expected_data = { "policies" => { "install-example" => { "revision_id" => revision_id } } }
254
-
255
- expect(exported_policy_group_data).to eq(expected_data)
256
- end
257
-
258
- it "copies the policyfile lock in standard format to Policyfile.lock.json" do
259
- policyfile_lock_path = File.join(export_dir, "Policyfile.lock.json")
260
- policyfile_lock_data = FFI_Yajl::Parser.parse(IO.read(policyfile_lock_path))
261
- expected_lock_data = export_service.policyfile_lock.to_lock
262
-
263
- # stringify keys in source_options
264
- path = expected_lock_data["cookbook_locks"]["local-cookbook"]["source_options"].delete(:path)
265
- expected_lock_data["cookbook_locks"]["local-cookbook"]["source_options"]["path"] = path
266
-
267
- expect(policyfile_lock_data).to eq(expected_lock_data)
268
- end
269
-
270
- it "creates a working local mode configuration file" do
271
- expected_config_text = <<~CONFIG
272
- ### Chef Client Configuration ###
273
- # The settings in this file will configure chef to apply the exported policy in
274
- # this directory. To use it, run:
275
- #
276
- # chef-client -z
277
- #
278
-
279
- policy_name 'install-example'
280
- policy_group 'local'
281
-
282
- use_policyfile true
283
- policy_document_native_api true
284
-
285
- # In order to use this repo, you need a version of Chef Client and Chef Zero
286
- # that supports policyfile "native mode" APIs:
287
- current_version = Gem::Version.new(Chef::VERSION)
288
- unless Gem::Requirement.new(">= 12.7").satisfied_by?(current_version)
289
- puts("!" * 80)
290
- puts(<<-MESSAGE)
291
- This Chef Repo requires features introduced in Chef 12.7, but you are using
292
- Chef \#{Chef::VERSION}. Please upgrade to Chef 12.7 or later.
293
- MESSAGE
294
- puts("!" * 80)
295
- exit!(1)
296
- end
297
-
298
- CONFIG
299
- config_path = File.join(export_dir, ".chef", "config.rb")
300
- expect(File).to exist(config_path)
301
- expect(IO.read(config_path)).to eq(expected_config_text)
302
- end
303
-
304
- it "generates a README.md in the exported repo" do
305
- readme_path = File.join(export_dir, "README.md")
306
- expect(File).to exist(readme_path)
307
- end
308
-
309
- end
310
-
311
- context "when the export dir is empty" do
312
-
313
- include_examples "successful_export"
314
- end
315
-
316
- context "When an error occurs creating the export" do
317
-
318
- before do
319
- allow(export_service.policyfile_lock).to receive(:validate_cookbooks!).and_return(true)
320
- expect(export_service).to receive(:create_repo_structure)
321
- .and_raise(Errno::EACCES.new("Permission denied @ rb_sysopen - /etc/foobarbaz.txt"))
322
- end
323
-
324
- it "wraps the error in a custom error class" do
325
- message = "Failed to export policy (in #{expanded_policyfile_path}) to #{export_dir}"
326
- expect { export_service.run }.to raise_error(ChefDK::PolicyfileExportRepoError, message)
327
- end
328
-
329
- end
330
-
331
- context "When the export dir has non-conflicting content" do
332
-
333
- let(:file_in_export_dir) { File.join(export_dir, "some_random_cruft") }
334
-
335
- let(:extra_data_bag_dir) { File.join(export_dir, "data_bags", "extraneous") }
336
-
337
- let(:extra_data_bag_item) { File.join(extra_data_bag_dir, "an_item.json") }
338
-
339
- before do
340
- FileUtils.mkdir_p(export_dir)
341
- File.open(file_in_export_dir, "wb+") { |f| f.print "some random cruft" }
342
- FileUtils.mkdir_p(extra_data_bag_dir)
343
- File.open(extra_data_bag_item, "wb+") { |f| f.print "some random cruft" }
344
- end
345
-
346
- it "ignores the non-conflicting content and exports" do
347
- expect(File).to exist(file_in_export_dir)
348
- expect(File).to exist(extra_data_bag_item)
349
-
350
- expect(File).to be_directory(File.join(export_dir, "cookbook_artifacts"))
351
- expect(File).to be_directory(File.join(export_dir, "policies"))
352
- expect(File).to be_directory(File.join(export_dir, "policy_groups"))
353
- end
354
-
355
- include_examples "successful_export"
356
-
357
- end
358
-
359
- context "When the export dir has conflicting content" do
360
-
361
- let(:non_conflicting_file_in_export_dir) { File.join(export_dir, "some_random_cruft") }
362
-
363
- let(:cookbook_artifacts_dir) { File.join(export_dir, "cookbook_artifacts") }
364
-
365
- let(:file_in_cookbook_artifacts_dir) { File.join(cookbook_artifacts_dir, "some_random_cruft") }
366
-
367
- let(:policies_dir) { File.join(export_dir, "policies") }
368
-
369
- let(:policy_groups_dir) { File.join(export_dir, "policy_groups") }
370
-
371
- let(:extra_policy_item) { File.join(policies_dir, "leftover-policy.json") }
372
-
373
- let(:extra_policy_group_item) { File.join(policy_groups_dir, "leftover-policy-group.json") }
374
-
375
- let(:conflicting_policyfile_lock) { File.join(export_dir, "Policyfile.lock.json") }
376
-
377
- before do
378
- FileUtils.mkdir_p(export_dir)
379
- FileUtils.mkdir_p(cookbook_artifacts_dir)
380
- FileUtils.mkdir_p(policies_dir)
381
- FileUtils.mkdir_p(policy_groups_dir)
382
- File.open(non_conflicting_file_in_export_dir, "wb+") { |f| f.print "some random cruft" }
383
- File.open(file_in_cookbook_artifacts_dir, "wb+") { |f| f.print "some random cruft" }
384
- File.open(extra_policy_item, "wb+") { |f| f.print "some random cruft" }
385
- File.open(extra_policy_group_item, "wb+") { |f| f.print "some random cruft" }
386
- File.open(conflicting_policyfile_lock, "wb+") { |f| f.print "some random cruft" }
387
- end
388
-
389
- it "raises a PolicyfileExportRepoError" do
390
- message = "Export dir (#{export_dir}) not clean. Refusing to export. (Conflicting files: #{file_in_cookbook_artifacts_dir}, #{extra_policy_item}, #{extra_policy_group_item}, #{conflicting_policyfile_lock})"
391
- expect { export_service.run }.to raise_error(ChefDK::ExportDirNotEmpty, message)
392
- expect(File).to exist(non_conflicting_file_in_export_dir)
393
- expect(File).to exist(file_in_cookbook_artifacts_dir)
394
- expect(File).to exist(extra_policy_item)
395
- expect(File).to exist(extra_policy_group_item)
396
- end
397
-
398
- context "and the force option is set" do
399
-
400
- let(:force_export) { true }
401
-
402
- it "clears the export dir and exports" do
403
- export_service.run
404
-
405
- expect(File).to_not exist(file_in_cookbook_artifacts_dir)
406
- expect(File).to_not exist(extra_policy_item)
407
- expect(File).to_not exist(extra_policy_group_item)
408
-
409
- expect(File).to exist(non_conflicting_file_in_export_dir)
410
-
411
- expect(File).to be_directory(File.join(export_dir, "cookbook_artifacts"))
412
- expect(File).to be_directory(File.join(export_dir, "policies"))
413
- expect(File).to be_directory(File.join(export_dir, "policy_groups"))
414
- end
415
-
416
- end
417
-
418
- end # When the export dir has conflicting content
419
-
420
- context "when archive mode is enabled" do
421
-
422
- let(:archive) { true }
423
-
424
- let(:expected_archive_path) do
425
- File.join(export_dir, "install-example-7da81d2c7bb97f904637f97e7f8b487fa4bb1ed682edea7087743dec84c254ec.tgz")
426
- end
427
-
428
- it "exports the repo as a tgz archive" do
429
- expect(File).to exist(expected_archive_path)
430
- end
431
-
432
- include_examples "successful_export" do
433
-
434
- # explode the tarball so the assertions can find the files
435
- before do
436
- Mixlib::Archive.new(expected_archive_path).extract(export_dir)
437
- end
438
-
439
- end
440
-
441
- context "when the target dir has a cookbooks or data_bags dir" do
442
-
443
- let(:cookbooks_dir) { File.join(export_dir, "cookbooks") }
444
-
445
- let(:file_in_cookbooks_dir) { File.join(cookbooks_dir, "some_random_cruft") }
446
-
447
- let(:policyfiles_data_bag_dir) { File.join(export_dir, "data_bags", "policyfiles") }
448
-
449
- let(:extra_policyfile_data_item) { File.join(policyfiles_data_bag_dir, "leftover-policy.json") }
450
-
451
- before do
452
- FileUtils.mkdir_p(export_dir)
453
- FileUtils.mkdir_p(cookbooks_dir)
454
- FileUtils.mkdir_p(policyfiles_data_bag_dir)
455
- File.open(file_in_cookbooks_dir, "wb+") { |f| f.print "some random cruft" }
456
- File.open(extra_policyfile_data_item, "wb+") { |f| f.print "some random cruft" }
457
- end
458
-
459
- it "exports successfully" do
460
- expect { export_service.run }.to_not raise_error
461
- expect(File).to exist(expected_archive_path)
462
- end
463
-
464
- end
465
-
466
- end # when archive mode is enabled
467
-
468
- end # copying the cookbooks to the export dir
469
- end
470
-
471
- end
472
-
473
- 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 "spec_helper"
19
+ require "chef-dk/policyfile_services/export_repo"
20
+
21
+ describe ChefDK::PolicyfileServices::ExportRepo do
22
+
23
+ let(:working_dir) do
24
+ path = File.join(tempdir, "policyfile_services_test_working_dir")
25
+ Dir.mkdir(path)
26
+ path
27
+ end
28
+
29
+ let(:export_dir) { File.join(tempdir, "export_repo_export_dir") }
30
+
31
+ let(:policyfile_rb_explicit_name) { nil }
32
+
33
+ let(:policyfile_rb_name) { policyfile_rb_explicit_name || "Policyfile.rb" }
34
+
35
+ let(:expanded_policyfile_path) { File.join(working_dir, policyfile_rb_name) }
36
+
37
+ let(:policyfile_lock_name) { "Policyfile.lock.json" }
38
+
39
+ let(:policyfile_lock_path) { File.join(working_dir, policyfile_lock_name) }
40
+
41
+ let(:force_export) { false }
42
+
43
+ let(:archive) { false }
44
+
45
+ subject(:export_service) do
46
+ described_class.new(policyfile: policyfile_rb_explicit_name,
47
+ root_dir: working_dir,
48
+ export_dir: export_dir,
49
+ archive: archive,
50
+ force: force_export)
51
+ end
52
+
53
+ it "uses Policyfile.rb as the default Policyfile name" do
54
+ expect(export_service.policyfile_filename).to eq(expanded_policyfile_path)
55
+ end
56
+
57
+ context "when given an explicit Policyfile name" do
58
+
59
+ let(:policyfile_rb_explicit_name) { "MyPolicy.rb" }
60
+
61
+ it "uses the given Policyfile name" do
62
+ expect(export_service.policyfile_filename).to eq(expanded_policyfile_path)
63
+ end
64
+
65
+ end
66
+
67
+ it "has a destination directory for the export" do
68
+ expect(export_service.export_dir).to eq(export_dir)
69
+ end
70
+
71
+ context "when the policyfile lock is missing" do
72
+
73
+ it "raises an error that suggests you run `chef install'" do
74
+ expect { export_service.run }.to raise_error(ChefDK::LockfileNotFound)
75
+ end
76
+
77
+ end
78
+
79
+ context "when a lockfile is present" do
80
+
81
+ before do
82
+ File.open(policyfile_lock_path, "w+") { |f| f.print(lockfile_content) }
83
+ end
84
+
85
+ context "and the lockfile has invalid JSON" do
86
+
87
+ let(:lockfile_content) { ":::" }
88
+
89
+ it "errors out" do
90
+ expect { export_service.run }.to raise_error(ChefDK::PolicyfileExportRepoError, /Error reading lockfile/)
91
+ end
92
+
93
+ end
94
+
95
+ context "and the lockfile is semantically invalid" do
96
+
97
+ let(:lockfile_content) { "{ }" }
98
+
99
+ it "errors out" do
100
+ expect { export_service.run }.to raise_error(ChefDK::PolicyfileExportRepoError, /Invalid lockfile data/)
101
+ end
102
+
103
+ end
104
+
105
+ context "and the lockfile is valid" do
106
+
107
+ let(:local_cookbook_path) { File.join(fixtures_path, "local_path_cookbooks/local-cookbook") }
108
+
109
+ let(:revision_id) { "7da81d2c7bb97f904637f97e7f8b487fa4bb1ed682edea7087743dec84c254ec" }
110
+
111
+ let(:lockfile_content) do
112
+ <<~E
113
+ {
114
+ "revision_id": "#{revision_id}",
115
+ "name": "install-example",
116
+ "run_list": [
117
+ "recipe[local-cookbook::default]"
118
+ ],
119
+ "cookbook_locks": {
120
+ "local-cookbook": {
121
+ "version": "2.3.4",
122
+ "identifier": "1e9dfd1134735385b425c056cb5decef9081b92c",
123
+ "dotted_decimal_identifier": "42704157235437826.6970356709321892.63549625984142",
124
+ "source": "#{local_cookbook_path}",
125
+ "cache_key": null,
126
+ "scm_info": null,
127
+ "source_options": {
128
+ "path": "#{local_cookbook_path}"
129
+ }
130
+ }
131
+ },
132
+ "default_attributes": {},
133
+ "override_attributes": {},
134
+ "solution_dependencies": {
135
+ "Policyfile": [
136
+ [
137
+ "local-cookbook",
138
+ ">= 0.0.0"
139
+ ]
140
+ ],
141
+ "dependencies": {
142
+ "local-cookbook (2.3.4)": [
143
+
144
+ ]
145
+ }
146
+ }
147
+ }
148
+ E
149
+ end
150
+
151
+ it "reads the lockfile data" do
152
+ lock = export_service.policyfile_lock
153
+ expect(lock).to be_an_instance_of(ChefDK::PolicyfileLock)
154
+ expect(lock.name).to eq("install-example")
155
+ expect(lock.cookbook_locks.size).to eq(1)
156
+ expect(lock.cookbook_locks).to have_key("local-cookbook")
157
+ end
158
+
159
+ it "delegates #policy_name to the lockfile" do
160
+ expect(export_service.policy_name).to eq("install-example")
161
+ end
162
+
163
+ context "when using archive mode" do
164
+
165
+ let(:archive) { true }
166
+
167
+ # TODO: also support a full file name
168
+ context "when the given 'export_dir' is a directory" do
169
+
170
+ it "sets the archive file location to $policy_name-$revision.tgz" do
171
+ expected = File.join(export_dir, "install-example-7da81d2c7bb97f904637f97e7f8b487fa4bb1ed682edea7087743dec84c254ec.tgz")
172
+ expect(export_service.archive_file_location).to eq(expected)
173
+ end
174
+
175
+ end
176
+
177
+ end
178
+
179
+ describe "writing updates to the policyfile lock" do
180
+
181
+ let(:updated_lockfile_io) { StringIO.new }
182
+
183
+ it "validates the lockfile and writes updates to disk" do
184
+ allow(File).to receive(:open).and_call_original
185
+ expect(File).to receive(:open).with(policyfile_lock_path, "wb+").and_yield(updated_lockfile_io)
186
+
187
+ export_service.run
188
+ end
189
+
190
+ end
191
+
192
+ context "copying the cookbooks to the export dir" do
193
+
194
+ shared_examples_for "successful_export" do
195
+ before do
196
+ allow(export_service.policyfile_lock).to receive(:validate_cookbooks!).and_return(true)
197
+ export_service.run
198
+ end
199
+
200
+ let(:cookbook_files) do
201
+ base_pathname = Pathname.new(local_cookbook_path)
202
+ Dir.glob("#{local_cookbook_path}/**/*").map do |full_path|
203
+ Pathname.new(full_path).relative_path_from(base_pathname)
204
+ end
205
+ end
206
+
207
+ let(:expected_files_relative) do
208
+ metadata_rb = Pathname.new("metadata.rb")
209
+ expected = cookbook_files.delete_if { |p| p == metadata_rb }
210
+
211
+ # Berksfile is chefignored
212
+ berksfile = Pathname.new("Berksfile")
213
+ expected = expected.delete_if { |p| p == berksfile }
214
+
215
+ expected << Pathname.new("metadata.json")
216
+ end
217
+
218
+ let(:cookbook_with_version) { "local-cookbook-1e9dfd1134735385b425c056cb5decef9081b92c" }
219
+
220
+ let(:exported_cookbook_root) { Pathname.new(File.join(export_dir, "cookbook_artifacts", cookbook_with_version)) }
221
+
222
+ let(:expected_files) do
223
+ expected_files_relative.map do |file_rel_path|
224
+ exported_cookbook_root + file_rel_path
225
+ end
226
+ end
227
+
228
+ it "copies cookbooks to the target dir in versioned_cookbooks format" do
229
+ expected_files.each do |expected_file|
230
+ expect(expected_file).to exist
231
+ end
232
+ end
233
+
234
+ # Using JSON form of metadata ensures that we don't rely on anything
235
+ # in the ruby code in metadata.rb; commonly folks will do things like
236
+ # shell out to git for the version number, etc.
237
+ it "writes metadata.json in the exported cookbook, removing metadata.rb" do
238
+ metadata_json_path = File.join(exported_cookbook_root, "metadata.json")
239
+ metadata_json = FFI_Yajl::Parser.parse(IO.read(metadata_json_path))
240
+ expect(metadata_json["version"]).to eq("2.3.4")
241
+ end
242
+
243
+ it "copies the policyfile lock to policies/POLICY_NAME.json" do
244
+ exported_policy_path = File.join(export_dir, "policies", "install-example-#{revision_id}.json")
245
+ exported_policy_json = IO.read(exported_policy_path)
246
+ expect(exported_policy_json).to eq(FFI_Yajl::Encoder.encode(export_service.policyfile_lock.to_lock, pretty: true))
247
+ end
248
+
249
+ it "creates a policy_group file for the local policy group with the revision id of the exported policy" do
250
+ exported_policy_group_path = File.join(export_dir, "policy_groups", "local.json")
251
+ exported_policy_group_data = FFI_Yajl::Parser.parse(IO.read(exported_policy_group_path))
252
+
253
+ expected_data = { "policies" => { "install-example" => { "revision_id" => revision_id } } }
254
+
255
+ expect(exported_policy_group_data).to eq(expected_data)
256
+ end
257
+
258
+ it "copies the policyfile lock in standard format to Policyfile.lock.json" do
259
+ policyfile_lock_path = File.join(export_dir, "Policyfile.lock.json")
260
+ policyfile_lock_data = FFI_Yajl::Parser.parse(IO.read(policyfile_lock_path))
261
+ expected_lock_data = export_service.policyfile_lock.to_lock
262
+
263
+ # stringify keys in source_options
264
+ path = expected_lock_data["cookbook_locks"]["local-cookbook"]["source_options"].delete(:path)
265
+ expected_lock_data["cookbook_locks"]["local-cookbook"]["source_options"]["path"] = path
266
+
267
+ expect(policyfile_lock_data).to eq(expected_lock_data)
268
+ end
269
+
270
+ it "creates a working local mode configuration file" do
271
+ expected_config_text = <<~CONFIG
272
+ ### Chef Client Configuration ###
273
+ # The settings in this file will configure chef to apply the exported policy in
274
+ # this directory. To use it, run:
275
+ #
276
+ # chef-client -z
277
+ #
278
+
279
+ policy_name 'install-example'
280
+ policy_group 'local'
281
+
282
+ use_policyfile true
283
+ policy_document_native_api true
284
+
285
+ # In order to use this repo, you need a version of Chef Client and Chef Zero
286
+ # that supports policyfile "native mode" APIs:
287
+ current_version = Gem::Version.new(Chef::VERSION)
288
+ unless Gem::Requirement.new(">= 12.7").satisfied_by?(current_version)
289
+ puts("!" * 80)
290
+ puts(<<-MESSAGE)
291
+ This Chef Repo requires features introduced in Chef 12.7, but you are using
292
+ Chef \#{Chef::VERSION}. Please upgrade to Chef 12.7 or later.
293
+ MESSAGE
294
+ puts("!" * 80)
295
+ exit!(1)
296
+ end
297
+
298
+ CONFIG
299
+ config_path = File.join(export_dir, ".chef", "config.rb")
300
+ expect(File).to exist(config_path)
301
+ expect(IO.read(config_path)).to eq(expected_config_text)
302
+ end
303
+
304
+ it "generates a README.md in the exported repo" do
305
+ readme_path = File.join(export_dir, "README.md")
306
+ expect(File).to exist(readme_path)
307
+ end
308
+
309
+ end
310
+
311
+ context "when the export dir is empty" do
312
+
313
+ include_examples "successful_export"
314
+ end
315
+
316
+ context "When an error occurs creating the export" do
317
+
318
+ before do
319
+ allow(export_service.policyfile_lock).to receive(:validate_cookbooks!).and_return(true)
320
+ expect(export_service).to receive(:create_repo_structure)
321
+ .and_raise(Errno::EACCES.new("Permission denied @ rb_sysopen - /etc/foobarbaz.txt"))
322
+ end
323
+
324
+ it "wraps the error in a custom error class" do
325
+ message = "Failed to export policy (in #{expanded_policyfile_path}) to #{export_dir}"
326
+ expect { export_service.run }.to raise_error(ChefDK::PolicyfileExportRepoError, message)
327
+ end
328
+
329
+ end
330
+
331
+ context "When the export dir has non-conflicting content" do
332
+
333
+ let(:file_in_export_dir) { File.join(export_dir, "some_random_cruft") }
334
+
335
+ let(:extra_data_bag_dir) { File.join(export_dir, "data_bags", "extraneous") }
336
+
337
+ let(:extra_data_bag_item) { File.join(extra_data_bag_dir, "an_item.json") }
338
+
339
+ before do
340
+ FileUtils.mkdir_p(export_dir)
341
+ File.open(file_in_export_dir, "wb+") { |f| f.print "some random cruft" }
342
+ FileUtils.mkdir_p(extra_data_bag_dir)
343
+ File.open(extra_data_bag_item, "wb+") { |f| f.print "some random cruft" }
344
+ end
345
+
346
+ it "ignores the non-conflicting content and exports" do
347
+ expect(File).to exist(file_in_export_dir)
348
+ expect(File).to exist(extra_data_bag_item)
349
+
350
+ expect(File).to be_directory(File.join(export_dir, "cookbook_artifacts"))
351
+ expect(File).to be_directory(File.join(export_dir, "policies"))
352
+ expect(File).to be_directory(File.join(export_dir, "policy_groups"))
353
+ end
354
+
355
+ include_examples "successful_export"
356
+
357
+ end
358
+
359
+ context "When the export dir has conflicting content" do
360
+
361
+ let(:non_conflicting_file_in_export_dir) { File.join(export_dir, "some_random_cruft") }
362
+
363
+ let(:cookbook_artifacts_dir) { File.join(export_dir, "cookbook_artifacts") }
364
+
365
+ let(:file_in_cookbook_artifacts_dir) { File.join(cookbook_artifacts_dir, "some_random_cruft") }
366
+
367
+ let(:policies_dir) { File.join(export_dir, "policies") }
368
+
369
+ let(:policy_groups_dir) { File.join(export_dir, "policy_groups") }
370
+
371
+ let(:extra_policy_item) { File.join(policies_dir, "leftover-policy.json") }
372
+
373
+ let(:extra_policy_group_item) { File.join(policy_groups_dir, "leftover-policy-group.json") }
374
+
375
+ let(:conflicting_policyfile_lock) { File.join(export_dir, "Policyfile.lock.json") }
376
+
377
+ before do
378
+ FileUtils.mkdir_p(export_dir)
379
+ FileUtils.mkdir_p(cookbook_artifacts_dir)
380
+ FileUtils.mkdir_p(policies_dir)
381
+ FileUtils.mkdir_p(policy_groups_dir)
382
+ File.open(non_conflicting_file_in_export_dir, "wb+") { |f| f.print "some random cruft" }
383
+ File.open(file_in_cookbook_artifacts_dir, "wb+") { |f| f.print "some random cruft" }
384
+ File.open(extra_policy_item, "wb+") { |f| f.print "some random cruft" }
385
+ File.open(extra_policy_group_item, "wb+") { |f| f.print "some random cruft" }
386
+ File.open(conflicting_policyfile_lock, "wb+") { |f| f.print "some random cruft" }
387
+ end
388
+
389
+ it "raises a PolicyfileExportRepoError" do
390
+ message = "Export dir (#{export_dir}) not clean. Refusing to export. (Conflicting files: #{file_in_cookbook_artifacts_dir}, #{extra_policy_item}, #{extra_policy_group_item}, #{conflicting_policyfile_lock})"
391
+ expect { export_service.run }.to raise_error(ChefDK::ExportDirNotEmpty, message)
392
+ expect(File).to exist(non_conflicting_file_in_export_dir)
393
+ expect(File).to exist(file_in_cookbook_artifacts_dir)
394
+ expect(File).to exist(extra_policy_item)
395
+ expect(File).to exist(extra_policy_group_item)
396
+ end
397
+
398
+ context "and the force option is set" do
399
+
400
+ let(:force_export) { true }
401
+
402
+ it "clears the export dir and exports" do
403
+ export_service.run
404
+
405
+ expect(File).to_not exist(file_in_cookbook_artifacts_dir)
406
+ expect(File).to_not exist(extra_policy_item)
407
+ expect(File).to_not exist(extra_policy_group_item)
408
+
409
+ expect(File).to exist(non_conflicting_file_in_export_dir)
410
+
411
+ expect(File).to be_directory(File.join(export_dir, "cookbook_artifacts"))
412
+ expect(File).to be_directory(File.join(export_dir, "policies"))
413
+ expect(File).to be_directory(File.join(export_dir, "policy_groups"))
414
+ end
415
+
416
+ end
417
+
418
+ end # When the export dir has conflicting content
419
+
420
+ context "when archive mode is enabled" do
421
+
422
+ let(:archive) { true }
423
+
424
+ let(:expected_archive_path) do
425
+ File.join(export_dir, "install-example-7da81d2c7bb97f904637f97e7f8b487fa4bb1ed682edea7087743dec84c254ec.tgz")
426
+ end
427
+
428
+ it "exports the repo as a tgz archive" do
429
+ expect(File).to exist(expected_archive_path)
430
+ end
431
+
432
+ include_examples "successful_export" do
433
+
434
+ # explode the tarball so the assertions can find the files
435
+ before do
436
+ Mixlib::Archive.new(expected_archive_path).extract(export_dir)
437
+ end
438
+
439
+ end
440
+
441
+ context "when the target dir has a cookbooks or data_bags dir" do
442
+
443
+ let(:cookbooks_dir) { File.join(export_dir, "cookbooks") }
444
+
445
+ let(:file_in_cookbooks_dir) { File.join(cookbooks_dir, "some_random_cruft") }
446
+
447
+ let(:policyfiles_data_bag_dir) { File.join(export_dir, "data_bags", "policyfiles") }
448
+
449
+ let(:extra_policyfile_data_item) { File.join(policyfiles_data_bag_dir, "leftover-policy.json") }
450
+
451
+ before do
452
+ FileUtils.mkdir_p(export_dir)
453
+ FileUtils.mkdir_p(cookbooks_dir)
454
+ FileUtils.mkdir_p(policyfiles_data_bag_dir)
455
+ File.open(file_in_cookbooks_dir, "wb+") { |f| f.print "some random cruft" }
456
+ File.open(extra_policyfile_data_item, "wb+") { |f| f.print "some random cruft" }
457
+ end
458
+
459
+ it "exports successfully" do
460
+ expect { export_service.run }.to_not raise_error
461
+ expect(File).to exist(expected_archive_path)
462
+ end
463
+
464
+ end
465
+
466
+ end # when archive mode is enabled
467
+
468
+ end # copying the cookbooks to the export dir
469
+ end
470
+
471
+ end
472
+
473
+ end