diff --git a/.rubocop_todo.yml b/.rubocop_todo.yml index 1d93e24..3b2c6c2 100644 --- a/.rubocop_todo.yml +++ b/.rubocop_todo.yml @@ -14,12 +14,12 @@ Metrics/AbcSize: # Offense count: 1 # Configuration parameters: CountComments, CountAsOne. Metrics/ClassLength: - Max: 138 + Max: 200 # Offense count: 9 # Configuration parameters: AllowedMethods, AllowedPatterns. Metrics/CyclomaticComplexity: - Max: 20 + Max: 25 # Offense count: 19 # Configuration parameters: CountComments, CountAsOne, AllowedMethods, AllowedPatterns. @@ -34,7 +34,7 @@ Metrics/ParameterLists: # Offense count: 7 # Configuration parameters: AllowedMethods, AllowedPatterns. Metrics/PerceivedComplexity: - Max: 24 + Max: 30 # Offense count: 2 # Configuration parameters: IgnoredMetadata. @@ -51,7 +51,7 @@ RSpec/DescribeClass: # Offense count: 4 # Configuration parameters: CountAsOne. RSpec/ExampleLength: - Max: 13 + Max: 30 # Offense count: 6 RSpec/MultipleExpectations: diff --git a/spec/tasks/abs_spec.rb b/spec/tasks/abs_spec.rb index 69c55de..a191aab 100644 --- a/spec/tasks/abs_spec.rb +++ b/spec/tasks/abs_spec.rb @@ -67,8 +67,21 @@ def with_env(env_vars) expect { ABSProvision.run }.to raise_error(RuntimeError, %r{specify a platform when provisioning}) end - it 'raises an error when node_name not given for tear_down' - it 'raises an error if both node_name and platform are given' + it 'raises an error when node_name not given for tear_down' do + expect($stdin).to receive(:read).and_return('{"action":"teardown"}') + + expect { ABSProvision.run }.to raise_error(RuntimeError, %r{specify only one of: node_name, platform}) + end + + it 'raises an error if both node_name and platform are given' do + expect($stdin).to receive(:read).and_return('{"action":"teardown","platform":"centos-9"}') + + expect { ABSProvision.run }.to( + raise_error(SystemExit) do |e| + expect(e.status).to eq(0) + end, + ) + end end context 'when provisioning' do diff --git a/spec/tasks/provision_service_spec.rb b/spec/tasks/provision_service_spec.rb new file mode 100644 index 0000000..9e98d08 --- /dev/null +++ b/spec/tasks/provision_service_spec.rb @@ -0,0 +1,153 @@ +# frozen_string_literal: true + +require 'spec_helper' +require 'webmock/rspec' +require_relative '../../tasks/provision_service' + +ENV['GITHUB_RUN_ID'] = '1234567890' +ENV['GITHUB_URL'] = 'https://api.github.com/repos/puppetlabs/puppetlabs-iis/actions/runs/1234567890' + +describe 'ProvisionService' do + describe '.run' do + context 'when inputs are invalid' do + it 'return exception' do + json_input = '{}' + allow($stdin).to receive(:read).and_return(json_input) + + expect { ProvisionService.run }.to( + raise_error(SystemExit) { |e| + expect(e.status).to eq(0) + }.and( + output(%r{Unknown action}).to_stdout, + ), + ) + end + + it 'return exception about invalid action' do + json_input = '{"action":"foo","platform":"bar"}' + allow($stdin).to receive(:read).and_return(json_input) + + expect { ProvisionService.run }.to( + raise_error(SystemExit) { |e| + expect(e.status).to eq(0) + }.and( + output(%r{Unknown action 'foo'}).to_stdout, + ), + ) + end + + it 'return exception for missing platform' do + json_input = '{"action":"provision"}' + allow($stdin).to receive(:read).and_return(json_input) + + expect { ProvisionService.run }.to( + raise_error(SystemExit) { |e| + expect(e.status).to eq(1) + }.and( + output(%r{specify a platform when provisioning}).to_stdout, + ), + ) + end + + it 'return exception for missing node_name' do + json_input = '{"action":"tear_down"}' + allow($stdin).to receive(:read).and_return(json_input) + + expect { ProvisionService.run }.to( + raise_error(SystemExit) { |e| + expect(e.status).to eq(1) + }.and( + output(%r{specify a node_name when tearing down}).to_stdout, + ), + ) + end + end + end + + describe '#provision' do + let(:inventory_location) { "#{Dir.pwd}/litmus_inventory.yaml" } + let(:vars) { nil } + let(:platform) { 'centos-8' } + let(:retry_attempts) { 8 } + let(:response_body) do + { + 'groups' => [ + 'targets' => { + 'uri' => '127.0.0.1' + }, + ] + } + end + let(:provision_service) { ProvisionService.new } + + context 'when response is empty' do + it 'return exception' do + stub_request(:post, 'https://facade-release-6f3kfepqcq-ew.a.run.app/v1/provision') + .with( + body: '{"url":"https://api.github.com/repos/puppetlabs/puppetlabs-iis/actions/runs/1234567890","VMs":[{"cloud":null,"region":null,"zone":null,"images":["centos-8"]}]}', + headers: { + 'Accept' => 'application/json', + 'Accept-Encoding' => 'gzip;q=1.0,deflate;q=0.6,identity;q=0.3', + 'Content-Type' => 'application/json', + 'Host' => 'facade-release-6f3kfepqcq-ew.a.run.app', + 'User-Agent' => 'Ruby' + }, + ) + .to_return(status: 200, body: '', headers: {}) + expect { provision_service.provision(platform, inventory_location, vars, retry_attempts) }.to raise_error(RuntimeError) + end + end + + context 'when successive retry success' do + it 'return valid response' do + stub_request(:post, 'https://facade-release-6f3kfepqcq-ew.a.run.app/v1/provision') + .with( + body: '{"url":"https://api.github.com/repos/puppetlabs/puppetlabs-iis/actions/runs/1234567890","VMs":[{"cloud":null,"region":null,"zone":null,"images":["centos-8"]}]}', + headers: { + 'Accept' => 'application/json', + 'Accept-Encoding' => 'gzip;q=1.0,deflate;q=0.6,identity;q=0.3', + 'Content-Type' => 'application/json', + 'Host' => 'facade-release-6f3kfepqcq-ew.a.run.app', + 'User-Agent' => 'Ruby' + }, + ) + .to_return(status: 200, body: '', headers: {}) + expect { provision_service.provision(platform, inventory_location, vars, retry_attempts) }.to raise_error(RuntimeError) + stub_request(:post, 'https://facade-release-6f3kfepqcq-ew.a.run.app/v1/provision') + .with( + body: '{"url":"https://api.github.com/repos/puppetlabs/puppetlabs-iis/actions/runs/1234567890","VMs":[{"cloud":null,"region":null,"zone":null,"images":["centos-8"]}]}', + headers: { + 'Accept' => 'application/json', + 'Accept-Encoding' => 'gzip;q=1.0,deflate;q=0.6,identity;q=0.3', + 'Content-Type' => 'application/json', + 'Host' => 'facade-release-6f3kfepqcq-ew.a.run.app', + 'User-Agent' => 'Ruby' + }, + ) + .to_return(status: 200, body: response_body.to_json, headers: {}) + allow(File).to receive(:open) + expect(provision_service.provision(platform, inventory_location, vars, retry_attempts)[:status]).to eq('ok') + end + end + + context 'when response is avlid' do + it 'return valid response' do + stub_request(:post, 'https://facade-release-6f3kfepqcq-ew.a.run.app/v1/provision') + .with( + body: '{"url":"https://api.github.com/repos/puppetlabs/puppetlabs-iis/actions/runs/1234567890","VMs":[{"cloud":null,"region":null,"zone":null,"images":["centos-8"]}]}', + headers: { + 'Accept' => 'application/json', + 'Accept-Encoding' => 'gzip;q=1.0,deflate;q=0.6,identity;q=0.3', + 'Content-Type' => 'application/json', + 'Host' => 'facade-release-6f3kfepqcq-ew.a.run.app', + 'User-Agent' => 'Ruby' + }, + ) + .to_return(status: 200, body: response_body.to_json, headers: {}) + + allow(File).to receive(:open) + expect(provision_service.provision(platform, inventory_location, vars, retry_attempts)[:status]).to eq('ok') + end + end + end +end diff --git a/tasks/provision_service.rb b/tasks/provision_service.rb index 9aa5cd2..bd5c099 100755 --- a/tasks/provision_service.rb +++ b/tasks/provision_service.rb @@ -7,174 +7,190 @@ require 'puppet_litmus' require 'etc' require_relative '../lib/task_helper' -include PuppetLitmus::InventoryManipulation -def default_uri - 'https://facade-release-6f3kfepqcq-ew.a.run.app/v1/provision' -end +# Provision and teardown vms through provision service. +class ProvisionService + RETRY_COUNT = 3 -def platform_to_cloud_request_parameters(platform, cloud, region, zone) - case platform - when String - { cloud: cloud, region: region, zone: zone, images: [platform] } - when Array - { cloud: cloud, region: region, zone: zone, images: platform } - else - platform[:cloud] = cloud unless cloud.nil? - platform[:images] = [platform[:images]] if platform[:images].is_a?(String) - platform - end -end + include PuppetLitmus::InventoryManipulation -# curl -X POST https://facade-validation-6f3kfepqcq-ew.a.run.app/v1/provision --data @test_machines.json -def invoke_cloud_request(params, uri, job_url, verb, retry_attempts) - headers = { - 'Accept' => 'application/json', - 'Content-Type' => 'application/json' - } - headers['X-Honeycomb-Trace'] = ENV['HTTP_X_HONEYCOMB_TRACE'] if ENV['HTTP_X_HONEYCOMB_TRACE'] # legacy variable - headers['X-Honeycomb-Trace'] = ENV['HONEYCOMB_TRACE'] if ENV['HONEYCOMB_TRACE'] - - case verb.downcase - when 'post' - request = Net::HTTP::Post.new(uri, headers) - machines = [] - machines << params - request.body = if job_url - { url: job_url, VMs: machines }.to_json - else - { github_token: ENV['GITHUB_TOKEN'], VMs: machines }.to_json - end - when 'delete' - request = Net::HTTP::Delete.new(uri, headers) - request.body = { uuid: params }.to_json - else - raise StandardError "Unknown verb: '#{verb}'" + def default_uri + 'https://facade-release-6f3kfepqcq-ew.a.run.app/v1/provision' end - if job_url - File.open('request.json', 'wb') do |f| - f.write(request.body) + def platform_to_cloud_request_parameters(platform, cloud, region, zone) + case platform + when String + { cloud: cloud, region: region, zone: zone, images: [platform] } + when Array + { cloud: cloud, region: region, zone: zone, images: platform } + else + platform[:cloud] = cloud unless cloud.nil? + platform[:images] = [platform[:images]] if platform[:images].is_a?(String) + platform end end - req_options = { - use_ssl: uri.scheme == 'https', - read_timeout: 60 * 5, # timeout reads after 5 minutes - that's longer than the backend service would keep the request open - max_retries: retry_attempts # retry up to 5 times before throwing an error - } + # curl -X POST https://facade-validation-6f3kfepqcq-ew.a.run.app/v1/provision --data @test_machines.json + def invoke_cloud_request(params, uri, job_url, verb, retry_attempts) + headers = { + 'Accept' => 'application/json', + 'Content-Type' => 'application/json' + } + + case verb.downcase + when 'post' + request = Net::HTTP::Post.new(uri, headers) + machines = [] + machines << params + request.body = if job_url + { url: job_url, VMs: machines }.to_json + else + { github_token: ENV['GITHUB_TOKEN'], VMs: machines }.to_json + end + when 'delete' + request = Net::HTTP::Delete.new(uri, headers) + request.body = { uuid: params }.to_json + else + raise StandardError "Unknown verb: '#{verb}'" + end - response = Net::HTTP.start(uri.hostname, uri.port, req_options) do |http| - http.request(request) - end - if response.code == '200' - response.body - else - begin - body = JSON.parse(response.body) - body_json = true - rescue JSON::ParserError - body = response.body - body_json = false + if job_url + File.open('request.json', 'wb') do |f| + f.write(request.body) + end end - puts({ _error: { kind: 'provision_service/service_error', msg: 'provision service returned an error', code: response.code, body: body, body_json: body_json } }.to_json) - exit 1 - end -end -def provision(platform, inventory_location, vars, retry_attempts) - # Call the provision service with the information necessary and write the inventory file locally + req_options = { + use_ssl: uri.scheme == 'https', + read_timeout: 60 * 5, # timeout reads after 5 minutes - that's longer than the backend service would keep the request open + max_retries: retry_attempts # retry up to 5 times before throwing an error + } - if ENV['GITHUB_RUN_ID'] - job_url = ENV['GITHUB_URL'] || "https://api.github.com/repos/#{ENV['GITHUB_REPOSITORY']}/actions/runs/#{ENV['GITHUB_RUN_ID']}" - else - puts 'Using GITHUB_TOKEN as no GITHHUB_RUN_ID found' - end - uri = URI.parse(ENV['SERVICE_URL'] || default_uri) - cloud = ENV['CLOUD'] - region = ENV['REGION'] - zone = ENV['ZONE'] - if job_url.nil? && vars - data = JSON.parse(vars.tr(';', ',')) - job_url = data['job_url'] + response = Net::HTTP.start(uri.hostname, uri.port, req_options) do |http| + http.request(request) + end + if response.code == '200' + response.body + else + begin + body = JSON.parse(response.body) + body_json = true + rescue JSON::ParserError + body = response.body + body_json = false + end + puts({ _error: { kind: 'provision_service/service_error', msg: 'provision service returned an error', code: response.code, body: body, body_json: body_json } }.to_json) + exit 1 + end end - inventory_full_path = File.join(inventory_location, '/spec/fixtures/litmus_inventory.yaml') - params = platform_to_cloud_request_parameters(platform, cloud, region, zone) - response = invoke_cloud_request(params, uri, job_url, 'post', retry_attempts) - response_hash = YAML.safe_load(response) + def provision(platform, inventory_location, vars, retry_attempts) + # Call the provision service with the information necessary and write the inventory file locally - unless vars.nil? - var_hash = YAML.safe_load(vars) - response_hash['groups'].each do |bg| - bg['targets'].each do |trgts| - trgts['vars'] = var_hash - end + if ENV['GITHUB_RUN_ID'] + job_url = ENV['GITHUB_URL'] || "https://api.github.com/repos/#{ENV['GITHUB_REPOSITORY']}/actions/runs/#{ENV['GITHUB_RUN_ID']}" + else + puts 'Using GITHUB_TOKEN as no GITHHUB_RUN_ID found' + end + uri = URI.parse(ENV['SERVICE_URL'] || default_uri) + cloud = ENV['CLOUD'] + region = ENV['REGION'] + zone = ENV['ZONE'] + if job_url.nil? && vars + data = JSON.parse(vars.tr(';', ',')) + job_url = data['job_url'] + end + inventory_full_path = File.join(inventory_location, '/spec/fixtures/litmus_inventory.yaml') + currnet_retry_count = 0 + begin + params = platform_to_cloud_request_parameters(platform, cloud, region, zone) + response = invoke_cloud_request(params, uri, job_url, 'post', retry_attempts) + response_hash = YAML.safe_load(response) + # Knock the response for validity to make sure return payload is expected. + # Have seen multiple occurances of nil:NilClass error where the response code is 200 but return payload is empty + raise if response_hash.nil? || response_hash.empty? + rescue StandardError => e + currnet_retry_count += 1 + raise e if currnet_retry_count >= RETRY_COUNT + + puts "Failed while provisioning the resource with response :\n #{response_hash}\nHence retrying #{currnet_retry_count} of #{RETRY_COUNT}" + retry end - end - if File.file?(inventory_full_path) - inventory_hash = inventory_hash_from_inventory_file(inventory_full_path) - inventory_hash['groups'].each do |g| + unless vars.nil? + var_hash = YAML.safe_load(vars) response_hash['groups'].each do |bg| - g['targets'] = g['targets'] + bg['targets'] if g['name'] == bg['name'] + bg['targets'].each do |trgts| + trgts['vars'] = var_hash + end end end - File.open(inventory_full_path, 'w') { |f| f.write inventory_hash.to_yaml } - else - FileUtils.mkdir_p(File.join(Dir.pwd, '/spec/fixtures')) - File.open(inventory_full_path, 'wb') do |f| - f.write(YAML.dump(response_hash)) + + if File.file?(inventory_full_path) + inventory_hash = inventory_hash_from_inventory_file(inventory_full_path) + inventory_hash['groups'].each do |g| + response_hash['groups'].each do |bg| + g['targets'] = g['targets'] + bg['targets'] if g['name'] == bg['name'] + end + end + File.open(inventory_full_path, 'w') { |f| f.write inventory_hash.to_yaml } + else + FileUtils.mkdir_p(File.join(Dir.pwd, '/spec/fixtures')) + File.open(inventory_full_path, 'wb') do |f| + f.write(YAML.dump(response_hash)) + end end - end - { - status: 'ok', - node_name: platform, - target_names: response_hash['groups']&.each { |g| g['targets'] }&.map { |t| t['uri'] }&.flatten&.uniq - } -end + { + status: 'ok', + node_name: platform, + target_names: response_hash['groups']&.each { |g| g['targets'] }&.map { |t| t['uri'] }&.flatten&.uniq + } + end -def tear_down(platform, inventory_location, _vars, retry_attempts) - # remove all provisioned resources - uri = URI.parse(ENV['SERVICE_URL'] || default_uri) - - inventory_full_path = File.join(inventory_location, '/spec/fixtures/litmus_inventory.yaml') - # rubocop:disable Style/GuardClause - if File.file?(inventory_full_path) - inventory_hash = inventory_hash_from_inventory_file(inventory_full_path) - facts = facts_from_node(inventory_hash, platform) - job_id = facts['uuid'] - response = invoke_cloud_request(job_id, uri, '', 'delete', retry_attempts) - response.to_json + def tear_down(platform, inventory_location, _vars, retry_attempts) + # remove all provisioned resources + uri = URI.parse(ENV['SERVICE_URL'] || default_uri) + + inventory_full_path = File.join(inventory_location, '/spec/fixtures/litmus_inventory.yaml') + # rubocop:disable Style/GuardClause + if File.file?(inventory_full_path) + inventory_hash = inventory_hash_from_inventory_file(inventory_full_path) + facts = facts_from_node(inventory_hash, platform) + job_id = facts['uuid'] + response = invoke_cloud_request(job_id, uri, '', 'delete', retry_attempts) + response.to_json + end + # rubocop:enable Style/GuardClause end - # rubocop:enable Style/GuardClause -end -params = JSON.parse($stdin.read) -platform = params['platform'] -action = params['action'] -vars = params['vars'] -node_name = params['node_name'] -retry_attempts = params['retry_attempts'] -inventory_location = sanitise_inventory_location(params['inventory']) - -begin - case action - when 'provision' - raise 'specify a platform when provisioning' if platform.nil? - - result = provision(platform, inventory_location, vars, retry_attempts) - when 'tear_down' - raise 'specify a node_name when tearing down' if node_name.nil? - - result = tear_down(node_name, inventory_location, vars, retry_attempts) - else - result = { _error: { kind: 'provision_service/argument_error', msg: "Unknown action '#{action}'" } } + def self.run + params = JSON.parse($stdin.read) + params.transform_keys!(&:to_sym) + action, node_name, platform, vars, retry_attempts, inventory_location = params.values_at(:action, :node_name, :platform, :vars, :retry_attempts, :inventory) + + runner = new + begin + case action + when 'provision' + raise 'specify a platform when provisioning' if platform.to_s.empty? + + result = runner.provision(platform, inventory_location, vars, retry_attempts) + when 'tear_down' + raise 'specify a node_name when tearing down' if node_name.nil? + + result = runner.tear_down(node_name, inventory_location, vars, retry_attempts) + else + result = { _error: { kind: 'provision_service/argument_error', msg: "Unknown action '#{action}'" } } + end + puts result.to_json + exit 0 + rescue StandardError => e + puts({ _error: { kind: 'provision_service/failure', msg: e.message, details: { backtrace: e.backtrace } } }.to_json) + exit 1 + end end - puts result.to_json - exit 0 -rescue StandardError => e - puts({ _error: { kind: 'provision_service/failure', msg: e.message, details: { backtrace: e.backtrace } } }.to_json) - exit 1 end + +ProvisionService.run if __FILE__ == $PROGRAM_NAME