chef-dk 0.11.2 → 0.12.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (325) hide show
  1. checksums.yaml +4 -4
  2. data/CONTRIBUTING.md +186 -190
  3. data/Gemfile +30 -26
  4. data/LICENSE +201 -201
  5. data/README.md +307 -307
  6. data/Rakefile +29 -27
  7. data/acceptance/.shared/kitchen_acceptance/.kitchen.digitalocean.yml +27 -0
  8. data/acceptance/.shared/kitchen_acceptance/.kitchen.ec2.yml +284 -0
  9. data/acceptance/.shared/kitchen_acceptance/.kitchen.vagrant.yml +52 -0
  10. data/acceptance/.shared/kitchen_acceptance/libraries/kitchen.rb +51 -0
  11. data/acceptance/.shared/kitchen_acceptance/metadata.rb +1 -0
  12. data/acceptance/Gemfile +11 -0
  13. data/acceptance/README.md +132 -0
  14. data/acceptance/trivial/.acceptance/acceptance-cookbook/.gitignore +2 -0
  15. data/acceptance/trivial/.acceptance/acceptance-cookbook/metadata.rb +2 -0
  16. data/acceptance/trivial/.acceptance/acceptance-cookbook/recipes/destroy.rb +1 -0
  17. data/acceptance/trivial/.acceptance/acceptance-cookbook/recipes/provision.rb +1 -0
  18. data/acceptance/trivial/.acceptance/acceptance-cookbook/recipes/verify.rb +1 -0
  19. data/acceptance/trivial/.kitchen.yml +7 -0
  20. data/acceptance/trivial/test/integration/chefdk-current-install/inspec/chef_client_spec.rb +5 -0
  21. data/bin/chef +25 -25
  22. data/chef-dk.gemspec +64 -64
  23. data/lib/chef-dk.rb +19 -19
  24. data/lib/chef-dk/authenticated_http.rb +40 -40
  25. data/lib/chef-dk/builtin_commands.rb +60 -60
  26. data/lib/chef-dk/chef_runner.rb +107 -107
  27. data/lib/chef-dk/cli.rb +200 -200
  28. data/lib/chef-dk/command/base.rb +79 -79
  29. data/lib/chef-dk/command/clean_policy_cookbooks.rb +116 -116
  30. data/lib/chef-dk/command/clean_policy_revisions.rb +113 -113
  31. data/lib/chef-dk/command/delete_policy.rb +122 -122
  32. data/lib/chef-dk/command/delete_policy_group.rb +122 -122
  33. data/lib/chef-dk/command/diff.rb +316 -316
  34. data/lib/chef-dk/command/env.rb +90 -90
  35. data/lib/chef-dk/command/exec.rb +45 -45
  36. data/lib/chef-dk/command/export.rb +157 -157
  37. data/lib/chef-dk/command/gem.rb +47 -47
  38. data/lib/chef-dk/command/generate.rb +120 -120
  39. data/lib/chef-dk/command/generator_commands.rb +80 -80
  40. data/lib/chef-dk/command/generator_commands/app.rb +107 -107
  41. data/lib/chef-dk/command/generator_commands/attribute.rb +37 -37
  42. data/lib/chef-dk/command/generator_commands/base.rb +148 -148
  43. data/lib/chef-dk/command/generator_commands/cookbook.rb +153 -153
  44. data/lib/chef-dk/command/generator_commands/cookbook_code_file.rb +100 -100
  45. data/lib/chef-dk/command/generator_commands/cookbook_file.rb +45 -45
  46. data/lib/chef-dk/command/generator_commands/generator_generator.rb +177 -177
  47. data/lib/chef-dk/command/generator_commands/lwrp.rb +36 -36
  48. data/lib/chef-dk/command/generator_commands/policyfile.rb +127 -127
  49. data/lib/chef-dk/command/generator_commands/recipe.rb +36 -36
  50. data/lib/chef-dk/command/generator_commands/repo.rb +125 -125
  51. data/lib/chef-dk/command/generator_commands/template.rb +46 -46
  52. data/lib/chef-dk/command/install.rb +121 -121
  53. data/lib/chef-dk/command/provision.rb +438 -438
  54. data/lib/chef-dk/command/push.rb +118 -118
  55. data/lib/chef-dk/command/push_archive.rb +126 -126
  56. data/lib/chef-dk/command/shell_init.rb +180 -180
  57. data/lib/chef-dk/command/show_policy.rb +165 -165
  58. data/lib/chef-dk/command/undelete.rb +155 -155
  59. data/lib/chef-dk/command/update.rb +148 -148
  60. data/lib/chef-dk/command/verify.rb +534 -534
  61. data/lib/chef-dk/commands_map.rb +115 -115
  62. data/lib/chef-dk/completions/bash.sh.erb +5 -5
  63. data/lib/chef-dk/completions/chef.fish.erb +10 -10
  64. data/lib/chef-dk/completions/zsh.zsh.erb +21 -21
  65. data/lib/chef-dk/component_test.rb +194 -173
  66. data/lib/chef-dk/configurable.rb +69 -69
  67. data/lib/chef-dk/cookbook_metadata.rb +45 -45
  68. data/lib/chef-dk/cookbook_omnifetch.rb +32 -32
  69. data/lib/chef-dk/cookbook_profiler/git.rb +151 -151
  70. data/lib/chef-dk/cookbook_profiler/identifiers.rb +72 -72
  71. data/lib/chef-dk/cookbook_profiler/null_scm.rb +32 -32
  72. data/lib/chef-dk/exceptions.rb +129 -129
  73. data/lib/chef-dk/generator.rb +163 -163
  74. data/lib/chef-dk/helpers.rb +159 -159
  75. data/lib/chef-dk/pager.rb +106 -106
  76. data/lib/chef-dk/policyfile/chef_repo_cookbook_source.rb +138 -138
  77. data/lib/chef-dk/policyfile/chef_server_cookbook_source.rb +54 -54
  78. data/lib/chef-dk/policyfile/community_cookbook_source.rb +97 -97
  79. data/lib/chef-dk/policyfile/comparison_base.rb +124 -124
  80. data/lib/chef-dk/policyfile/cookbook_location_specification.rb +154 -154
  81. data/lib/chef-dk/policyfile/cookbook_locks.rb +466 -466
  82. data/lib/chef-dk/policyfile/cookbook_sources.rb +22 -22
  83. data/lib/chef-dk/policyfile/delivery_supermarket_source.rb +90 -90
  84. data/lib/chef-dk/policyfile/differ.rb +266 -266
  85. data/lib/chef-dk/policyfile/dsl.rb +261 -261
  86. data/lib/chef-dk/policyfile/lister.rb +232 -232
  87. data/lib/chef-dk/policyfile/null_cookbook_source.rb +45 -45
  88. data/lib/chef-dk/policyfile/read_cookbook_for_compat_mode_upload.rb +124 -124
  89. data/lib/chef-dk/policyfile/reports/install.rb +70 -70
  90. data/lib/chef-dk/policyfile/reports/table_printer.rb +58 -58
  91. data/lib/chef-dk/policyfile/reports/upload.rb +70 -70
  92. data/lib/chef-dk/policyfile/solution_dependencies.rb +298 -298
  93. data/lib/chef-dk/policyfile/storage_config.rb +100 -100
  94. data/lib/chef-dk/policyfile/undo_record.rb +142 -142
  95. data/lib/chef-dk/policyfile/undo_stack.rb +130 -130
  96. data/lib/chef-dk/policyfile/uploader.rb +213 -213
  97. data/lib/chef-dk/policyfile_compiler.rb +419 -419
  98. data/lib/chef-dk/policyfile_lock.rb +552 -552
  99. data/lib/chef-dk/policyfile_services/clean_policies.rb +95 -95
  100. data/lib/chef-dk/policyfile_services/clean_policy_cookbooks.rb +125 -125
  101. data/lib/chef-dk/policyfile_services/export_repo.rb +421 -414
  102. data/lib/chef-dk/policyfile_services/install.rb +126 -126
  103. data/lib/chef-dk/policyfile_services/push.rb +114 -114
  104. data/lib/chef-dk/policyfile_services/push_archive.rb +204 -204
  105. data/lib/chef-dk/policyfile_services/rm_policy.rb +142 -142
  106. data/lib/chef-dk/policyfile_services/rm_policy_group.rb +86 -86
  107. data/lib/chef-dk/policyfile_services/show_policy.rb +237 -237
  108. data/lib/chef-dk/policyfile_services/undelete.rb +108 -108
  109. data/lib/chef-dk/policyfile_services/update_attributes.rb +104 -104
  110. data/lib/chef-dk/service_exception_inspectors.rb +25 -25
  111. data/lib/chef-dk/service_exception_inspectors/base.rb +40 -40
  112. data/lib/chef-dk/service_exception_inspectors/http.rb +121 -121
  113. data/lib/chef-dk/service_exceptions.rb +143 -143
  114. data/lib/chef-dk/shell_out.rb +36 -36
  115. data/lib/chef-dk/skeletons/code_generator/files/default/Berksfile +3 -3
  116. data/lib/chef-dk/skeletons/code_generator/files/default/chefignore +102 -102
  117. data/lib/chef-dk/skeletons/code_generator/files/default/cookbook_readmes/README-policy.md +9 -9
  118. data/lib/chef-dk/skeletons/code_generator/files/default/cookbook_readmes/README.md +54 -54
  119. data/lib/chef-dk/skeletons/code_generator/files/default/gitignore +16 -16
  120. data/lib/chef-dk/skeletons/code_generator/files/default/repo/README.md +28 -28
  121. data/lib/chef-dk/skeletons/code_generator/files/default/repo/cookbooks/example/README.md +27 -27
  122. data/lib/chef-dk/skeletons/code_generator/files/default/repo/cookbooks/example/attributes/default.rb +7 -7
  123. data/lib/chef-dk/skeletons/code_generator/files/default/repo/cookbooks/example/metadata.rb +3 -3
  124. data/lib/chef-dk/skeletons/code_generator/files/default/repo/cookbooks/example/recipes/default.rb +8 -8
  125. data/lib/chef-dk/skeletons/code_generator/files/default/repo/data_bags/README.md +58 -58
  126. data/lib/chef-dk/skeletons/code_generator/files/default/repo/data_bags/example/example_item.json +3 -3
  127. data/lib/chef-dk/skeletons/code_generator/files/default/repo/dot-chef-repo.txt +6 -6
  128. data/lib/chef-dk/skeletons/code_generator/files/default/repo/environments/README.md +9 -9
  129. data/lib/chef-dk/skeletons/code_generator/files/default/repo/environments/example.json +12 -12
  130. data/lib/chef-dk/skeletons/code_generator/files/default/repo/policies/README.md +24 -24
  131. data/lib/chef-dk/skeletons/code_generator/files/default/repo/roles/README.md +9 -9
  132. data/lib/chef-dk/skeletons/code_generator/files/default/repo/roles/example.json +12 -12
  133. data/lib/chef-dk/skeletons/code_generator/files/default/serverspec_spec_helper.rb +8 -8
  134. data/lib/chef-dk/skeletons/code_generator/files/default/spec_helper.rb +2 -2
  135. data/lib/chef-dk/skeletons/code_generator/files/default/spec_helper_policyfile.rb +2 -2
  136. data/lib/chef-dk/skeletons/code_generator/metadata.rb +8 -8
  137. data/lib/chef-dk/skeletons/code_generator/recipes/app.rb +97 -97
  138. data/lib/chef-dk/skeletons/code_generator/recipes/attribute.rb +12 -12
  139. data/lib/chef-dk/skeletons/code_generator/recipes/cookbook.rb +117 -117
  140. data/lib/chef-dk/skeletons/code_generator/recipes/cookbook_file.rb +24 -24
  141. data/lib/chef-dk/skeletons/code_generator/recipes/lwrp.rb +23 -23
  142. data/lib/chef-dk/skeletons/code_generator/recipes/policyfile.rb +8 -8
  143. data/lib/chef-dk/skeletons/code_generator/recipes/recipe.rb +27 -27
  144. data/lib/chef-dk/skeletons/code_generator/recipes/repo.rb +67 -67
  145. data/lib/chef-dk/skeletons/code_generator/recipes/template.rb +32 -32
  146. data/lib/chef-dk/skeletons/code_generator/templates/default/LICENSE.all_rights.erb +3 -3
  147. data/lib/chef-dk/skeletons/code_generator/templates/default/LICENSE.apache2.erb +201 -201
  148. data/lib/chef-dk/skeletons/code_generator/templates/default/LICENSE.gplv2.erb +339 -339
  149. data/lib/chef-dk/skeletons/code_generator/templates/default/LICENSE.gplv3.erb +674 -674
  150. data/lib/chef-dk/skeletons/code_generator/templates/default/LICENSE.mit.erb +21 -21
  151. data/lib/chef-dk/skeletons/code_generator/templates/default/Policyfile.rb.erb +25 -25
  152. data/lib/chef-dk/skeletons/code_generator/templates/default/README.md.erb +4 -4
  153. data/lib/chef-dk/skeletons/code_generator/templates/default/kitchen.yml.erb +21 -21
  154. data/lib/chef-dk/skeletons/code_generator/templates/default/kitchen_policyfile.yml.erb +32 -32
  155. data/lib/chef-dk/skeletons/code_generator/templates/default/metadata.rb.erb +7 -7
  156. data/lib/chef-dk/skeletons/code_generator/templates/default/recipe.rb.erb +5 -5
  157. data/lib/chef-dk/skeletons/code_generator/templates/default/recipe_spec.rb.erb +20 -20
  158. data/lib/chef-dk/skeletons/code_generator/templates/default/repo/gitignore.erb +11 -11
  159. data/lib/chef-dk/skeletons/code_generator/templates/default/serverspec_default_spec.rb.erb +9 -9
  160. data/lib/chef-dk/ui.rb +58 -58
  161. data/lib/chef-dk/version.rb +20 -20
  162. data/lib/kitchen/provisioner/policyfile_zero.rb +194 -194
  163. data/spec/shared/a_file_generator.rb +125 -125
  164. data/spec/shared/a_generated_file.rb +12 -12
  165. data/spec/shared/command_with_ui_object.rb +11 -11
  166. data/spec/shared/custom_generator_cookbook.rb +130 -130
  167. data/spec/shared/fixture_cookbook_checksums.rb +47 -47
  168. data/spec/shared/setup_git_cookbooks.rb +53 -53
  169. data/spec/spec_helper.rb +51 -50
  170. data/spec/test_helpers.rb +84 -84
  171. data/spec/unit/chef_runner_spec.rb +139 -139
  172. data/spec/unit/cli_spec.rb +357 -357
  173. data/spec/unit/command/base_spec.rb +169 -169
  174. data/spec/unit/command/clean_policy_cookbooks_spec.rb +181 -181
  175. data/spec/unit/command/clean_policy_revisions_spec.rb +181 -181
  176. data/spec/unit/command/delete_policy_group_spec.rb +207 -207
  177. data/spec/unit/command/delete_policy_spec.rb +207 -207
  178. data/spec/unit/command/diff_spec.rb +312 -312
  179. data/spec/unit/command/env_spec.rb +52 -52
  180. data/spec/unit/command/exec_spec.rb +179 -179
  181. data/spec/unit/command/export_spec.rb +200 -200
  182. data/spec/unit/command/generate_spec.rb +142 -142
  183. data/spec/unit/command/generator_commands/app_spec.rb +169 -169
  184. data/spec/unit/command/generator_commands/attribute_spec.rb +32 -32
  185. data/spec/unit/command/generator_commands/base_spec.rb +136 -136
  186. data/spec/unit/command/generator_commands/cookbook_file_spec.rb +32 -32
  187. data/spec/unit/command/generator_commands/cookbook_spec.rb +450 -450
  188. data/spec/unit/command/generator_commands/generator_generator_spec.rb +229 -229
  189. data/spec/unit/command/generator_commands/lwrp_spec.rb +32 -32
  190. data/spec/unit/command/generator_commands/policyfile_spec.rb +225 -225
  191. data/spec/unit/command/generator_commands/recipe_spec.rb +34 -34
  192. data/spec/unit/command/generator_commands/repo_spec.rb +367 -367
  193. data/spec/unit/command/generator_commands/template_spec.rb +32 -32
  194. data/spec/unit/command/install_spec.rb +179 -179
  195. data/spec/unit/command/provision_spec.rb +592 -592
  196. data/spec/unit/command/push_archive_spec.rb +153 -153
  197. data/spec/unit/command/push_spec.rb +199 -199
  198. data/spec/unit/command/shell_init_spec.rb +329 -329
  199. data/spec/unit/command/show_policy_spec.rb +235 -235
  200. data/spec/unit/command/undelete_spec.rb +246 -246
  201. data/spec/unit/command/update_spec.rb +275 -275
  202. data/spec/unit/command/verify_spec.rb +325 -325
  203. data/spec/unit/commands_map_spec.rb +57 -57
  204. data/spec/unit/component_test_spec.rb +126 -126
  205. data/spec/unit/configurable_spec.rb +41 -41
  206. data/spec/unit/cookbook_metadata_spec.rb +98 -98
  207. data/spec/unit/cookbook_profiler/git_spec.rb +176 -176
  208. data/spec/unit/cookbook_profiler/identifiers_spec.rb +83 -83
  209. data/spec/unit/fixtures/chef-runner-cookbooks/test_cookbook/recipes/recipe_one.rb +9 -9
  210. data/spec/unit/fixtures/chef-runner-cookbooks/test_cookbook/recipes/recipe_two.rb +9 -9
  211. data/spec/unit/fixtures/command/cli_test_command.rb +26 -26
  212. data/spec/unit/fixtures/command/explicit_path_example.rb +7 -7
  213. data/spec/unit/fixtures/configurable/test_config_loader.rb +5 -5
  214. data/spec/unit/fixtures/configurable/test_configurable.rb +10 -10
  215. data/spec/unit/fixtures/cookbook_cache/baz-f59ee7a5bca6a4e606b67f7f856b768d847c39bb/.kitchen.yml +16 -16
  216. data/spec/unit/fixtures/cookbook_cache/baz-f59ee7a5bca6a4e606b67f7f856b768d847c39bb/Berksfile +3 -3
  217. data/spec/unit/fixtures/cookbook_cache/baz-f59ee7a5bca6a4e606b67f7f856b768d847c39bb/README.md +4 -4
  218. data/spec/unit/fixtures/cookbook_cache/baz-f59ee7a5bca6a4e606b67f7f856b768d847c39bb/chefignore +96 -96
  219. data/spec/unit/fixtures/cookbook_cache/baz-f59ee7a5bca6a4e606b67f7f856b768d847c39bb/metadata.rb +8 -8
  220. data/spec/unit/fixtures/cookbook_cache/baz-f59ee7a5bca6a4e606b67f7f856b768d847c39bb/recipes/default.rb +8 -8
  221. data/spec/unit/fixtures/cookbook_cache/dep_of_bar-1.2.3/.kitchen.yml +16 -16
  222. data/spec/unit/fixtures/cookbook_cache/dep_of_bar-1.2.3/Berksfile +3 -3
  223. data/spec/unit/fixtures/cookbook_cache/dep_of_bar-1.2.3/README.md +4 -4
  224. data/spec/unit/fixtures/cookbook_cache/dep_of_bar-1.2.3/chefignore +96 -96
  225. data/spec/unit/fixtures/cookbook_cache/dep_of_bar-1.2.3/metadata.rb +8 -8
  226. data/spec/unit/fixtures/cookbook_cache/dep_of_bar-1.2.3/recipes/default.rb +8 -8
  227. data/spec/unit/fixtures/cookbook_cache/foo-1.0.0/.kitchen.yml +16 -16
  228. data/spec/unit/fixtures/cookbook_cache/foo-1.0.0/Berksfile +3 -3
  229. data/spec/unit/fixtures/cookbook_cache/foo-1.0.0/README.md +4 -4
  230. data/spec/unit/fixtures/cookbook_cache/foo-1.0.0/chefignore +96 -96
  231. data/spec/unit/fixtures/cookbook_cache/foo-1.0.0/metadata.rb +8 -8
  232. data/spec/unit/fixtures/cookbook_cache/foo-1.0.0/recipes/default.rb +8 -8
  233. data/spec/unit/fixtures/cookbooks_api/pruned_small_universe.json +1321 -1321
  234. data/spec/unit/fixtures/cookbooks_api/small_universe.json +2987 -2987
  235. data/spec/unit/fixtures/cookbooks_api/universe.json +1 -1
  236. data/spec/unit/fixtures/cookbooks_api/update_fixtures.rb +36 -36
  237. data/spec/unit/fixtures/dev_cookbooks/README.md +16 -16
  238. data/spec/unit/fixtures/eg_omnibus_dir/valid/embedded/apps/berkshelf/integration_test +2 -2
  239. data/spec/unit/fixtures/eg_omnibus_dir/valid/embedded/apps/berkshelf/verify_me +5 -5
  240. data/spec/unit/fixtures/eg_omnibus_dir/valid/embedded/apps/chef/verify_me +3 -3
  241. data/spec/unit/fixtures/eg_omnibus_dir/valid/embedded/apps/test-kitchen/verify_me +2 -2
  242. data/spec/unit/fixtures/example_cookbook/.gitignore +17 -17
  243. data/spec/unit/fixtures/example_cookbook/.kitchen.yml +16 -16
  244. data/spec/unit/fixtures/example_cookbook/Berksfile +3 -3
  245. data/spec/unit/fixtures/example_cookbook/README.md +4 -4
  246. data/spec/unit/fixtures/example_cookbook/chefignore +96 -96
  247. data/spec/unit/fixtures/example_cookbook/metadata.rb +8 -8
  248. data/spec/unit/fixtures/example_cookbook/recipes/default.rb +8 -8
  249. data/spec/unit/fixtures/example_cookbook_metadata_json_only/.gitignore +17 -17
  250. data/spec/unit/fixtures/example_cookbook_metadata_json_only/.kitchen.yml +16 -16
  251. data/spec/unit/fixtures/example_cookbook_metadata_json_only/Berksfile +3 -3
  252. data/spec/unit/fixtures/example_cookbook_metadata_json_only/README.md +4 -4
  253. data/spec/unit/fixtures/example_cookbook_metadata_json_only/chefignore +96 -96
  254. data/spec/unit/fixtures/example_cookbook_metadata_json_only/metadata.json +5 -5
  255. data/spec/unit/fixtures/example_cookbook_metadata_json_only/recipes/default.rb +8 -8
  256. data/spec/unit/fixtures/example_cookbook_no_metadata/.gitignore +17 -17
  257. data/spec/unit/fixtures/example_cookbook_no_metadata/.kitchen.yml +16 -16
  258. data/spec/unit/fixtures/example_cookbook_no_metadata/Berksfile +3 -3
  259. data/spec/unit/fixtures/example_cookbook_no_metadata/README.md +4 -4
  260. data/spec/unit/fixtures/example_cookbook_no_metadata/chefignore +96 -96
  261. data/spec/unit/fixtures/example_cookbook_no_metadata/recipes/default.rb +8 -8
  262. data/spec/unit/fixtures/local_path_cookbooks/another-local-cookbook/README.md +4 -4
  263. data/spec/unit/fixtures/local_path_cookbooks/another-local-cookbook/chefignore +96 -96
  264. data/spec/unit/fixtures/local_path_cookbooks/another-local-cookbook/metadata.rb +8 -8
  265. data/spec/unit/fixtures/local_path_cookbooks/another-local-cookbook/recipes/default.rb +8 -8
  266. data/spec/unit/fixtures/local_path_cookbooks/cookbook-with-a-dep/Berksfile +3 -3
  267. data/spec/unit/fixtures/local_path_cookbooks/cookbook-with-a-dep/README.md +4 -4
  268. data/spec/unit/fixtures/local_path_cookbooks/cookbook-with-a-dep/chefignore +96 -96
  269. data/spec/unit/fixtures/local_path_cookbooks/cookbook-with-a-dep/metadata.rb +9 -9
  270. data/spec/unit/fixtures/local_path_cookbooks/cookbook-with-a-dep/recipes/default.rb +8 -8
  271. data/spec/unit/fixtures/local_path_cookbooks/local-cookbook/.kitchen.yml +16 -16
  272. data/spec/unit/fixtures/local_path_cookbooks/local-cookbook/Berksfile +3 -3
  273. data/spec/unit/fixtures/local_path_cookbooks/local-cookbook/README.md +4 -4
  274. data/spec/unit/fixtures/local_path_cookbooks/local-cookbook/chefignore +96 -96
  275. data/spec/unit/fixtures/local_path_cookbooks/local-cookbook/extra/extra_file.txt +0 -0
  276. data/spec/unit/fixtures/local_path_cookbooks/local-cookbook/metadata.rb +8 -8
  277. data/spec/unit/fixtures/local_path_cookbooks/local-cookbook/recipes/default.rb +8 -8
  278. data/spec/unit/fixtures/local_path_cookbooks/metadata-missing/README.md +2 -2
  279. data/spec/unit/fixtures/local_path_cookbooks/noignore-f59ee7a5bca6a4e606b67f7f856b768d847c39bb/.kitchen.yml +16 -16
  280. data/spec/unit/fixtures/local_path_cookbooks/noignore-f59ee7a5bca6a4e606b67f7f856b768d847c39bb/README.md +4 -4
  281. data/spec/unit/fixtures/local_path_cookbooks/noignore-f59ee7a5bca6a4e606b67f7f856b768d847c39bb/metadata.rb +8 -8
  282. data/spec/unit/fixtures/local_path_cookbooks/noignore-f59ee7a5bca6a4e606b67f7f856b768d847c39bb/recipes/default.rb +8 -8
  283. data/spec/unit/generator_spec.rb +120 -120
  284. data/spec/unit/helpers_spec.rb +92 -92
  285. data/spec/unit/pager_spec.rb +119 -119
  286. data/spec/unit/policyfile/chef_repo_cookbook_source_spec.rb +93 -93
  287. data/spec/unit/policyfile/chef_server_cookbook_source_spec.rb +34 -34
  288. data/spec/unit/policyfile/community_cookbook_source_spec.rb +84 -84
  289. data/spec/unit/policyfile/comparison_base_spec.rb +343 -343
  290. data/spec/unit/policyfile/cookbook_location_specification_spec.rb +277 -277
  291. data/spec/unit/policyfile/cookbook_locks_spec.rb +529 -529
  292. data/spec/unit/policyfile/delivery_supermarket_source_spec.rb +130 -130
  293. data/spec/unit/policyfile/differ_spec.rb +687 -687
  294. data/spec/unit/policyfile/lister_spec.rb +272 -272
  295. data/spec/unit/policyfile/null_cookbook_source_spec.rb +35 -35
  296. data/spec/unit/policyfile/read_cookbook_for_compat_mode_upload_spec.rb +92 -92
  297. data/spec/unit/policyfile/reports/install_spec.rb +115 -115
  298. data/spec/unit/policyfile/reports/upload_spec.rb +96 -96
  299. data/spec/unit/policyfile/solution_dependencies_spec.rb +145 -145
  300. data/spec/unit/policyfile/storage_config_spec.rb +172 -172
  301. data/spec/unit/policyfile/undo_record_spec.rb +260 -260
  302. data/spec/unit/policyfile/undo_stack_spec.rb +266 -266
  303. data/spec/unit/policyfile/uploader_spec.rb +410 -410
  304. data/spec/unit/policyfile_demands_spec.rb +1203 -1203
  305. data/spec/unit/policyfile_evaluation_spec.rb +642 -642
  306. data/spec/unit/policyfile_lock_build_spec.rb +1056 -1056
  307. data/spec/unit/policyfile_lock_install_spec.rb +138 -138
  308. data/spec/unit/policyfile_lock_serialization_spec.rb +425 -425
  309. data/spec/unit/policyfile_lock_validation_spec.rb +611 -611
  310. data/spec/unit/policyfile_services/clean_policies_spec.rb +236 -236
  311. data/spec/unit/policyfile_services/clean_policy_cookbooks_spec.rb +275 -275
  312. data/spec/unit/policyfile_services/export_repo_spec.rb +481 -481
  313. data/spec/unit/policyfile_services/install_spec.rb +211 -211
  314. data/spec/unit/policyfile_services/push_archive_spec.rb +378 -378
  315. data/spec/unit/policyfile_services/push_spec.rb +233 -233
  316. data/spec/unit/policyfile_services/rm_policy_group_spec.rb +241 -241
  317. data/spec/unit/policyfile_services/rm_policy_spec.rb +266 -266
  318. data/spec/unit/policyfile_services/show_policy_spec.rb +889 -889
  319. data/spec/unit/policyfile_services/undelete_spec.rb +304 -304
  320. data/spec/unit/policyfile_services/update_attributes_spec.rb +217 -217
  321. data/spec/unit/service_exception_inspectors/base_spec.rb +43 -43
  322. data/spec/unit/service_exception_inspectors/http_spec.rb +140 -140
  323. data/spec/unit/shell_out_spec.rb +34 -34
  324. data/warning.txt +9 -9
  325. metadata +24 -19
@@ -1,95 +1,95 @@
1
- #
2
- # Copyright:: Copyright (c) 2015 Chef Software Inc.
3
- # License:: Apache License, Version 2.0
4
- #
5
- # Licensed under the Apache License, Version 2.0 (the "License");
6
- # you may not use this file except in compliance with the License.
7
- # You may obtain a copy of the License at
8
- #
9
- # http://www.apache.org/licenses/LICENSE-2.0
10
- #
11
- # Unless required by applicable law or agreed to in writing, software
12
- # distributed under the License is distributed on an "AS IS" BASIS,
13
- # WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
14
- # See the License for the specific language governing permissions and
15
- # limitations under the License.
16
- #
17
-
18
- require 'chef-dk/exceptions'
19
- require 'chef-dk/service_exceptions'
20
- require 'chef-dk/policyfile/lister'
21
-
22
- module ChefDK
23
- module PolicyfileServices
24
- class CleanPolicies
25
-
26
- Orphan = Struct.new(:policy_name, :revision_id)
27
-
28
- attr_reader :chef_config
29
- attr_reader :ui
30
-
31
- def initialize(config: nil, ui: nil)
32
- @chef_config = config
33
- @ui = ui
34
- end
35
-
36
- def run
37
- revisions_to_remove = orphaned_policies
38
-
39
- if revisions_to_remove.empty?
40
- ui.err("No policy revisions deleted")
41
- return true
42
- end
43
-
44
- results = revisions_to_remove.map do |policy|
45
- [ remove_policy(policy), policy ]
46
- end
47
-
48
- failures = results.select { |result, _policy| result.kind_of?(Exception) }
49
-
50
- unless failures.empty?
51
- details = failures.map do |result, policy|
52
- "- #{policy.policy_name} (#{policy.revision_id}): #{result.class} #{result}"
53
- end
54
-
55
- message = "Failed to delete some policy revisions:\n" + details.join("\n") + "\n"
56
-
57
- raise PolicyfileCleanError.new(message, MultipleErrors.new("multiple errors"))
58
- end
59
-
60
- true
61
- end
62
-
63
- def orphaned_policies
64
- policy_lister.policies_by_name.keys.inject([]) do |orphans, policy_name|
65
- orphans + policy_lister.orphaned_revisions(policy_name).map do |revision_id|
66
- Orphan.new(policy_name, revision_id)
67
- end
68
- end
69
- rescue => e
70
- raise PolicyfileCleanError.new("Failed to list policies for cleaning.", e)
71
- end
72
-
73
- def policy_lister
74
- @policy_lister ||= Policyfile::Lister.new(config: chef_config)
75
- end
76
-
77
- def http_client
78
- @http_client ||= ChefDK::AuthenticatedHTTP.new(chef_config.chef_server_url,
79
- signing_key_filename: chef_config.client_key,
80
- client_name: chef_config.node_name)
81
- end
82
-
83
- private
84
-
85
- def remove_policy(policy)
86
- ui.msg("DELETE #{policy.policy_name} #{policy.revision_id}")
87
- http_client.delete("/policies/#{policy.policy_name}/revisions/#{policy.revision_id}")
88
- :ok
89
- rescue => e
90
- e
91
- end
92
-
93
- end
94
- end
95
- end
1
+ #
2
+ # Copyright:: Copyright (c) 2015 Chef Software Inc.
3
+ # License:: Apache License, Version 2.0
4
+ #
5
+ # Licensed under the Apache License, Version 2.0 (the "License");
6
+ # you may not use this file except in compliance with the License.
7
+ # You may obtain a copy of the License at
8
+ #
9
+ # http://www.apache.org/licenses/LICENSE-2.0
10
+ #
11
+ # Unless required by applicable law or agreed to in writing, software
12
+ # distributed under the License is distributed on an "AS IS" BASIS,
13
+ # WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
14
+ # See the License for the specific language governing permissions and
15
+ # limitations under the License.
16
+ #
17
+
18
+ require 'chef-dk/exceptions'
19
+ require 'chef-dk/service_exceptions'
20
+ require 'chef-dk/policyfile/lister'
21
+
22
+ module ChefDK
23
+ module PolicyfileServices
24
+ class CleanPolicies
25
+
26
+ Orphan = Struct.new(:policy_name, :revision_id)
27
+
28
+ attr_reader :chef_config
29
+ attr_reader :ui
30
+
31
+ def initialize(config: nil, ui: nil)
32
+ @chef_config = config
33
+ @ui = ui
34
+ end
35
+
36
+ def run
37
+ revisions_to_remove = orphaned_policies
38
+
39
+ if revisions_to_remove.empty?
40
+ ui.err("No policy revisions deleted")
41
+ return true
42
+ end
43
+
44
+ results = revisions_to_remove.map do |policy|
45
+ [ remove_policy(policy), policy ]
46
+ end
47
+
48
+ failures = results.select { |result, _policy| result.kind_of?(Exception) }
49
+
50
+ unless failures.empty?
51
+ details = failures.map do |result, policy|
52
+ "- #{policy.policy_name} (#{policy.revision_id}): #{result.class} #{result}"
53
+ end
54
+
55
+ message = "Failed to delete some policy revisions:\n" + details.join("\n") + "\n"
56
+
57
+ raise PolicyfileCleanError.new(message, MultipleErrors.new("multiple errors"))
58
+ end
59
+
60
+ true
61
+ end
62
+
63
+ def orphaned_policies
64
+ policy_lister.policies_by_name.keys.inject([]) do |orphans, policy_name|
65
+ orphans + policy_lister.orphaned_revisions(policy_name).map do |revision_id|
66
+ Orphan.new(policy_name, revision_id)
67
+ end
68
+ end
69
+ rescue => e
70
+ raise PolicyfileCleanError.new("Failed to list policies for cleaning.", e)
71
+ end
72
+
73
+ def policy_lister
74
+ @policy_lister ||= Policyfile::Lister.new(config: chef_config)
75
+ end
76
+
77
+ def http_client
78
+ @http_client ||= ChefDK::AuthenticatedHTTP.new(chef_config.chef_server_url,
79
+ signing_key_filename: chef_config.client_key,
80
+ client_name: chef_config.node_name)
81
+ end
82
+
83
+ private
84
+
85
+ def remove_policy(policy)
86
+ ui.msg("DELETE #{policy.policy_name} #{policy.revision_id}")
87
+ http_client.delete("/policies/#{policy.policy_name}/revisions/#{policy.revision_id}")
88
+ :ok
89
+ rescue => e
90
+ e
91
+ end
92
+
93
+ end
94
+ end
95
+ end
@@ -1,125 +1,125 @@
1
- #
2
- # Copyright:: Copyright (c) 2015 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 'set'
19
-
20
- require 'chef-dk/authenticated_http'
21
- require 'chef-dk/service_exceptions'
22
-
23
- module ChefDK
24
- module PolicyfileServices
25
-
26
- class CleanPolicyCookbooks
27
-
28
- attr_reader :chef_config
29
-
30
- attr_reader :ui
31
-
32
- def initialize(config: nil, ui: nil)
33
- @chef_config = config
34
- @ui = ui
35
-
36
- @all_cookbooks = nil
37
- @active_cookbooks = nil
38
- @all_policies = nil
39
- end
40
-
41
- def run
42
- gc_cookbooks
43
- rescue => e
44
- raise PolicyCookbookCleanError.new("Failed to cleanup policy cookbooks", e)
45
- end
46
-
47
- def gc_cookbooks
48
- cookbooks = cookbooks_to_clean
49
-
50
- if cookbooks.empty?
51
- ui.msg("No cookbooks deleted.")
52
- end
53
-
54
- cookbooks.each do |name, identifiers|
55
- identifiers.each do |identifier|
56
- http_client.delete("/cookbook_artifacts/#{name}/#{identifier}")
57
- ui.msg("DELETE #{name} #{identifier}")
58
- end
59
- end
60
- end
61
-
62
-
63
- def all_cookbooks
64
- cookbook_list = http_client.get("/cookbook_artifacts")
65
- cookbook_list.inject({}) do |cb_map, (name, cb_info)|
66
- cb_map[name] = cb_info["versions"].map { |v| v["identifier"] }
67
- cb_map
68
- end
69
- end
70
-
71
- def active_cookbooks
72
- policy_revisions_by_name.inject({}) do |cb_map, (policy_name, revision_ids)|
73
- revision_ids.each do |revision_id|
74
- cookbook_revisions_in_policy(policy_name, revision_id).each do |cb_name, identifier|
75
- cb_map[cb_name] ||= Set.new
76
- cb_map[cb_name] << identifier
77
- end
78
- end
79
- cb_map
80
- end
81
- end
82
-
83
- def cookbooks_to_clean
84
- active_cbs = active_cookbooks
85
-
86
- all_cookbooks.inject({}) do |cb_map, (cb_name, revisions)|
87
- active_revs = active_cbs[cb_name] || Set.new
88
- inactive_revs = Set.new(revisions) - active_revs
89
- cb_map[cb_name] = inactive_revs unless inactive_revs.empty?
90
-
91
- cb_map
92
- end
93
- end
94
-
95
- # @api private
96
- def policy_revisions_by_name
97
- policies_list = http_client.get("/policies")
98
- policies_list.inject({}) do |policies_map, (name, policy_info)|
99
- policies_map[name] = policy_info["revisions"].keys
100
- policies_map
101
- end
102
- end
103
-
104
- # @api private
105
- def cookbook_revisions_in_policy(name, revision_id)
106
- policy_revision_data = http_client.get("/policies/#{name}/revisions/#{revision_id}")
107
-
108
- policy_revision_data["cookbook_locks"].inject({}) do |cb_map, (cb_name, lock_info)|
109
- cb_map[cb_name] = lock_info["identifier"]
110
- cb_map
111
- end
112
- end
113
-
114
- # @api private
115
- # An instance of ChefDK::AuthenticatedHTTP configured with the user's
116
- # server URL and credentials.
117
- def http_client
118
- @http_client ||= ChefDK::AuthenticatedHTTP.new(chef_config.chef_server_url,
119
- signing_key_filename: chef_config.client_key,
120
- client_name: chef_config.node_name)
121
- end
122
- end
123
- end
124
- end
125
-
1
+ #
2
+ # Copyright:: Copyright (c) 2015 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 'set'
19
+
20
+ require 'chef-dk/authenticated_http'
21
+ require 'chef-dk/service_exceptions'
22
+
23
+ module ChefDK
24
+ module PolicyfileServices
25
+
26
+ class CleanPolicyCookbooks
27
+
28
+ attr_reader :chef_config
29
+
30
+ attr_reader :ui
31
+
32
+ def initialize(config: nil, ui: nil)
33
+ @chef_config = config
34
+ @ui = ui
35
+
36
+ @all_cookbooks = nil
37
+ @active_cookbooks = nil
38
+ @all_policies = nil
39
+ end
40
+
41
+ def run
42
+ gc_cookbooks
43
+ rescue => e
44
+ raise PolicyCookbookCleanError.new("Failed to cleanup policy cookbooks", e)
45
+ end
46
+
47
+ def gc_cookbooks
48
+ cookbooks = cookbooks_to_clean
49
+
50
+ if cookbooks.empty?
51
+ ui.msg("No cookbooks deleted.")
52
+ end
53
+
54
+ cookbooks.each do |name, identifiers|
55
+ identifiers.each do |identifier|
56
+ http_client.delete("/cookbook_artifacts/#{name}/#{identifier}")
57
+ ui.msg("DELETE #{name} #{identifier}")
58
+ end
59
+ end
60
+ end
61
+
62
+
63
+ def all_cookbooks
64
+ cookbook_list = http_client.get("/cookbook_artifacts")
65
+ cookbook_list.inject({}) do |cb_map, (name, cb_info)|
66
+ cb_map[name] = cb_info["versions"].map { |v| v["identifier"] }
67
+ cb_map
68
+ end
69
+ end
70
+
71
+ def active_cookbooks
72
+ policy_revisions_by_name.inject({}) do |cb_map, (policy_name, revision_ids)|
73
+ revision_ids.each do |revision_id|
74
+ cookbook_revisions_in_policy(policy_name, revision_id).each do |cb_name, identifier|
75
+ cb_map[cb_name] ||= Set.new
76
+ cb_map[cb_name] << identifier
77
+ end
78
+ end
79
+ cb_map
80
+ end
81
+ end
82
+
83
+ def cookbooks_to_clean
84
+ active_cbs = active_cookbooks
85
+
86
+ all_cookbooks.inject({}) do |cb_map, (cb_name, revisions)|
87
+ active_revs = active_cbs[cb_name] || Set.new
88
+ inactive_revs = Set.new(revisions) - active_revs
89
+ cb_map[cb_name] = inactive_revs unless inactive_revs.empty?
90
+
91
+ cb_map
92
+ end
93
+ end
94
+
95
+ # @api private
96
+ def policy_revisions_by_name
97
+ policies_list = http_client.get("/policies")
98
+ policies_list.inject({}) do |policies_map, (name, policy_info)|
99
+ policies_map[name] = policy_info["revisions"].keys
100
+ policies_map
101
+ end
102
+ end
103
+
104
+ # @api private
105
+ def cookbook_revisions_in_policy(name, revision_id)
106
+ policy_revision_data = http_client.get("/policies/#{name}/revisions/#{revision_id}")
107
+
108
+ policy_revision_data["cookbook_locks"].inject({}) do |cb_map, (cb_name, lock_info)|
109
+ cb_map[cb_name] = lock_info["identifier"]
110
+ cb_map
111
+ end
112
+ end
113
+
114
+ # @api private
115
+ # An instance of ChefDK::AuthenticatedHTTP configured with the user's
116
+ # server URL and credentials.
117
+ def http_client
118
+ @http_client ||= ChefDK::AuthenticatedHTTP.new(chef_config.chef_server_url,
119
+ signing_key_filename: chef_config.client_key,
120
+ client_name: chef_config.node_name)
121
+ end
122
+ end
123
+ end
124
+ end
125
+
@@ -1,414 +1,421 @@
1
- #
2
- # Copyright:: Copyright (c) 2014 Chef Software Inc.
3
- # License:: Apache License, Version 2.0
4
- #
5
- # Licensed under the Apache License, Version 2.0 (the "License");
6
- # you may not use this file except in compliance with the License.
7
- # You may obtain a copy of the License at
8
- #
9
- # http://www.apache.org/licenses/LICENSE-2.0
10
- #
11
- # Unless required by applicable law or agreed to in writing, software
12
- # distributed under the License is distributed on an "AS IS" BASIS,
13
- # WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
14
- # See the License for the specific language governing permissions and
15
- # limitations under the License.
16
- #
17
-
18
- require 'fileutils'
19
- require 'tmpdir'
20
- require 'zlib'
21
-
22
- require 'archive/tar/minitar'
23
-
24
- require 'chef/cookbook/chefignore'
25
-
26
- require 'chef-dk/service_exceptions'
27
- require 'chef-dk/policyfile_lock'
28
- require 'chef-dk/policyfile/storage_config'
29
-
30
- module ChefDK
31
- module PolicyfileServices
32
-
33
- class ExportRepo
34
-
35
- # Policy groups provide namespaces for policies so that a Chef Server can
36
- # have multiple active iterations of a policy at once, but we don't need
37
- # this when serving a single exported policy via Chef Zero, so hardcode
38
- # it to a "well known" value:
39
- POLICY_GROUP = 'local'.freeze
40
-
41
- include Policyfile::StorageConfigDelegation
42
-
43
- attr_reader :storage_config
44
- attr_reader :root_dir
45
- attr_reader :export_dir
46
-
47
- def initialize(policyfile: nil, export_dir: nil, root_dir: nil, archive: false, force: false)
48
- @root_dir = root_dir
49
- @export_dir = File.expand_path(export_dir)
50
- @archive = archive
51
- @force_export = force
52
-
53
- @policy_data = nil
54
- @policyfile_lock = nil
55
-
56
- policyfile_rel_path = policyfile || "Policyfile.rb"
57
- policyfile_full_path = File.expand_path(policyfile_rel_path, root_dir)
58
- @storage_config = Policyfile::StorageConfig.new.use_policyfile(policyfile_full_path)
59
-
60
- @staging_dir = nil
61
- end
62
-
63
- def archive?
64
- @archive
65
- end
66
-
67
- def policy_name
68
- policyfile_lock.name
69
- end
70
-
71
- def run
72
- assert_lockfile_exists!
73
- assert_export_dir_clean!
74
-
75
- validate_lockfile
76
- write_updated_lockfile
77
- export
78
- end
79
-
80
- def policy_data
81
- @policy_data ||= FFI_Yajl::Parser.parse(IO.read(policyfile_lock_expanded_path))
82
- rescue => error
83
- raise PolicyfileExportRepoError.new("Error reading lockfile #{policyfile_lock_expanded_path}", error)
84
- end
85
-
86
- def policyfile_lock
87
- @policyfile_lock || validate_lockfile
88
- end
89
-
90
- def archive_file_location
91
- return nil unless archive?
92
- filename = "#{policyfile_lock.name}-#{policyfile_lock.revision_id}.tgz"
93
- File.join(export_dir, filename)
94
- end
95
-
96
- def export
97
- with_staging_dir do
98
- create_repo_structure
99
- copy_cookbooks
100
- create_policyfile_repo_item
101
- create_policy_group_repo_item
102
- copy_policyfile_lock
103
- create_client_rb
104
- create_readme_md
105
- if archive?
106
- create_archive
107
- else
108
- mv_staged_repo
109
- end
110
- end
111
- rescue => error
112
- msg = "Failed to export policy (in #{policyfile_filename}) to #{export_dir}"
113
- raise PolicyfileExportRepoError.new(msg, error)
114
- end
115
-
116
- private
117
-
118
- def with_staging_dir
119
- p = Process.pid
120
- t = Time.new.utc.strftime("%Y%m%d%H%M%S")
121
- Dir.mktmpdir("chefdk-export-#{p}-#{t}") do |d|
122
- begin
123
- @staging_dir = d
124
- yield
125
- ensure
126
- @staging_dir = nil
127
- end
128
- end
129
- end
130
-
131
- def create_archive
132
- Zlib::GzipWriter.open(archive_file_location) do |gz_file|
133
- Dir.chdir(staging_dir) do
134
- Archive::Tar::Minitar.pack(".", gz_file)
135
- end
136
- end
137
- end
138
-
139
- def staging_dir
140
- @staging_dir
141
- end
142
-
143
- def create_repo_structure
144
- FileUtils.mkdir_p(export_dir)
145
- FileUtils.mkdir_p(dot_chef_staging_dir)
146
- FileUtils.mkdir_p(cookbook_artifacts_staging_dir)
147
- FileUtils.mkdir_p(policies_staging_dir)
148
- FileUtils.mkdir_p(policy_groups_staging_dir)
149
- end
150
-
151
- def copy_cookbooks
152
- policyfile_lock.cookbook_locks.each do |name, lock|
153
- copy_cookbook(lock)
154
- end
155
- end
156
-
157
- def copy_cookbook(lock)
158
- dirname = "#{lock.name}-#{lock.identifier}"
159
- export_path = File.join(staging_dir, "cookbook_artifacts", dirname)
160
- metadata_rb_path = File.join(export_path, "metadata.rb")
161
- FileUtils.mkdir(export_path) if not File.directory?(export_path)
162
- copy_unignored_cookbook_files(lock, export_path)
163
- FileUtils.rm_f(metadata_rb_path)
164
- metadata = lock.cookbook_version.metadata
165
-
166
- metadata_json_path = File.join(export_path, "metadata.json")
167
-
168
- File.open(metadata_json_path, "wb+") do |f|
169
- f.print(FFI_Yajl::Encoder.encode(metadata.to_hash, pretty: true ))
170
- end
171
- end
172
-
173
- def copy_unignored_cookbook_files(lock, export_path)
174
- cookbook_files_to_copy(lock.cookbook_path).each do |rel_path|
175
- full_source_path = File.join(lock.cookbook_path, rel_path)
176
- full_dest_path = File.join(export_path, rel_path)
177
- dest_dirname = File.dirname(full_dest_path)
178
- FileUtils.mkdir_p(dest_dirname) unless File.directory?(dest_dirname)
179
- FileUtils.cp(full_source_path, full_dest_path)
180
- end
181
- end
182
-
183
- def cookbook_files_to_copy(cookbook_path)
184
- cookbook_loader_for(cookbook_path).cookbook_version.manifest_records_by_path.keys
185
- end
186
-
187
- def cookbook_loader_for(cookbook_path)
188
- loader = Chef::Cookbook::CookbookVersionLoader.new(cookbook_path, chefignore_for(cookbook_path))
189
- loader.load!
190
- loader
191
- end
192
-
193
- def chefignore_for(cookbook_path)
194
- Chef::Cookbook::Chefignore.new(File.join(cookbook_path, "chefignore"))
195
- end
196
-
197
- def create_policyfile_repo_item
198
- File.open(policyfile_repo_item_path, "wb+") do |f|
199
- f.print(FFI_Yajl::Encoder.encode(policyfile_lock.to_lock, pretty: true ))
200
- end
201
- end
202
-
203
- def create_policy_group_repo_item
204
- data = {
205
- "policies" => {
206
- policyfile_lock.name => {
207
- "revision_id" => policyfile_lock.revision_id
208
- }
209
- }
210
- }
211
-
212
- File.open(policy_group_repo_item_path, "wb+") do |f|
213
- f.print(FFI_Yajl::Encoder.encode(data, pretty: true ))
214
- end
215
- end
216
-
217
- def copy_policyfile_lock
218
- File.open(lockfile_staging_path, "wb+") do |f|
219
- f.print(FFI_Yajl::Encoder.encode(policyfile_lock.to_lock, pretty: true ))
220
- end
221
- end
222
-
223
- def create_client_rb
224
- File.open(client_rb_staging_path, "wb+") do |f|
225
- f.print( <<-CONFIG )
226
- ### Chef Client Configuration ###
227
- # The settings in this file will configure chef to apply the exported policy in
228
- # this directory. To use it, run:
229
- #
230
- # chef-client -z
231
- #
232
-
233
- policy_name '#{policy_name}'
234
- policy_group 'local'
235
-
236
- use_policyfile true
237
- policy_document_native_api true
238
-
239
- # In order to use this repo, you need a version of Chef Client and Chef Zero
240
- # that supports policyfile "native mode" APIs:
241
- current_version = Gem::Version.new(Chef::VERSION)
242
- unless Gem::Requirement.new(">= 12.7").satisfied_by?(current_version)
243
- puts("!" * 80)
244
- puts(<<-MESSAGE)
245
- This Chef Repo requires features introduced in Chef 12.7, but you are using
246
- Chef \#{Chef::VERSION}. Please upgrade to Chef 12.7 or later.
247
- MESSAGE
248
- puts("!" * 80)
249
- exit!(1)
250
- end
251
-
252
- CONFIG
253
- end
254
- end
255
-
256
- def create_readme_md
257
- File.open(readme_staging_path, "wb+") do |f|
258
- f.print( <<-README )
259
- # Exported Chef Repository for Policy '#{policy_name}'
260
-
261
- Policy revision: #{policyfile_lock.revision_id}
262
-
263
- This directory contains all the cookbooks and configuration necessary for Chef
264
- to converge a system using this exported policy. To converge a system with the
265
- exported policy, use a privileged account to run `chef-client -z` from the
266
- directory containing the exported policy.
267
-
268
- ## Contents:
269
-
270
- ### Policyfile.lock.json
271
-
272
- A copy of the exported policy, used by the `chef push-archive` command.
273
-
274
- ### .chef/config.rb
275
-
276
- A configuration file for Chef Client. This file configures Chef Client to use
277
- the correct `policy_name` and `policy_group` for this exported repository. Chef
278
- Client will use this configuration automatically if you've set your working
279
- directory properly.
280
-
281
- ### cookbook_artifacts/
282
-
283
- All of the cookbooks required by the policy will be stored in this directory.
284
-
285
- ### policies/
286
-
287
- A different copy of the exported policy, used by the `chef-client` command.
288
-
289
- ### policy_groups/
290
-
291
- Policy groups are used by Chef Server to manage multiple revisions of the same
292
- policy. However, exported policies contain only a single policy revision, so
293
- this policy group name is hardcoded to "local" and should not be changed.
294
-
295
- README
296
- end
297
- end
298
-
299
- def mv_staged_repo
300
- # If we got here, either these dirs are empty/don't exist or force is
301
- # set to true.
302
- FileUtils.rm_rf(cookbook_artifacts_dir)
303
- FileUtils.rm_rf(policies_dir)
304
- FileUtils.rm_rf(policy_groups_dir)
305
- FileUtils.rm_rf(dot_chef_dir)
306
-
307
- FileUtils.mv(cookbook_artifacts_staging_dir, export_dir)
308
- FileUtils.mv(policies_staging_dir, export_dir)
309
- FileUtils.mv(policy_groups_staging_dir, export_dir)
310
- FileUtils.mv(lockfile_staging_path, export_dir)
311
- FileUtils.mv(dot_chef_staging_dir, export_dir)
312
- FileUtils.mv(readme_staging_path, export_dir)
313
- end
314
-
315
- def validate_lockfile
316
- return @policyfile_lock if @policyfile_lock
317
- @policyfile_lock = ChefDK::PolicyfileLock.new(storage_config).build_from_lock_data(policy_data)
318
- # TODO: enumerate any cookbook that have been updated
319
- @policyfile_lock.validate_cookbooks!
320
- @policyfile_lock
321
- rescue PolicyfileExportRepoError
322
- raise
323
- rescue => error
324
- raise PolicyfileExportRepoError.new("Invalid lockfile data", error)
325
- end
326
-
327
- def write_updated_lockfile
328
- File.open(policyfile_lock_expanded_path, "wb+") do |f|
329
- f.print(FFI_Yajl::Encoder.encode(policyfile_lock.to_lock, pretty: true ))
330
- end
331
- end
332
-
333
- def assert_lockfile_exists!
334
- unless File.exist?(policyfile_lock_expanded_path)
335
- raise LockfileNotFound, "No lockfile at #{policyfile_lock_expanded_path} - you need to run `install` before `push`"
336
- end
337
- end
338
-
339
- def assert_export_dir_clean!
340
- if !force_export? && !conflicting_fs_entries.empty? && !archive?
341
- msg = "Export dir (#{export_dir}) not clean. Refusing to export. (Conflicting files: #{conflicting_fs_entries.join(', ')})"
342
- raise ExportDirNotEmpty, msg
343
- end
344
- end
345
-
346
- def force_export?
347
- @force_export
348
- end
349
-
350
- def conflicting_fs_entries
351
- Dir.glob(File.join(cookbook_artifacts_dir, "*")) +
352
- Dir.glob(File.join(policies_dir, "*")) +
353
- Dir.glob(File.join(policy_groups_dir, "*")) +
354
- Dir.glob(File.join(export_dir, "Policyfile.lock.json"))
355
- end
356
-
357
- def cookbook_artifacts_dir
358
- File.join(export_dir, "cookbook_artifacts")
359
- end
360
-
361
- def policies_dir
362
- File.join(export_dir, "policies")
363
- end
364
-
365
- def policy_groups_dir
366
- File.join(export_dir, "policy_groups")
367
- end
368
-
369
- def dot_chef_dir
370
- File.join(export_dir, ".chef")
371
- end
372
-
373
- def policyfile_repo_item_path
374
- basename = "#{policyfile_lock.name}-#{policyfile_lock.revision_id}"
375
- File.join(staging_dir, "policies", "#{basename}.json")
376
- end
377
-
378
- def policy_group_repo_item_path
379
- File.join(staging_dir, "policy_groups", "local.json")
380
- end
381
-
382
- def dot_chef_staging_dir
383
- File.join(staging_dir, ".chef")
384
- end
385
-
386
- def cookbook_artifacts_staging_dir
387
- File.join(staging_dir, "cookbook_artifacts")
388
- end
389
-
390
- def policies_staging_dir
391
- File.join(staging_dir, "policies")
392
- end
393
-
394
- def policy_groups_staging_dir
395
- File.join(staging_dir, "policy_groups")
396
- end
397
-
398
- def lockfile_staging_path
399
- File.join(staging_dir, "Policyfile.lock.json")
400
- end
401
-
402
- def client_rb_staging_path
403
- File.join(dot_chef_staging_dir, "config.rb")
404
- end
405
-
406
- def readme_staging_path
407
- File.join(staging_dir, "README.md")
408
- end
409
-
410
- end
411
-
412
- end
413
- end
414
-
1
+ #
2
+ # Copyright:: Copyright (c) 2014 Chef Software Inc.
3
+ # License:: Apache License, Version 2.0
4
+ #
5
+ # Licensed under the Apache License, Version 2.0 (the "License");
6
+ # you may not use this file except in compliance with the License.
7
+ # You may obtain a copy of the License at
8
+ #
9
+ # http://www.apache.org/licenses/LICENSE-2.0
10
+ #
11
+ # Unless required by applicable law or agreed to in writing, software
12
+ # distributed under the License is distributed on an "AS IS" BASIS,
13
+ # WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
14
+ # See the License for the specific language governing permissions and
15
+ # limitations under the License.
16
+ #
17
+
18
+ require 'pathname'
19
+ require 'fileutils'
20
+ require 'tmpdir'
21
+ require 'zlib'
22
+
23
+ require 'archive/tar/minitar'
24
+
25
+ require 'chef/cookbook/chefignore'
26
+
27
+ require 'chef-dk/service_exceptions'
28
+ require 'chef-dk/policyfile_lock'
29
+ require 'chef-dk/policyfile/storage_config'
30
+
31
+ module ChefDK
32
+ module PolicyfileServices
33
+
34
+ class ExportRepo
35
+
36
+ # Policy groups provide namespaces for policies so that a Chef Server can
37
+ # have multiple active iterations of a policy at once, but we don't need
38
+ # this when serving a single exported policy via Chef Zero, so hardcode
39
+ # it to a "well known" value:
40
+ POLICY_GROUP = 'local'.freeze
41
+
42
+ include Policyfile::StorageConfigDelegation
43
+
44
+ attr_reader :storage_config
45
+ attr_reader :root_dir
46
+ attr_reader :export_dir
47
+
48
+ def initialize(policyfile: nil, export_dir: nil, root_dir: nil, archive: false, force: false)
49
+ @root_dir = root_dir
50
+ @export_dir = File.expand_path(export_dir)
51
+ @archive = archive
52
+ @force_export = force
53
+
54
+ @policy_data = nil
55
+ @policyfile_lock = nil
56
+
57
+ policyfile_rel_path = policyfile || "Policyfile.rb"
58
+ policyfile_full_path = File.expand_path(policyfile_rel_path, root_dir)
59
+ @storage_config = Policyfile::StorageConfig.new.use_policyfile(policyfile_full_path)
60
+
61
+ @staging_dir = nil
62
+ end
63
+
64
+ def archive?
65
+ @archive
66
+ end
67
+
68
+ def policy_name
69
+ policyfile_lock.name
70
+ end
71
+
72
+ def run
73
+ assert_lockfile_exists!
74
+ assert_export_dir_clean!
75
+
76
+ validate_lockfile
77
+ write_updated_lockfile
78
+ export
79
+ end
80
+
81
+ def policy_data
82
+ @policy_data ||= FFI_Yajl::Parser.parse(IO.read(policyfile_lock_expanded_path))
83
+ rescue => error
84
+ raise PolicyfileExportRepoError.new("Error reading lockfile #{policyfile_lock_expanded_path}", error)
85
+ end
86
+
87
+ def policyfile_lock
88
+ @policyfile_lock || validate_lockfile
89
+ end
90
+
91
+ def archive_file_location
92
+ return nil unless archive?
93
+ filename = "#{policyfile_lock.name}-#{policyfile_lock.revision_id}.tgz"
94
+ File.join(export_dir, filename)
95
+ end
96
+
97
+ def export
98
+ with_staging_dir do
99
+ create_repo_structure
100
+ copy_cookbooks
101
+ create_policyfile_repo_item
102
+ create_policy_group_repo_item
103
+ copy_policyfile_lock
104
+ create_client_rb
105
+ create_readme_md
106
+ if archive?
107
+ create_archive
108
+ else
109
+ mv_staged_repo
110
+ end
111
+ end
112
+ rescue => error
113
+ msg = "Failed to export policy (in #{policyfile_filename}) to #{export_dir}"
114
+ raise PolicyfileExportRepoError.new(msg, error)
115
+ end
116
+
117
+ private
118
+
119
+ def with_staging_dir
120
+ p = Process.pid
121
+ t = Time.new.utc.strftime("%Y%m%d%H%M%S")
122
+ Dir.mktmpdir("chefdk-export-#{p}-#{t}") do |d|
123
+ begin
124
+ @staging_dir = d
125
+ yield
126
+ ensure
127
+ @staging_dir = nil
128
+ end
129
+ end
130
+ end
131
+
132
+ def create_archive
133
+ Zlib::GzipWriter.open(archive_file_location) do |gz_file|
134
+ Dir.chdir(staging_dir) do
135
+ Archive::Tar::Minitar.pack(".", gz_file)
136
+ end
137
+ end
138
+ end
139
+
140
+ def staging_dir
141
+ @staging_dir
142
+ end
143
+
144
+ def create_repo_structure
145
+ FileUtils.mkdir_p(export_dir)
146
+ FileUtils.mkdir_p(dot_chef_staging_dir)
147
+ FileUtils.mkdir_p(cookbook_artifacts_staging_dir)
148
+ FileUtils.mkdir_p(policies_staging_dir)
149
+ FileUtils.mkdir_p(policy_groups_staging_dir)
150
+ end
151
+
152
+ def copy_cookbooks
153
+ policyfile_lock.cookbook_locks.each do |name, lock|
154
+ copy_cookbook(lock)
155
+ end
156
+ end
157
+
158
+ def copy_cookbook(lock)
159
+ dirname = "#{lock.name}-#{lock.identifier}"
160
+ export_path = File.join(staging_dir, "cookbook_artifacts", dirname)
161
+ metadata_rb_path = File.join(export_path, "metadata.rb")
162
+ FileUtils.mkdir(export_path) if not File.directory?(export_path)
163
+ copy_unignored_cookbook_files(lock, export_path)
164
+ FileUtils.rm_f(metadata_rb_path)
165
+ metadata = lock.cookbook_version.metadata
166
+
167
+ metadata_json_path = File.join(export_path, "metadata.json")
168
+
169
+ File.open(metadata_json_path, "wb+") do |f|
170
+ f.print(FFI_Yajl::Encoder.encode(metadata.to_hash, pretty: true ))
171
+ end
172
+ end
173
+
174
+ def copy_unignored_cookbook_files(lock, export_path)
175
+ cookbook_files_to_copy(lock.cookbook_path).each do |rel_path|
176
+ full_source_path = File.join(lock.cookbook_path, rel_path)
177
+ full_dest_path = File.join(export_path, rel_path)
178
+ dest_dirname = File.dirname(full_dest_path)
179
+ FileUtils.mkdir_p(dest_dirname) unless File.directory?(dest_dirname)
180
+ FileUtils.cp(full_source_path, full_dest_path)
181
+ end
182
+ end
183
+
184
+ def cookbook_files_to_copy(cookbook_path)
185
+ cookbook = cookbook_loader_for(cookbook_path).cookbook_version
186
+
187
+ root = Pathname.new(cookbook.root_dir)
188
+
189
+ cookbook.all_files.map do |full_path|
190
+ Pathname.new(full_path).relative_path_from(root).to_s
191
+ end
192
+ end
193
+
194
+ def cookbook_loader_for(cookbook_path)
195
+ loader = Chef::Cookbook::CookbookVersionLoader.new(cookbook_path, chefignore_for(cookbook_path))
196
+ loader.load!
197
+ loader
198
+ end
199
+
200
+ def chefignore_for(cookbook_path)
201
+ Chef::Cookbook::Chefignore.new(File.join(cookbook_path, "chefignore"))
202
+ end
203
+
204
+ def create_policyfile_repo_item
205
+ File.open(policyfile_repo_item_path, "wb+") do |f|
206
+ f.print(FFI_Yajl::Encoder.encode(policyfile_lock.to_lock, pretty: true ))
207
+ end
208
+ end
209
+
210
+ def create_policy_group_repo_item
211
+ data = {
212
+ "policies" => {
213
+ policyfile_lock.name => {
214
+ "revision_id" => policyfile_lock.revision_id
215
+ }
216
+ }
217
+ }
218
+
219
+ File.open(policy_group_repo_item_path, "wb+") do |f|
220
+ f.print(FFI_Yajl::Encoder.encode(data, pretty: true ))
221
+ end
222
+ end
223
+
224
+ def copy_policyfile_lock
225
+ File.open(lockfile_staging_path, "wb+") do |f|
226
+ f.print(FFI_Yajl::Encoder.encode(policyfile_lock.to_lock, pretty: true ))
227
+ end
228
+ end
229
+
230
+ def create_client_rb
231
+ File.open(client_rb_staging_path, "wb+") do |f|
232
+ f.print( <<-CONFIG )
233
+ ### Chef Client Configuration ###
234
+ # The settings in this file will configure chef to apply the exported policy in
235
+ # this directory. To use it, run:
236
+ #
237
+ # chef-client -z
238
+ #
239
+
240
+ policy_name '#{policy_name}'
241
+ policy_group 'local'
242
+
243
+ use_policyfile true
244
+ policy_document_native_api true
245
+
246
+ # In order to use this repo, you need a version of Chef Client and Chef Zero
247
+ # that supports policyfile "native mode" APIs:
248
+ current_version = Gem::Version.new(Chef::VERSION)
249
+ unless Gem::Requirement.new(">= 12.7").satisfied_by?(current_version)
250
+ puts("!" * 80)
251
+ puts(<<-MESSAGE)
252
+ This Chef Repo requires features introduced in Chef 12.7, but you are using
253
+ Chef \#{Chef::VERSION}. Please upgrade to Chef 12.7 or later.
254
+ MESSAGE
255
+ puts("!" * 80)
256
+ exit!(1)
257
+ end
258
+
259
+ CONFIG
260
+ end
261
+ end
262
+
263
+ def create_readme_md
264
+ File.open(readme_staging_path, "wb+") do |f|
265
+ f.print( <<-README )
266
+ # Exported Chef Repository for Policy '#{policy_name}'
267
+
268
+ Policy revision: #{policyfile_lock.revision_id}
269
+
270
+ This directory contains all the cookbooks and configuration necessary for Chef
271
+ to converge a system using this exported policy. To converge a system with the
272
+ exported policy, use a privileged account to run `chef-client -z` from the
273
+ directory containing the exported policy.
274
+
275
+ ## Contents:
276
+
277
+ ### Policyfile.lock.json
278
+
279
+ A copy of the exported policy, used by the `chef push-archive` command.
280
+
281
+ ### .chef/config.rb
282
+
283
+ A configuration file for Chef Client. This file configures Chef Client to use
284
+ the correct `policy_name` and `policy_group` for this exported repository. Chef
285
+ Client will use this configuration automatically if you've set your working
286
+ directory properly.
287
+
288
+ ### cookbook_artifacts/
289
+
290
+ All of the cookbooks required by the policy will be stored in this directory.
291
+
292
+ ### policies/
293
+
294
+ A different copy of the exported policy, used by the `chef-client` command.
295
+
296
+ ### policy_groups/
297
+
298
+ Policy groups are used by Chef Server to manage multiple revisions of the same
299
+ policy. However, exported policies contain only a single policy revision, so
300
+ this policy group name is hardcoded to "local" and should not be changed.
301
+
302
+ README
303
+ end
304
+ end
305
+
306
+ def mv_staged_repo
307
+ # If we got here, either these dirs are empty/don't exist or force is
308
+ # set to true.
309
+ FileUtils.rm_rf(cookbook_artifacts_dir)
310
+ FileUtils.rm_rf(policies_dir)
311
+ FileUtils.rm_rf(policy_groups_dir)
312
+ FileUtils.rm_rf(dot_chef_dir)
313
+
314
+ FileUtils.mv(cookbook_artifacts_staging_dir, export_dir)
315
+ FileUtils.mv(policies_staging_dir, export_dir)
316
+ FileUtils.mv(policy_groups_staging_dir, export_dir)
317
+ FileUtils.mv(lockfile_staging_path, export_dir)
318
+ FileUtils.mv(dot_chef_staging_dir, export_dir)
319
+ FileUtils.mv(readme_staging_path, export_dir)
320
+ end
321
+
322
+ def validate_lockfile
323
+ return @policyfile_lock if @policyfile_lock
324
+ @policyfile_lock = ChefDK::PolicyfileLock.new(storage_config).build_from_lock_data(policy_data)
325
+ # TODO: enumerate any cookbook that have been updated
326
+ @policyfile_lock.validate_cookbooks!
327
+ @policyfile_lock
328
+ rescue PolicyfileExportRepoError
329
+ raise
330
+ rescue => error
331
+ raise PolicyfileExportRepoError.new("Invalid lockfile data", error)
332
+ end
333
+
334
+ def write_updated_lockfile
335
+ File.open(policyfile_lock_expanded_path, "wb+") do |f|
336
+ f.print(FFI_Yajl::Encoder.encode(policyfile_lock.to_lock, pretty: true ))
337
+ end
338
+ end
339
+
340
+ def assert_lockfile_exists!
341
+ unless File.exist?(policyfile_lock_expanded_path)
342
+ raise LockfileNotFound, "No lockfile at #{policyfile_lock_expanded_path} - you need to run `install` before `push`"
343
+ end
344
+ end
345
+
346
+ def assert_export_dir_clean!
347
+ if !force_export? && !conflicting_fs_entries.empty? && !archive?
348
+ msg = "Export dir (#{export_dir}) not clean. Refusing to export. (Conflicting files: #{conflicting_fs_entries.join(', ')})"
349
+ raise ExportDirNotEmpty, msg
350
+ end
351
+ end
352
+
353
+ def force_export?
354
+ @force_export
355
+ end
356
+
357
+ def conflicting_fs_entries
358
+ Dir.glob(File.join(cookbook_artifacts_dir, "*")) +
359
+ Dir.glob(File.join(policies_dir, "*")) +
360
+ Dir.glob(File.join(policy_groups_dir, "*")) +
361
+ Dir.glob(File.join(export_dir, "Policyfile.lock.json"))
362
+ end
363
+
364
+ def cookbook_artifacts_dir
365
+ File.join(export_dir, "cookbook_artifacts")
366
+ end
367
+
368
+ def policies_dir
369
+ File.join(export_dir, "policies")
370
+ end
371
+
372
+ def policy_groups_dir
373
+ File.join(export_dir, "policy_groups")
374
+ end
375
+
376
+ def dot_chef_dir
377
+ File.join(export_dir, ".chef")
378
+ end
379
+
380
+ def policyfile_repo_item_path
381
+ basename = "#{policyfile_lock.name}-#{policyfile_lock.revision_id}"
382
+ File.join(staging_dir, "policies", "#{basename}.json")
383
+ end
384
+
385
+ def policy_group_repo_item_path
386
+ File.join(staging_dir, "policy_groups", "local.json")
387
+ end
388
+
389
+ def dot_chef_staging_dir
390
+ File.join(staging_dir, ".chef")
391
+ end
392
+
393
+ def cookbook_artifacts_staging_dir
394
+ File.join(staging_dir, "cookbook_artifacts")
395
+ end
396
+
397
+ def policies_staging_dir
398
+ File.join(staging_dir, "policies")
399
+ end
400
+
401
+ def policy_groups_staging_dir
402
+ File.join(staging_dir, "policy_groups")
403
+ end
404
+
405
+ def lockfile_staging_path
406
+ File.join(staging_dir, "Policyfile.lock.json")
407
+ end
408
+
409
+ def client_rb_staging_path
410
+ File.join(dot_chef_staging_dir, "config.rb")
411
+ end
412
+
413
+ def readme_staging_path
414
+ File.join(staging_dir, "README.md")
415
+ end
416
+
417
+ end
418
+
419
+ end
420
+ end
421
+