Class: ChefApply::CLI
- Inherits:
-
Object
- Object
- ChefApply::CLI
- Includes:
- Help, Options, Validation, Mixlib::CLI
- Defined in:
- lib/chef_apply/cli.rb,
lib/chef_apply/cli/help.rb,
lib/chef_apply/cli/options.rb,
lib/chef_apply/cli/validation.rb
Defined Under Namespace
Modules: Help, Options, Validation Classes: OptionValidationError
Constant Summary collapse
- RC_OK =
0
- RC_COMMAND_FAILED =
1
- RC_UNHANDLED_ERROR =
32
- RC_ERROR_HANDLING_FAILED =
64
Constants included from Help
Constants included from Validation
Validation::CB_MATCHER, Validation::PROPERTY_MATCHER
Constants included from Options
Instance Attribute Summary collapse
-
#archive_file_location ⇒ Object
readonly
Returns the value of attribute archive_file_location.
-
#target_hosts ⇒ Object
readonly
Returns the value of attribute target_hosts.
-
#temp_cookbook ⇒ Object
readonly
Returns the value of attribute temp_cookbook.
Instance Method Summary collapse
- #capture_exception_backtrace(e) ⇒ Object
-
#connect_target(target_host, reporter) ⇒ Object
Accepts a target_host and establishes the connection to that host while providing visual feedback via the Terminal API.
-
#converge(reporter, local_policy_path, target_host) ⇒ Object
Runs the Converge action and renders UI updates as the action reports back.
- #do_connect(target_host, reporter) ⇒ Object
-
#generate_local_policy(reporter) ⇒ Object
Runs the GenerateLocalPolicy action and renders UI updates as the action reports back.
-
#generate_temp_cookbook(arguments, reporter) ⇒ Object
Runs a GenerateCookbook action based on recipe/resource infoprovided and renders UI updates as the action reports back.
- #handle_failed_job(job) ⇒ Object
-
#handle_failed_jobs(jobs) ⇒ Object
When running multiple jobs, exceptions are captured to the job to avoid interrupting other jobs in process.
-
#handle_message(message, data, reporter) ⇒ Object
A handler for common action messages.
- #handle_perform_error(e) ⇒ Object
- #handle_run_error(e) ⇒ Object
-
#initialize(argv) ⇒ CLI
constructor
A new instance of CLI.
- #install(target_host, reporter) ⇒ Object
- #perform_run ⇒ Object
- #render_converge(target_hosts) ⇒ Object
- #render_cookbook_setup(arguments) ⇒ Object
- #resolve_targets(host_spec, opts) ⇒ Object
- #run ⇒ Object
Methods included from Help
#format_flags, #format_help, #show_help, #show_version, #usage
Methods included from Validation
#properties_from_string, #transform_property_value, #validate_params
Methods included from Options
Constructor Details
Instance Attribute Details
#archive_file_location ⇒ Object (readonly)
Returns the value of attribute archive_file_location.
42 43 44 |
# File 'lib/chef_apply/cli.rb', line 42 def archive_file_location @archive_file_location end |
#target_hosts ⇒ Object (readonly)
Returns the value of attribute target_hosts.
42 43 44 |
# File 'lib/chef_apply/cli.rb', line 42 def target_hosts @target_hosts end |
#temp_cookbook ⇒ Object (readonly)
Returns the value of attribute temp_cookbook.
42 43 44 |
# File 'lib/chef_apply/cli.rb', line 42 def temp_cookbook @temp_cookbook end |
Instance Method Details
#capture_exception_backtrace(e) ⇒ Object
314 315 316 |
# File 'lib/chef_apply/cli.rb', line 314 def capture_exception_backtrace(e) UI::ErrorPrinter.write_backtrace(e, @argv) end |
#connect_target(target_host, reporter) ⇒ Object
Accepts a target_host and establishes the connection to that host while providing visual feedback via the Terminal API.
164 165 166 167 168 |
# File 'lib/chef_apply/cli.rb', line 164 def connect_target(target_host, reporter) = T.status.connecting(target_host.user) reporter.update() do_connect(target_host, reporter) end |
#converge(reporter, local_policy_path, target_host) ⇒ Object
Runs the Converge action and renders UI updates as the action reports back
248 249 250 251 252 253 254 255 256 257 258 259 260 261 262 263 264 265 266 267 268 269 270 |
# File 'lib/chef_apply/cli.rb', line 248 def converge(reporter, local_policy_path, target_host) reporter.update(TS.converge.converging(temp_cookbook.descriptor)) converge_args = { local_policy_path: local_policy_path, target_host: target_host } converger = Action::ConvergeTarget.new(converge_args) converger.run do |event, data| case event when :success reporter.success(TS.converge.success(temp_cookbook.descriptor)) when :converge_error reporter.error(TS.converge.failure(temp_cookbook.descriptor)) when :creating_remote_policy reporter.update(TS.converge.creating_remote_policy) when :uploading_trusted_certs reporter.update(TS.converge.uploading_trusted_certs) when :running_chef reporter.update(TS.converge.converging(temp_cookbook.descriptor)) when :reboot reporter.success(TS.converge.reboot) else (event, data, reporter) end end end |
#do_connect(target_host, reporter) ⇒ Object
318 319 320 321 322 323 324 325 |
# File 'lib/chef_apply/cli.rb', line 318 def do_connect(target_host, reporter) target_host.connect! reporter.update(T.status.connected) rescue StandardError => e = ChefApply::UI::ErrorPrinter.error_summary(e) reporter.error() raise end |
#generate_local_policy(reporter) ⇒ Object
Runs the GenerateLocalPolicy action and renders UI updates as the action reports back
229 230 231 232 233 234 235 236 237 238 239 240 241 242 243 244 |
# File 'lib/chef_apply/cli.rb', line 229 def generate_local_policy(reporter) action = Action::GenerateLocalPolicy.new(cookbook: temp_cookbook) action.run do |event, data| case event when :generating reporter.update(TS.generate_local_policy.) when :exporting reporter.update(TS.generate_local_policy.exporting) when :success reporter.success(TS.generate_local_policy.success) else (event, data, reporter) end end action.archive_file_location end |
#generate_temp_cookbook(arguments, reporter) ⇒ Object
Runs a GenerateCookbook action based on recipe/resource infoprovided and renders UI updates as the action reports back
204 205 206 207 208 209 210 211 212 213 214 215 216 217 218 219 220 221 222 223 224 225 |
# File 'lib/chef_apply/cli.rb', line 204 def generate_temp_cookbook(arguments, reporter) opts = if arguments.length == 1 { recipe_spec: arguments.shift, cookbook_repo_paths: [:cookbook_repo_paths] } else { resource_type: arguments.shift, resource_name: arguments.shift, resource_properties: properties_from_string(arguments) } end action = ChefApply::Action::GenerateTempCookbook.(opts) action.run do |event, data| case event when :generating reporter.update(TS.generate_temp_cookbook.) when :success reporter.success(TS.generate_temp_cookbook.success) else (event, data, reporter) end end action.generated_cookbook end |
#handle_failed_job(job) ⇒ Object
302 303 304 |
# File 'lib/chef_apply/cli.rb', line 302 def handle_failed_job(job) raise job.exception unless job.exception.nil? end |
#handle_failed_jobs(jobs) ⇒ Object
When running multiple jobs, exceptions are captured to the job to avoid interrupting other jobs in process. This function collects them and raises directly (in the case of just one job in the list) or raises a MultiJobFailure (when more than one job was being run)
291 292 293 294 295 296 297 298 299 300 |
# File 'lib/chef_apply/cli.rb', line 291 def handle_failed_jobs(jobs) failed_jobs = jobs.select { |j| !j.exception.nil? } return if failed_jobs.empty? if jobs.length == 1 # Don't provide a bad UX by showing a 'one or more jobs has failed' # message when there was only one job. raise jobs.first.exception end raise ChefApply::MultiJobFailure.new(failed_jobs) end |
#handle_message(message, data, reporter) ⇒ Object
A handler for common action messages
307 308 309 310 311 312 |
# File 'lib/chef_apply/cli.rb', line 307 def (, data, reporter) if == :error # data[0] = exception # Mark the current task as failed with whatever data is available to us reporter.error(ChefApply::UI::ErrorPrinter.error_summary(data[0])) end end |
#handle_perform_error(e) ⇒ Object
272 273 274 275 276 277 278 279 280 281 282 283 284 285 |
# File 'lib/chef_apply/cli.rb', line 272 def handle_perform_error(e) require "chef_apply/errors/standard_error_resolver" id = e.respond_to?(:id) ? e.id : e.class.to_s # TODO: This is currently sending host information for certain ssh errors # post release we need to scrub this data. For now I'm redacting the # whole message. # message = e.respond_to?(:message) ? e.message : e.to_s Telemeter.capture(:error, exception: { id: id, message: "redacted" }) wrapper = ChefApply::Errors::StandardErrorResolver.wrap_exception(e) capture_exception_backtrace(wrapper) # Now that our housekeeping is done, allow user-facing handling/formatting # in `run` to execute by re-raising raise wrapper end |
#handle_run_error(e) ⇒ Object
84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 |
# File 'lib/chef_apply/cli.rb', line 84 def handle_run_error(e) case e when nil RC_OK when WrappedError UI::ErrorPrinter.show_error(e) RC_COMMAND_FAILED when SystemExit e.status when Exception UI::ErrorPrinter.dump_unexpected_error(e) RC_ERROR_HANDLING_FAILED else UI::ErrorPrinter.dump_unexpected_error(e) RC_UNHANDLED_ERROR end end |
#install(target_host, reporter) ⇒ Object
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 198 199 200 |
# File 'lib/chef_apply/cli.rb', line 170 def install(target_host, reporter) context = TS.install_chef reporter.update(context.) installer = Action::InstallChef.instance_for_target(target_host, check_only: ![:install]) installer.run do |event, data| case event when :installing if installer.upgrading? = context.upgrading(target_host.installed_chef_version, installer.version_to_install) else = context.installing(installer.version_to_install) end reporter.update() when :uploading reporter.update(context.uploading) when :downloading reporter.update(context.downloading) when :already_installed reporter.update(context.already_present(target_host.installed_chef_version)) when :install_complete if installer.upgrading? = context.upgrade_success(target_host.installed_chef_version, installer.version_to_install) else = context.install_success(installer.version_to_install) end reporter.update() else (event, data, reporter) end end end |
#perform_run ⇒ Object
102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 |
# File 'lib/chef_apply/cli.rb', line 102 def perform_run (@argv) if @argv.empty? || [:help] show_help elsif [:version] show_version else validate_params(cli_arguments) target_hosts = resolve_targets(cli_arguments.shift, ) render_cookbook_setup(cli_arguments) render_converge(target_hosts) end rescue OptionParser::InvalidOption => e # from parse_options # Using nil here is a bit gross but it prevents usage from printing. ove = OptionValidationError.new("CHEFVAL010", nil, e..split(":")[1].strip, # only want the flag format_flags.lines[1..-1].join # remove 'FLAGS:' header ) handle_perform_error(ove) rescue => e handle_perform_error(e) ensure temp_cookbook.delete unless temp_cookbook.nil? end |
#render_converge(target_hosts) ⇒ Object
148 149 150 151 152 153 154 155 156 157 158 159 160 |
# File 'lib/chef_apply/cli.rb', line 148 def render_converge(target_hosts) jobs = target_hosts.map do |target_host| # Each block will run in its own thread during render. UI::Terminal::Job.new("[#{target_host.hostname}]", target_host) do |reporter| connect_target(target_host, reporter) install(target_host, reporter) converge(reporter, archive_file_location, target_host) end end header = TS.converge.header(target_hosts.length, temp_cookbook.descriptor, temp_cookbook.from) UI::Terminal.render_parallel_jobs(header, jobs) handle_failed_jobs(jobs) end |
#render_cookbook_setup(arguments) ⇒ Object
133 134 135 136 137 138 139 140 141 142 143 144 145 146 |
# File 'lib/chef_apply/cli.rb', line 133 def render_cookbook_setup(arguments) # TODO update Job so that it doesn't require prefix and host. As a data container, # should these attributes even be required? job = UI::Terminal::Job.new("", nil) do |reporter| @temp_cookbook = generate_temp_cookbook(arguments, reporter) end UI::Terminal.render_job("...", job) handle_failed_job(job) job = UI::Terminal::Job.new("", nil) do |reporter| @archive_file_location = generate_local_policy(reporter) end UI::Terminal.render_job("...", job) handle_failed_job(job) end |
#resolve_targets(host_spec, opts) ⇒ Object
127 128 129 130 131 |
# File 'lib/chef_apply/cli.rb', line 127 def resolve_targets(host_spec, opts) @target_hosts = TargetResolver.new(host_spec, opts.delete(:protocol), opts).targets end |
#run ⇒ Object
63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 |
# File 'lib/chef_apply/cli.rb', line 63 def run # Perform a timing and capture of the run. Individual methods and actions may perform # nested Telemeter.timed_*_capture or Telemeter.capture calls in their operation, and # they will be captured in the same telemetry session. # NOTE: We're not currently sending arguments to telemetry because we have not implemented # pre-parsing of arguments to eliminate potentially sensitive data such as # passwords in host name, or in ad-hoc converge properties. Telemeter.timed_run_capture([:redacted]) do begin perform_run rescue Exception => e @rc = handle_run_error(e) end end rescue => e @rc = handle_run_error(e) ensure Telemeter.commit exit @rc end |