Class: Ufo::Stack

Inherits:
Object
  • Object
show all
Extended by:
Memoist
Includes:
Helper
Defined in:
lib/ufo/stack/status.rb,
lib/ufo/stack.rb,
lib/ufo/stack/helper.rb,
lib/ufo/stack/context.rb

Overview

CloudFormation status codes, full list:

https://docs.aws.amazon.com/AWSCloudFormation/latest/UserGuide/using-cfn-describing-stacks.html

CREATE_COMPLETE
ROLLBACK_COMPLETE
DELETE_COMPLETE
UPDATE_COMPLETE
UPDATE_ROLLBACK_COMPLETE

CREATE_FAILED
DELETE_FAILED
ROLLBACK_FAILED
UPDATE_ROLLBACK_FAILED

CREATE_IN_PROGRESS
DELETE_IN_PROGRESS
REVIEW_IN_PROGRESS
ROLLBACK_IN_PROGRESS
UPDATE_COMPLETE_CLEANUP_IN_PROGRESS
UPDATE_IN_PROGRESS
UPDATE_ROLLBACK_COMPLETE_CLEANUP_IN_PROGRESS
UPDATE_ROLLBACK_IN_PROGRESS

Defined Under Namespace

Modules: Helper Classes: Context, Status

Instance Method Summary collapse

Methods included from Helper

#adjust_stack_name, #cfn, #find_stack, #status

Methods included from Util

#default_cluster, #display_params, #execute, #pretty_time, #settings, #task_definition_arns, #user_params

Methods included from AwsService

#cloudformation, #cloudwatchlogs, #ec2, #ecr, #ecs, #elb

Constructor Details

#initialize(options) ⇒ Stack

Returns a new instance of Stack.



29
30
31
32
33
34
35
# File 'lib/ufo/stack.rb', line 29

def initialize(options)
  @options = options
  @task_definition = options[:task_definition]
  @service = options[:service]
  @cluster = @options[:cluster] || default_cluster(@service)
  @stack_name = adjust_stack_name(@cluster, options[:service])
end

Instance Method Details

#contextObject



130
131
132
133
134
135
136
# File 'lib/ufo/stack.rb', line 130

def context
  Context.new(@options.merge(
    cluster: @cluster,
    stack_name: @stack_name,
    stack: @stack,
  ))
end

#current_desired_countObject



144
145
146
147
148
149
150
151
152
153
154
155
# File 'lib/ufo/stack.rb', line 144

def current_desired_count
  # Cannot set ECS desired count when is scheduling_strategy DAEMON
  return '' if scheduling_strategy == "DAEMON"

  info = Info.new(@service, @options)
  service = info.service
  if service
    service.desired_count.to_s
  else
    "1" # new service default
  end
end

#deployObject



37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
# File 'lib/ufo/stack.rb', line 37

def deploy
  @stack = find_stack(@stack_name)
  if @stack && rollback_complete?(@stack)
    puts "Existing stack in ROLLBACK_COMPLETE state. Deleting stack before continuing."
    cloudformation.delete_stack(stack_name: @stack_name)
    status.wait
    status.reset
    @stack = nil # at this point stack has been deleted
  end

  exit_with_message(@stack) if @stack && !updatable?(@stack)

  @stack ? perform(:update) : perform(:create)

  stop_old_tasks if @options[:stop_old_task]

  return unless @options[:wait]
  status.wait

  puts status.rollback_error_message if status.update_rollback?

  status.success?
end

#exit_with_message(stack) ⇒ Object



175
176
177
178
179
180
181
# File 'lib/ufo/stack.rb', line 175

def exit_with_message(stack)
  region = `aws configure get region`.strip rescue "us-east-1"
  url = "https://console.aws.amazon.com/cloudformation/home?region=#{region}#/stacks"
  puts "The stack is not in an updateable state: #{stack.stack_status.color(:yellow)}."
  puts "Here's the CloudFormation url to check for more details #{url}"
  exit 1
end

#handle_stack_error(e) ⇒ Object

Assume only first container_definition to get the container info. Stack:arn:aws:cloudformation:… is in ROLLBACK_COMPLETE state and can not be updated.



185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
# File 'lib/ufo/stack.rb', line 185

def handle_stack_error(e)
  case e.message
  when /state and can not be updated/
    puts "The #{@stack_name} stack is in a state that cannot be updated. Deleted the stack and try again."
    puts "ERROR: #{e.message}"
    if message.include?('UPDATE_ROLLBACK_FAILED')
      puts "You might be able to do a 'Continue Update Rollback' and skip some resources to get the stack back into a good state."
    end
    region = `aws configure get region`.strip rescue 'us-east-1'
    url = "https://console.aws.amazon.com/cloudformation/home?region=#{region}"
    puts "Here's the CloudFormation console url: #{url}"
    exit 1
  when /No updates are to be performed/
    puts "There are no updates to be performed. Exiting.".color(:yellow)
    exit 1
  else
    raise
  end
end

#parametersObject



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
121
122
123
124
125
126
127
# File 'lib/ufo/stack.rb', line 94

def parameters
  create_elb, elb_target_group = context.elb_options

  network = Setting::Profile.new(:network, settings[:network_profile]).data
  # pp network
  elb_subnets = network[:elb_subnets] && !network[:elb_subnets].empty? ?
                network[:elb_subnets] :
                network[:ecs_subnets]

  hash = {
    Vpc: network[:vpc],
    ElbSubnets: elb_subnets.join(','),
    EcsSubnets: network[:ecs_subnets].join(','),

    CreateElb: create_elb,
    ElbTargetGroup: elb_target_group,
    ElbEipIds: context.elb_eip_ids,

    EcsDesiredCount: current_desired_count,
    EcsTaskDefinition: task_definition_arn,
    EcsSchedulingStrategy: scheduling_strategy,
  }

  hash[:EcsSecurityGroups] = network[:ecs_security_groups].join(',') if network[:ecs_security_groups]
  hash[:ElbSecurityGroups] = network[:elb_security_groups].join(',') if network[:elb_security_groups]

  hash.map do |k,v|
    if v == :use_previous_value
      { parameter_key: k, use_previous_value: true }
    else
      { parameter_key: k, parameter_value: v }
    end
  end
end

#perform(action) ⇒ Object



61
62
63
64
65
66
67
# File 'lib/ufo/stack.rb', line 61

def perform(action)
  puts "#{action[0..-2].capitalize}ing stack #{@stack_name.color(:green)}..."
  # Example: cloudformation.send("update_stack", stack_options)
  cloudformation.send("#{action}_stack", stack_options)
rescue Aws::CloudFormation::Errors::ValidationError => e
  handle_stack_error(e)
end

#rollback_complete?(stack) ⇒ Boolean

Returns:

  • (Boolean)


205
206
207
# File 'lib/ufo/stack.rb', line 205

def rollback_complete?(stack)
  stack.stack_status == 'ROLLBACK_COMPLETE'
end

#save_templateObject

Store template in tmp in case for debugging



164
165
166
167
168
169
170
171
172
173
# File 'lib/ufo/stack.rb', line 164

def save_template
  path = "/tmp/ufo/#{@stack_name}/stack.yml"
  FileUtils.mkdir_p(File.dirname(path))
  IO.write(path, template_body)
  puts "Generated template saved at: #{path}"

  path = "/tmp/ufo/#{@stack_name}/parameters.yml"
  IO.write(path, JSON.pretty_generate(parameters))
  puts "Generated parameters saved at: #{path}"
end

#scheduling_strategyObject



139
140
141
142
# File 'lib/ufo/stack.rb', line 139

def scheduling_strategy
  strategy = @options[:scheduling_strategy] || context.scheduling_strategy
  strategy.upcase
end

#stack_optionsObject



81
82
83
84
85
86
87
88
89
90
91
92
# File 'lib/ufo/stack.rb', line 81

def stack_options
  save_template
  if ENV['SAVE_TEMPLATE_EXIT']
    puts "Template saved. Exiting."
    exit 1
  end
  {
    parameters: parameters,
    stack_name: @stack_name,
    template_body: template_body,
  }
end

#task_definition_arnObject



157
158
159
160
# File 'lib/ufo/stack.rb', line 157

def task_definition_arn
  resp = ecs.describe_task_definition(task_definition: @task_definition)
  resp.task_definition.task_definition_arn
end

#template_bodyObject

do not memoize template_body it can change for a rename retry



70
71
72
73
74
75
76
77
78
79
# File 'lib/ufo/stack.rb', line 70

def template_body
  custom_template = "#{Ufo.root}/.ufo/settings/cfn/stack.yml"
  path = if File.exist?(custom_template)
           custom_template
         else
           # built-in default
           File.expand_path("../cfn/stack.yml", File.dirname(__FILE__))
         end
  RenderMePretty.result(path, context: context.scope)
end

#updatable?(stack) ⇒ Boolean

Returns:

  • (Boolean)


209
210
211
# File 'lib/ufo/stack.rb', line 209

def updatable?(stack)
  stack.stack_status =~ /_COMPLETE$/ || stack.stack_status == 'UPDATE_ROLLBACK_FAILED'
end