Module: SparkleFormation::Provider::Aws

Defined in:
lib/sparkle_formation/provider/aws.rb

Overview

AWS specific implementation

Instance Method Summary collapse

Instance Method Details

#apply_deep_nesting(*args) {|stack, resource, s_name| ... } ⇒ Hash

Apply deeply nested stacks. This is the new nesting approach and does not bubble parameters up to the root stack. Parameters are isolated to the stack resource itself and output mapping is automatically applied.

Yield Parameters:

  • stack (SparkleFormation)

    stack instance

  • resource (AttributeStruct)

    the stack resource

  • s_name (String)

    stack resource name

Yield Returns:

  • (Hash)

    key/values to be merged into resource properties

Returns:

  • (Hash)

    dumped stack



51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
# File 'lib/sparkle_formation/provider/aws.rb', line 51

def apply_deep_nesting(*args, &block)
  outputs = collect_outputs
  nested_stacks(:with_resource).each do |stack, resource|
    unless(stack.nested_stacks.empty?)
      stack.apply_deep_nesting(*args)
    end
    stack.compile.parameters.keys!.each do |parameter_name|
      if(output_name = output_matched?(parameter_name, outputs.keys))
        next if outputs[output_name] == stack
        stack_output = stack.make_output_available(output_name, outputs)
        resource.properties.parameters.set!(parameter_name, stack_output)
      end
    end
  end
  if(block_given?)
    extract_templates(&block)
  end
  compile.dump!
end

#apply_shallow_nesting(*args) {|resource_name, stack| ... } ⇒ Hash

Apply shallow nesting. This style of nesting will bubble parameters up to the root stack. This type of nesting is the original and now deprecated, but remains for compat issues so any existing usage won’t be automatically busted.

Yield Parameters:

  • resource_name (String)

    name of stack resource

  • stack (SparkleFormation)

    nested stack

Yield Returns:

  • (String)

    Remote URL storage for template

Returns:

  • (Hash)


80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
# File 'lib/sparkle_formation/provider/aws.rb', line 80

def apply_shallow_nesting(*args, &block)
  parameters = compile[:parameters] ? compile[:parameters]._dump : {}
  output_map = {}
  nested_stacks(:with_resource, :with_name).each do |_stack, stack_resource, stack_name|
    remap_nested_parameters(compile, parameters, stack_name, stack_resource, output_map)
  end
  extract_templates(&block)
  compile.parameters parameters
  if(args.include?(:bubble_outputs))
    outputs_hash = Hash[
      output_map do |name, value|
        [name, {'Value' => {'Fn::GetAtt' => value}}]
      end
    ]
    if(compile.outputs)
      compile._merge(compile._klass_new(outputs_hash))
    else
      compile.outputs output_hash
    end
  end
  compile.dump!
end

#generate_policyHash

Generate policy for stack

Returns:

  • (Hash)


16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
# File 'lib/sparkle_formation/provider/aws.rb', line 16

def generate_policy
  statements = []
  compile.resources.keys!.each do |r_name|
    r_object = compile.resources[r_name]
    if(r_object['Policy'])
      r_object['Policy'].keys!.each do |effect|
        statements.push(
          'Effect' => effect.to_s.capitalize,
          'Action' => [r_object['Policy'][effect]].flatten.compact.map{|i| "Update:#{i}"},
          'Resource' => "LogicalResourceId/#{r_name}",
          'Principal' => '*'
        )
      end
      r_object.delete!('Policy')
    end
  end
  statements.push(
    'Effect' => 'Allow',
    'Action' => 'Update:*',
    'Resource' => '*',
    'Principal' => '*'
  )
  Smash.new('Statement' => statements)
end

#make_output_available(output_name, outputs) ⇒ Object

Extract output to make available for stack parameter usage at the current depth

Parameters:

  • output_name (String)

    name of output

  • outputs (Hash)

    listing of outputs



109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
# File 'lib/sparkle_formation/provider/aws.rb', line 109

def make_output_available(output_name, outputs)
  bubble_path = outputs[output_name].root_path - root_path
  drip_path = root_path - outputs[output_name].root_path
  bubble_path.each_slice(2) do |base_sparkle, ref_sparkle|
    next unless ref_sparkle
    base_sparkle.compile.outputs.set!(output_name).set!(
      :value, base_sparkle.compile.attr!(
        ref_sparkle.name, "Outputs.#{output_name}"
      )
    )
  end
  if(bubble_path.empty?)
    if(drip_path.size == 1)
      parent = drip_path.first.parent
      if(parent && parent.compile.parameters.data![output_name])
        return compile.ref!(output_name)
      end
    end
    raise ArgumentError.new "Failed to detect available bubbling path for output `#{output_name}`. " <<
      'This may be due to a circular dependency! ' <<
      "(Output Path: #{outputs[output_name].root_path.map(&:name).join(' > ')} " <<
      "Requester Path: #{root_path.map(&:name).join(' > ')})"
  end
  result = compile.attr!(bubble_path.first.name, "Outputs.#{output_name}")
  if(drip_path.size > 1)
    parent = drip_path.first.parent
    drip_path.unshift(parent) if parent
    drip_path.each_slice(2) do |base_sparkle, ref_sparkle|
      next unless ref_sparkle
      base_sparkle.compile.resources[ref_sparkle.name].properties.parameters.set!(output_name, result)
      ref_sparkle.compile.parameters.set!(output_name){ type 'String' } # TODO: <<<<------ type check and prop
      result = compile.ref!(output_name)
    end
  end
  result
end

#remap_nested_parameters(template, parameters, stack_name, stack_resource, output_map) ⇒ TrueClass

Note:

if parameter has includes ‘StackUnique` a new parameter will be added to container stack and it will not use outputs

Extract parameters from nested stacks. Check for previous nested stack outputs that match parameter. If match, set parameter to use output. If no match, check container stack parameters for match. If match, set to use ref. If no match, add parameter to container stack parameters and set to use ref.

Parameters:

  • template (Hash)

    template being processed

  • parameters (Hash)

    top level parameter set being built

  • stack_name (String)

    name of stack resource

  • stack_resource (Hash)

    duplicate of stack resource contents

  • output_map (Hash)

    mapping of output names to required stack output access

Returns:

  • (TrueClass)


160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
# File 'lib/sparkle_formation/provider/aws.rb', line 160

def remap_nested_parameters(template, parameters, stack_name, stack_resource, output_map)
  stack_parameters = stack_resource.properties.stack.compile.parameters
  unless(stack_parameters.nil?)
    stack_parameters._dump.each do |pname, pval|
      if(pval['StackUnique'])
        check_name = [stack_name, pname].join
      else
        check_name = pname
      end
      if(parameters.keys.include?(check_name))
        if(parameters[check_name]['Type'] == 'CommaDelimitedList')
          new_val = {'Fn::Join' => [',', {'Ref' => check_name}]}
        else
          new_val = {'Ref' => check_name}
        end
        template.resources.set!(stack_name).properties.parameters.set!(pname, new_val)
      elsif(output_map[check_name])
        template.resources.set!(stack_name).properties.parameters.set!(
          pname, 'Fn::GetAtt' => output_map[check_name]
        )
      else
        if(pval['Type'] == 'CommaDelimitedList')
          new_val = {'Fn::Join' => [',', {'Ref' => check_name}]}
        else
          new_val = {'Ref' => check_name}
        end
        template.resources.set!(stack_name).properties.parameters.set!(pname, new_val)
        parameters[check_name] = pval
      end
    end
  end
  unless(stack_resource.properties.stack.compile.outputs.nil?)
    stack_resource.properties.stack.compile.outputs.keys!.each do |oname|
      output_map[oname] = [stack_name, "Outputs.#{oname}"]
    end
  end
  true
end

#stack_resource_typeString

Returns Type string for AWS CFN stack resource.

Returns:

  • (String)

    Type string for AWS CFN stack resource



9
10
11
# File 'lib/sparkle_formation/provider/aws.rb', line 9

def stack_resource_type
  'AWS::CloudFormation::Stack'
end