chef-cli 1.0.3

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 (372) hide show
  1. checksums.yaml +7 -0
  2. data/Gemfile +32 -0
  3. data/LICENSE +201 -0
  4. data/Rakefile +70 -0
  5. data/bin/chef +25 -0
  6. data/chef-cli.gemspec +53 -0
  7. data/lib/chef-cli.rb +19 -0
  8. data/lib/chef-cli/authenticated_http.rb +22 -0
  9. data/lib/chef-cli/builtin_commands.rb +62 -0
  10. data/lib/chef-cli/chef_runner.rb +114 -0
  11. data/lib/chef-cli/chef_server_api_multi.rb +73 -0
  12. data/lib/chef-cli/cli.rb +206 -0
  13. data/lib/chef-cli/command/base.rb +89 -0
  14. data/lib/chef-cli/command/clean_policy_cookbooks.rb +115 -0
  15. data/lib/chef-cli/command/clean_policy_revisions.rb +112 -0
  16. data/lib/chef-cli/command/delete_policy.rb +121 -0
  17. data/lib/chef-cli/command/delete_policy_group.rb +121 -0
  18. data/lib/chef-cli/command/describe_cookbook.rb +98 -0
  19. data/lib/chef-cli/command/diff.rb +316 -0
  20. data/lib/chef-cli/command/env.rb +99 -0
  21. data/lib/chef-cli/command/exec.rb +45 -0
  22. data/lib/chef-cli/command/export.rb +156 -0
  23. data/lib/chef-cli/command/gem.rb +48 -0
  24. data/lib/chef-cli/command/generate.rb +123 -0
  25. data/lib/chef-cli/command/generator_commands.rb +83 -0
  26. data/lib/chef-cli/command/generator_commands/attribute.rb +37 -0
  27. data/lib/chef-cli/command/generator_commands/base.rb +157 -0
  28. data/lib/chef-cli/command/generator_commands/build_cookbook.rb +126 -0
  29. data/lib/chef-cli/command/generator_commands/chef_exts/generator_desc_resource.rb +40 -0
  30. data/lib/chef-cli/command/generator_commands/chef_exts/quieter_doc_formatter.rb +38 -0
  31. data/lib/chef-cli/command/generator_commands/chef_exts/recipe_dsl_ext.rb +39 -0
  32. data/lib/chef-cli/command/generator_commands/cookbook.rb +251 -0
  33. data/lib/chef-cli/command/generator_commands/cookbook_code_file.rb +100 -0
  34. data/lib/chef-cli/command/generator_commands/cookbook_file.rb +46 -0
  35. data/lib/chef-cli/command/generator_commands/generator_generator.rb +175 -0
  36. data/lib/chef-cli/command/generator_commands/helpers.rb +37 -0
  37. data/lib/chef-cli/command/generator_commands/policyfile.rb +125 -0
  38. data/lib/chef-cli/command/generator_commands/recipe.rb +37 -0
  39. data/lib/chef-cli/command/generator_commands/repo.rb +140 -0
  40. data/lib/chef-cli/command/generator_commands/resource.rb +37 -0
  41. data/lib/chef-cli/command/generator_commands/template.rb +47 -0
  42. data/lib/chef-cli/command/install.rb +121 -0
  43. data/lib/chef-cli/command/provision.rb +38 -0
  44. data/lib/chef-cli/command/push.rb +118 -0
  45. data/lib/chef-cli/command/push_archive.rb +126 -0
  46. data/lib/chef-cli/command/shell_init.rb +185 -0
  47. data/lib/chef-cli/command/show_policy.rb +164 -0
  48. data/lib/chef-cli/command/undelete.rb +155 -0
  49. data/lib/chef-cli/command/update.rb +140 -0
  50. data/lib/chef-cli/command/verify.rb +548 -0
  51. data/lib/chef-cli/commands_map.rb +113 -0
  52. data/lib/chef-cli/completions/bash.sh.erb +5 -0
  53. data/lib/chef-cli/completions/chef.fish.erb +10 -0
  54. data/lib/chef-cli/completions/zsh.zsh.erb +21 -0
  55. data/lib/chef-cli/component_test.rb +226 -0
  56. data/lib/chef-cli/configurable.rb +88 -0
  57. data/lib/chef-cli/cookbook_metadata.rb +45 -0
  58. data/lib/chef-cli/cookbook_omnifetch.rb +32 -0
  59. data/lib/chef-cli/cookbook_profiler/git.rb +152 -0
  60. data/lib/chef-cli/cookbook_profiler/identifiers.rb +72 -0
  61. data/lib/chef-cli/cookbook_profiler/null_scm.rb +31 -0
  62. data/lib/chef-cli/dist.rb +31 -0
  63. data/lib/chef-cli/exceptions.rb +153 -0
  64. data/lib/chef-cli/generator.rb +165 -0
  65. data/lib/chef-cli/helpers.rb +170 -0
  66. data/lib/chef-cli/pager.rb +104 -0
  67. data/lib/chef-cli/policyfile/artifactory_cookbook_source.rb +102 -0
  68. data/lib/chef-cli/policyfile/attribute_merge_checker.rb +110 -0
  69. data/lib/chef-cli/policyfile/chef_repo_cookbook_source.rb +138 -0
  70. data/lib/chef-cli/policyfile/chef_server_cookbook_source.rb +99 -0
  71. data/lib/chef-cli/policyfile/chef_server_lock_fetcher.rb +167 -0
  72. data/lib/chef-cli/policyfile/community_cookbook_source.rb +95 -0
  73. data/lib/chef-cli/policyfile/comparison_base.rb +123 -0
  74. data/lib/chef-cli/policyfile/cookbook_location_specification.rb +154 -0
  75. data/lib/chef-cli/policyfile/cookbook_locks.rb +466 -0
  76. data/lib/chef-cli/policyfile/cookbook_sources.rb +23 -0
  77. data/lib/chef-cli/policyfile/delivery_supermarket_source.rb +89 -0
  78. data/lib/chef-cli/policyfile/differ.rb +263 -0
  79. data/lib/chef-cli/policyfile/dsl.rb +288 -0
  80. data/lib/chef-cli/policyfile/git_lock_fetcher.rb +265 -0
  81. data/lib/chef-cli/policyfile/included_policies_cookbook_source.rb +156 -0
  82. data/lib/chef-cli/policyfile/lister.rb +229 -0
  83. data/lib/chef-cli/policyfile/local_lock_fetcher.rb +132 -0
  84. data/lib/chef-cli/policyfile/lock_applier.rb +80 -0
  85. data/lib/chef-cli/policyfile/lock_fetcher_mixin.rb +37 -0
  86. data/lib/chef-cli/policyfile/null_cookbook_source.rb +49 -0
  87. data/lib/chef-cli/policyfile/policyfile_location_specification.rb +128 -0
  88. data/lib/chef-cli/policyfile/read_cookbook_for_compat_mode_upload.rb +124 -0
  89. data/lib/chef-cli/policyfile/remote_lock_fetcher.rb +108 -0
  90. data/lib/chef-cli/policyfile/reports/install.rb +69 -0
  91. data/lib/chef-cli/policyfile/reports/table_printer.rb +57 -0
  92. data/lib/chef-cli/policyfile/reports/upload.rb +70 -0
  93. data/lib/chef-cli/policyfile/solution_dependencies.rb +311 -0
  94. data/lib/chef-cli/policyfile/source_uri.rb +57 -0
  95. data/lib/chef-cli/policyfile/storage_config.rb +112 -0
  96. data/lib/chef-cli/policyfile/undo_record.rb +139 -0
  97. data/lib/chef-cli/policyfile/undo_stack.rb +128 -0
  98. data/lib/chef-cli/policyfile/uploader.rb +222 -0
  99. data/lib/chef-cli/policyfile_compiler.rb +528 -0
  100. data/lib/chef-cli/policyfile_lock.rb +581 -0
  101. data/lib/chef-cli/policyfile_services/clean_policies.rb +95 -0
  102. data/lib/chef-cli/policyfile_services/clean_policy_cookbooks.rb +123 -0
  103. data/lib/chef-cli/policyfile_services/export_repo.rb +419 -0
  104. data/lib/chef-cli/policyfile_services/install.rb +167 -0
  105. data/lib/chef-cli/policyfile_services/push.rb +112 -0
  106. data/lib/chef-cli/policyfile_services/push_archive.rb +164 -0
  107. data/lib/chef-cli/policyfile_services/rm_policy.rb +141 -0
  108. data/lib/chef-cli/policyfile_services/rm_policy_group.rb +85 -0
  109. data/lib/chef-cli/policyfile_services/show_policy.rb +234 -0
  110. data/lib/chef-cli/policyfile_services/undelete.rb +108 -0
  111. data/lib/chef-cli/policyfile_services/update_attributes.rb +110 -0
  112. data/lib/chef-cli/service_exception_inspectors.rb +24 -0
  113. data/lib/chef-cli/service_exception_inspectors/base.rb +39 -0
  114. data/lib/chef-cli/service_exception_inspectors/http.rb +119 -0
  115. data/lib/chef-cli/service_exceptions.rb +142 -0
  116. data/lib/chef-cli/shell_out.rb +36 -0
  117. data/lib/chef-cli/skeletons/code_generator/files/default/Berksfile +3 -0
  118. data/lib/chef-cli/skeletons/code_generator/files/default/build_cookbook/README.md +146 -0
  119. data/lib/chef-cli/skeletons/code_generator/files/default/build_cookbook/kitchen.yml +21 -0
  120. data/lib/chef-cli/skeletons/code_generator/files/default/build_cookbook/test-fixture-recipe.rb +8 -0
  121. data/lib/chef-cli/skeletons/code_generator/files/default/chefignore +110 -0
  122. data/lib/chef-cli/skeletons/code_generator/files/default/cookbook_readmes/README-policy.md +9 -0
  123. data/lib/chef-cli/skeletons/code_generator/files/default/cookbook_readmes/README.md +66 -0
  124. data/lib/chef-cli/skeletons/code_generator/files/default/delivery-config.json +17 -0
  125. data/lib/chef-cli/skeletons/code_generator/files/default/delivery-project.toml +34 -0
  126. data/lib/chef-cli/skeletons/code_generator/files/default/gitignore +22 -0
  127. data/lib/chef-cli/skeletons/code_generator/files/default/repo/README.md +20 -0
  128. data/lib/chef-cli/skeletons/code_generator/files/default/repo/cookbooks/example/README.md +27 -0
  129. data/lib/chef-cli/skeletons/code_generator/files/default/repo/cookbooks/example/attributes/default.rb +7 -0
  130. data/lib/chef-cli/skeletons/code_generator/files/default/repo/cookbooks/example/metadata.rb +6 -0
  131. data/lib/chef-cli/skeletons/code_generator/files/default/repo/cookbooks/example/recipes/default.rb +8 -0
  132. data/lib/chef-cli/skeletons/code_generator/files/default/repo/data_bags/README.md +56 -0
  133. data/lib/chef-cli/skeletons/code_generator/files/default/repo/data_bags/example/example_item.json +4 -0
  134. data/lib/chef-cli/skeletons/code_generator/files/default/repo/dot-chef-repo.txt +6 -0
  135. data/lib/chef-cli/skeletons/code_generator/files/default/repo/environments/README.md +9 -0
  136. data/lib/chef-cli/skeletons/code_generator/files/default/repo/environments/example.json +13 -0
  137. data/lib/chef-cli/skeletons/code_generator/files/default/repo/policyfiles/README.md +24 -0
  138. data/lib/chef-cli/skeletons/code_generator/files/default/repo/roles/README.md +9 -0
  139. data/lib/chef-cli/skeletons/code_generator/files/default/repo/roles/example.json +13 -0
  140. data/lib/chef-cli/skeletons/code_generator/files/default/spec_helper.rb +2 -0
  141. data/lib/chef-cli/skeletons/code_generator/files/default/spec_helper_policyfile.rb +2 -0
  142. data/lib/chef-cli/skeletons/code_generator/metadata.rb +8 -0
  143. data/lib/chef-cli/skeletons/code_generator/recipes/attribute.rb +11 -0
  144. data/lib/chef-cli/skeletons/code_generator/recipes/build_cookbook.rb +175 -0
  145. data/lib/chef-cli/skeletons/code_generator/recipes/cookbook.rb +167 -0
  146. data/lib/chef-cli/skeletons/code_generator/recipes/cookbook_file.rb +23 -0
  147. data/lib/chef-cli/skeletons/code_generator/recipes/helpers.rb +19 -0
  148. data/lib/chef-cli/skeletons/code_generator/recipes/policyfile.rb +7 -0
  149. data/lib/chef-cli/skeletons/code_generator/recipes/recipe.rb +50 -0
  150. data/lib/chef-cli/skeletons/code_generator/recipes/repo.rb +71 -0
  151. data/lib/chef-cli/skeletons/code_generator/recipes/resource.rb +12 -0
  152. data/lib/chef-cli/skeletons/code_generator/recipes/template.rb +31 -0
  153. data/lib/chef-cli/skeletons/code_generator/templates/default/CHANGELOG.md.erb +11 -0
  154. data/lib/chef-cli/skeletons/code_generator/templates/default/LICENSE.all_rights.erb +3 -0
  155. data/lib/chef-cli/skeletons/code_generator/templates/default/LICENSE.apachev2.erb +201 -0
  156. data/lib/chef-cli/skeletons/code_generator/templates/default/LICENSE.gplv2.erb +339 -0
  157. data/lib/chef-cli/skeletons/code_generator/templates/default/LICENSE.gplv3.erb +674 -0
  158. data/lib/chef-cli/skeletons/code_generator/templates/default/LICENSE.mit.erb +21 -0
  159. data/lib/chef-cli/skeletons/code_generator/templates/default/Policyfile.rb.erb +25 -0
  160. data/lib/chef-cli/skeletons/code_generator/templates/default/README.md.erb +4 -0
  161. data/lib/chef-cli/skeletons/code_generator/templates/default/attribute.rb.erb +0 -0
  162. data/lib/chef-cli/skeletons/code_generator/templates/default/build_cookbook/Berksfile.erb +7 -0
  163. data/lib/chef-cli/skeletons/code_generator/templates/default/build_cookbook/metadata.rb.erb +10 -0
  164. data/lib/chef-cli/skeletons/code_generator/templates/default/build_cookbook/recipe.rb.erb +9 -0
  165. data/lib/chef-cli/skeletons/code_generator/templates/default/cookbook_file.erb +0 -0
  166. data/lib/chef-cli/skeletons/code_generator/templates/default/helpers.rb.erb +39 -0
  167. data/lib/chef-cli/skeletons/code_generator/templates/default/inspec_default_test.rb.erb +16 -0
  168. data/lib/chef-cli/skeletons/code_generator/templates/default/kitchen.yml.erb +38 -0
  169. data/lib/chef-cli/skeletons/code_generator/templates/default/kitchen_dokken.yml.erb +36 -0
  170. data/lib/chef-cli/skeletons/code_generator/templates/default/kitchen_policyfile.yml.erb +32 -0
  171. data/lib/chef-cli/skeletons/code_generator/templates/default/metadata.rb.erb +20 -0
  172. data/lib/chef-cli/skeletons/code_generator/templates/default/recipe.rb.erb +5 -0
  173. data/lib/chef-cli/skeletons/code_generator/templates/default/recipe_spec.rb.erb +29 -0
  174. data/lib/chef-cli/skeletons/code_generator/templates/default/repo/gitignore.erb +128 -0
  175. data/lib/chef-cli/skeletons/code_generator/templates/default/resource.rb.erb +1 -0
  176. data/lib/chef-cli/skeletons/code_generator/templates/default/template.erb +0 -0
  177. data/lib/chef-cli/ui.rb +57 -0
  178. data/lib/chef-cli/version.rb +20 -0
  179. data/lib/kitchen/provisioner/policyfile_zero.rb +195 -0
  180. data/spec/shared/a_file_generator.rb +125 -0
  181. data/spec/shared/a_generated_file.rb +12 -0
  182. data/spec/shared/command_with_ui_object.rb +11 -0
  183. data/spec/shared/custom_generator_cookbook.rb +136 -0
  184. data/spec/shared/fixture_cookbook_checksums.rb +46 -0
  185. data/spec/shared/setup_git_committer_config.rb +54 -0
  186. data/spec/shared/setup_git_cookbooks.rb +53 -0
  187. data/spec/spec_helper.rb +51 -0
  188. data/spec/test_helpers.rb +84 -0
  189. data/spec/unit/chef_runner_spec.rb +139 -0
  190. data/spec/unit/chef_server_api_multi_spec.rb +120 -0
  191. data/spec/unit/cli_spec.rb +375 -0
  192. data/spec/unit/command/base_spec.rb +195 -0
  193. data/spec/unit/command/clean_policy_cookbooks_spec.rb +180 -0
  194. data/spec/unit/command/clean_policy_revisions_spec.rb +180 -0
  195. data/spec/unit/command/delete_policy_group_spec.rb +206 -0
  196. data/spec/unit/command/delete_policy_spec.rb +206 -0
  197. data/spec/unit/command/diff_spec.rb +311 -0
  198. data/spec/unit/command/env_spec.rb +86 -0
  199. data/spec/unit/command/exec_spec.rb +178 -0
  200. data/spec/unit/command/export_spec.rb +199 -0
  201. data/spec/unit/command/generate_spec.rb +142 -0
  202. data/spec/unit/command/generator_commands/attribute_spec.rb +31 -0
  203. data/spec/unit/command/generator_commands/base_spec.rb +180 -0
  204. data/spec/unit/command/generator_commands/build_cookbook_spec.rb +377 -0
  205. data/spec/unit/command/generator_commands/chef_exts/generator_desc_resource_spec.rb +77 -0
  206. data/spec/unit/command/generator_commands/chef_exts/recipe_dsl_ext_spec.rb +111 -0
  207. data/spec/unit/command/generator_commands/cookbook_file_spec.rb +31 -0
  208. data/spec/unit/command/generator_commands/cookbook_spec.rb +769 -0
  209. data/spec/unit/command/generator_commands/generator_generator_spec.rb +227 -0
  210. data/spec/unit/command/generator_commands/helpers_spec.rb +31 -0
  211. data/spec/unit/command/generator_commands/policyfile_spec.rb +223 -0
  212. data/spec/unit/command/generator_commands/recipe_spec.rb +37 -0
  213. data/spec/unit/command/generator_commands/repo_spec.rb +374 -0
  214. data/spec/unit/command/generator_commands/resource_spec.rb +31 -0
  215. data/spec/unit/command/generator_commands/template_spec.rb +31 -0
  216. data/spec/unit/command/install_spec.rb +179 -0
  217. data/spec/unit/command/push_archive_spec.rb +153 -0
  218. data/spec/unit/command/push_spec.rb +198 -0
  219. data/spec/unit/command/shell_init_spec.rb +339 -0
  220. data/spec/unit/command/show_policy_spec.rb +234 -0
  221. data/spec/unit/command/undelete_spec.rb +244 -0
  222. data/spec/unit/command/update_spec.rb +283 -0
  223. data/spec/unit/command/verify_spec.rb +341 -0
  224. data/spec/unit/commands_map_spec.rb +57 -0
  225. data/spec/unit/component_test_spec.rb +128 -0
  226. data/spec/unit/configurable_spec.rb +68 -0
  227. data/spec/unit/cookbook_metadata_spec.rb +96 -0
  228. data/spec/unit/cookbook_profiler/git_spec.rb +176 -0
  229. data/spec/unit/cookbook_profiler/identifiers_spec.rb +81 -0
  230. data/spec/unit/fixtures/chef-runner-cookbooks/test_cookbook/recipes/recipe_one.rb +9 -0
  231. data/spec/unit/fixtures/chef-runner-cookbooks/test_cookbook/recipes/recipe_two.rb +9 -0
  232. data/spec/unit/fixtures/command/cli_test_command.rb +26 -0
  233. data/spec/unit/fixtures/command/explicit_path_example.rb +7 -0
  234. data/spec/unit/fixtures/configurable/test_config_loader.rb +5 -0
  235. data/spec/unit/fixtures/configurable/test_configurable.rb +10 -0
  236. data/spec/unit/fixtures/cookbook_cache/baz-f59ee7a5bca6a4e606b67f7f856b768d847c39bb/.kitchen.yml +16 -0
  237. data/spec/unit/fixtures/cookbook_cache/baz-f59ee7a5bca6a4e606b67f7f856b768d847c39bb/Berksfile +3 -0
  238. data/spec/unit/fixtures/cookbook_cache/baz-f59ee7a5bca6a4e606b67f7f856b768d847c39bb/README.md +4 -0
  239. data/spec/unit/fixtures/cookbook_cache/baz-f59ee7a5bca6a4e606b67f7f856b768d847c39bb/chefignore +96 -0
  240. data/spec/unit/fixtures/cookbook_cache/baz-f59ee7a5bca6a4e606b67f7f856b768d847c39bb/metadata.rb +8 -0
  241. data/spec/unit/fixtures/cookbook_cache/baz-f59ee7a5bca6a4e606b67f7f856b768d847c39bb/recipes/default.rb +8 -0
  242. data/spec/unit/fixtures/cookbook_cache/dep_of_bar-1.2.3/.kitchen.yml +16 -0
  243. data/spec/unit/fixtures/cookbook_cache/dep_of_bar-1.2.3/Berksfile +3 -0
  244. data/spec/unit/fixtures/cookbook_cache/dep_of_bar-1.2.3/README.md +4 -0
  245. data/spec/unit/fixtures/cookbook_cache/dep_of_bar-1.2.3/chefignore +96 -0
  246. data/spec/unit/fixtures/cookbook_cache/dep_of_bar-1.2.3/metadata.rb +8 -0
  247. data/spec/unit/fixtures/cookbook_cache/dep_of_bar-1.2.3/recipes/default.rb +8 -0
  248. data/spec/unit/fixtures/cookbook_cache/foo-1.0.0/Berksfile +3 -0
  249. data/spec/unit/fixtures/cookbook_cache/foo-1.0.0/README.md +4 -0
  250. data/spec/unit/fixtures/cookbook_cache/foo-1.0.0/chefignore +96 -0
  251. data/spec/unit/fixtures/cookbook_cache/foo-1.0.0/kitchen.yml +16 -0
  252. data/spec/unit/fixtures/cookbook_cache/foo-1.0.0/metadata.rb +8 -0
  253. data/spec/unit/fixtures/cookbook_cache/foo-1.0.0/recipes/default.rb +8 -0
  254. data/spec/unit/fixtures/cookbooks_api/chef_server_universe.json +56 -0
  255. data/spec/unit/fixtures/cookbooks_api/pruned_chef_server_universe.json +30 -0
  256. data/spec/unit/fixtures/cookbooks_api/pruned_small_universe.json +1322 -0
  257. data/spec/unit/fixtures/cookbooks_api/small_universe.json +2987 -0
  258. data/spec/unit/fixtures/cookbooks_api/universe.json +1 -0
  259. data/spec/unit/fixtures/cookbooks_api/update_fixtures.rb +33 -0
  260. data/spec/unit/fixtures/dev_cookbooks/README.md +16 -0
  261. data/spec/unit/fixtures/dev_cookbooks/bar-cookbook.gitbundle +0 -0
  262. data/spec/unit/fixtures/eg_omnibus_dir/missing_apps/bin/.keep +0 -0
  263. data/spec/unit/fixtures/eg_omnibus_dir/missing_apps/embedded/.keep +0 -0
  264. data/spec/unit/fixtures/eg_omnibus_dir/missing_apps/embedded/bin/.keep +0 -0
  265. data/spec/unit/fixtures/eg_omnibus_dir/missing_component/bin/.keep +0 -0
  266. data/spec/unit/fixtures/eg_omnibus_dir/missing_component/embedded/apps/berkshelf/.keep +0 -0
  267. data/spec/unit/fixtures/eg_omnibus_dir/missing_component/embedded/apps/test-kitchen/.keep +0 -0
  268. data/spec/unit/fixtures/eg_omnibus_dir/missing_component/embedded/bin/.keep +0 -0
  269. data/spec/unit/fixtures/eg_omnibus_dir/valid/bin/.keep +0 -0
  270. data/spec/unit/fixtures/eg_omnibus_dir/valid/embedded/apps/berkshelf/integration_test +2 -0
  271. data/spec/unit/fixtures/eg_omnibus_dir/valid/embedded/apps/berkshelf/verify_me +5 -0
  272. data/spec/unit/fixtures/eg_omnibus_dir/valid/embedded/apps/chef-dk/.keep +0 -0
  273. data/spec/unit/fixtures/eg_omnibus_dir/valid/embedded/apps/chef/verify_me +3 -0
  274. data/spec/unit/fixtures/eg_omnibus_dir/valid/embedded/apps/test-kitchen/verify_me +2 -0
  275. data/spec/unit/fixtures/eg_omnibus_dir/valid/embedded/bin/.keep +0 -0
  276. data/spec/unit/fixtures/example_app/Policyfile.rb +0 -0
  277. data/spec/unit/fixtures/example_cookbook/.gitignore +17 -0
  278. data/spec/unit/fixtures/example_cookbook/.kitchen.yml +16 -0
  279. data/spec/unit/fixtures/example_cookbook/Berksfile +3 -0
  280. data/spec/unit/fixtures/example_cookbook/README.md +4 -0
  281. data/spec/unit/fixtures/example_cookbook/chefignore +96 -0
  282. data/spec/unit/fixtures/example_cookbook/metadata.rb +8 -0
  283. data/spec/unit/fixtures/example_cookbook/recipes/default.rb +8 -0
  284. data/spec/unit/fixtures/example_cookbook_metadata_json_only/.gitignore +17 -0
  285. data/spec/unit/fixtures/example_cookbook_metadata_json_only/.kitchen.yml +16 -0
  286. data/spec/unit/fixtures/example_cookbook_metadata_json_only/Berksfile +3 -0
  287. data/spec/unit/fixtures/example_cookbook_metadata_json_only/README.md +4 -0
  288. data/spec/unit/fixtures/example_cookbook_metadata_json_only/chefignore +96 -0
  289. data/spec/unit/fixtures/example_cookbook_metadata_json_only/metadata.json +5 -0
  290. data/spec/unit/fixtures/example_cookbook_metadata_json_only/recipes/default.rb +8 -0
  291. data/spec/unit/fixtures/example_cookbook_no_metadata/.gitignore +17 -0
  292. data/spec/unit/fixtures/example_cookbook_no_metadata/.kitchen.yml +16 -0
  293. data/spec/unit/fixtures/example_cookbook_no_metadata/Berksfile +3 -0
  294. data/spec/unit/fixtures/example_cookbook_no_metadata/README.md +4 -0
  295. data/spec/unit/fixtures/example_cookbook_no_metadata/chefignore +96 -0
  296. data/spec/unit/fixtures/example_cookbook_no_metadata/recipes/default.rb +8 -0
  297. data/spec/unit/fixtures/local_path_cookbooks/another-local-cookbook/README.md +4 -0
  298. data/spec/unit/fixtures/local_path_cookbooks/another-local-cookbook/chefignore +96 -0
  299. data/spec/unit/fixtures/local_path_cookbooks/another-local-cookbook/metadata.rb +8 -0
  300. data/spec/unit/fixtures/local_path_cookbooks/another-local-cookbook/recipes/default.rb +8 -0
  301. data/spec/unit/fixtures/local_path_cookbooks/cookbook-with-a-dep/Berksfile +3 -0
  302. data/spec/unit/fixtures/local_path_cookbooks/cookbook-with-a-dep/README.md +4 -0
  303. data/spec/unit/fixtures/local_path_cookbooks/cookbook-with-a-dep/chefignore +96 -0
  304. data/spec/unit/fixtures/local_path_cookbooks/cookbook-with-a-dep/metadata.rb +9 -0
  305. data/spec/unit/fixtures/local_path_cookbooks/cookbook-with-a-dep/recipes/default.rb +8 -0
  306. data/spec/unit/fixtures/local_path_cookbooks/local-cookbook/.kitchen.yml +16 -0
  307. data/spec/unit/fixtures/local_path_cookbooks/local-cookbook/Berksfile +3 -0
  308. data/spec/unit/fixtures/local_path_cookbooks/local-cookbook/README.md +4 -0
  309. data/spec/unit/fixtures/local_path_cookbooks/local-cookbook/chefignore +96 -0
  310. data/spec/unit/fixtures/local_path_cookbooks/local-cookbook/extra/extra_file.txt +0 -0
  311. data/spec/unit/fixtures/local_path_cookbooks/local-cookbook/metadata.rb +8 -0
  312. data/spec/unit/fixtures/local_path_cookbooks/local-cookbook/recipes/default.rb +8 -0
  313. data/spec/unit/fixtures/local_path_cookbooks/metadata-missing/README.md +2 -0
  314. data/spec/unit/fixtures/local_path_cookbooks/noignore-f59ee7a5bca6a4e606b67f7f856b768d847c39bb/.kitchen.yml +16 -0
  315. data/spec/unit/fixtures/local_path_cookbooks/noignore-f59ee7a5bca6a4e606b67f7f856b768d847c39bb/README.md +4 -0
  316. data/spec/unit/fixtures/local_path_cookbooks/noignore-f59ee7a5bca6a4e606b67f7f856b768d847c39bb/metadata.rb +8 -0
  317. data/spec/unit/fixtures/local_path_cookbooks/noignore-f59ee7a5bca6a4e606b67f7f856b768d847c39bb/recipes/default.rb +8 -0
  318. data/spec/unit/generator_spec.rb +119 -0
  319. data/spec/unit/pager_spec.rb +117 -0
  320. data/spec/unit/policyfile/artifactory_cookbook_source_spec.rb +59 -0
  321. data/spec/unit/policyfile/attribute_merge_checker_spec.rb +80 -0
  322. data/spec/unit/policyfile/chef_repo_cookbook_source_spec.rb +93 -0
  323. data/spec/unit/policyfile/chef_server_cookbook_source_spec.rb +55 -0
  324. data/spec/unit/policyfile/chef_server_lock_fetcher_spec.rb +161 -0
  325. data/spec/unit/policyfile/community_cookbook_source_spec.rb +83 -0
  326. data/spec/unit/policyfile/comparison_base_spec.rb +340 -0
  327. data/spec/unit/policyfile/cookbook_location_specification_spec.rb +347 -0
  328. data/spec/unit/policyfile/cookbook_locks_spec.rb +527 -0
  329. data/spec/unit/policyfile/delivery_supermarket_source_spec.rb +129 -0
  330. data/spec/unit/policyfile/differ_spec.rb +686 -0
  331. data/spec/unit/policyfile/git_lock_fetcher_spec.rb +155 -0
  332. data/spec/unit/policyfile/included_policies_cookbook_source_spec.rb +242 -0
  333. data/spec/unit/policyfile/lister_spec.rb +268 -0
  334. data/spec/unit/policyfile/local_lock_fetcher_spec.rb +199 -0
  335. data/spec/unit/policyfile/lock_applier_spec.rb +100 -0
  336. data/spec/unit/policyfile/lock_fetcher_mixin_spec.rb +60 -0
  337. data/spec/unit/policyfile/null_cookbook_source_spec.rb +34 -0
  338. data/spec/unit/policyfile/read_cookbook_for_compat_mode_upload_spec.rb +92 -0
  339. data/spec/unit/policyfile/remote_lock_fetcher_spec.rb +129 -0
  340. data/spec/unit/policyfile/reports/install_spec.rb +114 -0
  341. data/spec/unit/policyfile/reports/upload_spec.rb +94 -0
  342. data/spec/unit/policyfile/solution_dependencies_spec.rb +170 -0
  343. data/spec/unit/policyfile/source_uri_spec.rb +36 -0
  344. data/spec/unit/policyfile/storage_config_spec.rb +180 -0
  345. data/spec/unit/policyfile/undo_record_spec.rb +258 -0
  346. data/spec/unit/policyfile/undo_stack_spec.rb +265 -0
  347. data/spec/unit/policyfile/uploader_spec.rb +410 -0
  348. data/spec/unit/policyfile_demands_spec.rb +1197 -0
  349. data/spec/unit/policyfile_evaluation_spec.rb +628 -0
  350. data/spec/unit/policyfile_includes_dsl_spec.rb +220 -0
  351. data/spec/unit/policyfile_includes_spec.rb +720 -0
  352. data/spec/unit/policyfile_install_with_includes_spec.rb +232 -0
  353. data/spec/unit/policyfile_lock_build_spec.rb +1065 -0
  354. data/spec/unit/policyfile_lock_install_spec.rb +137 -0
  355. data/spec/unit/policyfile_lock_serialization_spec.rb +424 -0
  356. data/spec/unit/policyfile_lock_validation_spec.rb +608 -0
  357. data/spec/unit/policyfile_services/clean_policies_spec.rb +236 -0
  358. data/spec/unit/policyfile_services/clean_policy_cookbooks_spec.rb +272 -0
  359. data/spec/unit/policyfile_services/export_repo_spec.rb +473 -0
  360. data/spec/unit/policyfile_services/install_spec.rb +209 -0
  361. data/spec/unit/policyfile_services/push_archive_spec.rb +359 -0
  362. data/spec/unit/policyfile_services/push_spec.rb +249 -0
  363. data/spec/unit/policyfile_services/rm_policy_group_spec.rb +237 -0
  364. data/spec/unit/policyfile_services/rm_policy_spec.rb +263 -0
  365. data/spec/unit/policyfile_services/show_policy_spec.rb +887 -0
  366. data/spec/unit/policyfile_services/undelete_spec.rb +302 -0
  367. data/spec/unit/policyfile_services/update_attributes_spec.rb +229 -0
  368. data/spec/unit/policyfile_services/update_spec.rb +162 -0
  369. data/spec/unit/service_exception_inspectors/base_spec.rb +41 -0
  370. data/spec/unit/service_exception_inspectors/http_spec.rb +138 -0
  371. data/spec/unit/shell_out_spec.rb +34 -0
  372. metadata +796 -0
@@ -0,0 +1,265 @@
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 "spec_helper"
19
+ require "chef-cli/policyfile/undo_stack"
20
+
21
+ describe ChefCLI::Policyfile::UndoStack do
22
+
23
+ let(:package_home) { File.join(tempdir, "package_home", ".chefcli") }
24
+
25
+ let(:policy_revision) do
26
+ {
27
+ "name" => "appserver",
28
+ "revision_id" => "1111111111111111111111111111111111111111111111111111111111111111",
29
+ }
30
+ end
31
+
32
+ let(:undo_record) do
33
+ ChefCLI::Policyfile::UndoRecord.new.tap do |undo_record|
34
+ undo_record.add_policy_group("preprod")
35
+ undo_record.add_policy_revision("appserver", "preprod", policy_revision)
36
+ end
37
+ end
38
+
39
+ # Probably takes a Chef::Config as an arg?
40
+ subject(:undo_stack) { described_class.new }
41
+
42
+ let(:expected_undo_dir) { File.join(package_home, "undo") }
43
+
44
+ def undo_stack_files
45
+ Dir[File.join(expected_undo_dir, "*")]
46
+ end
47
+
48
+ before do
49
+ clear_tempdir
50
+ allow(ChefCLI::Helpers).to receive(:package_home).and_return(package_home)
51
+ end
52
+
53
+ after(:all) do
54
+ clear_tempdir
55
+ end
56
+
57
+ it "uses package_home to infer the location of the undo directory" do
58
+ expect(undo_stack.undo_dir).to eq(expected_undo_dir)
59
+ end
60
+
61
+ context "when there are no undo records" do
62
+
63
+ it "has zero items" do
64
+ expect(undo_stack.size).to eq(0)
65
+ end
66
+
67
+ it "is empty" do
68
+ expect(undo_stack).to be_empty
69
+ end
70
+
71
+ it "has no items to iterate over" do
72
+ expect { |b| undo_stack.each_with_id(&b) }.to_not yield_control
73
+ end
74
+
75
+ it "has an empty list of undo records" do
76
+ expect(undo_stack.undo_records).to eq([])
77
+ end
78
+
79
+ it "raises an error when attempting to pop an item from the stack" do
80
+ expect { undo_stack.pop }.to raise_error(ChefCLI::CantUndo)
81
+ end
82
+
83
+ describe "pushing an undo record" do
84
+
85
+ before do
86
+ expect(File.exist?(package_home)).to be(false)
87
+ expect(File.exist?(expected_undo_dir)).to be(false)
88
+
89
+ undo_stack.push(undo_record)
90
+ end
91
+
92
+ it "creates the undo directory" do
93
+ expect(File.exist?(package_home)).to be(true)
94
+ expect(File.exist?(expected_undo_dir)).to be(true)
95
+ end
96
+
97
+ it "creates the undo record" do
98
+ expect(undo_stack_files.size).to eq(1)
99
+
100
+ undo_record_json = IO.read(undo_stack_files.first)
101
+ undo_record_data = FFI_Yajl::Parser.parse(undo_record_json)
102
+ expect(undo_record_data).to eq(undo_record.for_serialization)
103
+ end
104
+
105
+ end
106
+
107
+ end
108
+
109
+ context "when there is one undo record" do
110
+
111
+ # `Time.new` is stubbed later on, need to force it to be evaluated before
112
+ # then.
113
+ let!(:start_time) { Time.new }
114
+
115
+ let(:expected_id) { start_time.utc.strftime("%Y%m%d%H%M%S") }
116
+
117
+ let(:missing_id) { (start_time + 1).utc.strftime("%Y%m%d%H%M%S") }
118
+
119
+ before do
120
+ allow(Time).to receive(:new).and_return(start_time)
121
+ undo_stack.push(undo_record)
122
+ end
123
+
124
+ it "creates the item on disk" do
125
+ expect(File).to be_directory(undo_stack.undo_dir)
126
+ expect(undo_stack_files.size).to eq(1)
127
+ end
128
+
129
+ it "has one item" do
130
+ expect(undo_stack.size).to eq(1)
131
+ end
132
+
133
+ it "is not empty" do
134
+ expect(undo_stack).to_not be_empty
135
+ end
136
+
137
+ it "checks whether a record exists by id" do
138
+ expect(undo_stack).to have_id(expected_id)
139
+ expect(undo_stack).to_not have_id(missing_id)
140
+ end
141
+
142
+ it "deletes a record by id" do
143
+ expect(undo_stack.delete(expected_id)).to eq(undo_record)
144
+ end
145
+
146
+ it "deletes a record by id and yields it" do
147
+ expect { |b| undo_stack.delete(expected_id, &b) }.to yield_with_args(undo_record)
148
+ end
149
+
150
+ it "fails to delete a record that doesn't exist" do
151
+ expect { undo_stack.delete(missing_id) }.to raise_error(ChefCLI::UndoRecordNotFound)
152
+ end
153
+
154
+ it "pops the last record" do
155
+ expect(undo_stack.pop).to eq(undo_record)
156
+ end
157
+
158
+ it "pops the last record and yields it" do
159
+ expect { |b| undo_stack.pop(&b) }.to yield_with_args(undo_record)
160
+ end
161
+
162
+ it "iterates over the records" do
163
+ expect { |b| undo_stack.each_with_id(&b) }.to yield_successive_args([expected_id, undo_record])
164
+ end
165
+
166
+ it "has the undo record that was pushed" do
167
+ expect(undo_stack.undo_records.size).to eq(1)
168
+ expect(undo_stack.undo_records).to eq( [ undo_record ] )
169
+ end
170
+
171
+ context "and the record is removed" do
172
+
173
+ let!(:popped_record) { undo_stack.pop }
174
+
175
+ it "has no items" do
176
+ expect(undo_stack_files.size).to eq(0)
177
+ end
178
+
179
+ end
180
+
181
+ end
182
+
183
+ context "when the stack is at the maximum configured size" do
184
+ # `Time.new` is stubbed later on, need to force it to be evaluated before
185
+ # then.
186
+ let!(:start_time) { Time.new }
187
+
188
+ def next_time
189
+ @increment ||= 0
190
+ @increment += 1
191
+
192
+ start_time + @increment
193
+ end
194
+
195
+ def incremented_undo_record(i)
196
+ record = {
197
+ "name" => "appserver",
198
+ "revision_id" => i.to_s * 64,
199
+ }
200
+
201
+ ChefCLI::Policyfile::UndoRecord.new.tap do |undo_record|
202
+ undo_record.description = "delete-policy-group preprod-#{i}"
203
+ undo_record.add_policy_group("preprod-#{i}")
204
+ undo_record.add_policy_revision("appserver", "preprod-#{i}", record)
205
+ end
206
+ end
207
+
208
+ before do
209
+ ## # UndoStack assumes you're not creating more than 1 undo record/second,
210
+ ## # which is reasonable in the real world but won't work here.
211
+ allow(Time).to receive(:new) { next_time }
212
+
213
+ 10.times { |i| undo_stack.push(incremented_undo_record(i)) }
214
+
215
+ expect(undo_stack_files.size).to eq(10)
216
+ end
217
+
218
+ describe "pushing a new undo record" do
219
+ it "does not exceed the maximum size" do
220
+ undo_stack.push(incremented_undo_record(11))
221
+
222
+ expect(undo_stack_files.size).to eq(10)
223
+ end
224
+
225
+ it "removes the oldest record" do
226
+ oldest_record_file = undo_stack_files.sort.first
227
+
228
+ undo_stack.push(incremented_undo_record(11))
229
+
230
+ expect(File.exist?(oldest_record_file)).to be(false)
231
+ end
232
+ end
233
+
234
+ context "when the stack is above maximum configured size" do
235
+
236
+ let(:older_record_path) do
237
+ record_id = (Time.new - (3600 * 24)).utc.strftime("%Y%m%d%H%M%S")
238
+ File.join(expected_undo_dir, record_id)
239
+ end
240
+
241
+ before do
242
+ FileUtils.touch(older_record_path)
243
+ expect(undo_stack_files.size).to eq(11)
244
+ end
245
+
246
+ describe "pushing a new undo record" do
247
+ it "does not exceed the maximum size" do
248
+ undo_stack.push(incremented_undo_record(11))
249
+
250
+ expect(undo_stack_files.size).to eq(10)
251
+ end
252
+
253
+ it "removes the oldest record" do
254
+ oldest_record_file = undo_stack_files.sort.first
255
+
256
+ undo_stack.push(incremented_undo_record(11))
257
+
258
+ expect(File.exist?(oldest_record_file)).to be(false)
259
+ end
260
+ end
261
+
262
+ end
263
+ end
264
+
265
+ end
@@ -0,0 +1,410 @@
1
+ #
2
+ # Copyright:: Copyright (c) 2014-2018 Chef Software Inc.
3
+ # License:: Apache License, Version 2.0
4
+ #
5
+ # Licensed under the Apache License, Version 2.0 (the "License");
6
+ # you may not use this file except in compliance with the License.
7
+ # You may obtain a copy of the License at
8
+ #
9
+ # http://www.apache.org/licenses/LICENSE-2.0
10
+ #
11
+ # Unless required by applicable law or agreed to in writing, software
12
+ # distributed under the License is distributed on an "AS IS" BASIS,
13
+ # WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
14
+ # See the License for the specific language governing permissions and
15
+ # limitations under the License.
16
+ #
17
+
18
+ require "spec_helper"
19
+ require "chef-cli/policyfile/uploader"
20
+
21
+ # We load this here to ensure we get the "verifying doubles" behavior from
22
+ # RSpec. It's not used by Policyfile::Uploader, but it's a collaborator.
23
+ require "chef/server_api"
24
+
25
+ describe ChefCLI::Policyfile::Uploader do
26
+
27
+ let(:policyfile_lock_data) do
28
+ {
29
+ "name" => "example",
30
+ "run_list" => [ "recipe[omnibus::default]" ],
31
+ "revision_id" => "f4b29d87c36de67cbfd9aa3147df77cebf9f719e8c884036b3cf34ba94773ca5",
32
+ "cookbook_locks" => {
33
+ "omnibus" => {
34
+ "version" => "2.2.0",
35
+ "identifier" => "64b3e64306cff223206348e46af545b19032b170",
36
+ "dotted_decimal_identifier" => "28345299219435506.9887234981653237.76628930769264",
37
+ "cache_key" => "omnibus-2cf98f9797cacce9c8688fc4e74858b858e2bc14",
38
+ "origin" => "git@github.com:opscode-cookbooks/omnibus.git",
39
+ "source_options" => {
40
+ "git" => "git@github.com:opscode-cookbooks/omnibus.git",
41
+ "revision" => "2cf98f9797cacce9c8688fc4e74858b858e2bc14",
42
+ "branch" => "master",
43
+ },
44
+ },
45
+ },
46
+ }
47
+ end
48
+
49
+ let(:policyfile_lock) do
50
+ instance_double("ChefCLI::PolicyfileLock", name: "example",
51
+ to_lock: policyfile_lock_data) end
52
+
53
+ let(:policy_group) { "unit-test" }
54
+
55
+ let(:http_client) { instance_double("Chef::ServerAPI") }
56
+
57
+ let(:policy_document_native_api) { false }
58
+
59
+ let(:uploader) do
60
+ described_class.new(policyfile_lock,
61
+ policy_group,
62
+ http_client: http_client,
63
+ policy_document_native_api: policy_document_native_api)
64
+ end
65
+
66
+ let(:policyfile_as_data_bag_item) do
67
+
68
+ policyfile_as_data_bag_item = {
69
+ "id" => "example-unit-test",
70
+ "name" => "data_bag_item_policyfiles_example-unit-test",
71
+ "data_bag" => "policyfiles",
72
+ }
73
+ policyfile_as_data_bag_item["raw_data"] = policyfile_lock_data.dup
74
+ policyfile_as_data_bag_item["raw_data"]["id"] = "example-unit-test"
75
+ policyfile_as_data_bag_item["json_class"] = "Chef::DataBagItem"
76
+ policyfile_as_data_bag_item
77
+ end
78
+
79
+ it "has a lockfile" do
80
+ expect(uploader.policyfile_lock).to eq(policyfile_lock)
81
+ end
82
+
83
+ it "has a policy group" do
84
+ expect(uploader.policy_group).to eq(policy_group)
85
+ end
86
+
87
+ it "has an HTTP client" do
88
+ expect(uploader.http_client).to eq(http_client)
89
+ end
90
+
91
+ describe "uploading policies and cookbooks" do
92
+
93
+ let(:cookbook_locks) { {} }
94
+ let(:cookbook_versions) { {} }
95
+
96
+ before do
97
+ allow(policyfile_lock).to receive(:cookbook_locks).and_return(cookbook_locks)
98
+ end
99
+
100
+ def lock_double(name, identifier, dotted_decimal_id)
101
+ cache_path = "/home/user/cache_path/#{name}"
102
+
103
+ lock = instance_double("ChefCLI::Policyfile::CookbookLock",
104
+ name: name,
105
+ version: "1.0.0",
106
+ identifier: identifier,
107
+ dotted_decimal_identifier: dotted_decimal_id,
108
+ cookbook_path: cache_path)
109
+
110
+ cookbook_version = instance_double("Chef::CookbookVersion",
111
+ name: name,
112
+ identifier: lock.identifier,
113
+ version: dotted_decimal_id)
114
+
115
+ allow(cookbook_version).to receive(:identifier=).with(lock.identifier)
116
+
117
+ allow(ChefCLI::Policyfile::ReadCookbookForCompatModeUpload)
118
+ .to receive(:load)
119
+ .with(name, dotted_decimal_id, cache_path)
120
+ .and_return(cookbook_version)
121
+
122
+ allow(ChefCLI::Policyfile::CookbookLoaderWithChefignore)
123
+ .to receive(:load)
124
+ .with(name, cache_path)
125
+ .and_return(cookbook_version)
126
+
127
+ cookbook_versions[name] = cookbook_version
128
+ cookbook_locks[name] = lock
129
+
130
+ lock
131
+ end
132
+
133
+ shared_examples_for "uploading cookbooks" do
134
+
135
+ describe "uploading cookbooks" do
136
+
137
+ it "enumerates the cookbooks already on the server" do
138
+ expect(http_client).to receive(:get).with(list_cookbooks_url).and_return(existing_cookbook_on_remote)
139
+ expect(uploader.existing_cookbook_on_remote).to eq(existing_cookbook_on_remote)
140
+ end
141
+
142
+ context "with an empty policyfile lock" do
143
+
144
+ it "has an empty list of cookbooks for possible upload" do
145
+ expect(policyfile_lock).to receive(:validate_cookbooks!)
146
+
147
+ expect(uploader.cookbook_versions_for_policy).to eq([])
148
+ end
149
+
150
+ it "has an empty list of cookbooks that need to be uploaded" do
151
+ expect(policyfile_lock).to receive(:validate_cookbooks!)
152
+
153
+ expect(uploader.cookbook_versions_to_upload).to eq([])
154
+ end
155
+
156
+ end
157
+
158
+ context "with a set of cookbooks that don't exist on the server" do
159
+
160
+ before do
161
+ lock_double("my_apache2", "1111111111111111111111111111111111111111", "123.456.789")
162
+ lock_double("my_jenkins", "2222222222222222222222222222222222222222", "321.654.987")
163
+ end
164
+
165
+ it "lists the cookbooks in the lock as possibly needing to be uploaded" do
166
+ expect(policyfile_lock).to receive(:validate_cookbooks!)
167
+
168
+ expected_versions_for_policy = cookbook_versions.keys.map do |cb_name|
169
+ cb = cookbook_versions[cb_name]
170
+ lock = cookbook_locks[cb_name]
171
+ ChefCLI::Policyfile::Uploader::LockedCookbookForUpload.new(cb, lock)
172
+ end
173
+
174
+ expect(uploader.cookbook_versions_for_policy).to eq(expected_versions_for_policy)
175
+ end
176
+
177
+ it "lists all cookbooks in the lock as needing to be uploaded" do
178
+ expect(policyfile_lock).to receive(:validate_cookbooks!)
179
+ expect(http_client).to receive(:get).with(list_cookbooks_url).and_return(existing_cookbook_on_remote)
180
+
181
+ expect(uploader.cookbook_versions_to_upload).to eq(cookbook_versions.values)
182
+ end
183
+
184
+ it "uploads the cookbooks and then the policy" do
185
+ expect(policyfile_lock).to receive(:validate_cookbooks!)
186
+ expect(http_client).to receive(:get).with(list_cookbooks_url).and_return(existing_cookbook_on_remote)
187
+
188
+ cookbook_uploader = instance_double("Chef::CookbookUploader")
189
+ expect(Chef::CookbookUploader).to receive(:new)
190
+ .with(cookbook_versions.values, rest: http_client, policy_mode: policy_document_native_api)
191
+ .and_return(cookbook_uploader)
192
+ expect(cookbook_uploader).to receive(:upload_cookbooks)
193
+
194
+ expect_policyfile_upload
195
+
196
+ uploader.upload
197
+ end
198
+
199
+ end
200
+
201
+ context "with a set of cookbooks where some already exist on the server" do
202
+
203
+ before do
204
+ # These are new:
205
+ lock_double("my_apache2", "1111111111111111111111111111111111111111", "123.456.789")
206
+ lock_double("my_jenkins", "2222222222222222222222222222222222222222", "321.654.987")
207
+
208
+ # Have this one:
209
+ lock_double("build-essential", "571d8ebd02b296fe90b2e4d68754af7e8e185f28", "67369247788170534.26353953100055918.55660493423796")
210
+ end
211
+
212
+ let(:expected_cookbooks_for_upload) do
213
+ [
214
+ cookbook_versions["my_apache2"],
215
+ cookbook_versions["my_jenkins"],
216
+ ]
217
+ end
218
+
219
+ it "lists only cookbooks not on the server as needing to be uploaded" do
220
+ expect(policyfile_lock).to receive(:validate_cookbooks!)
221
+ expect(http_client).to receive(:get).with(list_cookbooks_url).and_return(existing_cookbook_on_remote)
222
+
223
+ expect(uploader.cookbook_versions_to_upload).to eq(expected_cookbooks_for_upload)
224
+ end
225
+
226
+ it "uploads the cookbooks and then the policy" do
227
+ expect(policyfile_lock).to receive(:validate_cookbooks!)
228
+ expect(http_client).to receive(:get).with(list_cookbooks_url).and_return(existing_cookbook_on_remote)
229
+
230
+ cookbook_uploader = instance_double("Chef::CookbookUploader")
231
+ expect(Chef::CookbookUploader).to receive(:new)
232
+ .with(expected_cookbooks_for_upload, rest: http_client, policy_mode: policy_document_native_api)
233
+ .and_return(cookbook_uploader)
234
+ expect(cookbook_uploader).to receive(:upload_cookbooks)
235
+
236
+ expect_policyfile_upload
237
+
238
+ uploader.upload
239
+ end
240
+
241
+ end
242
+
243
+ context "with a set of cookbooks that all exist on the server" do
244
+
245
+ before do
246
+ # Have this one:
247
+ lock_double("build-essential", "571d8ebd02b296fe90b2e4d68754af7e8e185f28", "67369247788170534.26353953100055918.55660493423796")
248
+ end
249
+
250
+ let(:expected_cookbooks_for_upload) do
251
+ []
252
+ end
253
+
254
+ it "lists no cookbooks as needing to be uploaded" do
255
+ expect(policyfile_lock).to receive(:validate_cookbooks!)
256
+ expect(http_client).to receive(:get).with(list_cookbooks_url).and_return(existing_cookbook_on_remote)
257
+
258
+ expect(uploader.cookbook_versions_to_upload).to eq(expected_cookbooks_for_upload)
259
+ end
260
+
261
+ it "skips cookbooks uploads, then uploads the policy" do
262
+ expect(policyfile_lock).to receive(:validate_cookbooks!)
263
+ expect(http_client).to receive(:get).with(list_cookbooks_url).and_return(existing_cookbook_on_remote)
264
+
265
+ expect(uploader.uploader).to_not receive(:upload_cookbooks)
266
+
267
+ expect_policyfile_upload
268
+
269
+ uploader.upload
270
+ end
271
+ end
272
+ end
273
+ end # uploading cookbooks shared examples
274
+
275
+ context "when configured for policy document compat mode" do
276
+
277
+ let(:policyfiles_data_bag) { { "name" => "policyfiles" } }
278
+
279
+ let(:list_cookbooks_url) { "cookbooks?num_versions=all" }
280
+
281
+ let(:existing_cookbook_on_remote) do
282
+ { "apt" =>
283
+ { "url" => "http://localhost:8889/cookbooks/apt",
284
+ "versions" =>
285
+ [{ "url" =>
286
+ "http://localhost:8889/cookbooks/apt/46097674477573307.43471642740453733.243606720748315",
287
+ "version" => "46097674477573307.43471642740453733.243606720748315" }] },
288
+ "build-essential" =>
289
+ { "url" => "http://localhost:8889/cookbooks/build-essential",
290
+ "versions" =>
291
+ [{ "url" =>
292
+ "http://localhost:8889/cookbooks/build-essential/67369247788170534.26353953100055918.55660493423796",
293
+ "version" => "67369247788170534.26353953100055918.55660493423796" }] },
294
+ "java" =>
295
+ { "url" => "http://localhost:8889/cookbooks/java",
296
+ "versions" =>
297
+ [{ "url" =>
298
+ "http://localhost:8889/cookbooks/java/5664982062912610.52588194571203830.6215746262253",
299
+ "version" => "5664982062912610.52588194571203830.6215746262253" }] },
300
+ "jenkins" =>
301
+ { "url" => "http://localhost:8889/cookbooks/jenkins",
302
+ "versions" =>
303
+ [{ "url" =>
304
+ "http://localhost:8889/cookbooks/jenkins/69194928762630300.30177357398946006.269829039948647",
305
+ "version" => "69194928762630300.30177357398946006.269829039948647" }] },
306
+ }
307
+ end
308
+
309
+ def expect_policyfile_upload
310
+ expect(uploader).to receive(:data_bag_create)
311
+ expect(uploader).to receive(:data_bag_item_create)
312
+ end
313
+
314
+ it "ensures a data bag named 'policyfiles' exists" do
315
+ expect(http_client).to receive(:post).with("data", policyfiles_data_bag)
316
+ uploader.data_bag_create
317
+ end
318
+
319
+ it "does not error when the 'policyfiles' data bag exists" do
320
+ response = double("Net::HTTP response", code: "409")
321
+ error = Net::HTTPServerException.new("conflict", response)
322
+ expect(http_client).to receive(:post).with("data", { "name" => "policyfiles" }).and_raise(error)
323
+ expect { uploader.data_bag_create }.to_not raise_error
324
+ end
325
+
326
+ it "uploads the policyfile as a data bag item" do
327
+ response = double("Net::HTTP response", code: "404")
328
+ error = Net::HTTPServerException.new("Not Found", response)
329
+ expect(http_client).to receive(:put)
330
+ .with("data/policyfiles/example-unit-test", policyfile_as_data_bag_item)
331
+ .and_raise(error)
332
+ expect(http_client).to receive(:post)
333
+ .with("data/policyfiles", policyfile_as_data_bag_item)
334
+
335
+ uploader.data_bag_item_create
336
+ end
337
+
338
+ it "replaces an existing policyfile on the server if it exists" do
339
+ expect(http_client).to receive(:put)
340
+ .with("data/policyfiles/example-unit-test", policyfile_as_data_bag_item)
341
+ uploader.data_bag_item_create
342
+ end
343
+
344
+ it "creates the data bag and item to upload the policy" do
345
+ expect(http_client).to receive(:post).with("data", policyfiles_data_bag)
346
+ expect(http_client).to receive(:put)
347
+ .with("data/policyfiles/example-unit-test", policyfile_as_data_bag_item)
348
+ uploader.upload_policy
349
+ end
350
+
351
+ include_examples "uploading cookbooks"
352
+
353
+ end
354
+
355
+ context "when configured for policy document native mode" do
356
+
357
+ let(:policy_document_native_api) { true }
358
+
359
+ let(:list_cookbooks_url) { "cookbook_artifacts?num_versions=all" }
360
+
361
+ let(:existing_cookbook_on_remote) do
362
+ { "apt" =>
363
+ { "url" => "http://localhost:8889/cookbook_artifacts/apt",
364
+ "versions" =>
365
+ [{ "url" =>
366
+ "http://localhost:8889/cookbook_artifacts/apt/5f7045a8aeaf6ccda3b3594258df9ee982b3a023",
367
+ "identifier" => "5f7045a8aeaf6ccda3b3594258df9ee982b3a023" }] },
368
+ "build-essential" =>
369
+ { "url" => "http://localhost:8889/cookbook_artifacts/build-essential",
370
+ "versions" =>
371
+ [{ "url" =>
372
+ "http://localhost:8889/cookbook_artifacts/build-essential/571d8ebd02b296fe90b2e4d68754af7e8e185f28",
373
+ "identifier" => "571d8ebd02b296fe90b2e4d68754af7e8e185f28" }] },
374
+ "java" =>
375
+ { "url" => "http://localhost:8889/cookbook_artifacts/java",
376
+ "versions" =>
377
+ [{ "url" =>
378
+ "http://localhost:8889/cookbook_artifacts/java/9178a38ad3e3baa55b49c1b8d9f4bf6a43dbc358",
379
+ "identifier" => "9178a38ad3e3baa55b49c1b8d9f4bf6a43dbc358" }] },
380
+ "jenkins" =>
381
+ { "url" => "http://localhost:8889/cookbook_artifacts/jenkins",
382
+ "versions" =>
383
+ [{ "url" =>
384
+ "http://localhost:8889/cookbook_artifacts/jenkins/0be380429add00d189b4431059ac967a60052323",
385
+ "identifier" => "0be380429add00d189b4431059ac967a60052323" }] },
386
+ }
387
+ end
388
+ def expect_policyfile_upload
389
+ expect(http_client).to receive(:put)
390
+ .with("/policy_groups/unit-test/policies/example", policyfile_lock_data)
391
+ end
392
+
393
+ it "enables native document mode for policyfiles" do
394
+ expect(uploader.using_policy_document_native_api?).to be(true)
395
+ end
396
+
397
+ it "uploads the policyfile to the native API" do
398
+ expect(http_client).to receive(:put)
399
+ .with("/policy_groups/unit-test/policies/example", policyfile_lock_data)
400
+
401
+ uploader.upload_policy
402
+ end
403
+
404
+ include_examples "uploading cookbooks"
405
+
406
+ end
407
+
408
+ end
409
+
410
+ end