lono 8.0.0.pre.rc1 → 8.0.0.pre.rc4

Sign up to get free protection for your applications and to get access to all the features.
Files changed (218) hide show
  1. checksums.yaml +4 -4
  2. data/.cody/acceptance/bin/build.sh +5 -5
  3. data/.github/ISSUE_TEMPLATE/bug_report.md +84 -0
  4. data/.github/ISSUE_TEMPLATE/documentation.md +12 -0
  5. data/.github/ISSUE_TEMPLATE/feature_request.md +64 -0
  6. data/.github/ISSUE_TEMPLATE/question.md +14 -0
  7. data/.github/ISSUE_TEMPLATE.md +7 -0
  8. data/.github/PULL_REQUEST_TEMPLATE.md +50 -0
  9. data/CHANGELOG.md +59 -2
  10. data/Gemfile +1 -3
  11. data/README.md +2 -7
  12. data/lib/lono/app/callable_option/concern.rb +12 -0
  13. data/lib/lono/app/callable_option.rb +56 -0
  14. data/lib/lono/app.rb +56 -20
  15. data/lib/lono/autoloader.rb +1 -0
  16. data/lib/lono/aws_services/helper.rb +0 -2
  17. data/lib/lono/blueprint.rb +1 -18
  18. data/lib/lono/builder/allow/base.rb +54 -0
  19. data/lib/lono/builder/allow/env.rb +17 -0
  20. data/lib/lono/builder/allow/region.rb +20 -0
  21. data/lib/lono/builder/allow.rb +8 -0
  22. data/lib/lono/builder/configset/combiner.rb +145 -0
  23. data/lib/lono/builder/configset/definition/base.rb +47 -0
  24. data/lib/lono/builder/configset/definition/context.rb +70 -0
  25. data/lib/lono/{configset/strategy/helpers/dsl → builder/configset/definition/dsl/syntax}/auth.rb +1 -1
  26. data/lib/lono/builder/configset/definition/dsl/syntax/content.rb +22 -0
  27. data/lib/lono/{configset/strategy/helpers/dsl/syntax.rb → builder/configset/definition/dsl/syntax/core.rb} +15 -9
  28. data/lib/lono/{configset/strategy/helpers/dsl → builder/configset/definition/dsl/syntax}/package.rb +1 -1
  29. data/lib/lono/builder/configset/definition/dsl/syntax.rb +5 -0
  30. data/lib/lono/{configset/strategy → builder/configset/definition}/dsl.rb +10 -14
  31. data/lib/lono/{configset/strategy → builder/configset/definition}/erb.rb +13 -16
  32. data/lib/lono/builder/configset/definition.rb +18 -0
  33. data/lib/lono/builder/configset/evaluator.rb +10 -0
  34. data/lib/lono/builder/configset/registration.rb +35 -0
  35. data/lib/lono/builder/context.rb +33 -18
  36. data/lib/lono/builder/dsl/evaluator.rb +45 -0
  37. data/lib/lono/builder/dsl/finalizer/base.rb +8 -0
  38. data/lib/lono/builder/dsl/finalizer/configsets.rb +20 -0
  39. data/lib/lono/builder/dsl/finalizer/files/base.rb +4 -0
  40. data/lib/lono/builder/dsl/finalizer/files/build.rb +50 -0
  41. data/lib/lono/builder/dsl/finalizer/files/replace.rb +36 -0
  42. data/lib/lono/builder/dsl/finalizer/files.rb +9 -0
  43. data/lib/lono/builder/{template/dsl → dsl}/finalizer/parameter_groups.rb +6 -5
  44. data/lib/lono/builder/dsl/finalizer.rb +15 -0
  45. data/lib/lono/builder/{template/dsl/evaluator/helpers/ec2_helper.rb → dsl/helpers/ec2.rb} +3 -3
  46. data/lib/lono/builder/dsl/helpers/files.rb +7 -0
  47. data/lib/lono/builder/{template/helpers.rb → dsl/helpers/partials.rb} +48 -57
  48. data/lib/lono/builder/dsl/helpers/s3.rb +14 -0
  49. data/lib/lono/builder/{context/ssm_fetcher.rb → dsl/helpers/ssm/fetcher.rb} +5 -3
  50. data/lib/lono/builder/{context/helpers.rb → dsl/helpers/ssm.rb} +3 -3
  51. data/lib/lono/builder/{template/dsl/evaluator/helpers/stack_helper.rb → dsl/helpers/stack.rb} +6 -6
  52. data/lib/lono/builder/{template/dsl/evaluator/helpers/tags_helper.rb → dsl/helpers/tags.rb} +2 -2
  53. data/lib/lono/builder/dsl/helpers/template_file.rb +66 -0
  54. data/lib/lono/builder/dsl/helpers.rb +5 -0
  55. data/lib/lono/builder/{template/dsl/evaluator/section → dsl/syntax/core}/base.rb +4 -5
  56. data/lib/lono/builder/{template/dsl/evaluator/section → dsl/syntax/core}/condition.rb +1 -1
  57. data/lib/lono/builder/{template/dsl/evaluator/section → dsl/syntax/core}/mapping.rb +1 -1
  58. data/lib/lono/builder/{template/dsl/evaluator/section → dsl/syntax/core}/output.rb +1 -1
  59. data/lib/lono/builder/{template/dsl/evaluator/section → dsl/syntax/core}/parameter.rb +1 -1
  60. data/lib/lono/builder/{template/dsl/evaluator/section → dsl/syntax/core}/resource/property_mover.rb +12 -2
  61. data/lib/lono/builder/{template/dsl/evaluator/section → dsl/syntax/core}/resource.rb +1 -1
  62. data/lib/lono/builder/{template/dsl/evaluator/section → dsl/syntax/core}/section.rb +1 -1
  63. data/lib/lono/builder/dsl/syntax/core/squeezer.rb +40 -0
  64. data/lib/lono/builder/{template/dsl/evaluator/section/methods.rb → dsl/syntax/core.rb} +2 -2
  65. data/lib/lono/builder/{template/dsl/evaluator → dsl/syntax}/fn.rb +1 -3
  66. data/lib/lono/builder/{template/dsl/evaluator → dsl}/syntax/parameter_group.rb +1 -1
  67. data/lib/lono/builder/dsl/syntax.rb +9 -0
  68. data/lib/lono/builder/{template/dsl.rb → dsl.rb} +2 -2
  69. data/lib/lono/builder/param.rb +5 -2
  70. data/lib/lono/builder/template/upload.rb +2 -15
  71. data/lib/lono/builder/template.rb +1 -3
  72. data/lib/lono/builder/{template/dsl/evaluator → util}/stringify.rb +1 -1
  73. data/lib/lono/builder.rb +39 -0
  74. data/lib/lono/bundler/component/props/extension.rb +0 -1
  75. data/lib/lono/bundler/extract/tar.rb +0 -1
  76. data/lib/lono/bundler/util/git.rb +18 -16
  77. data/lib/lono/cfn/base.rb +2 -2
  78. data/lib/lono/cfn/cancel.rb +5 -5
  79. data/lib/lono/cfn/concerns/build.rb +1 -1
  80. data/lib/lono/cfn/delete.rb +5 -2
  81. data/lib/lono/cfn/deploy/iam.rb +1 -2
  82. data/lib/lono/cfn/deploy/rollback.rb +1 -3
  83. data/lib/lono/cfn/deploy.rb +46 -5
  84. data/lib/lono/cfn/download.rb +0 -1
  85. data/lib/lono/cfn/plan/changeset.rb +3 -3
  86. data/lib/lono/cfn/plan/diff/data.rb +11 -1
  87. data/lib/lono/cfn/plan/diff/file.rb +2 -1
  88. data/lib/lono/cfn/plan/param.rb +1 -1
  89. data/lib/lono/cfn/plan/template.rb +2 -2
  90. data/lib/lono/cfn/plan.rb +1 -0
  91. data/lib/lono/cli/abstract.rb +0 -6
  92. data/lib/lono/cli/base.rb +6 -3
  93. data/lib/lono/cli/clean.rb +3 -2
  94. data/lib/lono/cli/help/cfn/status.md +0 -1
  95. data/lib/lono/cli/help/new/helper.md +27 -0
  96. data/lib/lono/cli/help/new/hook.md +14 -0
  97. data/lib/lono/cli/help/new/project.md +3 -10
  98. data/lib/lono/cli/help/seed.md +22 -21
  99. data/lib/lono/cli/iam.rb +27 -0
  100. data/lib/lono/cli/list.rb +3 -6
  101. data/lib/lono/cli/new/blueprint.rb +3 -9
  102. data/lib/lono/cli/new/configset.rb +11 -44
  103. data/lib/lono/cli/new/helper.rb +19 -11
  104. data/lib/lono/cli/new/hook.rb +32 -0
  105. data/lib/lono/cli/new/sequence.rb +6 -1
  106. data/lib/lono/cli/new.rb +9 -9
  107. data/lib/lono/cli/s3.rb +3 -3
  108. data/lib/lono/cli/script/build.rb +0 -5
  109. data/lib/lono/cli/status.rb +18 -0
  110. data/lib/lono/{inspector → cli}/summary.rb +23 -33
  111. data/lib/lono/cli.rb +16 -16
  112. data/lib/lono/command.rb +28 -2
  113. data/lib/lono/component.rb +29 -0
  114. data/lib/lono/concerns/aws_info.rb +14 -0
  115. data/lib/lono/concerns/names.rb +9 -0
  116. data/lib/lono/configset.rb +17 -0
  117. data/lib/lono/core.rb +11 -1
  118. data/lib/lono/ext/bundler.rb +7 -0
  119. data/lib/lono/ext/core/module.rb +31 -0
  120. data/lib/lono/ext/core/object.rb +32 -0
  121. data/lib/lono/ext/core/string.rb +9 -0
  122. data/lib/lono/ext.rb +4 -0
  123. data/lib/lono/files/base.rb +12 -0
  124. data/lib/lono/{utils → files/builder/lambda_layer}/rsync.rb +3 -5
  125. data/lib/lono/{app_file/build → files/builder}/lambda_layer/ruby_packager.rb +104 -79
  126. data/lib/lono/files/builder/lambda_layer/ruby_version.rb +38 -0
  127. data/lib/lono/files/builder/lambda_layer.rb +17 -0
  128. data/lib/lono/files/builder.rb +37 -0
  129. data/lib/lono/files/compressor.rb +53 -0
  130. data/lib/lono/files/concerns/post_processing.rb +52 -0
  131. data/lib/lono/files/concerns/registration.rb +16 -0
  132. data/lib/lono/files/registry.rb +6 -0
  133. data/lib/lono/files.rb +41 -0
  134. data/lib/lono/hooks/builder.rb +50 -0
  135. data/lib/lono/hooks/concern.rb +9 -0
  136. data/lib/lono/hooks/dsl.rb +20 -0
  137. data/lib/lono/hooks/runner.rb +46 -0
  138. data/lib/lono/layering/layer.rb +64 -56
  139. data/lib/lono/logger.rb +2 -3
  140. data/lib/lono/names.rb +7 -10
  141. data/lib/lono/s3/bucket.rb +43 -37
  142. data/lib/lono/s3/rollback.rb +8 -0
  143. data/lib/lono/s3/uploader.rb +33 -28
  144. data/lib/lono/script/build.rb +0 -5
  145. data/lib/lono/seeder.rb +38 -29
  146. data/lib/lono/user_data.rb +3 -1
  147. data/lib/lono/utils/sh.rb +32 -0
  148. data/lib/lono/utils.rb +5 -0
  149. data/lib/lono/version.rb +1 -1
  150. data/lib/lono/yamler/validator.rb +7 -22
  151. data/lib/lono.rb +8 -0
  152. data/lib/templates/configset/configset.rb +2 -16
  153. data/lib/templates/examples/configset/configset.rb +16 -0
  154. data/lib/templates/helper/%underscore_name%_helper.rb.tt +1 -1
  155. data/lib/templates/hook/config/hooks.rb.tt +14 -0
  156. data/lono.gemspec +1 -1
  157. metadata +99 -93
  158. data/lib/lono/app_file/base.rb +0 -28
  159. data/lib/lono/app_file/build/lambda_layer.rb +0 -20
  160. data/lib/lono/app_file/build.rb +0 -79
  161. data/lib/lono/app_file/registry/item.rb +0 -24
  162. data/lib/lono/app_file/registry.rb +0 -16
  163. data/lib/lono/app_file/upload.rb +0 -12
  164. data/lib/lono/builder/context/loader/load_files.rb +0 -23
  165. data/lib/lono/builder/context/loader.rb +0 -35
  166. data/lib/lono/builder/context/params.rb +0 -7
  167. data/lib/lono/builder/template/configset_injector.rb +0 -52
  168. data/lib/lono/builder/template/dsl/evaluator/helpers/core_helper.rb +0 -14
  169. data/lib/lono/builder/template/dsl/evaluator/helpers/file_helper.rb +0 -44
  170. data/lib/lono/builder/template/dsl/evaluator/helpers/s3_helper.rb +0 -16
  171. data/lib/lono/builder/template/dsl/evaluator/helpers.rb +0 -16
  172. data/lib/lono/builder/template/dsl/evaluator/squeezer.rb +0 -24
  173. data/lib/lono/builder/template/dsl/evaluator/syntax/extend_with.rb +0 -9
  174. data/lib/lono/builder/template/dsl/evaluator/syntax.rb +0 -11
  175. data/lib/lono/builder/template/dsl/evaluator.rb +0 -45
  176. data/lib/lono/builder/template/dsl/finalizer.rb +0 -12
  177. data/lib/lono/builder/template/evaluate.rb +0 -52
  178. data/lib/lono/builder/template/post_processor.rb +0 -67
  179. data/lib/lono/cli/build.rb +0 -101
  180. data/lib/lono/cli/code.rb +0 -22
  181. data/lib/lono/cli/help/blueprint/new.md +0 -56
  182. data/lib/lono/cli/new/extension.rb +0 -57
  183. data/lib/lono/configset/builder.rb +0 -59
  184. data/lib/lono/configset/combiner.rb +0 -164
  185. data/lib/lono/configset/evaluate_file.rb +0 -8
  186. data/lib/lono/configset/meta/dsl.rb +0 -12
  187. data/lib/lono/configset/meta.rb +0 -19
  188. data/lib/lono/configset/s3_file/build.rb +0 -34
  189. data/lib/lono/configset/s3_file/item.rb +0 -38
  190. data/lib/lono/configset/s3_file/registry.rb +0 -12
  191. data/lib/lono/configset/s3_file/upload.rb +0 -12
  192. data/lib/lono/configset/strategy/base.rb +0 -83
  193. data/lib/lono/configset/strategy/helpers/dsl/core.rb +0 -18
  194. data/lib/lono/configset/strategy/helpers/dsl.rb +0 -8
  195. data/lib/lono/configset/strategy/helpers/erb.rb +0 -9
  196. data/lib/lono/extensions/dsl.rb +0 -8
  197. data/lib/lono/extensions/loader.rb +0 -19
  198. data/lib/lono/extensions.rb +0 -18
  199. data/lib/lono/importer/base.rb +0 -48
  200. data/lib/lono/importer/converter.rb +0 -19
  201. data/lib/lono/importer/download.rb +0 -46
  202. data/lib/lono/importer/dsl.rb +0 -36
  203. data/lib/lono/importer/params.rb +0 -56
  204. data/lib/lono/importer/service/coder.rb +0 -81
  205. data/lib/lono/inspector/base.rb +0 -32
  206. data/lib/lono/lookup.rb +0 -12
  207. data/lib/lono/seeder/service_role.rb +0 -11
  208. data/lib/lono/utils/context.rb +0 -15
  209. data/lib/lono/utils/item/file_methods.rb +0 -29
  210. data/lib/lono/utils/item/zip.rb +0 -42
  211. data/lib/templates/extension/lib/%extension_name%/autoloader.rb.tt +0 -23
  212. data/lib/templates/extension/lib/%extension_name%/helpers/mappings.rb.tt +0 -24
  213. data/lib/templates/extension/lib/%extension_name%/helpers/outputs.rb.tt +0 -7
  214. data/lib/templates/extension/lib/%extension_name%/helpers/parameters.rb.tt +0 -10
  215. data/lib/templates/extension/lib/%extension_name%/helpers/resources/resource.rb.tt +0 -4
  216. data/lib/templates/extension/lib/%extension_name%/helpers/variables.rb.tt +0 -6
  217. data/lib/templates/extension/lib/%extension_name%/version.rb.tt +0 -3
  218. data/lib/templates/extension/lib/%extension_name%.rb.tt +0 -8
@@ -0,0 +1,20 @@
1
+ class Lono::Builder::Allow
2
+ class Region < Base
3
+ include Lono::Concerns::AwsInfo
4
+
5
+ # interface method
6
+ def message
7
+ messages = []
8
+ word = config_name.to_s # IE: regions or locations
9
+ messages << "This #{word.singularize} is not allowed to be used: Detected current #{word.singularize}=#{current_region}"
10
+ messages << "Allow #{word}: #{allows.join(', ')}" if allows
11
+ messages << "Deny #{word}: #{denys.join(', ')}" if denys
12
+ messages.join("\n")
13
+ end
14
+
15
+ # interface method
16
+ def check_value
17
+ region
18
+ end
19
+ end
20
+ end
@@ -0,0 +1,8 @@
1
+ class Lono::Builder
2
+ class Allow < Lono::CLI::Base
3
+ def check!
4
+ Env.new(@options).check!
5
+ Region.new(@options).check!
6
+ end
7
+ end
8
+ end
@@ -0,0 +1,145 @@
1
+ module Lono::Builder::Configset
2
+ class Combiner < Lono::CLI::Base
3
+ def initialize(options={})
4
+ super
5
+ @cfn = options[:cfn] # merge from Dsl::Finalizer::Configsets#run
6
+ @metas = options[:metas] # merge from Configset::Evaluator#evaluate
7
+ @configsets = []
8
+ @map = {} # stores resource logical id => metadata cfn-init
9
+ end
10
+
11
+ # Returns metadata map structure
12
+ #
13
+ # {"Instance"=>
14
+ # {"Metadata"=>
15
+ # {"AWS::CloudFormation::Init"=>
16
+ #
17
+ def combine
18
+ return @map if metas_empty?
19
+ add_existing
20
+ add_built
21
+ build_map # metadata map
22
+ end
23
+
24
+ def metas_empty?
25
+ @metas.empty?
26
+ end
27
+
28
+ def add_existing
29
+ existing_configsets.each do |configset|
30
+ add(configset)
31
+ end
32
+ end
33
+
34
+ def add_built
35
+ @metas.each do |meta|
36
+ definition = Definition.new(@options.merge(meta: meta))
37
+ configset = definition.evaluate
38
+ add(configset)
39
+ end
40
+ end
41
+
42
+ # Useful for specs
43
+ def add(configset)
44
+ found = @configsets.detect { |c| c.name == configset.name && c.resource == configset.resource }
45
+ @configsets << configset unless found
46
+ end
47
+
48
+ # Each metadata has this structure:
49
+ #
50
+ # {"Metadata"=>
51
+ # {"AWS::CloudFormation::Init"=>
52
+ # {"configSets"=>{"default"=>["aaa1", "aaa2"]},
53
+ # "aaa1"=>{"commands"=>{"test"=>{"command"=>"echo from-aaa1 > test1.txt"}}},
54
+ # "aaa2"=>
55
+ # {"commands"=>{"test"=>{"command"=>"echo from-aaa2 > test1.txt"}}}}}}
56
+ #
57
+ def build_map
58
+ metadata_map = {}
59
+ configsets_map = {}
60
+
61
+ @configsets.each_with_index do |configset, i|
62
+ padded_i = "%03d" % i
63
+
64
+ name, resource, metadata = configset.name, configset.resource, configset.metadata
65
+ configsets = configsets_map[resource] ||= {}
66
+
67
+ validate_structure!(name, metadata)
68
+
69
+ new_metadata = metadata["Metadata"].dup
70
+ init = new_metadata["AWS::CloudFormation::Init"] # important: adjust data by reference
71
+
72
+ if init.key?("configSets")
73
+ # validate_simple!(name, new_metadata["AWS::CloudFormation::Init"]["configSets"]) # validate original configset for only simple elements
74
+
75
+ # 1. expand each config as its own config, flattening to top-level
76
+ cs = init.delete("configSets") # Only support configSets with simple Array of Strings
77
+ new_config_set = {}
78
+ new_config_set[name] = cs["default"].map {|c| "#{padded_i}_#{c}" }
79
+ init.transform_keys! { |c| "#{padded_i}_#{c}" }
80
+
81
+ # Rebuild default configSet, append the new complex ConfigSet structure with each iteration
82
+ configsets["default"] ||= []
83
+ configsets["default"] << {"ConfigSet" => name}
84
+ configsets.merge!(new_config_set) # add each config from #1 to the top-level
85
+
86
+ init["configSets"] = configsets # replace new configset
87
+ else # simple config
88
+ init["configSets"] = configsets # adjust data by reference
89
+ configsets["default"] ||= []
90
+ configsets["default"] << {"ConfigSet" => name}
91
+
92
+ # build new config
93
+ config_key = "#{padded_i}_single_generated"
94
+ configsets[name] = [config_key]
95
+ new_config = {config_key => init["config"]}
96
+ # replace old config with new one
97
+ init.delete("config") # delete original simple config
98
+ init.merge!(new_config)
99
+ end
100
+
101
+ metadata_map[resource] ||= {"Metadata" => {}}
102
+ metadata_map[resource]["Metadata"].deep_merge!(new_metadata)
103
+ @map[resource] = metadata_map[resource]
104
+ end
105
+ @map
106
+ end
107
+
108
+ # Normalized/convert cfn template to mimic the registry format
109
+ def existing_configsets
110
+ configsets = []
111
+ @cfn["Resources"].each do |logical_id, attributes|
112
+ cloudformation_init = attributes.dig("Metadata", "AWS::CloudFormation::Init")
113
+ next unless cloudformation_init
114
+
115
+ configset = Lono::Configset.new(name: "#{logical_id}OriginalConfigset", resource: logical_id)
116
+ configset.metadata = {"Metadata" => attributes["Metadata"]} # wrap metadata to create right structure
117
+ configsets << configset
118
+ end
119
+ configsets
120
+ end
121
+
122
+ def validate_structure!(name, metadata)
123
+ return if metadata.is_a?(Hash) && metadata.dig("Metadata", "AWS::CloudFormation::Init")
124
+
125
+ puts "ERROR: The #{name} configset does not appear to have a AWS::CloudFormation::Init key".color(:red)
126
+ puts "Please double check the #{name} configset.yml structure"
127
+ exit 1
128
+ end
129
+
130
+ def validate_simple!(name, cs)
131
+ has_complex_type = cs["default"].detect { |s| !s.is_a?(String) }
132
+ return unless has_complex_type
133
+ message =<<~EOL
134
+ ERROR: The configset #{name} has a configSets property with a complex type.
135
+ configSets:
136
+
137
+ #{cs}
138
+
139
+ lono configsets only supports combining configSets with an Array of Strings.
140
+ EOL
141
+ puts message.color(:red)
142
+ exit 1
143
+ end
144
+ end
145
+ end
@@ -0,0 +1,47 @@
1
+ class Lono::Builder::Configset::Definition
2
+ class Base < Lono::CLI::Base
3
+ include DslEvaluator
4
+ include Lono::Builder::Dsl::Syntax
5
+ include Lono::Utils::Pretty
6
+ include Context
7
+
8
+ # Really only use @path in Configset DSL and ERB evaluation.
9
+ # However, configsets are built within the CloudFormation template and can use
10
+ # things instrinic functions like `ref` like would normally have access to.
11
+ # So configsets need the same context
12
+ #
13
+ # Configset::Definition::Base < Lono::CLI::Base
14
+ #
15
+ # for
16
+ #
17
+ # include Lono::Builder::Dsl::Syntax
18
+ #
19
+ def initialize(options={})
20
+ super
21
+ @meta = @options[:meta]
22
+ @configset = Lono::Configset.new(@meta)
23
+ expose_instance_variables
24
+ end
25
+
26
+ # This context is used by the DSL evaluation. Expose variables so user can use them in configset definitions.
27
+ # Example:
28
+ #
29
+ # "/etc/cfn/hooks.d/cfn-auto-reloader.conf":
30
+ # content:
31
+ # Fn::Sub:
32
+ # ...
33
+ # path=Resources.<%= @resource %>.Metadata.AWS::CloudFormation::Init
34
+ #
35
+ def expose_instance_variables
36
+ @name = @meta[:name]
37
+ @resource = @meta[:resource]
38
+ end
39
+
40
+ def wrap_with_metadata(cloudformation_init)
41
+ full = {"Metadata" => {}}
42
+ full["Metadata"]["AWS::CloudFormation::Init"] = cloudformation_init["AWS::CloudFormation::Init"]
43
+ full["Metadata"]["AWS::CloudFormation::Authentication"] = authentication if authentication # only on dsl
44
+ full.deep_stringify_keys.dup # metadata
45
+ end
46
+ end
47
+ end
@@ -0,0 +1,70 @@
1
+ class Lono::Builder::Configset::Definition
2
+ module Context
3
+ include DslEvaluator
4
+
5
+ def load_context
6
+ load_vars
7
+ load_helpers
8
+ end
9
+
10
+ # Docs: https://lono.cloud/docs/configsets/layering/
11
+ # Think layering is simple enough to keep in this method for now.
12
+ # Consider trying to combine with Lono::Layering::Layer for uniformity though.
13
+ def load_vars
14
+ layers.each do |layer|
15
+ evaluate_file(layer)
16
+ end
17
+ end
18
+
19
+ def layers
20
+ paths = [
21
+ "#{@configset.root}/vars.rb", # source defaults
22
+ "#{Lono.root}/config/configsets/#{@configset.name}/vars.rb", # source overrides
23
+ "#{@blueprint.root}/config/configsets/vars/#{@configset.name}.rb", # blueprint overrides
24
+ "#{Lono.root}/config/blueprints/#{@blueprint.name}/configsets/vars/#{@configset.name}.rb", # user overrides
25
+ ]
26
+ show_layers(paths)
27
+ paths
28
+ end
29
+
30
+ def show_layers(paths)
31
+ if ENV['LONO_LAYERS_ALL']
32
+ show_all_layers(paths)
33
+ else
34
+ show_existing_layers(paths)
35
+ end
36
+ end
37
+
38
+ def show_all_layers(paths)
39
+ show_header
40
+ paths.each do |path|
41
+ show_layer(path)
42
+ end
43
+ end
44
+
45
+ def show_existing_layers(paths)
46
+ existing = paths.select { |path| File.exist?(path) }
47
+ return if existing.empty?
48
+ show_header
49
+ existing.each do |path|
50
+ show_layer(path)
51
+ end
52
+ end
53
+
54
+ def show_header
55
+ logger.info " Configset Layers #{@configset.name}"
56
+ end
57
+
58
+ def show_layer(path)
59
+ logger.info " #{pretty_path(path)}"
60
+ end
61
+
62
+ def show_layers?
63
+ Lono.config.layering.show || ENV['LONO_LAYERS']
64
+ end
65
+
66
+ def load_helpers
67
+ load_helper_files("#{@configset.root}/helpers", type: :configset)
68
+ end
69
+ end
70
+ end
@@ -1,4 +1,4 @@
1
- module Lono::Configset::Strategy::Helpers::Dsl
1
+ module Lono::Builder::Configset::Definition::Dsl::Syntax
2
2
  module Auth
3
3
  def authentication(data=nil, force: false)
4
4
  if data.nil?
@@ -0,0 +1,22 @@
1
+ module Lono::Builder::Configset::Definition::Dsl::Syntax
2
+ module Content
3
+ def content_file(path)
4
+ content_path = "#{content_file_root}/content"
5
+ file = "#{content_path}/#{path}"
6
+ if File.exist?(file)
7
+ IO.read(file)
8
+ else
9
+ "File not found: #{file}"
10
+ end
11
+ end
12
+
13
+ private
14
+ def content_file_root
15
+ if @configset
16
+ @configset.root
17
+ else
18
+ @blueprint.root
19
+ end
20
+ end
21
+ end
22
+ end
@@ -1,7 +1,20 @@
1
- module Lono::Configset::Strategy::Helpers::Dsl
2
- module Syntax
1
+ module Lono::Builder::Configset::Definition::Dsl::Syntax
2
+ module Core
3
3
  extend Memoist
4
4
 
5
+ #
6
+ # The configset method is different with configset registration vs configset definition
7
+ #
8
+ # definition: app/configsets/httpd/configset.rb
9
+ # registration: app/blueprints/demo/configsets.rb
10
+ #
11
+ def configset(current)
12
+ @tracked << current
13
+ previous, @current = @current, current
14
+ yield
15
+ @current = previous
16
+ end
17
+
5
18
  %w[package group user file service].each do |meth|
6
19
  section = meth.pluralize
7
20
  define_method(meth) do |k, props={}|
@@ -53,13 +66,6 @@ module Lono::Configset::Strategy::Helpers::Dsl
53
66
  current_structure(@current)["sources"].deep_merge!(item)
54
67
  end
55
68
 
56
- def configset(current)
57
- @tracked << current
58
- previous, @current = @current, current
59
- yield
60
- @current = previous
61
- end
62
-
63
69
  private
64
70
  def current_structure(configset)
65
71
  @structure[configset] ||= {}
@@ -1,4 +1,4 @@
1
- module Lono::Configset::Strategy::Helpers::Dsl
1
+ module Lono::Builder::Configset::Definition::Dsl::Syntax
2
2
  module Package
3
3
  # In recent versions of the AmazonLinux2 cfnbootstrap, , the package command doesnt work with updated versions of rubygems.
4
4
  # Get this error
@@ -0,0 +1,5 @@
1
+ class Lono::Builder::Configset::Definition::Dsl
2
+ module Syntax
3
+ include_modules "syntax"
4
+ end
5
+ end
@@ -1,29 +1,25 @@
1
- module Lono::Configset::Strategy
1
+ class Lono::Builder::Configset::Definition
2
2
  class Dsl < Base
3
- # Configset helpers take higher precedence than Lono DSL Helpers. IE: s3_key is overwritten but s3_bucket is the same
4
- include Helpers::Dsl
3
+ include Syntax
5
4
 
6
5
  def initialize(options={})
7
6
  super
8
- @structure = {} # holds in memory the configset hash structure to build AWS::CloudFormation::Init
9
7
  @current = "main" # current configset
10
- @command_counts = Hash.new(0)
11
8
  @tracked = []
9
+ @structure = {} # holds in memory the configset hash structure to build AWS::CloudFormation::Init
10
+ @command_counts = Hash.new(0)
12
11
  # Also support ability to add AWS::CloudFormation::Authentication
13
12
  @authentication = nil # holds IAM policy info to build AWS::CloudFormation::Authentication
14
13
  end
15
14
 
16
- def find_evaluation_path
17
- "#{@root}/lib/configset.rb"
18
- end
19
-
20
- def load
21
- evaluate_file(@evaluation_path)
22
- configsets = @configsets || @tracked.uniq
15
+ def evaluate
16
+ load_context
17
+ evaluate_file(@configset.path)
18
+ configsets = configsets || @tracked.uniq
23
19
  configsets = ["main"] if configsets.empty?
24
20
  configsets_structure = {"configSets" => {"default" => configsets}}.merge(@structure)
25
- full = { "AWS::CloudFormation::Init" => configsets_structure }
26
- full.deep_stringify_keys!
21
+ cloudformation_init = { "AWS::CloudFormation::Init" => configsets_structure }
22
+ wrap_with_metadata(cloudformation_init)
27
23
  end
28
24
  end
29
25
  end
@@ -1,29 +1,22 @@
1
- require "json"
2
-
3
- module Lono::Configset::Strategy
1
+ class Lono::Builder::Configset::Definition
4
2
  class Erb < Base
5
- include Helpers::Erb
6
-
7
- def find_evaluation_path
8
- paths = %w[configset.yml configset.json].map { |p| "#{@root}/lib/#{p}" }
9
- paths.find { |p| File.exist?(p) }
3
+ def evaluate
4
+ load_context
5
+ cloudformation_init = build
6
+ wrap_with_metadata(cloudformation_init)
10
7
  end
11
8
 
12
- def load
13
- content = RenderMePretty.result(@evaluation_path, context: self)
14
- # init structure
15
- if File.extname(@evaluation_path) == ".yml"
9
+ def build
10
+ content = RenderMePretty.result(@configset.path, context: self)
11
+ if File.extname(@configset.path) == ".yml"
16
12
  load_yaml(content)
17
13
  else
18
14
  JSON.load(content)
19
15
  end
20
16
  end
21
17
 
22
- def authentication
23
- # noop
24
- end
25
-
26
18
  def load_yaml(content)
19
+ # init structure
27
20
  # Write to file so can use Yamler::Validator
28
21
  path = "/tmp/lono/configset.yml"
29
22
  FileUtils.mkdir_p(File.dirname(path))
@@ -31,5 +24,9 @@ module Lono::Configset::Strategy
31
24
  Lono::Yamler::Validator.new(path).validate!
32
25
  Lono::Yamler::Loader.new(content).load
33
26
  end
27
+
28
+ def authentication
29
+ # noop
30
+ end
34
31
  end
35
32
  end
@@ -0,0 +1,18 @@
1
+ module Lono::Builder::Configset
2
+ class Definition < Lono::CLI::Base
3
+ attr_reader :configset
4
+ def initialize(options={})
5
+ super
6
+ @meta = options[:meta]
7
+ @configset = Lono::Configset.new(@meta)
8
+ end
9
+
10
+ def evaluate
11
+ strategy_class = configset.path.include?('.rb') ? Dsl : Erb
12
+ strategy = strategy_class.new(@options.merge(path: configset.path))
13
+ metadata = strategy.evaluate
14
+ @configset.metadata = metadata
15
+ @configset
16
+ end
17
+ end
18
+ end
@@ -0,0 +1,10 @@
1
+ # Dsl or Erb
2
+ module Lono::Builder::Configset
3
+ class Evaluator < Lono::CLI::Base
4
+ def evaluate
5
+ Registration.new(@blueprint).evaluate
6
+ combiner = Combiner.new(@options.merge(metas: Registration.metas))
7
+ combiner.combine
8
+ end
9
+ end
10
+ end
@@ -0,0 +1,35 @@
1
+ module Lono::Builder::Configset
2
+ class Registration
3
+ include DslEvaluator
4
+
5
+ def initialize(blueprint)
6
+ @blueprint = blueprint
7
+ end
8
+
9
+ cattr_accessor :metas, default: []
10
+ def metas
11
+ self.class.metas
12
+ end
13
+
14
+ def evaluate
15
+ path = "#{@blueprint.root}/configsets.rb" # plural
16
+ evaluate_file(path)
17
+ end
18
+
19
+ # Only one syntax method so not in separate module.
20
+ #
21
+ # Register configset for later processing.
22
+ # By registering them all up front, can aggregate errors and show them together
23
+ # for a user-friendly experience.
24
+ #
25
+ # The configset method is different with configset registration vs configset definition
26
+ #
27
+ # definition: app/configsets/httpd/configset.rb
28
+ # registration: app/blueprints/demo/configsets.rb
29
+ #
30
+ def configset(name, options={})
31
+ found = metas.detect { |i| i[:name] == name && i[:resource] == options[:resource] }
32
+ metas << options.merge(name: name) unless found
33
+ end
34
+ end
35
+ end
@@ -1,25 +1,40 @@
1
- # Encapsulates helper methods and instance variables to be rendered in the ERB templates.
2
- module Lono::Builder
3
- class Context < Lono::CLI::Base
4
- extend Memoist
5
- include Lono::Builder::Template::Helpers
6
- include Loader
7
- include Helpers # ERB
8
- include Template::Dsl::Evaluator::Syntax # DSL
9
- include Lono::Utils::Context
1
+ class Lono::Builder
2
+ module Context
3
+ include DslEvaluator
10
4
 
11
- # Take a hash and makes them instance variables in the current scope.
12
- # Use this in custom helper methods to make variables accessible to ERB templates.
13
- def instance_variables!(variables)
14
- variables.each do |key, value|
15
- instance_variable_set('@' + key.to_s, value)
5
+ def load_context
6
+ load_variables
7
+ load_helpers
8
+ end
9
+
10
+ # Variables in base.rb are overridden by their environment specific variables
11
+ # file. Example, file LONO_ENV=dev:
12
+ #
13
+ # config/vars/base.rb
14
+ # config/vars/dev.rb - will override any variables in base.rb
15
+ # config/vars/base.rb
16
+ # config/vars/dev.rb - will override any variables in base.rb
17
+ #
18
+ def load_variables
19
+ return if seed?
20
+ layers = Lono::Layering::Layer.new(@blueprint, "vars").paths
21
+ layers.each do |layer|
22
+ evaluate_file(layer)
16
23
  end
17
24
  end
18
25
 
19
- # For Lono::AppFile::Build usage of Thor::Action directory
20
- # For some reason a send(:binding) doesnt work but get_binding like this works.
21
- def get_binding
22
- binding
26
+ # Load blueprint helpers
27
+ # blueprint helpers override extension helpers
28
+ def load_helpers
29
+ load_helper_files("#{Lono.root}/vendor/helpers", type: :project)
30
+ load_helper_files("#{Lono.root}/app/helpers", type: :project)
31
+ load_helper_files("#{@blueprint.root}/helpers", type: :blueprint) # takes higher precedence
32
+ end
33
+
34
+ # Dont want any existing files to prevent building the blueprint.
35
+ # This means that parameters cannot be based on vars. It's a trade-off.
36
+ def seed?
37
+ ARGV[0] == "seed"
23
38
  end
24
39
  end
25
40
  end
@@ -0,0 +1,45 @@
1
+ class Lono::Builder::Dsl
2
+ class Evaluator < Lono::CLI::Base
3
+ include DslEvaluator
4
+ include Lono::Builder::Context
5
+ include Lono::Builder::Dsl::Syntax
6
+ include Lono::Utils::Pretty
7
+
8
+ def initialize(options={})
9
+ super
10
+ @template_path = "#{@blueprint.root}/template.rb"
11
+ @parameters = [] # built by parameter_groups.rb
12
+ @cfn = {}
13
+ end
14
+
15
+ def evaluate
16
+ load_context
17
+ evaluate_template_paths(@template_path) # modifies @cfn
18
+ finalize
19
+ to_yaml
20
+ @cfn
21
+ end
22
+
23
+ def finalize
24
+ o = @options.merge(parameters: @parameters)
25
+ @cfn = Finalizer.new(@cfn, o).run
26
+ end
27
+
28
+ def to_yaml
29
+ # https://stackoverflow.com/questions/24508364/how-to-emit-yaml-in-ruby-expanding-aliases
30
+ # Trick to prevent YAML from emitting aliases
31
+ @cfn = YAML.load(@cfn.to_json)
32
+ end
33
+
34
+ # Example path: /full/path/to/project/app/blueprints/demo/template.rb
35
+ def evaluate_template_paths(path)
36
+ ext = File.extname(path)
37
+ folder = path.sub(ext, '')
38
+ expr = "#{folder}/**/*.rb"
39
+ evaluate_file(path) # process top-level template.rb first
40
+ Dir.glob(expr).each do |path|
41
+ evaluate_file(path)
42
+ end
43
+ end
44
+ end
45
+ end
@@ -0,0 +1,8 @@
1
+ class Lono::Builder::Dsl::Finalizer
2
+ class Base < Lono::CLI::Base
3
+ def initialize(options={})
4
+ super
5
+ @cfn = options[:cfn]
6
+ end
7
+ end
8
+ end
@@ -0,0 +1,20 @@
1
+ class Lono::Builder::Dsl::Finalizer
2
+ class Configsets < Base
3
+ # Replaces metadata under each logical id resource.
4
+ def run
5
+ dsl = Lono::Builder::Configset::Evaluator.new(@options.merge(cfn: @cfn))
6
+ metadata_map = dsl.evaluate
7
+ metadata_map.each do |logical_id, cs|
8
+ resource = @cfn["Resources"][logical_id]
9
+ unless resource
10
+ puts "WARN: Resources.#{logical_id} not found in the template. Are you sure you specified the correct resource logical id in your configsets.rb?".color(:yellow)
11
+ next
12
+ end
13
+
14
+ resource["Metadata"] = cs["Metadata"]
15
+ end
16
+
17
+ @cfn
18
+ end
19
+ end
20
+ end
@@ -0,0 +1,4 @@
1
+ class Lono::Builder::Dsl::Finalizer::Files
2
+ class Base < Lono::Builder::Dsl::Finalizer::Base
3
+ end
4
+ end