2018-12-13 13:39:08 +05:30
|
|
|
# frozen_string_literal: true
|
|
|
|
|
|
|
|
module Gitlab
|
|
|
|
module Ci
|
|
|
|
class Config
|
|
|
|
module External
|
|
|
|
class Mapper
|
2019-02-15 15:39:39 +05:30
|
|
|
include Gitlab::Utils::StrongMemoize
|
|
|
|
|
2019-05-18 00:54:41 +05:30
|
|
|
MAX_INCLUDES = 50
|
|
|
|
|
2019-02-15 15:39:39 +05:30
|
|
|
FILE_CLASSES = [
|
|
|
|
External::File::Remote,
|
|
|
|
External::File::Template,
|
|
|
|
External::File::Local,
|
|
|
|
External::File::Project
|
|
|
|
].freeze
|
|
|
|
|
2019-05-18 00:54:41 +05:30
|
|
|
Error = Class.new(StandardError)
|
|
|
|
AmbigiousSpecificationError = Class.new(Error)
|
|
|
|
DuplicateIncludesError = Class.new(Error)
|
|
|
|
TooManyIncludesError = Class.new(Error)
|
|
|
|
|
|
|
|
def initialize(values, project:, sha:, user:, expandset:)
|
|
|
|
raise Error, 'Expanded needs to be `Set`' unless expandset.is_a?(Set)
|
2019-02-15 15:39:39 +05:30
|
|
|
|
|
|
|
@locations = Array.wrap(values.fetch(:include, []))
|
2018-12-13 13:39:08 +05:30
|
|
|
@project = project
|
|
|
|
@sha = sha
|
2019-02-15 15:39:39 +05:30
|
|
|
@user = user
|
2019-05-18 00:54:41 +05:30
|
|
|
@expandset = expandset
|
2018-12-13 13:39:08 +05:30
|
|
|
end
|
|
|
|
|
|
|
|
def process
|
2019-05-18 00:54:41 +05:30
|
|
|
return [] if locations.empty?
|
|
|
|
|
2019-02-15 15:39:39 +05:30
|
|
|
locations
|
|
|
|
.compact
|
|
|
|
.map(&method(:normalize_location))
|
2019-05-18 00:54:41 +05:30
|
|
|
.each(&method(:verify_duplicates!))
|
2019-02-15 15:39:39 +05:30
|
|
|
.map(&method(:select_first_matching))
|
2018-12-13 13:39:08 +05:30
|
|
|
end
|
|
|
|
|
|
|
|
private
|
|
|
|
|
2019-05-18 00:54:41 +05:30
|
|
|
attr_reader :locations, :project, :sha, :user, :expandset
|
2019-02-15 15:39:39 +05:30
|
|
|
|
|
|
|
# convert location if String to canonical form
|
|
|
|
def normalize_location(location)
|
|
|
|
if location.is_a?(String)
|
|
|
|
normalize_location_string(location)
|
|
|
|
else
|
|
|
|
location.deep_symbolize_keys
|
|
|
|
end
|
|
|
|
end
|
2018-12-13 13:39:08 +05:30
|
|
|
|
2019-02-15 15:39:39 +05:30
|
|
|
def normalize_location_string(location)
|
2018-12-13 13:39:08 +05:30
|
|
|
if ::Gitlab::UrlSanitizer.valid?(location)
|
2019-02-15 15:39:39 +05:30
|
|
|
{ remote: location }
|
2018-12-13 13:39:08 +05:30
|
|
|
else
|
2019-02-15 15:39:39 +05:30
|
|
|
{ local: location }
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2019-05-18 00:54:41 +05:30
|
|
|
def verify_duplicates!(location)
|
|
|
|
if expandset.count >= MAX_INCLUDES
|
|
|
|
raise TooManyIncludesError, "Maximum of #{MAX_INCLUDES} nested includes are allowed!"
|
|
|
|
end
|
|
|
|
|
|
|
|
# We scope location to context, as this allows us to properly support
|
|
|
|
# relative incldues, and similarly looking relative in another project
|
|
|
|
# does not trigger duplicate error
|
|
|
|
scoped_location = location.merge(
|
|
|
|
context_project: project,
|
|
|
|
context_sha: sha)
|
|
|
|
|
|
|
|
unless expandset.add?(scoped_location)
|
|
|
|
raise DuplicateIncludesError, "Include `#{location.to_json}` was already included!"
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2019-02-15 15:39:39 +05:30
|
|
|
def select_first_matching(location)
|
|
|
|
matching = FILE_CLASSES.map do |file_class|
|
|
|
|
file_class.new(location, context)
|
|
|
|
end.select(&:matching?)
|
|
|
|
|
|
|
|
raise AmbigiousSpecificationError, "Include `#{location.to_json}` needs to match exactly one accessor!" unless matching.one?
|
|
|
|
|
|
|
|
matching.first
|
|
|
|
end
|
|
|
|
|
|
|
|
def context
|
|
|
|
strong_memoize(:context) do
|
2019-05-18 00:54:41 +05:30
|
|
|
External::File::Base::Context.new(project, sha, user, expandset)
|
2018-12-13 13:39:08 +05:30
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|