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.
- checksums.yaml +7 -0
- data/Gemfile +32 -0
- data/LICENSE +201 -0
- data/Rakefile +70 -0
- data/bin/chef +25 -0
- data/chef-cli.gemspec +53 -0
- data/lib/chef-cli.rb +19 -0
- data/lib/chef-cli/authenticated_http.rb +22 -0
- data/lib/chef-cli/builtin_commands.rb +62 -0
- data/lib/chef-cli/chef_runner.rb +114 -0
- data/lib/chef-cli/chef_server_api_multi.rb +73 -0
- data/lib/chef-cli/cli.rb +206 -0
- data/lib/chef-cli/command/base.rb +89 -0
- data/lib/chef-cli/command/clean_policy_cookbooks.rb +115 -0
- data/lib/chef-cli/command/clean_policy_revisions.rb +112 -0
- data/lib/chef-cli/command/delete_policy.rb +121 -0
- data/lib/chef-cli/command/delete_policy_group.rb +121 -0
- data/lib/chef-cli/command/describe_cookbook.rb +98 -0
- data/lib/chef-cli/command/diff.rb +316 -0
- data/lib/chef-cli/command/env.rb +99 -0
- data/lib/chef-cli/command/exec.rb +45 -0
- data/lib/chef-cli/command/export.rb +156 -0
- data/lib/chef-cli/command/gem.rb +48 -0
- data/lib/chef-cli/command/generate.rb +123 -0
- data/lib/chef-cli/command/generator_commands.rb +83 -0
- data/lib/chef-cli/command/generator_commands/attribute.rb +37 -0
- data/lib/chef-cli/command/generator_commands/base.rb +157 -0
- data/lib/chef-cli/command/generator_commands/build_cookbook.rb +126 -0
- data/lib/chef-cli/command/generator_commands/chef_exts/generator_desc_resource.rb +40 -0
- data/lib/chef-cli/command/generator_commands/chef_exts/quieter_doc_formatter.rb +38 -0
- data/lib/chef-cli/command/generator_commands/chef_exts/recipe_dsl_ext.rb +39 -0
- data/lib/chef-cli/command/generator_commands/cookbook.rb +251 -0
- data/lib/chef-cli/command/generator_commands/cookbook_code_file.rb +100 -0
- data/lib/chef-cli/command/generator_commands/cookbook_file.rb +46 -0
- data/lib/chef-cli/command/generator_commands/generator_generator.rb +175 -0
- data/lib/chef-cli/command/generator_commands/helpers.rb +37 -0
- data/lib/chef-cli/command/generator_commands/policyfile.rb +125 -0
- data/lib/chef-cli/command/generator_commands/recipe.rb +37 -0
- data/lib/chef-cli/command/generator_commands/repo.rb +140 -0
- data/lib/chef-cli/command/generator_commands/resource.rb +37 -0
- data/lib/chef-cli/command/generator_commands/template.rb +47 -0
- data/lib/chef-cli/command/install.rb +121 -0
- data/lib/chef-cli/command/provision.rb +38 -0
- data/lib/chef-cli/command/push.rb +118 -0
- data/lib/chef-cli/command/push_archive.rb +126 -0
- data/lib/chef-cli/command/shell_init.rb +185 -0
- data/lib/chef-cli/command/show_policy.rb +164 -0
- data/lib/chef-cli/command/undelete.rb +155 -0
- data/lib/chef-cli/command/update.rb +140 -0
- data/lib/chef-cli/command/verify.rb +548 -0
- data/lib/chef-cli/commands_map.rb +113 -0
- data/lib/chef-cli/completions/bash.sh.erb +5 -0
- data/lib/chef-cli/completions/chef.fish.erb +10 -0
- data/lib/chef-cli/completions/zsh.zsh.erb +21 -0
- data/lib/chef-cli/component_test.rb +226 -0
- data/lib/chef-cli/configurable.rb +88 -0
- data/lib/chef-cli/cookbook_metadata.rb +45 -0
- data/lib/chef-cli/cookbook_omnifetch.rb +32 -0
- data/lib/chef-cli/cookbook_profiler/git.rb +152 -0
- data/lib/chef-cli/cookbook_profiler/identifiers.rb +72 -0
- data/lib/chef-cli/cookbook_profiler/null_scm.rb +31 -0
- data/lib/chef-cli/dist.rb +31 -0
- data/lib/chef-cli/exceptions.rb +153 -0
- data/lib/chef-cli/generator.rb +165 -0
- data/lib/chef-cli/helpers.rb +170 -0
- data/lib/chef-cli/pager.rb +104 -0
- data/lib/chef-cli/policyfile/artifactory_cookbook_source.rb +102 -0
- data/lib/chef-cli/policyfile/attribute_merge_checker.rb +110 -0
- data/lib/chef-cli/policyfile/chef_repo_cookbook_source.rb +138 -0
- data/lib/chef-cli/policyfile/chef_server_cookbook_source.rb +99 -0
- data/lib/chef-cli/policyfile/chef_server_lock_fetcher.rb +167 -0
- data/lib/chef-cli/policyfile/community_cookbook_source.rb +95 -0
- data/lib/chef-cli/policyfile/comparison_base.rb +123 -0
- data/lib/chef-cli/policyfile/cookbook_location_specification.rb +154 -0
- data/lib/chef-cli/policyfile/cookbook_locks.rb +466 -0
- data/lib/chef-cli/policyfile/cookbook_sources.rb +23 -0
- data/lib/chef-cli/policyfile/delivery_supermarket_source.rb +89 -0
- data/lib/chef-cli/policyfile/differ.rb +263 -0
- data/lib/chef-cli/policyfile/dsl.rb +288 -0
- data/lib/chef-cli/policyfile/git_lock_fetcher.rb +265 -0
- data/lib/chef-cli/policyfile/included_policies_cookbook_source.rb +156 -0
- data/lib/chef-cli/policyfile/lister.rb +229 -0
- data/lib/chef-cli/policyfile/local_lock_fetcher.rb +132 -0
- data/lib/chef-cli/policyfile/lock_applier.rb +80 -0
- data/lib/chef-cli/policyfile/lock_fetcher_mixin.rb +37 -0
- data/lib/chef-cli/policyfile/null_cookbook_source.rb +49 -0
- data/lib/chef-cli/policyfile/policyfile_location_specification.rb +128 -0
- data/lib/chef-cli/policyfile/read_cookbook_for_compat_mode_upload.rb +124 -0
- data/lib/chef-cli/policyfile/remote_lock_fetcher.rb +108 -0
- data/lib/chef-cli/policyfile/reports/install.rb +69 -0
- data/lib/chef-cli/policyfile/reports/table_printer.rb +57 -0
- data/lib/chef-cli/policyfile/reports/upload.rb +70 -0
- data/lib/chef-cli/policyfile/solution_dependencies.rb +311 -0
- data/lib/chef-cli/policyfile/source_uri.rb +57 -0
- data/lib/chef-cli/policyfile/storage_config.rb +112 -0
- data/lib/chef-cli/policyfile/undo_record.rb +139 -0
- data/lib/chef-cli/policyfile/undo_stack.rb +128 -0
- data/lib/chef-cli/policyfile/uploader.rb +222 -0
- data/lib/chef-cli/policyfile_compiler.rb +528 -0
- data/lib/chef-cli/policyfile_lock.rb +581 -0
- data/lib/chef-cli/policyfile_services/clean_policies.rb +95 -0
- data/lib/chef-cli/policyfile_services/clean_policy_cookbooks.rb +123 -0
- data/lib/chef-cli/policyfile_services/export_repo.rb +419 -0
- data/lib/chef-cli/policyfile_services/install.rb +167 -0
- data/lib/chef-cli/policyfile_services/push.rb +112 -0
- data/lib/chef-cli/policyfile_services/push_archive.rb +164 -0
- data/lib/chef-cli/policyfile_services/rm_policy.rb +141 -0
- data/lib/chef-cli/policyfile_services/rm_policy_group.rb +85 -0
- data/lib/chef-cli/policyfile_services/show_policy.rb +234 -0
- data/lib/chef-cli/policyfile_services/undelete.rb +108 -0
- data/lib/chef-cli/policyfile_services/update_attributes.rb +110 -0
- data/lib/chef-cli/service_exception_inspectors.rb +24 -0
- data/lib/chef-cli/service_exception_inspectors/base.rb +39 -0
- data/lib/chef-cli/service_exception_inspectors/http.rb +119 -0
- data/lib/chef-cli/service_exceptions.rb +142 -0
- data/lib/chef-cli/shell_out.rb +36 -0
- data/lib/chef-cli/skeletons/code_generator/files/default/Berksfile +3 -0
- data/lib/chef-cli/skeletons/code_generator/files/default/build_cookbook/README.md +146 -0
- data/lib/chef-cli/skeletons/code_generator/files/default/build_cookbook/kitchen.yml +21 -0
- data/lib/chef-cli/skeletons/code_generator/files/default/build_cookbook/test-fixture-recipe.rb +8 -0
- data/lib/chef-cli/skeletons/code_generator/files/default/chefignore +110 -0
- data/lib/chef-cli/skeletons/code_generator/files/default/cookbook_readmes/README-policy.md +9 -0
- data/lib/chef-cli/skeletons/code_generator/files/default/cookbook_readmes/README.md +66 -0
- data/lib/chef-cli/skeletons/code_generator/files/default/delivery-config.json +17 -0
- data/lib/chef-cli/skeletons/code_generator/files/default/delivery-project.toml +34 -0
- data/lib/chef-cli/skeletons/code_generator/files/default/gitignore +22 -0
- data/lib/chef-cli/skeletons/code_generator/files/default/repo/README.md +20 -0
- data/lib/chef-cli/skeletons/code_generator/files/default/repo/cookbooks/example/README.md +27 -0
- data/lib/chef-cli/skeletons/code_generator/files/default/repo/cookbooks/example/attributes/default.rb +7 -0
- data/lib/chef-cli/skeletons/code_generator/files/default/repo/cookbooks/example/metadata.rb +6 -0
- data/lib/chef-cli/skeletons/code_generator/files/default/repo/cookbooks/example/recipes/default.rb +8 -0
- data/lib/chef-cli/skeletons/code_generator/files/default/repo/data_bags/README.md +56 -0
- data/lib/chef-cli/skeletons/code_generator/files/default/repo/data_bags/example/example_item.json +4 -0
- data/lib/chef-cli/skeletons/code_generator/files/default/repo/dot-chef-repo.txt +6 -0
- data/lib/chef-cli/skeletons/code_generator/files/default/repo/environments/README.md +9 -0
- data/lib/chef-cli/skeletons/code_generator/files/default/repo/environments/example.json +13 -0
- data/lib/chef-cli/skeletons/code_generator/files/default/repo/policyfiles/README.md +24 -0
- data/lib/chef-cli/skeletons/code_generator/files/default/repo/roles/README.md +9 -0
- data/lib/chef-cli/skeletons/code_generator/files/default/repo/roles/example.json +13 -0
- data/lib/chef-cli/skeletons/code_generator/files/default/spec_helper.rb +2 -0
- data/lib/chef-cli/skeletons/code_generator/files/default/spec_helper_policyfile.rb +2 -0
- data/lib/chef-cli/skeletons/code_generator/metadata.rb +8 -0
- data/lib/chef-cli/skeletons/code_generator/recipes/attribute.rb +11 -0
- data/lib/chef-cli/skeletons/code_generator/recipes/build_cookbook.rb +175 -0
- data/lib/chef-cli/skeletons/code_generator/recipes/cookbook.rb +167 -0
- data/lib/chef-cli/skeletons/code_generator/recipes/cookbook_file.rb +23 -0
- data/lib/chef-cli/skeletons/code_generator/recipes/helpers.rb +19 -0
- data/lib/chef-cli/skeletons/code_generator/recipes/policyfile.rb +7 -0
- data/lib/chef-cli/skeletons/code_generator/recipes/recipe.rb +50 -0
- data/lib/chef-cli/skeletons/code_generator/recipes/repo.rb +71 -0
- data/lib/chef-cli/skeletons/code_generator/recipes/resource.rb +12 -0
- data/lib/chef-cli/skeletons/code_generator/recipes/template.rb +31 -0
- data/lib/chef-cli/skeletons/code_generator/templates/default/CHANGELOG.md.erb +11 -0
- data/lib/chef-cli/skeletons/code_generator/templates/default/LICENSE.all_rights.erb +3 -0
- data/lib/chef-cli/skeletons/code_generator/templates/default/LICENSE.apachev2.erb +201 -0
- data/lib/chef-cli/skeletons/code_generator/templates/default/LICENSE.gplv2.erb +339 -0
- data/lib/chef-cli/skeletons/code_generator/templates/default/LICENSE.gplv3.erb +674 -0
- data/lib/chef-cli/skeletons/code_generator/templates/default/LICENSE.mit.erb +21 -0
- data/lib/chef-cli/skeletons/code_generator/templates/default/Policyfile.rb.erb +25 -0
- data/lib/chef-cli/skeletons/code_generator/templates/default/README.md.erb +4 -0
- data/lib/chef-cli/skeletons/code_generator/templates/default/attribute.rb.erb +0 -0
- data/lib/chef-cli/skeletons/code_generator/templates/default/build_cookbook/Berksfile.erb +7 -0
- data/lib/chef-cli/skeletons/code_generator/templates/default/build_cookbook/metadata.rb.erb +10 -0
- data/lib/chef-cli/skeletons/code_generator/templates/default/build_cookbook/recipe.rb.erb +9 -0
- data/lib/chef-cli/skeletons/code_generator/templates/default/cookbook_file.erb +0 -0
- data/lib/chef-cli/skeletons/code_generator/templates/default/helpers.rb.erb +39 -0
- data/lib/chef-cli/skeletons/code_generator/templates/default/inspec_default_test.rb.erb +16 -0
- data/lib/chef-cli/skeletons/code_generator/templates/default/kitchen.yml.erb +38 -0
- data/lib/chef-cli/skeletons/code_generator/templates/default/kitchen_dokken.yml.erb +36 -0
- data/lib/chef-cli/skeletons/code_generator/templates/default/kitchen_policyfile.yml.erb +32 -0
- data/lib/chef-cli/skeletons/code_generator/templates/default/metadata.rb.erb +20 -0
- data/lib/chef-cli/skeletons/code_generator/templates/default/recipe.rb.erb +5 -0
- data/lib/chef-cli/skeletons/code_generator/templates/default/recipe_spec.rb.erb +29 -0
- data/lib/chef-cli/skeletons/code_generator/templates/default/repo/gitignore.erb +128 -0
- data/lib/chef-cli/skeletons/code_generator/templates/default/resource.rb.erb +1 -0
- data/lib/chef-cli/skeletons/code_generator/templates/default/template.erb +0 -0
- data/lib/chef-cli/ui.rb +57 -0
- data/lib/chef-cli/version.rb +20 -0
- data/lib/kitchen/provisioner/policyfile_zero.rb +195 -0
- data/spec/shared/a_file_generator.rb +125 -0
- data/spec/shared/a_generated_file.rb +12 -0
- data/spec/shared/command_with_ui_object.rb +11 -0
- data/spec/shared/custom_generator_cookbook.rb +136 -0
- data/spec/shared/fixture_cookbook_checksums.rb +46 -0
- data/spec/shared/setup_git_committer_config.rb +54 -0
- data/spec/shared/setup_git_cookbooks.rb +53 -0
- data/spec/spec_helper.rb +51 -0
- data/spec/test_helpers.rb +84 -0
- data/spec/unit/chef_runner_spec.rb +139 -0
- data/spec/unit/chef_server_api_multi_spec.rb +120 -0
- data/spec/unit/cli_spec.rb +375 -0
- data/spec/unit/command/base_spec.rb +195 -0
- data/spec/unit/command/clean_policy_cookbooks_spec.rb +180 -0
- data/spec/unit/command/clean_policy_revisions_spec.rb +180 -0
- data/spec/unit/command/delete_policy_group_spec.rb +206 -0
- data/spec/unit/command/delete_policy_spec.rb +206 -0
- data/spec/unit/command/diff_spec.rb +311 -0
- data/spec/unit/command/env_spec.rb +86 -0
- data/spec/unit/command/exec_spec.rb +178 -0
- data/spec/unit/command/export_spec.rb +199 -0
- data/spec/unit/command/generate_spec.rb +142 -0
- data/spec/unit/command/generator_commands/attribute_spec.rb +31 -0
- data/spec/unit/command/generator_commands/base_spec.rb +180 -0
- data/spec/unit/command/generator_commands/build_cookbook_spec.rb +377 -0
- data/spec/unit/command/generator_commands/chef_exts/generator_desc_resource_spec.rb +77 -0
- data/spec/unit/command/generator_commands/chef_exts/recipe_dsl_ext_spec.rb +111 -0
- data/spec/unit/command/generator_commands/cookbook_file_spec.rb +31 -0
- data/spec/unit/command/generator_commands/cookbook_spec.rb +769 -0
- data/spec/unit/command/generator_commands/generator_generator_spec.rb +227 -0
- data/spec/unit/command/generator_commands/helpers_spec.rb +31 -0
- data/spec/unit/command/generator_commands/policyfile_spec.rb +223 -0
- data/spec/unit/command/generator_commands/recipe_spec.rb +37 -0
- data/spec/unit/command/generator_commands/repo_spec.rb +374 -0
- data/spec/unit/command/generator_commands/resource_spec.rb +31 -0
- data/spec/unit/command/generator_commands/template_spec.rb +31 -0
- data/spec/unit/command/install_spec.rb +179 -0
- data/spec/unit/command/push_archive_spec.rb +153 -0
- data/spec/unit/command/push_spec.rb +198 -0
- data/spec/unit/command/shell_init_spec.rb +339 -0
- data/spec/unit/command/show_policy_spec.rb +234 -0
- data/spec/unit/command/undelete_spec.rb +244 -0
- data/spec/unit/command/update_spec.rb +283 -0
- data/spec/unit/command/verify_spec.rb +341 -0
- data/spec/unit/commands_map_spec.rb +57 -0
- data/spec/unit/component_test_spec.rb +128 -0
- data/spec/unit/configurable_spec.rb +68 -0
- data/spec/unit/cookbook_metadata_spec.rb +96 -0
- data/spec/unit/cookbook_profiler/git_spec.rb +176 -0
- data/spec/unit/cookbook_profiler/identifiers_spec.rb +81 -0
- data/spec/unit/fixtures/chef-runner-cookbooks/test_cookbook/recipes/recipe_one.rb +9 -0
- data/spec/unit/fixtures/chef-runner-cookbooks/test_cookbook/recipes/recipe_two.rb +9 -0
- data/spec/unit/fixtures/command/cli_test_command.rb +26 -0
- data/spec/unit/fixtures/command/explicit_path_example.rb +7 -0
- data/spec/unit/fixtures/configurable/test_config_loader.rb +5 -0
- data/spec/unit/fixtures/configurable/test_configurable.rb +10 -0
- data/spec/unit/fixtures/cookbook_cache/baz-f59ee7a5bca6a4e606b67f7f856b768d847c39bb/.kitchen.yml +16 -0
- data/spec/unit/fixtures/cookbook_cache/baz-f59ee7a5bca6a4e606b67f7f856b768d847c39bb/Berksfile +3 -0
- data/spec/unit/fixtures/cookbook_cache/baz-f59ee7a5bca6a4e606b67f7f856b768d847c39bb/README.md +4 -0
- data/spec/unit/fixtures/cookbook_cache/baz-f59ee7a5bca6a4e606b67f7f856b768d847c39bb/chefignore +96 -0
- data/spec/unit/fixtures/cookbook_cache/baz-f59ee7a5bca6a4e606b67f7f856b768d847c39bb/metadata.rb +8 -0
- data/spec/unit/fixtures/cookbook_cache/baz-f59ee7a5bca6a4e606b67f7f856b768d847c39bb/recipes/default.rb +8 -0
- data/spec/unit/fixtures/cookbook_cache/dep_of_bar-1.2.3/.kitchen.yml +16 -0
- data/spec/unit/fixtures/cookbook_cache/dep_of_bar-1.2.3/Berksfile +3 -0
- data/spec/unit/fixtures/cookbook_cache/dep_of_bar-1.2.3/README.md +4 -0
- data/spec/unit/fixtures/cookbook_cache/dep_of_bar-1.2.3/chefignore +96 -0
- data/spec/unit/fixtures/cookbook_cache/dep_of_bar-1.2.3/metadata.rb +8 -0
- data/spec/unit/fixtures/cookbook_cache/dep_of_bar-1.2.3/recipes/default.rb +8 -0
- data/spec/unit/fixtures/cookbook_cache/foo-1.0.0/Berksfile +3 -0
- data/spec/unit/fixtures/cookbook_cache/foo-1.0.0/README.md +4 -0
- data/spec/unit/fixtures/cookbook_cache/foo-1.0.0/chefignore +96 -0
- data/spec/unit/fixtures/cookbook_cache/foo-1.0.0/kitchen.yml +16 -0
- data/spec/unit/fixtures/cookbook_cache/foo-1.0.0/metadata.rb +8 -0
- data/spec/unit/fixtures/cookbook_cache/foo-1.0.0/recipes/default.rb +8 -0
- data/spec/unit/fixtures/cookbooks_api/chef_server_universe.json +56 -0
- data/spec/unit/fixtures/cookbooks_api/pruned_chef_server_universe.json +30 -0
- data/spec/unit/fixtures/cookbooks_api/pruned_small_universe.json +1322 -0
- data/spec/unit/fixtures/cookbooks_api/small_universe.json +2987 -0
- data/spec/unit/fixtures/cookbooks_api/universe.json +1 -0
- data/spec/unit/fixtures/cookbooks_api/update_fixtures.rb +33 -0
- data/spec/unit/fixtures/dev_cookbooks/README.md +16 -0
- data/spec/unit/fixtures/dev_cookbooks/bar-cookbook.gitbundle +0 -0
- data/spec/unit/fixtures/eg_omnibus_dir/missing_apps/bin/.keep +0 -0
- data/spec/unit/fixtures/eg_omnibus_dir/missing_apps/embedded/.keep +0 -0
- data/spec/unit/fixtures/eg_omnibus_dir/missing_apps/embedded/bin/.keep +0 -0
- data/spec/unit/fixtures/eg_omnibus_dir/missing_component/bin/.keep +0 -0
- data/spec/unit/fixtures/eg_omnibus_dir/missing_component/embedded/apps/berkshelf/.keep +0 -0
- data/spec/unit/fixtures/eg_omnibus_dir/missing_component/embedded/apps/test-kitchen/.keep +0 -0
- data/spec/unit/fixtures/eg_omnibus_dir/missing_component/embedded/bin/.keep +0 -0
- data/spec/unit/fixtures/eg_omnibus_dir/valid/bin/.keep +0 -0
- data/spec/unit/fixtures/eg_omnibus_dir/valid/embedded/apps/berkshelf/integration_test +2 -0
- data/spec/unit/fixtures/eg_omnibus_dir/valid/embedded/apps/berkshelf/verify_me +5 -0
- data/spec/unit/fixtures/eg_omnibus_dir/valid/embedded/apps/chef-dk/.keep +0 -0
- data/spec/unit/fixtures/eg_omnibus_dir/valid/embedded/apps/chef/verify_me +3 -0
- data/spec/unit/fixtures/eg_omnibus_dir/valid/embedded/apps/test-kitchen/verify_me +2 -0
- data/spec/unit/fixtures/eg_omnibus_dir/valid/embedded/bin/.keep +0 -0
- data/spec/unit/fixtures/example_app/Policyfile.rb +0 -0
- data/spec/unit/fixtures/example_cookbook/.gitignore +17 -0
- data/spec/unit/fixtures/example_cookbook/.kitchen.yml +16 -0
- data/spec/unit/fixtures/example_cookbook/Berksfile +3 -0
- data/spec/unit/fixtures/example_cookbook/README.md +4 -0
- data/spec/unit/fixtures/example_cookbook/chefignore +96 -0
- data/spec/unit/fixtures/example_cookbook/metadata.rb +8 -0
- data/spec/unit/fixtures/example_cookbook/recipes/default.rb +8 -0
- data/spec/unit/fixtures/example_cookbook_metadata_json_only/.gitignore +17 -0
- data/spec/unit/fixtures/example_cookbook_metadata_json_only/.kitchen.yml +16 -0
- data/spec/unit/fixtures/example_cookbook_metadata_json_only/Berksfile +3 -0
- data/spec/unit/fixtures/example_cookbook_metadata_json_only/README.md +4 -0
- data/spec/unit/fixtures/example_cookbook_metadata_json_only/chefignore +96 -0
- data/spec/unit/fixtures/example_cookbook_metadata_json_only/metadata.json +5 -0
- data/spec/unit/fixtures/example_cookbook_metadata_json_only/recipes/default.rb +8 -0
- data/spec/unit/fixtures/example_cookbook_no_metadata/.gitignore +17 -0
- data/spec/unit/fixtures/example_cookbook_no_metadata/.kitchen.yml +16 -0
- data/spec/unit/fixtures/example_cookbook_no_metadata/Berksfile +3 -0
- data/spec/unit/fixtures/example_cookbook_no_metadata/README.md +4 -0
- data/spec/unit/fixtures/example_cookbook_no_metadata/chefignore +96 -0
- data/spec/unit/fixtures/example_cookbook_no_metadata/recipes/default.rb +8 -0
- data/spec/unit/fixtures/local_path_cookbooks/another-local-cookbook/README.md +4 -0
- data/spec/unit/fixtures/local_path_cookbooks/another-local-cookbook/chefignore +96 -0
- data/spec/unit/fixtures/local_path_cookbooks/another-local-cookbook/metadata.rb +8 -0
- data/spec/unit/fixtures/local_path_cookbooks/another-local-cookbook/recipes/default.rb +8 -0
- data/spec/unit/fixtures/local_path_cookbooks/cookbook-with-a-dep/Berksfile +3 -0
- data/spec/unit/fixtures/local_path_cookbooks/cookbook-with-a-dep/README.md +4 -0
- data/spec/unit/fixtures/local_path_cookbooks/cookbook-with-a-dep/chefignore +96 -0
- data/spec/unit/fixtures/local_path_cookbooks/cookbook-with-a-dep/metadata.rb +9 -0
- data/spec/unit/fixtures/local_path_cookbooks/cookbook-with-a-dep/recipes/default.rb +8 -0
- data/spec/unit/fixtures/local_path_cookbooks/local-cookbook/.kitchen.yml +16 -0
- data/spec/unit/fixtures/local_path_cookbooks/local-cookbook/Berksfile +3 -0
- data/spec/unit/fixtures/local_path_cookbooks/local-cookbook/README.md +4 -0
- data/spec/unit/fixtures/local_path_cookbooks/local-cookbook/chefignore +96 -0
- data/spec/unit/fixtures/local_path_cookbooks/local-cookbook/extra/extra_file.txt +0 -0
- data/spec/unit/fixtures/local_path_cookbooks/local-cookbook/metadata.rb +8 -0
- data/spec/unit/fixtures/local_path_cookbooks/local-cookbook/recipes/default.rb +8 -0
- data/spec/unit/fixtures/local_path_cookbooks/metadata-missing/README.md +2 -0
- data/spec/unit/fixtures/local_path_cookbooks/noignore-f59ee7a5bca6a4e606b67f7f856b768d847c39bb/.kitchen.yml +16 -0
- data/spec/unit/fixtures/local_path_cookbooks/noignore-f59ee7a5bca6a4e606b67f7f856b768d847c39bb/README.md +4 -0
- data/spec/unit/fixtures/local_path_cookbooks/noignore-f59ee7a5bca6a4e606b67f7f856b768d847c39bb/metadata.rb +8 -0
- data/spec/unit/fixtures/local_path_cookbooks/noignore-f59ee7a5bca6a4e606b67f7f856b768d847c39bb/recipes/default.rb +8 -0
- data/spec/unit/generator_spec.rb +119 -0
- data/spec/unit/pager_spec.rb +117 -0
- data/spec/unit/policyfile/artifactory_cookbook_source_spec.rb +59 -0
- data/spec/unit/policyfile/attribute_merge_checker_spec.rb +80 -0
- data/spec/unit/policyfile/chef_repo_cookbook_source_spec.rb +93 -0
- data/spec/unit/policyfile/chef_server_cookbook_source_spec.rb +55 -0
- data/spec/unit/policyfile/chef_server_lock_fetcher_spec.rb +161 -0
- data/spec/unit/policyfile/community_cookbook_source_spec.rb +83 -0
- data/spec/unit/policyfile/comparison_base_spec.rb +340 -0
- data/spec/unit/policyfile/cookbook_location_specification_spec.rb +347 -0
- data/spec/unit/policyfile/cookbook_locks_spec.rb +527 -0
- data/spec/unit/policyfile/delivery_supermarket_source_spec.rb +129 -0
- data/spec/unit/policyfile/differ_spec.rb +686 -0
- data/spec/unit/policyfile/git_lock_fetcher_spec.rb +155 -0
- data/spec/unit/policyfile/included_policies_cookbook_source_spec.rb +242 -0
- data/spec/unit/policyfile/lister_spec.rb +268 -0
- data/spec/unit/policyfile/local_lock_fetcher_spec.rb +199 -0
- data/spec/unit/policyfile/lock_applier_spec.rb +100 -0
- data/spec/unit/policyfile/lock_fetcher_mixin_spec.rb +60 -0
- data/spec/unit/policyfile/null_cookbook_source_spec.rb +34 -0
- data/spec/unit/policyfile/read_cookbook_for_compat_mode_upload_spec.rb +92 -0
- data/spec/unit/policyfile/remote_lock_fetcher_spec.rb +129 -0
- data/spec/unit/policyfile/reports/install_spec.rb +114 -0
- data/spec/unit/policyfile/reports/upload_spec.rb +94 -0
- data/spec/unit/policyfile/solution_dependencies_spec.rb +170 -0
- data/spec/unit/policyfile/source_uri_spec.rb +36 -0
- data/spec/unit/policyfile/storage_config_spec.rb +180 -0
- data/spec/unit/policyfile/undo_record_spec.rb +258 -0
- data/spec/unit/policyfile/undo_stack_spec.rb +265 -0
- data/spec/unit/policyfile/uploader_spec.rb +410 -0
- data/spec/unit/policyfile_demands_spec.rb +1197 -0
- data/spec/unit/policyfile_evaluation_spec.rb +628 -0
- data/spec/unit/policyfile_includes_dsl_spec.rb +220 -0
- data/spec/unit/policyfile_includes_spec.rb +720 -0
- data/spec/unit/policyfile_install_with_includes_spec.rb +232 -0
- data/spec/unit/policyfile_lock_build_spec.rb +1065 -0
- data/spec/unit/policyfile_lock_install_spec.rb +137 -0
- data/spec/unit/policyfile_lock_serialization_spec.rb +424 -0
- data/spec/unit/policyfile_lock_validation_spec.rb +608 -0
- data/spec/unit/policyfile_services/clean_policies_spec.rb +236 -0
- data/spec/unit/policyfile_services/clean_policy_cookbooks_spec.rb +272 -0
- data/spec/unit/policyfile_services/export_repo_spec.rb +473 -0
- data/spec/unit/policyfile_services/install_spec.rb +209 -0
- data/spec/unit/policyfile_services/push_archive_spec.rb +359 -0
- data/spec/unit/policyfile_services/push_spec.rb +249 -0
- data/spec/unit/policyfile_services/rm_policy_group_spec.rb +237 -0
- data/spec/unit/policyfile_services/rm_policy_spec.rb +263 -0
- data/spec/unit/policyfile_services/show_policy_spec.rb +887 -0
- data/spec/unit/policyfile_services/undelete_spec.rb +302 -0
- data/spec/unit/policyfile_services/update_attributes_spec.rb +229 -0
- data/spec/unit/policyfile_services/update_spec.rb +162 -0
- data/spec/unit/service_exception_inspectors/base_spec.rb +41 -0
- data/spec/unit/service_exception_inspectors/http_spec.rb +138 -0
- data/spec/unit/shell_out_spec.rb +34 -0
- metadata +796 -0
|
@@ -0,0 +1,57 @@
|
|
|
1
|
+
#
|
|
2
|
+
# Copyright:: Copyright (c) 2016 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 "addressable/uri" unless defined?(Addressable::URI)
|
|
19
|
+
require_relative "../exceptions"
|
|
20
|
+
|
|
21
|
+
module ChefCLI
|
|
22
|
+
module Policyfile
|
|
23
|
+
class SourceURI < Addressable::URI
|
|
24
|
+
class << self
|
|
25
|
+
# Returns a URI object based on the parsed string.
|
|
26
|
+
#
|
|
27
|
+
# @param [String, Addressable::URI, #to_str] uri The URI
|
|
28
|
+
# string to parse. No parsing is performed if the object
|
|
29
|
+
# is already an <code>Addressable::URI</code>.
|
|
30
|
+
#
|
|
31
|
+
# @raise [ChefCLI::InvalidPolicyfileSourceURI]
|
|
32
|
+
#
|
|
33
|
+
# @return [ChefCLI::Policyfile::SourceURI]
|
|
34
|
+
def parse(uri)
|
|
35
|
+
parsed_uri = super(uri)
|
|
36
|
+
parsed_uri.send(:validate)
|
|
37
|
+
parsed_uri
|
|
38
|
+
rescue TypeError, ArgumentError => ex
|
|
39
|
+
raise ChefCLI::InvalidPolicyfileSourceURI.new(uri, ex)
|
|
40
|
+
end
|
|
41
|
+
end
|
|
42
|
+
|
|
43
|
+
VALID_SCHEMES = %w{ https http }.freeze
|
|
44
|
+
|
|
45
|
+
# @raise [ChefCLI::InvalidPolicyfileSourceURI]
|
|
46
|
+
def validate
|
|
47
|
+
super
|
|
48
|
+
|
|
49
|
+
unless VALID_SCHEMES.include?(scheme)
|
|
50
|
+
raise InvalidPolicyfileSourceURI.new(self, "invalid URI scheme '#{scheme}'. Valid schemes: #{VALID_SCHEMES}")
|
|
51
|
+
end
|
|
52
|
+
rescue Addressable::URI::InvalidURIError => ex
|
|
53
|
+
raise InvalidPolicyfileSourceURI.new(self, ex)
|
|
54
|
+
end
|
|
55
|
+
end
|
|
56
|
+
end
|
|
57
|
+
end
|
|
@@ -0,0 +1,112 @@
|
|
|
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_relative "../cookbook_omnifetch"
|
|
19
|
+
require_relative "../exceptions"
|
|
20
|
+
|
|
21
|
+
module ChefCLI
|
|
22
|
+
module Policyfile
|
|
23
|
+
|
|
24
|
+
# Provides handling of relative paths, such as on disk cookbooks which
|
|
25
|
+
# are specified relative to the Policyfile and local caching of compiled
|
|
26
|
+
# Policyfile artifacts.
|
|
27
|
+
#
|
|
28
|
+
# @author Dan DeLeo
|
|
29
|
+
# @since 0.11.0
|
|
30
|
+
# @attr_accessor [String] relative_paths_root Root path relative to the named Policyfile
|
|
31
|
+
# @attr_accessor [String] cache_path Local cache directory for
|
|
32
|
+
# @attr_reader [String] policyfile_filename Uncompiled Policyfile filename.
|
|
33
|
+
# @attr_reader [String] policyfile_lock_filename Filename of compiled Policyfile.
|
|
34
|
+
class StorageConfig
|
|
35
|
+
|
|
36
|
+
attr_accessor :relative_paths_root
|
|
37
|
+
attr_accessor :cache_path
|
|
38
|
+
|
|
39
|
+
attr_reader :policyfile_filename
|
|
40
|
+
attr_reader :policyfile_lock_filename
|
|
41
|
+
|
|
42
|
+
def initialize(options = {})
|
|
43
|
+
@relative_paths_root = Dir.pwd
|
|
44
|
+
@cache_path = CookbookOmnifetch.storage_path
|
|
45
|
+
@policyfile_filename = "<< Policyfile filename not specified >>"
|
|
46
|
+
@policyfile_lock_filename = "<< Policyfile lock filename not specified >>"
|
|
47
|
+
handle_options(options)
|
|
48
|
+
end
|
|
49
|
+
|
|
50
|
+
def use_policyfile(policyfile_filename)
|
|
51
|
+
if policyfile_filename.end_with?(".lock.json")
|
|
52
|
+
return use_policyfile_lock(policyfile_filename)
|
|
53
|
+
end
|
|
54
|
+
unless policyfile_filename.end_with?(".rb")
|
|
55
|
+
raise InvalidPolicyfileFilename, "Policyfile filenames must end with `.rb' extension (you gave: `#{policyfile_filename}')"
|
|
56
|
+
end
|
|
57
|
+
@policyfile_filename = policyfile_filename
|
|
58
|
+
@policyfile_lock_filename = policyfile_filename.sub(/\.rb\Z/, ".lock.json")
|
|
59
|
+
@relative_paths_root = File.dirname(policyfile_filename)
|
|
60
|
+
self
|
|
61
|
+
end
|
|
62
|
+
|
|
63
|
+
def use_policyfile_lock(policyfile_lock_filename)
|
|
64
|
+
@policyfile_lock_filename = policyfile_lock_filename
|
|
65
|
+
@policyfile_filename = policyfile_lock_filename.sub(/\.lock\.json\Z/, ".rb")
|
|
66
|
+
@relative_paths_root = File.dirname(policyfile_lock_filename)
|
|
67
|
+
self
|
|
68
|
+
end
|
|
69
|
+
|
|
70
|
+
def policyfile_expanded_path
|
|
71
|
+
File.expand_path(policyfile_filename, relative_paths_root)
|
|
72
|
+
end
|
|
73
|
+
|
|
74
|
+
def policyfile_lock_expanded_path
|
|
75
|
+
File.expand_path(policyfile_lock_filename, relative_paths_root)
|
|
76
|
+
end
|
|
77
|
+
|
|
78
|
+
private
|
|
79
|
+
|
|
80
|
+
def handle_options(options)
|
|
81
|
+
@cache_path = options[:cache_path] if options[:cache_path]
|
|
82
|
+
@relative_paths_root = options[:relative_paths_root] if options.key?(:relative_paths_root)
|
|
83
|
+
end
|
|
84
|
+
|
|
85
|
+
end
|
|
86
|
+
|
|
87
|
+
module StorageConfigDelegation
|
|
88
|
+
|
|
89
|
+
def cache_path
|
|
90
|
+
storage_config.cache_path
|
|
91
|
+
end
|
|
92
|
+
|
|
93
|
+
def relative_paths_root
|
|
94
|
+
storage_config.relative_paths_root
|
|
95
|
+
end
|
|
96
|
+
|
|
97
|
+
def policyfile_filename
|
|
98
|
+
storage_config.policyfile_filename
|
|
99
|
+
end
|
|
100
|
+
|
|
101
|
+
def policyfile_expanded_path
|
|
102
|
+
storage_config.policyfile_expanded_path
|
|
103
|
+
end
|
|
104
|
+
|
|
105
|
+
def policyfile_lock_expanded_path
|
|
106
|
+
storage_config.policyfile_lock_expanded_path
|
|
107
|
+
end
|
|
108
|
+
|
|
109
|
+
end
|
|
110
|
+
|
|
111
|
+
end
|
|
112
|
+
end
|
|
@@ -0,0 +1,139 @@
|
|
|
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_relative "../exceptions"
|
|
19
|
+
|
|
20
|
+
module ChefCLI
|
|
21
|
+
module Policyfile
|
|
22
|
+
|
|
23
|
+
class UndoRecord
|
|
24
|
+
|
|
25
|
+
PolicyGroupRestoreData = Struct.new(:policy_name, :policy_group, :data) do
|
|
26
|
+
|
|
27
|
+
def load(data)
|
|
28
|
+
self.policy_name = data["policy_name"]
|
|
29
|
+
self.policy_group = data["policy_group"]
|
|
30
|
+
self.data = data["data"]
|
|
31
|
+
self
|
|
32
|
+
end
|
|
33
|
+
|
|
34
|
+
def for_serialization
|
|
35
|
+
{
|
|
36
|
+
"policy_name" => policy_name,
|
|
37
|
+
"policy_group" => policy_group,
|
|
38
|
+
"data" => data,
|
|
39
|
+
}
|
|
40
|
+
end
|
|
41
|
+
|
|
42
|
+
end
|
|
43
|
+
|
|
44
|
+
attr_reader :policy_groups
|
|
45
|
+
|
|
46
|
+
attr_reader :policy_revisions
|
|
47
|
+
|
|
48
|
+
attr_accessor :description
|
|
49
|
+
|
|
50
|
+
def initialize
|
|
51
|
+
reset!
|
|
52
|
+
end
|
|
53
|
+
|
|
54
|
+
def ==(other)
|
|
55
|
+
other.kind_of?(UndoRecord) &&
|
|
56
|
+
other.policy_groups == policy_groups &&
|
|
57
|
+
other.policy_revisions == policy_revisions
|
|
58
|
+
end
|
|
59
|
+
|
|
60
|
+
def add_policy_group(name)
|
|
61
|
+
@policy_groups << name
|
|
62
|
+
end
|
|
63
|
+
|
|
64
|
+
def add_policy_revision(policy_name, policy_group, data)
|
|
65
|
+
@policy_revisions << PolicyGroupRestoreData.new(policy_name, policy_group, data)
|
|
66
|
+
end
|
|
67
|
+
|
|
68
|
+
def load(data)
|
|
69
|
+
reset!
|
|
70
|
+
|
|
71
|
+
unless data.kind_of?(Hash)
|
|
72
|
+
raise InvalidUndoRecord, "Undo data is incorrectly formatted. Must be a Hash, got '#{data}'."
|
|
73
|
+
end
|
|
74
|
+
missing_fields = %w{ format_version description backup_data }.select { |key| !data.key?(key) }
|
|
75
|
+
unless missing_fields.empty?
|
|
76
|
+
raise InvalidUndoRecord, "Undo data is missing mandatory field(s) #{missing_fields.join(', ')}. Undo data: '#{data}'"
|
|
77
|
+
end
|
|
78
|
+
|
|
79
|
+
@description = data["description"]
|
|
80
|
+
|
|
81
|
+
policy_data = data["backup_data"]
|
|
82
|
+
unless policy_data.kind_of?(Hash)
|
|
83
|
+
raise InvalidUndoRecord, "'backup_data' in the undo record is incorrectly formatted. Must be a Hash, got '#{policy_data}'"
|
|
84
|
+
end
|
|
85
|
+
missing_policy_data_fields = %w{ policy_groups policy_revisions }.select { |key| !policy_data.key?(key) }
|
|
86
|
+
unless missing_policy_data_fields.empty?
|
|
87
|
+
raise InvalidUndoRecord,
|
|
88
|
+
"'backup_data' in the undo record is missing mandatory field(s) #{missing_policy_data_fields.join(', ')}. Backup data: #{policy_data}"
|
|
89
|
+
end
|
|
90
|
+
|
|
91
|
+
policy_groups = policy_data["policy_groups"]
|
|
92
|
+
|
|
93
|
+
unless policy_groups.kind_of?(Array)
|
|
94
|
+
raise InvalidUndoRecord,
|
|
95
|
+
"'policy_groups' data in the undo record is incorrectly formatted. Must be an Array, got '#{policy_groups}'"
|
|
96
|
+
end
|
|
97
|
+
|
|
98
|
+
@policy_groups = policy_groups
|
|
99
|
+
|
|
100
|
+
policy_revisions = policy_data["policy_revisions"]
|
|
101
|
+
unless policy_revisions.kind_of?(Array)
|
|
102
|
+
raise InvalidUndoRecord,
|
|
103
|
+
"'policy_revisions' data in the undo record is incorrectly formatted. Must be an Array, got '#{policy_revisions}'"
|
|
104
|
+
end
|
|
105
|
+
|
|
106
|
+
policy_revisions.each do |revision|
|
|
107
|
+
unless revision.kind_of?(Hash)
|
|
108
|
+
raise InvalidUndoRecord,
|
|
109
|
+
"Invalid item in 'policy_revisions' in the undo record. Must be a Hash, got '#{revision}'"
|
|
110
|
+
end
|
|
111
|
+
|
|
112
|
+
@policy_revisions << PolicyGroupRestoreData.new.load(revision)
|
|
113
|
+
end
|
|
114
|
+
|
|
115
|
+
self
|
|
116
|
+
end
|
|
117
|
+
|
|
118
|
+
def for_serialization
|
|
119
|
+
{
|
|
120
|
+
"format_version" => 1,
|
|
121
|
+
"description" => description,
|
|
122
|
+
"backup_data" => {
|
|
123
|
+
"policy_groups" => policy_groups,
|
|
124
|
+
"policy_revisions" => policy_revisions.map(&:for_serialization),
|
|
125
|
+
},
|
|
126
|
+
}
|
|
127
|
+
end
|
|
128
|
+
|
|
129
|
+
private
|
|
130
|
+
|
|
131
|
+
def reset!
|
|
132
|
+
@description = ""
|
|
133
|
+
@policy_groups = []
|
|
134
|
+
@policy_revisions = []
|
|
135
|
+
end
|
|
136
|
+
|
|
137
|
+
end
|
|
138
|
+
end
|
|
139
|
+
end
|
|
@@ -0,0 +1,128 @@
|
|
|
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 "fileutils" unless defined?(FileUtils)
|
|
19
|
+
|
|
20
|
+
require "ffi_yajl" unless defined?(FFI_Yajl)
|
|
21
|
+
|
|
22
|
+
require_relative "../helpers"
|
|
23
|
+
require_relative "undo_record"
|
|
24
|
+
|
|
25
|
+
module ChefCLI
|
|
26
|
+
module Policyfile
|
|
27
|
+
|
|
28
|
+
class UndoStack
|
|
29
|
+
|
|
30
|
+
MAX_SIZE = 10
|
|
31
|
+
|
|
32
|
+
include Helpers
|
|
33
|
+
|
|
34
|
+
def undo_dir
|
|
35
|
+
File.join(Helpers.package_home, "undo")
|
|
36
|
+
end
|
|
37
|
+
|
|
38
|
+
def size
|
|
39
|
+
undo_record_files.size
|
|
40
|
+
end
|
|
41
|
+
|
|
42
|
+
def empty?
|
|
43
|
+
size == 0
|
|
44
|
+
end
|
|
45
|
+
|
|
46
|
+
def has_id?(id)
|
|
47
|
+
File.exist?(undo_file_for(id))
|
|
48
|
+
end
|
|
49
|
+
|
|
50
|
+
def each_with_id
|
|
51
|
+
undo_record_files.each do |filename|
|
|
52
|
+
yield File.basename(filename), load_undo_record(filename)
|
|
53
|
+
end
|
|
54
|
+
end
|
|
55
|
+
|
|
56
|
+
def undo_records
|
|
57
|
+
undo_record_files.map { |f| load_undo_record(f) }
|
|
58
|
+
end
|
|
59
|
+
|
|
60
|
+
def push(undo_record)
|
|
61
|
+
ensure_undo_dir_exists
|
|
62
|
+
|
|
63
|
+
record_id = Time.new.utc.strftime("%Y%m%d%H%M%S")
|
|
64
|
+
path = File.join(undo_dir, record_id)
|
|
65
|
+
|
|
66
|
+
with_file(path) do |f|
|
|
67
|
+
f.print(FFI_Yajl::Encoder.encode(undo_record.for_serialization, pretty: true))
|
|
68
|
+
end
|
|
69
|
+
|
|
70
|
+
records_to_delete = undo_record_files.size - MAX_SIZE
|
|
71
|
+
if records_to_delete > 0
|
|
72
|
+
undo_record_files.take(records_to_delete).each do |file|
|
|
73
|
+
File.unlink(file)
|
|
74
|
+
end
|
|
75
|
+
end
|
|
76
|
+
|
|
77
|
+
self
|
|
78
|
+
end
|
|
79
|
+
|
|
80
|
+
def pop
|
|
81
|
+
file_to_pop = undo_record_files.last
|
|
82
|
+
if file_to_pop.nil?
|
|
83
|
+
raise CantUndo, "No undo records exist in #{undo_dir}"
|
|
84
|
+
end
|
|
85
|
+
|
|
86
|
+
record = load_undo_record(file_to_pop)
|
|
87
|
+
# if this hits an exception, we skip unlink
|
|
88
|
+
yield record if block_given?
|
|
89
|
+
File.unlink(file_to_pop)
|
|
90
|
+
record
|
|
91
|
+
end
|
|
92
|
+
|
|
93
|
+
def delete(id)
|
|
94
|
+
undo_file = undo_file_for(id)
|
|
95
|
+
unless File.exist?(undo_file)
|
|
96
|
+
raise UndoRecordNotFound, "No undo record for id '#{id}' exists at #{undo_file}"
|
|
97
|
+
end
|
|
98
|
+
|
|
99
|
+
record = load_undo_record(undo_file)
|
|
100
|
+
yield record if block_given?
|
|
101
|
+
File.unlink(undo_file)
|
|
102
|
+
record
|
|
103
|
+
end
|
|
104
|
+
|
|
105
|
+
private
|
|
106
|
+
|
|
107
|
+
def undo_file_for(id)
|
|
108
|
+
File.join(undo_dir, id)
|
|
109
|
+
end
|
|
110
|
+
|
|
111
|
+
def load_undo_record(file)
|
|
112
|
+
data = FFI_Yajl::Parser.parse(IO.read(file))
|
|
113
|
+
UndoRecord.new.load(data)
|
|
114
|
+
end
|
|
115
|
+
|
|
116
|
+
def undo_record_files
|
|
117
|
+
Dir[File.join(undo_dir, "*")].sort
|
|
118
|
+
end
|
|
119
|
+
|
|
120
|
+
def ensure_undo_dir_exists
|
|
121
|
+
return false if File.directory?(undo_dir)
|
|
122
|
+
|
|
123
|
+
FileUtils.mkdir_p(undo_dir)
|
|
124
|
+
end
|
|
125
|
+
end
|
|
126
|
+
|
|
127
|
+
end
|
|
128
|
+
end
|
|
@@ -0,0 +1,222 @@
|
|
|
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 "chef/cookbook_uploader"
|
|
19
|
+
require_relative "read_cookbook_for_compat_mode_upload"
|
|
20
|
+
|
|
21
|
+
require_relative "../ui"
|
|
22
|
+
require_relative "reports/upload"
|
|
23
|
+
|
|
24
|
+
module ChefCLI
|
|
25
|
+
module Policyfile
|
|
26
|
+
class Uploader
|
|
27
|
+
|
|
28
|
+
LockedCookbookForUpload = Struct.new(:cookbook, :lock)
|
|
29
|
+
|
|
30
|
+
COMPAT_MODE_DATA_BAG_NAME = "policyfiles".freeze
|
|
31
|
+
|
|
32
|
+
attr_reader :policyfile_lock
|
|
33
|
+
attr_reader :policy_group
|
|
34
|
+
attr_reader :http_client
|
|
35
|
+
attr_reader :ui
|
|
36
|
+
|
|
37
|
+
def initialize(policyfile_lock, policy_group, ui: nil, http_client: nil, policy_document_native_api: false)
|
|
38
|
+
@policyfile_lock = policyfile_lock
|
|
39
|
+
@policy_group = policy_group
|
|
40
|
+
@http_client = http_client
|
|
41
|
+
@ui = ui || UI.null
|
|
42
|
+
@policy_document_native_api = policy_document_native_api
|
|
43
|
+
|
|
44
|
+
@policy_lock_for_transport = nil
|
|
45
|
+
@cookbook_versions_for_policy = nil
|
|
46
|
+
end
|
|
47
|
+
|
|
48
|
+
def policy_name
|
|
49
|
+
policyfile_lock.name
|
|
50
|
+
end
|
|
51
|
+
|
|
52
|
+
def upload
|
|
53
|
+
ui.msg("Uploading policy #{policy_name} (#{short_revision_id}) to policy group #{policy_group}")
|
|
54
|
+
|
|
55
|
+
if !using_policy_document_native_api?
|
|
56
|
+
ui.msg(<<~DRAGONS)
|
|
57
|
+
WARN: Uploading policy to policy group #{policy_group} in compatibility mode.
|
|
58
|
+
Cookbooks will be uploaded with very large version numbers, which may be picked
|
|
59
|
+
up by existing nodes.
|
|
60
|
+
DRAGONS
|
|
61
|
+
end
|
|
62
|
+
|
|
63
|
+
upload_cookbooks
|
|
64
|
+
upload_policy
|
|
65
|
+
end
|
|
66
|
+
|
|
67
|
+
def upload_policy
|
|
68
|
+
if using_policy_document_native_api?
|
|
69
|
+
upload_policy_native
|
|
70
|
+
else
|
|
71
|
+
data_bag_create
|
|
72
|
+
data_bag_item_create
|
|
73
|
+
end
|
|
74
|
+
end
|
|
75
|
+
|
|
76
|
+
def upload_policy_native
|
|
77
|
+
http_client.put("/policy_groups/#{policy_group}/policies/#{policy_name}", policy_lock_for_transport)
|
|
78
|
+
end
|
|
79
|
+
|
|
80
|
+
def data_bag_create
|
|
81
|
+
http_client.post("data", { "name" => COMPAT_MODE_DATA_BAG_NAME })
|
|
82
|
+
rescue Net::HTTPServerException => e
|
|
83
|
+
raise e unless e.response.code == "409"
|
|
84
|
+
end
|
|
85
|
+
|
|
86
|
+
def data_bag_item_create
|
|
87
|
+
policy_id = "#{policy_name}-#{policy_group}"
|
|
88
|
+
lock_data = policy_lock_for_transport.dup
|
|
89
|
+
|
|
90
|
+
lock_data["id"] = policy_id
|
|
91
|
+
|
|
92
|
+
data_item = {
|
|
93
|
+
"id" => policy_id,
|
|
94
|
+
"name" => "data_bag_item_#{COMPAT_MODE_DATA_BAG_NAME}_#{policy_id}",
|
|
95
|
+
"data_bag" => COMPAT_MODE_DATA_BAG_NAME,
|
|
96
|
+
"raw_data" => lock_data,
|
|
97
|
+
# we'd prefer to leave this out, but the "compatibility mode"
|
|
98
|
+
# implementation in chef-client relies on magical class inflation
|
|
99
|
+
"json_class" => "Chef::DataBagItem",
|
|
100
|
+
}
|
|
101
|
+
|
|
102
|
+
upload_lockfile_as_data_bag_item(policy_id, data_item)
|
|
103
|
+
ui.msg("Policy uploaded as data bag item #{COMPAT_MODE_DATA_BAG_NAME}/#{policy_id}")
|
|
104
|
+
true
|
|
105
|
+
end
|
|
106
|
+
|
|
107
|
+
def uploader
|
|
108
|
+
# TODO: uploader runs cookbook validation; we want to do this at a different time.
|
|
109
|
+
@uploader ||= Chef::CookbookUploader.new(cookbook_versions_to_upload, rest: http_client, policy_mode: using_policy_document_native_api?)
|
|
110
|
+
end
|
|
111
|
+
|
|
112
|
+
def cookbook_versions_to_upload
|
|
113
|
+
cookbook_versions_for_policy.inject([]) do |versions_to_upload, cookbook_with_lock|
|
|
114
|
+
cb = cookbook_with_lock.cookbook
|
|
115
|
+
# When we abandon custom identifier support in favor of the one true
|
|
116
|
+
# hash, identifier generation code can be moved into chef proper and
|
|
117
|
+
# this can be removed.
|
|
118
|
+
cb.identifier = cookbook_with_lock.lock.identifier
|
|
119
|
+
|
|
120
|
+
versions_to_upload << cb unless remote_already_has_cookbook?(cb)
|
|
121
|
+
versions_to_upload
|
|
122
|
+
end
|
|
123
|
+
end
|
|
124
|
+
|
|
125
|
+
def remote_already_has_cookbook?(cookbook)
|
|
126
|
+
return false unless existing_cookbook_on_remote.key?(cookbook.name.to_s)
|
|
127
|
+
|
|
128
|
+
if using_policy_document_native_api?
|
|
129
|
+
native_mode_cookbook_exists_on_remote?(cookbook)
|
|
130
|
+
else
|
|
131
|
+
compat_mode_cookbook_exists_on_remote?(cookbook)
|
|
132
|
+
end
|
|
133
|
+
end
|
|
134
|
+
|
|
135
|
+
def native_mode_cookbook_exists_on_remote?(cookbook)
|
|
136
|
+
existing_cookbook_on_remote[cookbook.name.to_s]["versions"].any? do |cookbook_info|
|
|
137
|
+
cookbook_info["identifier"] == cookbook.identifier
|
|
138
|
+
end
|
|
139
|
+
end
|
|
140
|
+
|
|
141
|
+
def compat_mode_cookbook_exists_on_remote?(cookbook)
|
|
142
|
+
existing_cookbook_on_remote[cookbook.name.to_s]["versions"].any? do |cookbook_info|
|
|
143
|
+
cookbook_info["version"] == cookbook.version
|
|
144
|
+
end
|
|
145
|
+
end
|
|
146
|
+
|
|
147
|
+
def existing_cookbook_on_remote
|
|
148
|
+
@existing_cookbook_on_remote ||= http_client.get(list_cookbooks_url)
|
|
149
|
+
end
|
|
150
|
+
|
|
151
|
+
# An Array of Chef::CookbookVersion objects representing the full set that
|
|
152
|
+
# the policyfile lock requires.
|
|
153
|
+
def cookbook_versions_for_policy
|
|
154
|
+
return @cookbook_versions_for_policy if @cookbook_versions_for_policy
|
|
155
|
+
policyfile_lock.validate_cookbooks!
|
|
156
|
+
@cookbook_versions_for_policy =
|
|
157
|
+
if using_policy_document_native_api?
|
|
158
|
+
load_cookbooks_in_native_mode
|
|
159
|
+
else
|
|
160
|
+
load_cookbooks_in_compat_mode
|
|
161
|
+
end
|
|
162
|
+
end
|
|
163
|
+
|
|
164
|
+
def load_cookbooks_in_native_mode
|
|
165
|
+
policyfile_lock.cookbook_locks.map do |name, lock|
|
|
166
|
+
cb = CookbookLoaderWithChefignore.load(name, lock.cookbook_path)
|
|
167
|
+
LockedCookbookForUpload.new(cb, lock)
|
|
168
|
+
end
|
|
169
|
+
end
|
|
170
|
+
|
|
171
|
+
def load_cookbooks_in_compat_mode
|
|
172
|
+
policyfile_lock.cookbook_locks.map do |name, lock|
|
|
173
|
+
cb = ReadCookbookForCompatModeUpload.load(name, lock.dotted_decimal_identifier, lock.cookbook_path)
|
|
174
|
+
LockedCookbookForUpload.new(cb, lock)
|
|
175
|
+
end
|
|
176
|
+
end
|
|
177
|
+
|
|
178
|
+
def using_policy_document_native_api?
|
|
179
|
+
@policy_document_native_api
|
|
180
|
+
end
|
|
181
|
+
|
|
182
|
+
private
|
|
183
|
+
|
|
184
|
+
def short_revision_id
|
|
185
|
+
policy_lock_for_transport["revision_id"][0, 10]
|
|
186
|
+
end
|
|
187
|
+
|
|
188
|
+
def policy_lock_for_transport
|
|
189
|
+
@policy_lock_for_transport ||= policyfile_lock.to_lock
|
|
190
|
+
end
|
|
191
|
+
|
|
192
|
+
def list_cookbooks_url
|
|
193
|
+
if using_policy_document_native_api?
|
|
194
|
+
"cookbook_artifacts?num_versions=all"
|
|
195
|
+
else
|
|
196
|
+
"cookbooks?num_versions=all"
|
|
197
|
+
end
|
|
198
|
+
end
|
|
199
|
+
|
|
200
|
+
def upload_cookbooks
|
|
201
|
+
ui.msg("WARN: Uploading cookbooks using semver compat mode") unless using_policy_document_native_api?
|
|
202
|
+
|
|
203
|
+
uploader.upload_cookbooks unless cookbook_versions_to_upload.empty?
|
|
204
|
+
|
|
205
|
+
reused_cbs, uploaded_cbs = cookbook_versions_for_policy.partition do |cb_with_lock|
|
|
206
|
+
remote_already_has_cookbook?(cb_with_lock.cookbook)
|
|
207
|
+
end
|
|
208
|
+
|
|
209
|
+
Reports::Upload.new(reused_cbs: reused_cbs, uploaded_cbs: uploaded_cbs, ui: ui).show
|
|
210
|
+
|
|
211
|
+
true
|
|
212
|
+
end
|
|
213
|
+
|
|
214
|
+
def upload_lockfile_as_data_bag_item(policy_id, data_item)
|
|
215
|
+
http_client.put("data/#{COMPAT_MODE_DATA_BAG_NAME}/#{policy_id}", data_item)
|
|
216
|
+
rescue Net::HTTPServerException => e
|
|
217
|
+
raise e unless e.response.code == "404"
|
|
218
|
+
http_client.post("data/#{COMPAT_MODE_DATA_BAG_NAME}", data_item)
|
|
219
|
+
end
|
|
220
|
+
end
|
|
221
|
+
end
|
|
222
|
+
end
|