chef-dk 0.9.0 → 0.10.0

Sign up to get free protection for your applications and to get access to all the features.
Files changed (306) hide show
  1. checksums.yaml +4 -4
  2. data/CONTRIBUTING.md +190 -190
  3. data/Gemfile +26 -26
  4. data/LICENSE +201 -201
  5. data/README.md +292 -276
  6. data/Rakefile +18 -18
  7. data/bin/chef +25 -25
  8. data/chef-dk.gemspec +62 -0
  9. data/lib/chef-dk.rb +19 -19
  10. data/lib/chef-dk/authenticated_http.rb +40 -40
  11. data/lib/chef-dk/builtin_commands.rb +60 -60
  12. data/lib/chef-dk/chef_runner.rb +98 -98
  13. data/lib/chef-dk/cli.rb +200 -200
  14. data/lib/chef-dk/command/base.rb +79 -79
  15. data/lib/chef-dk/command/clean_policy_cookbooks.rb +116 -116
  16. data/lib/chef-dk/command/clean_policy_revisions.rb +113 -113
  17. data/lib/chef-dk/command/delete_policy.rb +122 -122
  18. data/lib/chef-dk/command/delete_policy_group.rb +122 -122
  19. data/lib/chef-dk/command/diff.rb +316 -316
  20. data/lib/chef-dk/command/env.rb +90 -90
  21. data/lib/chef-dk/command/exec.rb +45 -45
  22. data/lib/chef-dk/command/export.rb +151 -151
  23. data/lib/chef-dk/command/gem.rb +47 -47
  24. data/lib/chef-dk/command/generate.rb +120 -120
  25. data/lib/chef-dk/command/generator_commands.rb +80 -80
  26. data/lib/chef-dk/command/generator_commands/app.rb +107 -107
  27. data/lib/chef-dk/command/generator_commands/attribute.rb +37 -37
  28. data/lib/chef-dk/command/generator_commands/base.rb +121 -121
  29. data/lib/chef-dk/command/generator_commands/cookbook.rb +153 -119
  30. data/lib/chef-dk/command/generator_commands/cookbook_code_file.rb +100 -100
  31. data/lib/chef-dk/command/generator_commands/cookbook_file.rb +45 -45
  32. data/lib/chef-dk/command/generator_commands/generator_generator.rb +177 -177
  33. data/lib/chef-dk/command/generator_commands/lwrp.rb +36 -36
  34. data/lib/chef-dk/command/generator_commands/policyfile.rb +127 -86
  35. data/lib/chef-dk/command/generator_commands/recipe.rb +36 -36
  36. data/lib/chef-dk/command/generator_commands/repo.rb +125 -96
  37. data/lib/chef-dk/command/generator_commands/template.rb +46 -46
  38. data/lib/chef-dk/command/install.rb +121 -121
  39. data/lib/chef-dk/command/provision.rb +438 -438
  40. data/lib/chef-dk/command/push.rb +118 -118
  41. data/lib/chef-dk/command/push_archive.rb +126 -126
  42. data/lib/chef-dk/command/shell_init.rb +180 -180
  43. data/lib/chef-dk/command/show_policy.rb +165 -165
  44. data/lib/chef-dk/command/undelete.rb +155 -155
  45. data/lib/chef-dk/command/update.rb +129 -129
  46. data/lib/chef-dk/command/verify.rb +534 -490
  47. data/lib/chef-dk/commands_map.rb +115 -115
  48. data/lib/chef-dk/completions/bash.sh.erb +5 -5
  49. data/lib/chef-dk/completions/chef.fish.erb +10 -10
  50. data/lib/chef-dk/completions/zsh.zsh.erb +21 -21
  51. data/lib/chef-dk/component_test.rb +173 -171
  52. data/lib/chef-dk/configurable.rb +57 -57
  53. data/lib/chef-dk/cookbook_metadata.rb +45 -45
  54. data/lib/chef-dk/cookbook_omnifetch.rb +32 -32
  55. data/lib/chef-dk/cookbook_profiler/git.rb +151 -151
  56. data/lib/chef-dk/cookbook_profiler/identifiers.rb +72 -72
  57. data/lib/chef-dk/cookbook_profiler/null_scm.rb +32 -32
  58. data/lib/chef-dk/exceptions.rb +126 -113
  59. data/lib/chef-dk/generator.rb +163 -163
  60. data/lib/chef-dk/helpers.rb +159 -159
  61. data/lib/chef-dk/pager.rb +106 -106
  62. data/lib/chef-dk/policyfile/chef_repo_cookbook_source.rb +138 -122
  63. data/lib/chef-dk/policyfile/chef_server_cookbook_source.rb +54 -54
  64. data/lib/chef-dk/policyfile/community_cookbook_source.rb +97 -82
  65. data/lib/chef-dk/policyfile/comparison_base.rb +124 -124
  66. data/lib/chef-dk/policyfile/cookbook_location_specification.rb +141 -133
  67. data/lib/chef-dk/policyfile/cookbook_locks.rb +466 -466
  68. data/lib/chef-dk/policyfile/cookbook_sources.rb +22 -21
  69. data/lib/chef-dk/policyfile/delivery_supermarket_source.rb +90 -0
  70. data/lib/chef-dk/policyfile/differ.rb +266 -266
  71. data/lib/chef-dk/policyfile/dsl.rb +223 -197
  72. data/lib/chef-dk/policyfile/lister.rb +232 -232
  73. data/lib/chef-dk/policyfile/null_cookbook_source.rb +45 -45
  74. data/lib/chef-dk/policyfile/read_cookbook_for_compat_mode_upload.rb +124 -124
  75. data/lib/chef-dk/policyfile/reports/install.rb +70 -70
  76. data/lib/chef-dk/policyfile/reports/table_printer.rb +58 -58
  77. data/lib/chef-dk/policyfile/reports/upload.rb +70 -70
  78. data/lib/chef-dk/policyfile/solution_dependencies.rb +298 -298
  79. data/lib/chef-dk/policyfile/storage_config.rb +100 -100
  80. data/lib/chef-dk/policyfile/undo_record.rb +142 -142
  81. data/lib/chef-dk/policyfile/undo_stack.rb +130 -130
  82. data/lib/chef-dk/policyfile/uploader.rb +213 -213
  83. data/lib/chef-dk/policyfile_compiler.rb +380 -322
  84. data/lib/chef-dk/policyfile_lock.rb +552 -552
  85. data/lib/chef-dk/policyfile_services/clean_policies.rb +95 -95
  86. data/lib/chef-dk/policyfile_services/clean_policy_cookbooks.rb +125 -125
  87. data/lib/chef-dk/policyfile_services/export_repo.rb +309 -309
  88. data/lib/chef-dk/policyfile_services/install.rb +125 -125
  89. data/lib/chef-dk/policyfile_services/push.rb +114 -114
  90. data/lib/chef-dk/policyfile_services/push_archive.rb +173 -173
  91. data/lib/chef-dk/policyfile_services/rm_policy.rb +142 -142
  92. data/lib/chef-dk/policyfile_services/rm_policy_group.rb +86 -86
  93. data/lib/chef-dk/policyfile_services/show_policy.rb +237 -237
  94. data/lib/chef-dk/policyfile_services/undelete.rb +108 -108
  95. data/lib/chef-dk/policyfile_services/update_attributes.rb +104 -104
  96. data/lib/chef-dk/service_exception_inspectors.rb +25 -25
  97. data/lib/chef-dk/service_exception_inspectors/base.rb +40 -40
  98. data/lib/chef-dk/service_exception_inspectors/http.rb +121 -121
  99. data/lib/chef-dk/service_exceptions.rb +143 -143
  100. data/lib/chef-dk/shell_out.rb +36 -36
  101. data/lib/chef-dk/skeletons/code_generator/files/default/Berksfile +3 -3
  102. data/lib/chef-dk/skeletons/code_generator/files/default/chefignore +100 -100
  103. data/lib/chef-dk/skeletons/code_generator/files/default/cookbook_readmes/README-policy.md +9 -9
  104. data/lib/chef-dk/skeletons/code_generator/files/default/cookbook_readmes/README.md +54 -54
  105. data/lib/chef-dk/skeletons/code_generator/files/default/gitignore +16 -16
  106. data/lib/chef-dk/skeletons/code_generator/files/default/repo/README.md +28 -28
  107. data/lib/chef-dk/skeletons/code_generator/files/default/repo/cookbooks/example/README.md +27 -27
  108. data/lib/chef-dk/skeletons/code_generator/files/default/repo/cookbooks/example/attributes/default.rb +7 -7
  109. data/lib/chef-dk/skeletons/code_generator/files/default/repo/cookbooks/example/metadata.rb +3 -3
  110. data/lib/chef-dk/skeletons/code_generator/files/default/repo/cookbooks/example/recipes/default.rb +8 -8
  111. data/lib/chef-dk/skeletons/code_generator/files/default/repo/data_bags/README.md +58 -58
  112. data/lib/chef-dk/skeletons/code_generator/files/default/repo/data_bags/example/example_item.json +3 -3
  113. data/lib/chef-dk/skeletons/code_generator/files/default/repo/dot-chef-repo.txt +6 -0
  114. data/lib/chef-dk/skeletons/code_generator/files/default/repo/environments/README.md +9 -9
  115. data/lib/chef-dk/skeletons/code_generator/files/default/repo/environments/example.json +12 -12
  116. data/lib/chef-dk/skeletons/code_generator/files/default/repo/policies/README.md +24 -0
  117. data/lib/chef-dk/skeletons/code_generator/files/default/repo/roles/README.md +9 -9
  118. data/lib/chef-dk/skeletons/code_generator/files/default/repo/roles/example.json +12 -12
  119. data/lib/chef-dk/skeletons/code_generator/files/default/serverspec_spec_helper.rb +8 -8
  120. data/lib/chef-dk/skeletons/code_generator/files/default/spec_helper.rb +2 -2
  121. data/lib/chef-dk/skeletons/code_generator/files/default/spec_helper_policyfile.rb +2 -0
  122. data/lib/chef-dk/skeletons/code_generator/metadata.rb +8 -8
  123. data/lib/chef-dk/skeletons/code_generator/recipes/app.rb +97 -97
  124. data/lib/chef-dk/skeletons/code_generator/recipes/attribute.rb +12 -12
  125. data/lib/chef-dk/skeletons/code_generator/recipes/cookbook.rb +117 -104
  126. data/lib/chef-dk/skeletons/code_generator/recipes/cookbook_file.rb +24 -24
  127. data/lib/chef-dk/skeletons/code_generator/recipes/lwrp.rb +23 -23
  128. data/lib/chef-dk/skeletons/code_generator/recipes/policyfile.rb +8 -8
  129. data/lib/chef-dk/skeletons/code_generator/recipes/recipe.rb +27 -27
  130. data/lib/chef-dk/skeletons/code_generator/recipes/repo.rb +67 -48
  131. data/lib/chef-dk/skeletons/code_generator/recipes/template.rb +32 -32
  132. data/lib/chef-dk/skeletons/code_generator/templates/default/LICENSE.all_rights.erb +3 -3
  133. data/lib/chef-dk/skeletons/code_generator/templates/default/LICENSE.apache2.erb +201 -201
  134. data/lib/chef-dk/skeletons/code_generator/templates/default/LICENSE.gplv2.erb +339 -339
  135. data/lib/chef-dk/skeletons/code_generator/templates/default/LICENSE.gplv3.erb +674 -674
  136. data/lib/chef-dk/skeletons/code_generator/templates/default/LICENSE.mit.erb +21 -21
  137. data/lib/chef-dk/skeletons/code_generator/templates/default/Policyfile.rb.erb +25 -20
  138. data/lib/chef-dk/skeletons/code_generator/templates/default/README.md.erb +4 -4
  139. data/lib/chef-dk/skeletons/code_generator/templates/default/kitchen.yml.erb +21 -16
  140. data/lib/chef-dk/skeletons/code_generator/templates/default/kitchen_policyfile.yml.erb +32 -27
  141. data/lib/chef-dk/skeletons/code_generator/templates/default/metadata.rb.erb +7 -7
  142. data/lib/chef-dk/skeletons/code_generator/templates/default/recipe.rb.erb +5 -5
  143. data/lib/chef-dk/skeletons/code_generator/templates/default/recipe_spec.rb.erb +20 -20
  144. data/lib/chef-dk/skeletons/code_generator/templates/default/repo/gitignore.erb +11 -11
  145. data/lib/chef-dk/skeletons/code_generator/templates/default/serverspec_default_spec.rb.erb +9 -9
  146. data/lib/chef-dk/ui.rb +58 -58
  147. data/lib/chef-dk/version.rb +20 -20
  148. data/lib/kitchen/provisioner/policyfile_zero.rb +189 -193
  149. data/spec/shared/a_file_generator.rb +125 -125
  150. data/spec/shared/a_generated_file.rb +12 -12
  151. data/spec/shared/command_with_ui_object.rb +11 -11
  152. data/spec/shared/custom_generator_cookbook.rb +117 -117
  153. data/spec/shared/fixture_cookbook_checksums.rb +47 -47
  154. data/spec/shared/setup_git_cookbooks.rb +53 -53
  155. data/spec/spec_helper.rb +50 -49
  156. data/spec/test_helpers.rb +84 -84
  157. data/spec/unit/chef_runner_spec.rb +111 -111
  158. data/spec/unit/cli_spec.rb +357 -357
  159. data/spec/unit/command/base_spec.rb +169 -169
  160. data/spec/unit/command/clean_policy_cookbooks_spec.rb +181 -181
  161. data/spec/unit/command/clean_policy_revisions_spec.rb +181 -181
  162. data/spec/unit/command/delete_policy_group_spec.rb +207 -207
  163. data/spec/unit/command/delete_policy_spec.rb +207 -207
  164. data/spec/unit/command/diff_spec.rb +312 -312
  165. data/spec/unit/command/env_spec.rb +52 -52
  166. data/spec/unit/command/exec_spec.rb +179 -179
  167. data/spec/unit/command/export_spec.rb +189 -189
  168. data/spec/unit/command/generate_spec.rb +142 -142
  169. data/spec/unit/command/generator_commands/app_spec.rb +169 -169
  170. data/spec/unit/command/generator_commands/attribute_spec.rb +32 -32
  171. data/spec/unit/command/generator_commands/cookbook_file_spec.rb +32 -32
  172. data/spec/unit/command/generator_commands/cookbook_spec.rb +450 -320
  173. data/spec/unit/command/generator_commands/generator_generator_spec.rb +229 -229
  174. data/spec/unit/command/generator_commands/lwrp_spec.rb +32 -32
  175. data/spec/unit/command/generator_commands/policyfile_spec.rb +225 -125
  176. data/spec/unit/command/generator_commands/recipe_spec.rb +34 -34
  177. data/spec/unit/command/generator_commands/repo_spec.rb +367 -283
  178. data/spec/unit/command/generator_commands/template_spec.rb +32 -32
  179. data/spec/unit/command/install_spec.rb +179 -179
  180. data/spec/unit/command/provision_spec.rb +592 -592
  181. data/spec/unit/command/push_archive_spec.rb +153 -153
  182. data/spec/unit/command/push_spec.rb +199 -199
  183. data/spec/unit/command/shell_init_spec.rb +329 -329
  184. data/spec/unit/command/show_policy_spec.rb +235 -235
  185. data/spec/unit/command/undelete_spec.rb +246 -246
  186. data/spec/unit/command/update_spec.rb +251 -251
  187. data/spec/unit/command/verify_spec.rb +325 -323
  188. data/spec/unit/commands_map_spec.rb +57 -57
  189. data/spec/unit/component_test_spec.rb +126 -126
  190. data/spec/unit/cookbook_metadata_spec.rb +98 -98
  191. data/spec/unit/cookbook_profiler/git_spec.rb +176 -176
  192. data/spec/unit/cookbook_profiler/identifiers_spec.rb +83 -83
  193. data/spec/unit/fixtures/chef-runner-cookbooks/test_cookbook/recipes/recipe_one.rb +9 -9
  194. data/spec/unit/fixtures/chef-runner-cookbooks/test_cookbook/recipes/recipe_two.rb +9 -9
  195. data/spec/unit/fixtures/command/cli_test_command.rb +26 -26
  196. data/spec/unit/fixtures/command/explicit_path_example.rb +7 -7
  197. data/spec/unit/fixtures/cookbook_cache/baz-f59ee7a5bca6a4e606b67f7f856b768d847c39bb/.kitchen.yml +16 -16
  198. data/spec/unit/fixtures/cookbook_cache/baz-f59ee7a5bca6a4e606b67f7f856b768d847c39bb/Berksfile +3 -3
  199. data/spec/unit/fixtures/cookbook_cache/baz-f59ee7a5bca6a4e606b67f7f856b768d847c39bb/README.md +4 -4
  200. data/spec/unit/fixtures/cookbook_cache/baz-f59ee7a5bca6a4e606b67f7f856b768d847c39bb/chefignore +96 -96
  201. data/spec/unit/fixtures/cookbook_cache/baz-f59ee7a5bca6a4e606b67f7f856b768d847c39bb/metadata.rb +8 -8
  202. data/spec/unit/fixtures/cookbook_cache/baz-f59ee7a5bca6a4e606b67f7f856b768d847c39bb/recipes/default.rb +8 -8
  203. data/spec/unit/fixtures/cookbook_cache/dep_of_bar-1.2.3/.kitchen.yml +16 -16
  204. data/spec/unit/fixtures/cookbook_cache/dep_of_bar-1.2.3/Berksfile +3 -3
  205. data/spec/unit/fixtures/cookbook_cache/dep_of_bar-1.2.3/README.md +4 -4
  206. data/spec/unit/fixtures/cookbook_cache/dep_of_bar-1.2.3/chefignore +96 -96
  207. data/spec/unit/fixtures/cookbook_cache/dep_of_bar-1.2.3/metadata.rb +8 -8
  208. data/spec/unit/fixtures/cookbook_cache/dep_of_bar-1.2.3/recipes/default.rb +8 -8
  209. data/spec/unit/fixtures/cookbook_cache/foo-1.0.0/.kitchen.yml +16 -16
  210. data/spec/unit/fixtures/cookbook_cache/foo-1.0.0/Berksfile +3 -3
  211. data/spec/unit/fixtures/cookbook_cache/foo-1.0.0/README.md +4 -4
  212. data/spec/unit/fixtures/cookbook_cache/foo-1.0.0/chefignore +96 -96
  213. data/spec/unit/fixtures/cookbook_cache/foo-1.0.0/metadata.rb +8 -8
  214. data/spec/unit/fixtures/cookbook_cache/foo-1.0.0/recipes/default.rb +8 -8
  215. data/spec/unit/fixtures/cookbooks_api/pruned_small_universe.json +1321 -1321
  216. data/spec/unit/fixtures/cookbooks_api/small_universe.json +2987 -2987
  217. data/spec/unit/fixtures/cookbooks_api/universe.json +1 -1
  218. data/spec/unit/fixtures/cookbooks_api/update_fixtures.rb +36 -36
  219. data/spec/unit/fixtures/dev_cookbooks/README.md +16 -16
  220. data/spec/unit/fixtures/eg_omnibus_dir/valid/embedded/apps/berkshelf/integration_test +2 -2
  221. data/spec/unit/fixtures/eg_omnibus_dir/valid/embedded/apps/berkshelf/verify_me +5 -5
  222. data/spec/unit/fixtures/eg_omnibus_dir/valid/embedded/apps/chef/verify_me +3 -3
  223. data/spec/unit/fixtures/eg_omnibus_dir/valid/embedded/apps/test-kitchen/verify_me +2 -2
  224. data/spec/unit/fixtures/example_cookbook/.gitignore +17 -17
  225. data/spec/unit/fixtures/example_cookbook/.kitchen.yml +16 -16
  226. data/spec/unit/fixtures/example_cookbook/Berksfile +3 -3
  227. data/spec/unit/fixtures/example_cookbook/README.md +4 -4
  228. data/spec/unit/fixtures/example_cookbook/chefignore +96 -96
  229. data/spec/unit/fixtures/example_cookbook/metadata.rb +8 -8
  230. data/spec/unit/fixtures/example_cookbook/recipes/default.rb +8 -8
  231. data/spec/unit/fixtures/example_cookbook_metadata_json_only/.gitignore +17 -17
  232. data/spec/unit/fixtures/example_cookbook_metadata_json_only/.kitchen.yml +16 -16
  233. data/spec/unit/fixtures/example_cookbook_metadata_json_only/Berksfile +3 -3
  234. data/spec/unit/fixtures/example_cookbook_metadata_json_only/README.md +4 -4
  235. data/spec/unit/fixtures/example_cookbook_metadata_json_only/chefignore +96 -96
  236. data/spec/unit/fixtures/example_cookbook_metadata_json_only/metadata.json +5 -5
  237. data/spec/unit/fixtures/example_cookbook_metadata_json_only/recipes/default.rb +8 -8
  238. data/spec/unit/fixtures/example_cookbook_no_metadata/.gitignore +17 -17
  239. data/spec/unit/fixtures/example_cookbook_no_metadata/.kitchen.yml +16 -16
  240. data/spec/unit/fixtures/example_cookbook_no_metadata/Berksfile +3 -3
  241. data/spec/unit/fixtures/example_cookbook_no_metadata/README.md +4 -4
  242. data/spec/unit/fixtures/example_cookbook_no_metadata/chefignore +96 -96
  243. data/spec/unit/fixtures/example_cookbook_no_metadata/recipes/default.rb +8 -8
  244. data/spec/unit/fixtures/local_path_cookbooks/another-local-cookbook/README.md +4 -4
  245. data/spec/unit/fixtures/local_path_cookbooks/another-local-cookbook/chefignore +96 -96
  246. data/spec/unit/fixtures/local_path_cookbooks/another-local-cookbook/metadata.rb +8 -8
  247. data/spec/unit/fixtures/local_path_cookbooks/another-local-cookbook/recipes/default.rb +8 -8
  248. data/spec/unit/fixtures/local_path_cookbooks/cookbook-with-a-dep/Berksfile +3 -3
  249. data/spec/unit/fixtures/local_path_cookbooks/cookbook-with-a-dep/README.md +4 -4
  250. data/spec/unit/fixtures/local_path_cookbooks/cookbook-with-a-dep/chefignore +96 -96
  251. data/spec/unit/fixtures/local_path_cookbooks/cookbook-with-a-dep/metadata.rb +9 -9
  252. data/spec/unit/fixtures/local_path_cookbooks/cookbook-with-a-dep/recipes/default.rb +8 -8
  253. data/spec/unit/fixtures/local_path_cookbooks/local-cookbook/.kitchen.yml +16 -16
  254. data/spec/unit/fixtures/local_path_cookbooks/local-cookbook/Berksfile +3 -3
  255. data/spec/unit/fixtures/local_path_cookbooks/local-cookbook/README.md +4 -4
  256. data/spec/unit/fixtures/local_path_cookbooks/local-cookbook/chefignore +96 -96
  257. data/spec/unit/fixtures/local_path_cookbooks/local-cookbook/metadata.rb +8 -8
  258. data/spec/unit/fixtures/local_path_cookbooks/local-cookbook/recipes/default.rb +8 -8
  259. data/spec/unit/fixtures/local_path_cookbooks/metadata-missing/README.md +2 -2
  260. data/spec/unit/fixtures/local_path_cookbooks/noignore-f59ee7a5bca6a4e606b67f7f856b768d847c39bb/.kitchen.yml +16 -16
  261. data/spec/unit/fixtures/local_path_cookbooks/noignore-f59ee7a5bca6a4e606b67f7f856b768d847c39bb/README.md +4 -4
  262. data/spec/unit/fixtures/local_path_cookbooks/noignore-f59ee7a5bca6a4e606b67f7f856b768d847c39bb/metadata.rb +8 -8
  263. data/spec/unit/fixtures/local_path_cookbooks/noignore-f59ee7a5bca6a4e606b67f7f856b768d847c39bb/recipes/default.rb +8 -8
  264. data/spec/unit/generator_spec.rb +120 -120
  265. data/spec/unit/helpers_spec.rb +92 -92
  266. data/spec/unit/pager_spec.rb +119 -119
  267. data/spec/unit/policyfile/chef_repo_cookbook_source_spec.rb +93 -66
  268. data/spec/unit/policyfile/chef_server_cookbook_source_spec.rb +34 -34
  269. data/spec/unit/policyfile/community_cookbook_source_spec.rb +84 -51
  270. data/spec/unit/policyfile/comparison_base_spec.rb +343 -343
  271. data/spec/unit/policyfile/cookbook_location_specification_spec.rb +257 -252
  272. data/spec/unit/policyfile/cookbook_locks_spec.rb +529 -529
  273. data/spec/unit/policyfile/delivery_supermarket_source_spec.rb +130 -0
  274. data/spec/unit/policyfile/differ_spec.rb +687 -687
  275. data/spec/unit/policyfile/lister_spec.rb +272 -272
  276. data/spec/unit/policyfile/null_cookbook_source_spec.rb +35 -35
  277. data/spec/unit/policyfile/read_cookbook_for_compat_mode_upload_spec.rb +92 -92
  278. data/spec/unit/policyfile/reports/install_spec.rb +115 -115
  279. data/spec/unit/policyfile/reports/upload_spec.rb +96 -96
  280. data/spec/unit/policyfile/solution_dependencies_spec.rb +145 -145
  281. data/spec/unit/policyfile/storage_config_spec.rb +172 -172
  282. data/spec/unit/policyfile/undo_record_spec.rb +260 -260
  283. data/spec/unit/policyfile/undo_stack_spec.rb +266 -266
  284. data/spec/unit/policyfile/uploader_spec.rb +410 -410
  285. data/spec/unit/policyfile_demands_spec.rb +997 -876
  286. data/spec/unit/policyfile_evaluation_spec.rb +557 -441
  287. data/spec/unit/policyfile_lock_build_spec.rb +1056 -1056
  288. data/spec/unit/policyfile_lock_install_spec.rb +138 -138
  289. data/spec/unit/policyfile_lock_serialization_spec.rb +425 -425
  290. data/spec/unit/policyfile_lock_validation_spec.rb +611 -611
  291. data/spec/unit/policyfile_services/clean_policies_spec.rb +236 -236
  292. data/spec/unit/policyfile_services/clean_policy_cookbooks_spec.rb +275 -275
  293. data/spec/unit/policyfile_services/export_repo_spec.rb +439 -439
  294. data/spec/unit/policyfile_services/install_spec.rb +191 -191
  295. data/spec/unit/policyfile_services/push_archive_spec.rb +345 -345
  296. data/spec/unit/policyfile_services/push_spec.rb +233 -233
  297. data/spec/unit/policyfile_services/rm_policy_group_spec.rb +241 -241
  298. data/spec/unit/policyfile_services/rm_policy_spec.rb +266 -266
  299. data/spec/unit/policyfile_services/show_policy_spec.rb +889 -889
  300. data/spec/unit/policyfile_services/undelete_spec.rb +304 -304
  301. data/spec/unit/policyfile_services/update_attributes_spec.rb +217 -217
  302. data/spec/unit/service_exception_inspectors/base_spec.rb +43 -43
  303. data/spec/unit/service_exception_inspectors/http_spec.rb +140 -140
  304. data/spec/unit/shell_out_spec.rb +34 -34
  305. data/warning.txt +9 -0
  306. metadata +18 -5
@@ -1,439 +1,439 @@
1
- #
2
- # Copyright:: Copyright (c) 2014 Chef Software Inc.
3
- # License:: Apache License, Version 2.0
4
- #
5
- # Licensed under the Apache License, Version 2.0 (the "License");
6
- # you may not use this file except in compliance with the License.
7
- # You may obtain a copy of the License at
8
- #
9
- # http://www.apache.org/licenses/LICENSE-2.0
10
- #
11
- # Unless required by applicable law or agreed to in writing, software
12
- # distributed under the License is distributed on an "AS IS" BASIS,
13
- # WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
14
- # See the License for the specific language governing permissions and
15
- # limitations under the License.
16
- #
17
-
18
- require 'spec_helper'
19
- require 'chef-dk/policyfile_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(:lockfile_content) do
110
- <<-E
111
- {
112
- "name": "install-example",
113
- "run_list": [
114
- "recipe[local-cookbook::default]"
115
- ],
116
- "cookbook_locks": {
117
- "local-cookbook": {
118
- "version": "2.3.4",
119
- "identifier": "fab501cfaf747901bd82c1bc706beae7dc3a350c",
120
- "dotted_decimal_identifier": "70567763561641081.489844270461035.258281553147148",
121
- "source": "#{local_cookbook_path}",
122
- "cache_key": null,
123
- "scm_info": null,
124
- "source_options": {
125
- "path": "#{local_cookbook_path}"
126
- }
127
- }
128
- },
129
- "default_attributes": {},
130
- "override_attributes": {},
131
- "solution_dependencies": {
132
- "Policyfile": [
133
- [
134
- "local-cookbook",
135
- ">= 0.0.0"
136
- ]
137
- ],
138
- "dependencies": {
139
- "local-cookbook (2.3.4)": [
140
-
141
- ]
142
- }
143
- }
144
- }
145
- E
146
- end
147
-
148
- it "reads the lockfile data" do
149
- lock = export_service.policyfile_lock
150
- expect(lock).to be_an_instance_of(ChefDK::PolicyfileLock)
151
- expect(lock.name).to eq("install-example")
152
- expect(lock.cookbook_locks.size).to eq(1)
153
- expect(lock.cookbook_locks).to have_key("local-cookbook")
154
- end
155
-
156
- it "delegates #policy_name to the lockfile" do
157
- expect(export_service.policy_name).to eq("install-example")
158
- end
159
-
160
- context "when using archive mode" do
161
-
162
- let(:archive) { true }
163
-
164
- # TODO: also support a full file name
165
- context "when the given 'export_dir' is a directory" do
166
-
167
- it "sets the archive file location to $policy_name-$revision.tgz" do
168
- expected = File.join(export_dir, "install-example-60e5ad638dce219d8f87d589463ec4a9884007ba5e2adbb4c0a7021d67204f1a.tgz")
169
- expect(export_service.archive_file_location).to eq(expected)
170
- end
171
-
172
- end
173
-
174
- end
175
-
176
- describe "writing updates to the policyfile lock" do
177
-
178
- let(:updated_lockfile_io) { StringIO.new }
179
-
180
- it "validates the lockfile and writes updates to disk" do
181
- allow(File).to receive(:open).and_call_original
182
- expect(File).to receive(:open).with(policyfile_lock_path, "wb+").and_yield(updated_lockfile_io)
183
-
184
- export_service.run
185
- end
186
-
187
- end
188
-
189
- context "copying the cookbooks to the export dir" do
190
-
191
- shared_examples_for "successful_export" do
192
- before do
193
- allow(export_service.policyfile_lock).to receive(:validate_cookbooks!).and_return(true)
194
- export_service.run
195
- end
196
-
197
- let(:cookbook_files) do
198
- base_pathname = Pathname.new(local_cookbook_path)
199
- Dir.glob("#{local_cookbook_path}/**/*").map do |full_path|
200
- Pathname.new(full_path).relative_path_from(base_pathname)
201
- end
202
- end
203
-
204
- let(:expected_files_relative) do
205
- metadata_rb = Pathname.new("metadata.rb")
206
- expected = cookbook_files.delete_if { |p| p == metadata_rb }
207
- expected << Pathname.new("metadata.json")
208
- end
209
-
210
- let(:cookbook_with_version) { "local-cookbook-70567763561641081.489844270461035.258281553147148" }
211
-
212
- let(:exported_cookbook_root) { Pathname.new(File.join(export_dir, "cookbooks", cookbook_with_version)) }
213
-
214
- let(:expected_files) do
215
- expected_files_relative.map do |file_rel_path|
216
- exported_cookbook_root + file_rel_path
217
- end
218
- end
219
-
220
- it "copies cookbooks to the target dir in versioned_cookbooks format" do
221
- expected_files.each do |expected_file|
222
- expect(expected_file).to exist
223
- end
224
- end
225
-
226
- # This behavior does two things:
227
- # * ensures that Chef Zero uses our hacked version number
228
- # * works around external dependencies (e.g., using `git` in backticks)
229
- # in metadata.rb issue
230
- it "writes metadata.json in the exported cookbook, removing metadata.rb" do
231
- metadata_json_path = File.join(exported_cookbook_root, "metadata.json")
232
- metadata_json = FFI_Yajl::Parser.parse(IO.read(metadata_json_path))
233
- expect(metadata_json["version"]).to eq("70567763561641081.489844270461035.258281553147148")
234
- end
235
-
236
- it "copies the policyfile lock in data item format to data_bags/policyfiles" do
237
- data_bag_item_path = File.join(export_dir, "data_bags", "policyfiles", "install-example-local.json")
238
- data_item_json = FFI_Yajl::Parser.parse(IO.read(data_bag_item_path))
239
- expect(data_item_json["id"]).to eq("install-example-local")
240
- end
241
-
242
- it "copies the policyfile lock in standard format to Policyfile.lock.json" do
243
- policyfile_lock_path = File.join(export_dir, "Policyfile.lock.json")
244
- policyfile_lock_data = FFI_Yajl::Parser.parse(IO.read(policyfile_lock_path))
245
- expected_lock_data = export_service.policyfile_lock.to_lock
246
-
247
- # stringify keys in source_options
248
- path = expected_lock_data["cookbook_locks"]["local-cookbook"]["source_options"].delete(:path)
249
- expected_lock_data["cookbook_locks"]["local-cookbook"]["source_options"]["path"] = path
250
-
251
- expect(policyfile_lock_data).to eq(expected_lock_data)
252
- end
253
-
254
- it "creates a working local mode configuration file" do
255
- expected_config_text = <<-CONFIG
256
- ### Chef Client Configuration ###
257
- # The settings in this file will configure chef to apply the exported policy in
258
- # this directory. To use it, run:
259
- #
260
- # chef-client -c client.rb -z
261
- #
262
-
263
- use_policyfile true
264
-
265
- # compatibility mode settings are used because chef-zero doesn't yet support
266
- # native mode:
267
- deployment_group 'install-example-local'
268
- versioned_cookbooks true
269
- policy_document_native_api false
270
-
271
- CONFIG
272
- config_path = File.join(export_dir, "client.rb")
273
- expect(File).to exist(config_path)
274
- expect(IO.read(config_path)).to eq(expected_config_text)
275
- end
276
-
277
- end
278
-
279
- context "when the export dir is empty" do
280
-
281
- include_examples "successful_export"
282
- end
283
-
284
- context "When an error occurs creating the export" do
285
-
286
- before do
287
- allow(export_service.policyfile_lock).to receive(:validate_cookbooks!).and_return(true)
288
- expect(export_service).to receive(:create_repo_structure).
289
- and_raise(Errno::EACCES.new("Permission denied @ rb_sysopen - /etc/foobarbaz.txt"))
290
- end
291
-
292
- it "wraps the error in a custom error class" do
293
- message = "Failed to export policy (in #{expanded_policyfile_path}) to #{export_dir}"
294
- expect { export_service.run }.to raise_error(ChefDK::PolicyfileExportRepoError, message)
295
- end
296
-
297
- end
298
-
299
- context "When the export dir has non-conflicting content" do
300
-
301
- let(:file_in_export_dir) { File.join(export_dir, "some_random_cruft") }
302
-
303
- let(:extra_data_bag_dir) { File.join(export_dir, "data_bags", "extraneous") }
304
-
305
- let(:extra_data_bag_item) { File.join(extra_data_bag_dir, "an_item.json") }
306
-
307
- before do
308
- FileUtils.mkdir_p(export_dir)
309
- File.open(file_in_export_dir, "wb+") { |f| f.print "some random cruft" }
310
- FileUtils.mkdir_p(extra_data_bag_dir)
311
- File.open(extra_data_bag_item, "wb+") { |f| f.print "some random cruft" }
312
- end
313
-
314
- it "ignores the non-conflicting content and exports" do
315
- expect(File).to exist(file_in_export_dir)
316
- expect(File).to exist(extra_data_bag_item)
317
-
318
- expect(File).to be_directory(File.join(export_dir, "cookbooks"))
319
- expect(File).to be_directory(File.join(export_dir, "data_bags"))
320
- end
321
-
322
- include_examples "successful_export"
323
-
324
- end
325
-
326
- context "When the export dir has conflicting content" do
327
-
328
- let(:non_conflicting_file_in_export_dir) { File.join(export_dir, "some_random_cruft") }
329
-
330
- let(:cookbooks_dir) { File.join(export_dir, "cookbooks") }
331
-
332
- let(:file_in_cookbooks_dir) { File.join(cookbooks_dir, "some_random_cruft") }
333
-
334
- let(:policyfiles_data_bag_dir) { File.join(export_dir, "data_bags", "policyfiles") }
335
-
336
- let(:extra_policyfile_data_item) { File.join(policyfiles_data_bag_dir, "leftover-policy.json") }
337
-
338
- let(:conflicting_policyfile_lock) { File.join(export_dir, "Policyfile.lock.json") }
339
-
340
- before do
341
- FileUtils.mkdir_p(export_dir)
342
- FileUtils.mkdir_p(cookbooks_dir)
343
- FileUtils.mkdir_p(policyfiles_data_bag_dir)
344
- File.open(non_conflicting_file_in_export_dir, "wb+") { |f| f.print "some random cruft" }
345
- File.open(file_in_cookbooks_dir, "wb+") { |f| f.print "some random cruft" }
346
- File.open(extra_policyfile_data_item, "wb+") { |f| f.print "some random cruft" }
347
- File.open(conflicting_policyfile_lock, "wb+") { |f| f.print "some random cruft" }
348
- end
349
-
350
- it "raises a PolicyfileExportRepoError" do
351
- message = "Export dir (#{export_dir}) not clean. Refusing to export. (Conflicting files: #{file_in_cookbooks_dir}, #{extra_policyfile_data_item}, #{conflicting_policyfile_lock})"
352
- expect { export_service.run }.to raise_error(ChefDK::ExportDirNotEmpty, message)
353
- expect(File).to exist(non_conflicting_file_in_export_dir)
354
- expect(File).to exist(file_in_cookbooks_dir)
355
- expect(File).to exist(extra_policyfile_data_item)
356
- end
357
-
358
- context "and the force option is set" do
359
-
360
- let(:force_export) { true }
361
-
362
- it "clears the export dir and exports" do
363
- export_service.run
364
-
365
- expect(File).to_not exist(file_in_cookbooks_dir)
366
- expect(File).to_not exist(extra_policyfile_data_item)
367
-
368
- expect(File).to exist(non_conflicting_file_in_export_dir)
369
-
370
- expect(File).to be_directory(File.join(export_dir, "cookbooks"))
371
- expect(File).to be_directory(File.join(export_dir, "data_bags"))
372
- end
373
-
374
- end
375
-
376
- end # When the export dir has conflicting content
377
-
378
- context "when archive mode is enabled" do
379
-
380
- let(:archive) { true }
381
-
382
- let(:expected_archive_path) do
383
- File.join(export_dir, "install-example-60e5ad638dce219d8f87d589463ec4a9884007ba5e2adbb4c0a7021d67204f1a.tgz")
384
- end
385
-
386
- it "exports the repo as a tgz archive" do
387
- expect(File).to exist(expected_archive_path)
388
- end
389
-
390
- include_examples "successful_export" do
391
-
392
- # explode the tarball so the assertions can find the files
393
- before do
394
- Zlib::GzipReader.open(expected_archive_path) do |gz_file|
395
- tar = Archive::Tar::Minitar::Input.new(gz_file)
396
- tar.each do |e|
397
- tar.extract_entry(export_dir, e)
398
- end
399
- end
400
- end
401
-
402
- end
403
-
404
- context "when the target dir has a cookbooks or data_bags dir" do
405
-
406
- let(:cookbooks_dir) { File.join(export_dir, "cookbooks") }
407
-
408
- let(:file_in_cookbooks_dir) { File.join(cookbooks_dir, "some_random_cruft") }
409
-
410
- let(:policyfiles_data_bag_dir) { File.join(export_dir, "data_bags", "policyfiles") }
411
-
412
- let(:extra_policyfile_data_item) { File.join(policyfiles_data_bag_dir, "leftover-policy.json") }
413
-
414
- before do
415
- FileUtils.mkdir_p(export_dir)
416
- FileUtils.mkdir_p(cookbooks_dir)
417
- FileUtils.mkdir_p(policyfiles_data_bag_dir)
418
- File.open(file_in_cookbooks_dir, "wb+") { |f| f.print "some random cruft" }
419
- File.open(extra_policyfile_data_item, "wb+") { |f| f.print "some random cruft" }
420
- end
421
-
422
- it "exports successfully" do
423
- expect { export_service.run }.to_not raise_error
424
- expect(File).to exist(expected_archive_path)
425
- end
426
-
427
- end
428
-
429
- end # when archive mode is enabled
430
-
431
- end # copying the cookbooks to the export dir
432
- end
433
-
434
- end
435
-
436
-
437
-
438
- end
439
-
1
+ #
2
+ # Copyright:: Copyright (c) 2014 Chef Software Inc.
3
+ # License:: Apache License, Version 2.0
4
+ #
5
+ # Licensed under the Apache License, Version 2.0 (the "License");
6
+ # you may not use this file except in compliance with the License.
7
+ # You may obtain a copy of the License at
8
+ #
9
+ # http://www.apache.org/licenses/LICENSE-2.0
10
+ #
11
+ # Unless required by applicable law or agreed to in writing, software
12
+ # distributed under the License is distributed on an "AS IS" BASIS,
13
+ # WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
14
+ # See the License for the specific language governing permissions and
15
+ # limitations under the License.
16
+ #
17
+
18
+ require 'spec_helper'
19
+ require 'chef-dk/policyfile_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(:lockfile_content) do
110
+ <<-E
111
+ {
112
+ "name": "install-example",
113
+ "run_list": [
114
+ "recipe[local-cookbook::default]"
115
+ ],
116
+ "cookbook_locks": {
117
+ "local-cookbook": {
118
+ "version": "2.3.4",
119
+ "identifier": "fab501cfaf747901bd82c1bc706beae7dc3a350c",
120
+ "dotted_decimal_identifier": "70567763561641081.489844270461035.258281553147148",
121
+ "source": "#{local_cookbook_path}",
122
+ "cache_key": null,
123
+ "scm_info": null,
124
+ "source_options": {
125
+ "path": "#{local_cookbook_path}"
126
+ }
127
+ }
128
+ },
129
+ "default_attributes": {},
130
+ "override_attributes": {},
131
+ "solution_dependencies": {
132
+ "Policyfile": [
133
+ [
134
+ "local-cookbook",
135
+ ">= 0.0.0"
136
+ ]
137
+ ],
138
+ "dependencies": {
139
+ "local-cookbook (2.3.4)": [
140
+
141
+ ]
142
+ }
143
+ }
144
+ }
145
+ E
146
+ end
147
+
148
+ it "reads the lockfile data" do
149
+ lock = export_service.policyfile_lock
150
+ expect(lock).to be_an_instance_of(ChefDK::PolicyfileLock)
151
+ expect(lock.name).to eq("install-example")
152
+ expect(lock.cookbook_locks.size).to eq(1)
153
+ expect(lock.cookbook_locks).to have_key("local-cookbook")
154
+ end
155
+
156
+ it "delegates #policy_name to the lockfile" do
157
+ expect(export_service.policy_name).to eq("install-example")
158
+ end
159
+
160
+ context "when using archive mode" do
161
+
162
+ let(:archive) { true }
163
+
164
+ # TODO: also support a full file name
165
+ context "when the given 'export_dir' is a directory" do
166
+
167
+ it "sets the archive file location to $policy_name-$revision.tgz" do
168
+ expected = File.join(export_dir, "install-example-60e5ad638dce219d8f87d589463ec4a9884007ba5e2adbb4c0a7021d67204f1a.tgz")
169
+ expect(export_service.archive_file_location).to eq(expected)
170
+ end
171
+
172
+ end
173
+
174
+ end
175
+
176
+ describe "writing updates to the policyfile lock" do
177
+
178
+ let(:updated_lockfile_io) { StringIO.new }
179
+
180
+ it "validates the lockfile and writes updates to disk" do
181
+ allow(File).to receive(:open).and_call_original
182
+ expect(File).to receive(:open).with(policyfile_lock_path, "wb+").and_yield(updated_lockfile_io)
183
+
184
+ export_service.run
185
+ end
186
+
187
+ end
188
+
189
+ context "copying the cookbooks to the export dir" do
190
+
191
+ shared_examples_for "successful_export" do
192
+ before do
193
+ allow(export_service.policyfile_lock).to receive(:validate_cookbooks!).and_return(true)
194
+ export_service.run
195
+ end
196
+
197
+ let(:cookbook_files) do
198
+ base_pathname = Pathname.new(local_cookbook_path)
199
+ Dir.glob("#{local_cookbook_path}/**/*").map do |full_path|
200
+ Pathname.new(full_path).relative_path_from(base_pathname)
201
+ end
202
+ end
203
+
204
+ let(:expected_files_relative) do
205
+ metadata_rb = Pathname.new("metadata.rb")
206
+ expected = cookbook_files.delete_if { |p| p == metadata_rb }
207
+ expected << Pathname.new("metadata.json")
208
+ end
209
+
210
+ let(:cookbook_with_version) { "local-cookbook-70567763561641081.489844270461035.258281553147148" }
211
+
212
+ let(:exported_cookbook_root) { Pathname.new(File.join(export_dir, "cookbooks", cookbook_with_version)) }
213
+
214
+ let(:expected_files) do
215
+ expected_files_relative.map do |file_rel_path|
216
+ exported_cookbook_root + file_rel_path
217
+ end
218
+ end
219
+
220
+ it "copies cookbooks to the target dir in versioned_cookbooks format" do
221
+ expected_files.each do |expected_file|
222
+ expect(expected_file).to exist
223
+ end
224
+ end
225
+
226
+ # This behavior does two things:
227
+ # * ensures that Chef Zero uses our hacked version number
228
+ # * works around external dependencies (e.g., using `git` in backticks)
229
+ # in metadata.rb issue
230
+ it "writes metadata.json in the exported cookbook, removing metadata.rb" do
231
+ metadata_json_path = File.join(exported_cookbook_root, "metadata.json")
232
+ metadata_json = FFI_Yajl::Parser.parse(IO.read(metadata_json_path))
233
+ expect(metadata_json["version"]).to eq("70567763561641081.489844270461035.258281553147148")
234
+ end
235
+
236
+ it "copies the policyfile lock in data item format to data_bags/policyfiles" do
237
+ data_bag_item_path = File.join(export_dir, "data_bags", "policyfiles", "install-example-local.json")
238
+ data_item_json = FFI_Yajl::Parser.parse(IO.read(data_bag_item_path))
239
+ expect(data_item_json["id"]).to eq("install-example-local")
240
+ end
241
+
242
+ it "copies the policyfile lock in standard format to Policyfile.lock.json" do
243
+ policyfile_lock_path = File.join(export_dir, "Policyfile.lock.json")
244
+ policyfile_lock_data = FFI_Yajl::Parser.parse(IO.read(policyfile_lock_path))
245
+ expected_lock_data = export_service.policyfile_lock.to_lock
246
+
247
+ # stringify keys in source_options
248
+ path = expected_lock_data["cookbook_locks"]["local-cookbook"]["source_options"].delete(:path)
249
+ expected_lock_data["cookbook_locks"]["local-cookbook"]["source_options"]["path"] = path
250
+
251
+ expect(policyfile_lock_data).to eq(expected_lock_data)
252
+ end
253
+
254
+ it "creates a working local mode configuration file" do
255
+ expected_config_text = <<-CONFIG
256
+ ### Chef Client Configuration ###
257
+ # The settings in this file will configure chef to apply the exported policy in
258
+ # this directory. To use it, run:
259
+ #
260
+ # chef-client -c client.rb -z
261
+ #
262
+
263
+ use_policyfile true
264
+
265
+ # compatibility mode settings are used because chef-zero doesn't yet support
266
+ # native mode:
267
+ deployment_group 'install-example-local'
268
+ versioned_cookbooks true
269
+ policy_document_native_api false
270
+
271
+ CONFIG
272
+ config_path = File.join(export_dir, "client.rb")
273
+ expect(File).to exist(config_path)
274
+ expect(IO.read(config_path)).to eq(expected_config_text)
275
+ end
276
+
277
+ end
278
+
279
+ context "when the export dir is empty" do
280
+
281
+ include_examples "successful_export"
282
+ end
283
+
284
+ context "When an error occurs creating the export" do
285
+
286
+ before do
287
+ allow(export_service.policyfile_lock).to receive(:validate_cookbooks!).and_return(true)
288
+ expect(export_service).to receive(:create_repo_structure).
289
+ and_raise(Errno::EACCES.new("Permission denied @ rb_sysopen - /etc/foobarbaz.txt"))
290
+ end
291
+
292
+ it "wraps the error in a custom error class" do
293
+ message = "Failed to export policy (in #{expanded_policyfile_path}) to #{export_dir}"
294
+ expect { export_service.run }.to raise_error(ChefDK::PolicyfileExportRepoError, message)
295
+ end
296
+
297
+ end
298
+
299
+ context "When the export dir has non-conflicting content" do
300
+
301
+ let(:file_in_export_dir) { File.join(export_dir, "some_random_cruft") }
302
+
303
+ let(:extra_data_bag_dir) { File.join(export_dir, "data_bags", "extraneous") }
304
+
305
+ let(:extra_data_bag_item) { File.join(extra_data_bag_dir, "an_item.json") }
306
+
307
+ before do
308
+ FileUtils.mkdir_p(export_dir)
309
+ File.open(file_in_export_dir, "wb+") { |f| f.print "some random cruft" }
310
+ FileUtils.mkdir_p(extra_data_bag_dir)
311
+ File.open(extra_data_bag_item, "wb+") { |f| f.print "some random cruft" }
312
+ end
313
+
314
+ it "ignores the non-conflicting content and exports" do
315
+ expect(File).to exist(file_in_export_dir)
316
+ expect(File).to exist(extra_data_bag_item)
317
+
318
+ expect(File).to be_directory(File.join(export_dir, "cookbooks"))
319
+ expect(File).to be_directory(File.join(export_dir, "data_bags"))
320
+ end
321
+
322
+ include_examples "successful_export"
323
+
324
+ end
325
+
326
+ context "When the export dir has conflicting content" do
327
+
328
+ let(:non_conflicting_file_in_export_dir) { File.join(export_dir, "some_random_cruft") }
329
+
330
+ let(:cookbooks_dir) { File.join(export_dir, "cookbooks") }
331
+
332
+ let(:file_in_cookbooks_dir) { File.join(cookbooks_dir, "some_random_cruft") }
333
+
334
+ let(:policyfiles_data_bag_dir) { File.join(export_dir, "data_bags", "policyfiles") }
335
+
336
+ let(:extra_policyfile_data_item) { File.join(policyfiles_data_bag_dir, "leftover-policy.json") }
337
+
338
+ let(:conflicting_policyfile_lock) { File.join(export_dir, "Policyfile.lock.json") }
339
+
340
+ before do
341
+ FileUtils.mkdir_p(export_dir)
342
+ FileUtils.mkdir_p(cookbooks_dir)
343
+ FileUtils.mkdir_p(policyfiles_data_bag_dir)
344
+ File.open(non_conflicting_file_in_export_dir, "wb+") { |f| f.print "some random cruft" }
345
+ File.open(file_in_cookbooks_dir, "wb+") { |f| f.print "some random cruft" }
346
+ File.open(extra_policyfile_data_item, "wb+") { |f| f.print "some random cruft" }
347
+ File.open(conflicting_policyfile_lock, "wb+") { |f| f.print "some random cruft" }
348
+ end
349
+
350
+ it "raises a PolicyfileExportRepoError" do
351
+ message = "Export dir (#{export_dir}) not clean. Refusing to export. (Conflicting files: #{file_in_cookbooks_dir}, #{extra_policyfile_data_item}, #{conflicting_policyfile_lock})"
352
+ expect { export_service.run }.to raise_error(ChefDK::ExportDirNotEmpty, message)
353
+ expect(File).to exist(non_conflicting_file_in_export_dir)
354
+ expect(File).to exist(file_in_cookbooks_dir)
355
+ expect(File).to exist(extra_policyfile_data_item)
356
+ end
357
+
358
+ context "and the force option is set" do
359
+
360
+ let(:force_export) { true }
361
+
362
+ it "clears the export dir and exports" do
363
+ export_service.run
364
+
365
+ expect(File).to_not exist(file_in_cookbooks_dir)
366
+ expect(File).to_not exist(extra_policyfile_data_item)
367
+
368
+ expect(File).to exist(non_conflicting_file_in_export_dir)
369
+
370
+ expect(File).to be_directory(File.join(export_dir, "cookbooks"))
371
+ expect(File).to be_directory(File.join(export_dir, "data_bags"))
372
+ end
373
+
374
+ end
375
+
376
+ end # When the export dir has conflicting content
377
+
378
+ context "when archive mode is enabled" do
379
+
380
+ let(:archive) { true }
381
+
382
+ let(:expected_archive_path) do
383
+ File.join(export_dir, "install-example-60e5ad638dce219d8f87d589463ec4a9884007ba5e2adbb4c0a7021d67204f1a.tgz")
384
+ end
385
+
386
+ it "exports the repo as a tgz archive" do
387
+ expect(File).to exist(expected_archive_path)
388
+ end
389
+
390
+ include_examples "successful_export" do
391
+
392
+ # explode the tarball so the assertions can find the files
393
+ before do
394
+ Zlib::GzipReader.open(expected_archive_path) do |gz_file|
395
+ tar = Archive::Tar::Minitar::Input.new(gz_file)
396
+ tar.each do |e|
397
+ tar.extract_entry(export_dir, e)
398
+ end
399
+ end
400
+ end
401
+
402
+ end
403
+
404
+ context "when the target dir has a cookbooks or data_bags dir" do
405
+
406
+ let(:cookbooks_dir) { File.join(export_dir, "cookbooks") }
407
+
408
+ let(:file_in_cookbooks_dir) { File.join(cookbooks_dir, "some_random_cruft") }
409
+
410
+ let(:policyfiles_data_bag_dir) { File.join(export_dir, "data_bags", "policyfiles") }
411
+
412
+ let(:extra_policyfile_data_item) { File.join(policyfiles_data_bag_dir, "leftover-policy.json") }
413
+
414
+ before do
415
+ FileUtils.mkdir_p(export_dir)
416
+ FileUtils.mkdir_p(cookbooks_dir)
417
+ FileUtils.mkdir_p(policyfiles_data_bag_dir)
418
+ File.open(file_in_cookbooks_dir, "wb+") { |f| f.print "some random cruft" }
419
+ File.open(extra_policyfile_data_item, "wb+") { |f| f.print "some random cruft" }
420
+ end
421
+
422
+ it "exports successfully" do
423
+ expect { export_service.run }.to_not raise_error
424
+ expect(File).to exist(expected_archive_path)
425
+ end
426
+
427
+ end
428
+
429
+ end # when archive mode is enabled
430
+
431
+ end # copying the cookbooks to the export dir
432
+ end
433
+
434
+ end
435
+
436
+
437
+
438
+ end
439
+