blob: 0001a259281fdb5a1d5d0616d464d276dcb870b1 (
plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
|
# frozen_string_literal: true
module Gitlab
module Ci
class Config
class Extendable
class Entry
include Gitlab::Utils::StrongMemoize
InvalidExtensionError = Class.new(Extendable::ExtensionError)
CircularDependencyError = Class.new(Extendable::ExtensionError)
NestingTooDeepError = Class.new(Extendable::ExtensionError)
MAX_NESTING_LEVELS = 10
attr_reader :key
def initialize(key, context, parent = nil)
@key = key
@context = context
@parent = parent
unless @context.key?(@key)
raise StandardError, 'Invalid entry key!'
end
end
def extensible?
value.is_a?(Hash) && value.key?(:extends)
end
def value
strong_memoize(:value) do
@context.fetch(@key)
end
end
def base_hashes!
strong_memoize(:base_hashes) do
extends_keys.map do |key|
Extendable::Entry
.new(key, @context, self)
.extend!
end
end
end
def extends_keys
strong_memoize(:extends_keys) do
next unless extensible?
Array(value.fetch(:extends)).map(&:to_s).map(&:to_sym)
end
end
def ancestors
strong_memoize(:ancestors) do
Array(@parent&.ancestors) + Array(@parent&.key)
end
end
def extend!
return value unless extensible?
if unknown_extensions.any?
raise Entry::InvalidExtensionError,
"#{key}: unknown keys in `extends` (#{show_keys(unknown_extensions)})"
end
if invalid_bases.any?
raise Entry::InvalidExtensionError,
"#{key}: invalid base hashes in `extends` (#{show_keys(invalid_bases)})"
end
if nesting_too_deep?
raise Entry::NestingTooDeepError,
"#{key}: nesting too deep in `extends`"
end
if circular_dependency?
raise Entry::CircularDependencyError,
"#{key}: circular dependency detected in `extends`"
end
merged = {}
base_hashes!.each { |h| merged.deep_merge!(h) }
@context[key] = merged.deep_merge!(value)
end
private
def show_keys(keys)
keys.join(', ')
end
def nesting_too_deep?
ancestors.count > MAX_NESTING_LEVELS
end
def circular_dependency?
ancestors.include?(key)
end
def unknown_extensions
strong_memoize(:unknown_extensions) do
extends_keys.reject { |key| @context.key?(key) }
end
end
def invalid_bases
strong_memoize(:invalid_bases) do
extends_keys.reject { |key| @context[key].is_a?(Hash) }
end
end
end
end
end
end
end
|