debian-mirror-gitlab/lib/gitlab/ci/config/external/file/base.rb

Ignoring revisions in .git-blame-ignore-revs. Click here to bypass and see the normal blame view.

177 lines
5.1 KiB
Ruby
Raw Permalink Normal View History

2018-12-13 13:39:08 +05:30
# frozen_string_literal: true
module Gitlab
module Ci
class Config
module External
module File
class Base
include Gitlab::Utils::StrongMemoize
2019-02-15 15:39:39 +05:30
attr_reader :location, :params, :context, :errors
2018-12-13 13:39:08 +05:30
YAML_WHITELIST_EXTENSION = /.+\.(yml|yaml)$/i.freeze
2019-02-15 15:39:39 +05:30
def initialize(params, context)
@params = params
@context = context
2018-12-13 13:39:08 +05:30
@errors = []
end
2019-02-15 15:39:39 +05:30
def matching?
location.present?
end
2019-12-04 20:38:33 +05:30
def invalid_location_type?
!location.is_a?(String)
end
2018-12-13 13:39:08 +05:30
def invalid_extension?
2019-02-15 15:39:39 +05:30
location.nil? || !::File.basename(location).match?(YAML_WHITELIST_EXTENSION)
2018-12-13 13:39:08 +05:30
end
def valid?
errors.none?
end
def error_message
errors.first
end
def content
raise NotImplementedError, 'subclass must implement fetching raw content'
end
def to_hash
2019-07-07 11:18:12 +05:30
expanded_content_hash
2019-05-18 00:54:41 +05:30
end
2022-06-21 17:19:12 +05:30
def metadata
{
context_project: context.project&.full_path,
context_sha: context.sha
}
end
def eql?(other)
other.hash == hash
end
def hash
[params, context.project&.full_path, context.sha].hash
end
2023-05-27 22:25:52 +05:30
# This method is overridden to load context into the memoized result
# or to lazily load context via BatchLoader
def preload_context
# no-op
end
def preload_content
# calling the `content` method either loads content into the memoized result
# or lazily loads it via BatchLoader
content
end
2018-12-13 13:39:08 +05:30
def validate_location!
2019-12-04 20:38:33 +05:30
if invalid_location_type?
2022-04-01 21:47:47 +05:30
errors.push("Included file `#{masked_location}` needs to be a string")
2019-12-04 20:38:33 +05:30
elsif invalid_extension?
2022-04-01 21:47:47 +05:30
errors.push("Included file `#{masked_location}` does not have YAML extension!")
2018-12-13 13:39:08 +05:30
end
end
2023-01-13 00:05:48 +05:30
def validate_context!
2023-06-20 00:43:36 +05:30
raise NotImplementedError, 'subclass must implement `validate_context!`'
2023-01-13 00:05:48 +05:30
end
2023-04-23 21:23:45 +05:30
def validate_content!
2023-06-20 00:43:36 +05:30
errors.push("Included file `#{masked_location}` is empty or does not exist!") if content.blank?
end
def load_and_validate_expanded_hash!
context.logger.instrument(:config_file_fetch_content_hash) do
content_result # calling the method loads YAML then memoizes the content result
end
context.logger.instrument(:config_file_interpolate_result) do
interpolator.interpolate!
end
return validate_interpolation! unless interpolator.valid?
context.logger.instrument(:config_file_expand_content_includes) do
expanded_content_hash # calling the method expands then memoizes the result
2023-01-13 00:05:48 +05:30
end
2023-06-20 00:43:36 +05:30
validate_hash!
2023-01-13 00:05:48 +05:30
end
2023-04-23 21:23:45 +05:30
protected
2023-01-13 00:05:48 +05:30
2023-05-27 22:25:52 +05:30
def content_result
2023-06-20 00:43:36 +05:30
::Gitlab::Ci::Config::Yaml
.load_result!(content, project: context.project)
end
strong_memoize_attr :content_result
def content_inputs
2023-07-09 08:55:56 +05:30
# TODO: remove support for `with` syntax in 16.1, see https://gitlab.com/gitlab-org/gitlab/-/issues/408369
# In the interim prefer `inputs` over `with` while allow either syntax.
params.to_h.slice(:inputs, :with).each_value.first
2023-05-27 22:25:52 +05:30
end
2023-06-20 00:43:36 +05:30
strong_memoize_attr :content_inputs
2023-05-27 22:25:52 +05:30
def content_hash
2023-06-20 00:43:36 +05:30
interpolator.interpolate!
interpolator.to_hash
end
strong_memoize_attr :content_hash
2023-05-27 22:25:52 +05:30
2023-06-20 00:43:36 +05:30
def interpolator
External::Interpolator
.new(content_result, content_inputs, context)
2023-05-27 22:25:52 +05:30
end
2023-06-20 00:43:36 +05:30
strong_memoize_attr :interpolator
2023-05-27 22:25:52 +05:30
2023-04-23 21:23:45 +05:30
def expanded_content_hash
2023-06-20 00:43:36 +05:30
return if content_hash.blank?
2023-01-13 00:05:48 +05:30
2023-04-23 21:23:45 +05:30
strong_memoize(:expanded_content_hash) do
expand_includes(content_hash)
end
2023-01-13 00:05:48 +05:30
end
2018-12-13 13:39:08 +05:30
def validate_hash!
if to_hash.blank?
2022-04-01 21:47:47 +05:30
errors.push("Included file `#{masked_location}` does not have valid YAML syntax!")
2018-12-13 13:39:08 +05:30
end
end
2019-07-07 11:18:12 +05:30
2023-06-20 00:43:36 +05:30
def validate_interpolation!
return if interpolator.valid?
errors.push("`#{masked_location}`: #{interpolator.error_message}")
end
2019-07-07 11:18:12 +05:30
def expand_includes(hash)
2019-12-21 20:55:43 +05:30
External::Processor.new(hash, context.mutate(expand_context_attrs)).perform
2019-07-07 11:18:12 +05:30
end
2019-12-21 20:55:43 +05:30
def expand_context_attrs
{}
2019-07-07 11:18:12 +05:30
end
2022-04-01 21:47:47 +05:30
def masked_location
strong_memoize(:masked_location) do
context.mask_variables_from(location)
end
end
2018-12-13 13:39:08 +05:30
end
end
end
end
end
end