# # Copyright:: Copyright (c) 2014-2018 Chef Software Inc. # License:: Apache License, Version 2.0 # # Licensed under the Apache License, Version 2.0 (the "License"); # you may not use this file except in compliance with the License. # You may obtain a copy of the License at # # http://www.apache.org/licenses/LICENSE-2.0 # # Unless required by applicable law or agreed to in writing, software # distributed under the License is distributed on an "AS IS" BASIS, # WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. # See the License for the specific language governing permissions and # limitations under the License. # require "semverse" require_relative "../cookbook_omnifetch" require_relative "storage_config" module ChefCLI module Policyfile class CookbookLocationSpecification #-- # Provides #relative_paths_root, which is required by CookbookOmnifetch # API contract include StorageConfigDelegation SOURCE_TYPES = %i{git github path artifactserver chef_server chef_server_artifact artifactory}.freeze #-- # Required by CookbookOmnifetch API contract attr_reader :version_constraint #-- # Required by CookbookOmnifetch API contract attr_reader :name #-- # Required by CookbookOmnifetch API contract attr_reader :source_options attr_reader :source_type attr_reader :storage_config def initialize(name, version_constraint, source_options, storage_config) @name = name @version_constraint = Semverse::Constraint.new(version_constraint) @source_options = source_options @source_type = SOURCE_TYPES.find { |type| source_options.key?(type) } @storage_config = storage_config end def ==(other) other.is_a?(self.class) && other.name == name && other.version_constraint == version_constraint && other.source_options == source_options end def to_s # Note, this may appear in exceptions s = "Cookbook '#{name}'" s << " #{version_constraint}" s << " #{source_options}" unless source_options.empty? s end def mirrors_canonical_upstream? %i{git github artifactserver chef_server chef_server_artifact artifactory}.include?(source_type) end def installed? installer.installed? end def ensure_cached unless installer.installed? installer.install end end def valid? errors.empty? end def errors error_messages = [] if source_options_invalid? error_messages << "Cookbook `#{name}' has invalid source options `#{source_options.inspect}'" end error_messages end def installer # TODO: handle 'bad' return values here (invalid source_options, etc.) @installer ||= CookbookOmnifetch.init(self, source_options) end def cache_key installer.cache_key end def relative_path installer.relative_path.to_s end def uri installer.uri end def version_fixed? %i{git github path chef_server_artifact}.include?(@source_type) end def version cached_cookbook.version end def dependencies cached_cookbook.dependencies end def cookbook_has_recipe?(recipe_name) expected_path = cookbook_path.join("recipes/#{recipe_name}.rb") expected_path_yml = cookbook_path.join("recipes/#{recipe_name}.yml") expected_path.exist? || expected_path_yml.exist? end def cached_cookbook # TODO: handle 'bad' return values here (cookbook not installed yet) installer.cached_cookbook end def source_options_for_lock installer.lock_data end def source_options_invalid? !source_options.empty? && installer.nil? end def cookbook_path if installer.respond_to?(:expanded_path) installer.expanded_path else installer.install_path.expand_path end end end end end