ManageIQ/manageiq

View on GitHub
app/models/file_depot_s3.rb

Summary

Maintainability
A
0 mins
Test Coverage
F
26%
class FileDepotS3 < FileDepot
  attr_accessor :s3

  def self.uri_prefix
    "s3"
  end

  def self.validate_settings(settings)
    new(:uri => settings[:uri]).verify_credentials(nil, settings.slice(:username, :password))
  end

  def connect(options = {})
    require 'aws-sdk-s3'

    username = options[:username] || authentication_userid(options[:auth_type])
    password = options[:password] || authentication_password(options[:auth_type])
    # Note: The hard-coded aws_region will be removed after manageiq-ui-class implements region selection
    aws_region = options[:region] || "us-east-1"

    $aws_log ||= Vmdb::Loggers.create_logger("aws.log")

    Aws::S3::Resource.new(
      :access_key_id     => username,
      :secret_access_key => ManageIQ::Password.try_decrypt(password),
      :region            => aws_region,
      :logger            => $aws_log,
      :log_level         => :debug,
      :log_formatter     => Aws::Log::Formatter.new(Aws::Log::Formatter.default.pattern.chomp)
    )
  end

  def with_depot_connection(options = {})
    raise _("no block given") unless block_given?

    _log.info("Connecting through #{self.class.name}: [#{name}]")
    yield connect(options)
  end

  def verify_credentials(auth_type = nil, options = {})
    connection_rescue_block do
      # aws-sdk does Lazy Connections, so call a cheap function
      with_depot_connection(options.merge(:auth_type => auth_type)) do |s3|
        validate_connection(s3)
      end
    end

    true
  end

  def validate_connection(connection)
    connection_rescue_block do
      connection.client.list_buckets
    end
  end

  def connection_rescue_block
    yield
  rescue => err
    miq_exception = translate_exception(err)
    raise unless miq_exception

    _log.log_backtrace(err)
    _log.error("Error Class=#{err.class.name}, Message=#{err.message}")
    raise miq_exception
  end

  def translate_exception(err)
    require 'aws-sdk-s3'

    case err
    when Aws::S3::Errors::SignatureDoesNotMatch
      MiqException::MiqHostError.new("SignatureMismatch - check your AWS Secret Access Key and signing method")
    when Aws::S3::Errors::AuthFailure
      MiqException::MiqHostError.new("Login failed due to a bad username or password.")
    when Aws::Errors::MissingCredentialsError
      MiqException::MiqHostError.new("Missing credentials")
    else
      MiqException::MiqHostError.new("Unexpected response returned from system: #{err.message}")
    end
  end

  def self.display_name(number = 1)
    n_('AWS S3', 'AWS S3', number)
  end
end