miasma-aws 0.3.4 → 0.3.6
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- checksums.yaml +4 -4
- data/CHANGELOG.md +4 -0
- data/README.md +4 -0
- data/lib/miasma-aws/api.rb +1 -0
- data/lib/miasma-aws/api/iam.rb +3 -2
- data/lib/miasma-aws/api/sts.rb +3 -2
- data/lib/miasma-aws/version.rb +1 -1
- data/lib/miasma/contrib/aws.rb +133 -38
- data/lib/miasma/contrib/aws/auto_scale.rb +10 -5
- data/lib/miasma/contrib/aws/compute.rb +33 -15
- data/lib/miasma/contrib/aws/load_balancer.rb +26 -9
- data/lib/miasma/contrib/aws/orchestration.rb +47 -23
- data/lib/miasma/contrib/aws/storage.rb +9 -5
- data/miasma-aws.gemspec +3 -3
- metadata +15 -16
    
        checksums.yaml
    CHANGED
    
    | @@ -1,7 +1,7 @@ | |
| 1 1 | 
             
            ---
         | 
| 2 2 | 
             
            SHA1:
         | 
| 3 | 
            -
              metadata.gz:  | 
| 4 | 
            -
              data.tar.gz:  | 
| 3 | 
            +
              metadata.gz: caa36177f0ea2511cec15e3f84b55ab542845255
         | 
| 4 | 
            +
              data.tar.gz: c2e70dcbce33f6514fd110a2d1e90f2baf148675
         | 
| 5 5 | 
             
            SHA512:
         | 
| 6 | 
            -
              metadata.gz:  | 
| 7 | 
            -
              data.tar.gz:  | 
| 6 | 
            +
              metadata.gz: 7aff9a89bb222d9a3a247d167a3536759b4008aa1bb6fc83ab5baf536292452a7524319cbd113ea998c2be0bf3ed92833af6336f7d58f37f375b2e0ed46546c8
         | 
| 7 | 
            +
              data.tar.gz: 488ba3c0126a6d38795edd6d78f5e060b89370c9c34ababcd1f5996cff2bab4e428a924d0ecc947e0756e7686d8357bb3f4c60bbf02a357c1c717f866ce5df0d
         | 
    
        data/CHANGELOG.md
    CHANGED
    
    
    
        data/README.md
    CHANGED
    
    | @@ -33,6 +33,10 @@ Miasma.api( | |
| 33 33 |  | 
| 34 34 | 
             
            * `aws_iam_instance_profile` - Extract and use instance IAM credentials
         | 
| 35 35 |  | 
| 36 | 
            +
            ### ECS task related attributes
         | 
| 37 | 
            +
             | 
| 38 | 
            +
            * `aws_ecs_task_profile` - Extract and use ECS task IAM credentials
         | 
| 39 | 
            +
             | 
| 36 40 | 
             
            ### Secure Token Service related:
         | 
| 37 41 |  | 
| 38 42 | 
             
            * `aws_sts_token` - Set STS token to use with current key ID and secret
         | 
    
        data/lib/miasma-aws/api.rb
    CHANGED
    
    
    
        data/lib/miasma-aws/api/iam.rb
    CHANGED
    
    | @@ -4,12 +4,13 @@ module Miasma | |
| 4 4 | 
             
              module Contrib
         | 
| 5 5 | 
             
                module Aws
         | 
| 6 6 | 
             
                  module Api
         | 
| 7 | 
            +
                    # IAM helper class
         | 
| 7 8 | 
             
                    class Iam < Miasma::Types::Api
         | 
| 8 9 |  | 
| 9 10 | 
             
                      # Service name of the API
         | 
| 10 | 
            -
                      API_SERVICE = 'iam'
         | 
| 11 | 
            +
                      API_SERVICE = 'iam'.freeze
         | 
| 11 12 | 
             
                      # Supported version of the IAM API
         | 
| 12 | 
            -
                      API_VERSION = '2010-05-08'
         | 
| 13 | 
            +
                      API_VERSION = '2010-05-08'.freeze
         | 
| 13 14 |  | 
| 14 15 | 
             
                      include Contrib::AwsApiCore::ApiCommon
         | 
| 15 16 | 
             
                      include Contrib::AwsApiCore::RequestUtils
         | 
    
        data/lib/miasma-aws/api/sts.rb
    CHANGED
    
    | @@ -4,12 +4,13 @@ module Miasma | |
| 4 4 | 
             
              module Contrib
         | 
| 5 5 | 
             
                module Aws
         | 
| 6 6 | 
             
                  module Api
         | 
| 7 | 
            +
                    # STS helper class
         | 
| 7 8 | 
             
                    class Sts < Miasma::Types::Api
         | 
| 8 9 |  | 
| 9 10 | 
             
                      # Service name of the API
         | 
| 10 | 
            -
                      API_SERVICE = 'sts'
         | 
| 11 | 
            +
                      API_SERVICE = 'sts'.freeze
         | 
| 11 12 | 
             
                      # Supported version of the STS API
         | 
| 12 | 
            -
                      API_VERSION = '2011-06-15'
         | 
| 13 | 
            +
                      API_VERSION = '2011-06-15'.freeze
         | 
| 13 14 |  | 
| 14 15 | 
             
                      include Contrib::AwsApiCore::ApiCommon
         | 
| 15 16 | 
             
                      include Contrib::AwsApiCore::RequestUtils
         | 
    
        data/lib/miasma-aws/version.rb
    CHANGED
    
    
    
        data/lib/miasma/contrib/aws.rb
    CHANGED
    
    | @@ -4,14 +4,17 @@ require 'miasma/utils/smash' | |
| 4 4 | 
             
            require 'time'
         | 
| 5 5 | 
             
            require 'openssl'
         | 
| 6 6 |  | 
| 7 | 
            +
            # Miasma
         | 
| 7 8 | 
             
            module Miasma
         | 
| 8 9 | 
             
              module Contrib
         | 
| 10 | 
            +
                # AWS API implementations
         | 
| 9 11 | 
             
                module Aws
         | 
| 10 12 | 
             
                  autoload :Api, 'miasma-aws/api'
         | 
| 11 13 | 
             
                end
         | 
| 12 14 | 
             
                # Core API for AWS access
         | 
| 13 15 | 
             
                class AwsApiCore
         | 
| 14 16 |  | 
| 17 | 
            +
                  # Utility methods for API requests
         | 
| 15 18 | 
             
                  module RequestUtils
         | 
| 16 19 |  | 
| 17 20 | 
             
                    # Fetch all results when tokens are being used
         | 
| @@ -34,6 +37,11 @@ module Miasma | |
| 34 37 | 
             
                      end
         | 
| 35 38 | 
             
                      set = result.get(*result_key.slice(0, 3))
         | 
| 36 39 | 
             
                      if(set.is_a?(Hash) && set['NextToken'])
         | 
| 40 | 
            +
                        [content].flatten.compact.each do |item|
         | 
| 41 | 
            +
                          if(item.is_a?(Hash))
         | 
| 42 | 
            +
                            item['NextToken'] = set['NextToken']
         | 
| 43 | 
            +
                          end
         | 
| 44 | 
            +
                        end
         | 
| 37 45 | 
             
                        list += all_result_pages(set['NextToken'], *result_key, &block)
         | 
| 38 46 | 
             
                      end
         | 
| 39 47 | 
             
                      list.compact
         | 
| @@ -127,8 +135,8 @@ module Miasma | |
| 127 135 | 
             
                    # @param string [String] string to escape
         | 
| 128 136 | 
             
                    # @return [String] escaped string
         | 
| 129 137 | 
             
                    def safe_escape(string)
         | 
| 130 | 
            -
                      string.to_s.gsub(/([^a-zA-Z0-9_.\-~])/) do
         | 
| 131 | 
            -
                        '%' <<  | 
| 138 | 
            +
                      string.to_s.gsub(/([^a-zA-Z0-9_.\-~])/) do |match|
         | 
| 139 | 
            +
                        '%' << match.unpack('H2' * match.bytesize).join('%').upcase
         | 
| 132 140 | 
             
                      end
         | 
| 133 141 | 
             
                    end
         | 
| 134 142 |  | 
| @@ -168,7 +176,8 @@ module Miasma | |
| 168 176 | 
             
                    # @return [String] signature
         | 
| 169 177 | 
             
                    def generate(http_method, path, opts)
         | 
| 170 178 | 
             
                      signature = generate_signature(http_method, path, opts)
         | 
| 171 | 
            -
                      "#{algorithm} Credential=#{access_key}/#{credential_scope},  | 
| 179 | 
            +
                      "#{algorithm} Credential=#{access_key}/#{credential_scope}, " \
         | 
| 180 | 
            +
                        "SignedHeaders=#{signed_headers(opts[:headers])}, Signature=#{signature}"
         | 
| 172 181 | 
             
                    end
         | 
| 173 182 |  | 
| 174 183 | 
             
                    # Generate URL with signed params
         | 
| @@ -185,7 +194,10 @@ module Miasma | |
| 185 194 | 
             
                          'X-Amz-Credential' => "#{access_key}/#{credential_scope}"
         | 
| 186 195 | 
             
                        )
         | 
| 187 196 | 
             
                      )
         | 
| 188 | 
            -
                      signature = generate_signature( | 
| 197 | 
            +
                      signature = generate_signature(
         | 
| 198 | 
            +
                        http_method, path,
         | 
| 199 | 
            +
                        opts.merge(:body => 'UNSIGNED-PAYLOAD')
         | 
| 200 | 
            +
                      )
         | 
| 189 201 | 
             
                      params = opts[:params].merge('X-Amz-Signature' => signature)
         | 
| 190 202 | 
             
                      "https://#{opts[:headers]['Host']}/#{path}?#{canonical_query(params)}"
         | 
| 191 203 | 
             
                    end
         | 
| @@ -326,6 +338,7 @@ module Miasma | |
| 326 338 |  | 
| 327 339 | 
             
                  end
         | 
| 328 340 |  | 
| 341 | 
            +
                  # Common API setup
         | 
| 329 342 | 
             
                  module ApiCommon
         | 
| 330 343 |  | 
| 331 344 | 
             
                    def self.included(klass)
         | 
| @@ -340,17 +353,22 @@ module Miasma | |
| 340 353 | 
             
                        attribute :aws_sts_session_token, String
         | 
| 341 354 | 
             
                        attribute :aws_sts_session_token_code, [String, Proc, Method]
         | 
| 342 355 | 
             
                        attribute :aws_sts_mfa_serial_number, [String]
         | 
| 343 | 
            -
                        attribute :aws_credentials_file, String, :required => true, | 
| 344 | 
            -
             | 
| 356 | 
            +
                        attribute :aws_credentials_file, String, :required => true,
         | 
| 357 | 
            +
                          :default => File.join(Dir.home, '.aws/credentials')
         | 
| 358 | 
            +
                        attribute :aws_config_file, String, :required => true,
         | 
| 359 | 
            +
                          :default => File.join(Dir.home, '.aws/config')
         | 
| 345 360 | 
             
                        attribute :aws_access_key_id, String, :required => true
         | 
| 346 361 | 
             
                        attribute :aws_secret_access_key, String, :required => true
         | 
| 347 362 | 
             
                        attribute :aws_iam_instance_profile, [TrueClass, FalseClass], :default => false
         | 
| 363 | 
            +
                        attribute :aws_ecs_task_profile, [TrueClass, FalseClass], :default => false
         | 
| 348 364 | 
             
                        attribute :aws_region, String, :required => true
         | 
| 349 365 | 
             
                        attribute :aws_host, String
         | 
| 350 366 | 
             
                        attribute :aws_bucket_region, String
         | 
| 351 367 | 
             
                        attribute :api_endpoint, String, :required => true, :default => 'amazonaws.com'
         | 
| 352 | 
            -
                        attribute :euca_compat, Symbol, :allowed_values => [:path, :dns], | 
| 353 | 
            -
             | 
| 368 | 
            +
                        attribute :euca_compat, Symbol, :allowed_values => [:path, :dns],
         | 
| 369 | 
            +
                          :coerce => lambda{|v| v.is_a?(String) ? v.to_sym : v}
         | 
| 370 | 
            +
                        attribute :euca_dns_map, Smash, :coerce => lambda{|v| v.to_smash},
         | 
| 371 | 
            +
                          :default => Smash.new
         | 
| 354 372 | 
             
                        attribute :ssl_enabled, [TrueClass, FalseClass], :default => true
         | 
| 355 373 | 
             
                      end
         | 
| 356 374 |  | 
| @@ -361,11 +379,21 @@ module Miasma | |
| 361 379 | 
             
                          'role_arn' => 'aws_sts_role_arn',
         | 
| 362 380 | 
             
                          'aws_security_token' => 'aws_sts_token',
         | 
| 363 381 | 
             
                          'aws_session_token' => 'aws_sts_session_token'
         | 
| 364 | 
            -
                        )
         | 
| 382 | 
            +
                        ).to_smash.freeze
         | 
| 383 | 
            +
                      )
         | 
| 384 | 
            +
                      klass.const_set(:INSTANCE_PROFILE_HOST, 'http://169.254.169.254'.freeze)
         | 
| 385 | 
            +
                      klass.const_set(
         | 
| 386 | 
            +
                        :INSTANCE_PROFILE_PATH,
         | 
| 387 | 
            +
                        'latest/meta-data/iam/security-credentials'.freeze
         | 
| 388 | 
            +
                      )
         | 
| 389 | 
            +
                      klass.const_set(
         | 
| 390 | 
            +
                        :INSTANCE_PROFILE_AZ_PATH,
         | 
| 391 | 
            +
                        'latest/meta-data/placement/availability-zone'.freeze
         | 
| 392 | 
            +
                      )
         | 
| 393 | 
            +
                      klass.const_set(:ECS_TASK_PROFILE_HOST, 'http://169.254.170.2'.freeze)
         | 
| 394 | 
            +
                      klass.const_set(
         | 
| 395 | 
            +
                        :ECS_TASK_PROFILE_PATH, ENV['AWS_CONTAINER_CREDENTIALS_RELATIVE_URI']
         | 
| 365 396 | 
             
                      )
         | 
| 366 | 
            -
                      klass.const_set(:INSTANCE_PROFILE_HOST, 'http://169.254.169.254')
         | 
| 367 | 
            -
                      klass.const_set(:INSTANCE_PROFILE_PATH, 'latest/meta-data/iam/security-credentials')
         | 
| 368 | 
            -
                      klass.const_set(:INSTANCE_PROFILE_AZ_PATH, 'latest/meta-data/placement/availability-zone')
         | 
| 369 397 | 
             
                    end
         | 
| 370 398 |  | 
| 371 399 | 
             
                    # Build new API for specified type using current provider / creds
         | 
| @@ -406,7 +434,9 @@ module Miasma | |
| 406 434 | 
             
                        )
         | 
| 407 435 | 
             
                      end
         | 
| 408 436 | 
             
                      if(creds[:aws_iam_instance_profile])
         | 
| 409 | 
            -
                         | 
| 437 | 
            +
                        self.class.const_get(:ECS_TASK_PROFILE_PATH).nil? ?
         | 
| 438 | 
            +
                          load_instance_credentials!(creds) :
         | 
| 439 | 
            +
                          load_ecs_credentials(creds)
         | 
| 410 440 | 
             
                      end
         | 
| 411 441 | 
             
                      true
         | 
| 412 442 | 
             
                    end
         | 
| @@ -418,7 +448,7 @@ module Miasma | |
| 418 448 | 
             
                    # @return [TrueClass]
         | 
| 419 449 | 
             
                    def after_setup(creds)
         | 
| 420 450 | 
             
                      skip = self.class.attributes.keys.map(&:to_s)
         | 
| 421 | 
            -
                      creds.each do |k,v|
         | 
| 451 | 
            +
                      creds.each do |k, v|
         | 
| 422 452 | 
             
                        k = k.to_s
         | 
| 423 453 | 
             
                        if(k.start_with?('aws_') && !skip.include?(k))
         | 
| 424 454 | 
             
                          data[k] = v
         | 
| @@ -452,33 +482,82 @@ module Miasma | |
| 452 482 | 
             
                          data = {}
         | 
| 453 483 | 
             
                        end
         | 
| 454 484 | 
             
                      end
         | 
| 455 | 
            -
                      creds | 
| 456 | 
            -
                      creds[:aws_secret_access_key] = data['SecretAccessKey']
         | 
| 457 | 
            -
                      creds[:aws_sts_token] = data['Token']
         | 
| 458 | 
            -
                      creds[:aws_sts_token_expires] = Time.xmlschema(data['Expiration'])
         | 
| 485 | 
            +
                      creds.merge!(extract_creds(data))
         | 
| 459 486 | 
             
                      unless(creds[:aws_region])
         | 
| 460 | 
            -
                         | 
| 461 | 
            -
                          [
         | 
| 462 | 
            -
                            self.class.const_get(:INSTANCE_PROFILE_HOST),
         | 
| 463 | 
            -
                            self.class.const_get(:INSTANCE_PROFILE_AZ_PATH)
         | 
| 464 | 
            -
                          ].join('/')
         | 
| 465 | 
            -
                        ).body.to_s.strip
         | 
| 466 | 
            -
                        az.sub!(/[a-zA-Z]+$/, '')
         | 
| 467 | 
            -
                        creds[:aws_region] = az
         | 
| 487 | 
            +
                        creds[:aws_region] = get_region
         | 
| 468 488 | 
             
                      end
         | 
| 469 489 | 
             
                      true
         | 
| 470 490 | 
             
                    end
         | 
| 471 491 |  | 
| 492 | 
            +
                    # Attempt to load credentials from instance metadata
         | 
| 493 | 
            +
                    #
         | 
| 494 | 
            +
                    # @param creds [Hash]
         | 
| 495 | 
            +
                    # @return [TrueClass]
         | 
| 496 | 
            +
                    def load_ecs_credentials!(creds)
         | 
| 497 | 
            +
                      # As per docs ECS_TASK_PROFILE_PATH is defined as
         | 
| 498 | 
            +
                      # /credential_provider_version/credentials?id=task_UUID
         | 
| 499 | 
            +
                      # where AWS fills in the version and UUID.
         | 
| 500 | 
            +
                      # @see https://docs.aws.amazon.com/AmazonECS/latest/developerguide/task-iam-roles.html
         | 
| 501 | 
            +
                      data = HTTP.get(
         | 
| 502 | 
            +
                        [
         | 
| 503 | 
            +
                          self.class.const_get(:ECS_TASK_PROFILE_HOST),
         | 
| 504 | 
            +
                          self.class.const_get(:ECS_TASK_PROFILE_PATH)
         | 
| 505 | 
            +
                        ].join
         | 
| 506 | 
            +
                      ).body
         | 
| 507 | 
            +
                      unless(data.is_a?(Hash))
         | 
| 508 | 
            +
                        begin
         | 
| 509 | 
            +
                          data = MultiJson.load(data.to_s)
         | 
| 510 | 
            +
                        rescue MultiJson::ParseError
         | 
| 511 | 
            +
                          data = {}
         | 
| 512 | 
            +
                        end
         | 
| 513 | 
            +
                      end
         | 
| 514 | 
            +
                      creds.merge!(extract_creds(data))
         | 
| 515 | 
            +
                      unless(creds[:aws_region])
         | 
| 516 | 
            +
                        creds[:aws_region] = get_region
         | 
| 517 | 
            +
                      end
         | 
| 518 | 
            +
                      true
         | 
| 519 | 
            +
                    end
         | 
| 520 | 
            +
             | 
| 521 | 
            +
                    # Return hash with needed information to assume role
         | 
| 522 | 
            +
                    #
         | 
| 523 | 
            +
                    # @param data [Hash]
         | 
| 524 | 
            +
                    # @return [Hash]
         | 
| 525 | 
            +
                    def extract_creds(data)
         | 
| 526 | 
            +
                      c = Smash.new
         | 
| 527 | 
            +
                      c[:aws_access_key_id] = data['AccessKeyId']
         | 
| 528 | 
            +
                      c[:aws_secret_access_key] = data['SecretAccessKey']
         | 
| 529 | 
            +
                      c[:aws_sts_token] = data['Token']
         | 
| 530 | 
            +
                      c[:aws_sts_token_expires] = Time.xmlschema(data['Expiration'])
         | 
| 531 | 
            +
                      c[:aws_sts_role_arn] = data['RoleArn'] # used in ECS Role but not instance role
         | 
| 532 | 
            +
                      c
         | 
| 533 | 
            +
                    end
         | 
| 534 | 
            +
             | 
| 535 | 
            +
                    # Return region from meta-data service
         | 
| 536 | 
            +
                    #
         | 
| 537 | 
            +
                    # @return [String]
         | 
| 538 | 
            +
                    def get_region
         | 
| 539 | 
            +
                      az = HTTP.get(
         | 
| 540 | 
            +
                        [
         | 
| 541 | 
            +
                          self.class.const_get(:INSTANCE_PROFILE_HOST),
         | 
| 542 | 
            +
                          self.class.const_get(:INSTANCE_PROFILE_AZ_PATH)
         | 
| 543 | 
            +
                        ].join('/')
         | 
| 544 | 
            +
                      ).body.to_s.strip
         | 
| 545 | 
            +
                      az.sub!(/[a-zA-Z]+$/, '')
         | 
| 546 | 
            +
                      az
         | 
| 547 | 
            +
                    end
         | 
| 548 | 
            +
             | 
| 472 549 | 
             
                    def sts_mfa_session!(creds)
         | 
| 473 550 | 
             
                      if(sts_mfa_session_update_required?(creds))
         | 
| 474 551 | 
             
                        sts = Miasma::Contrib::Aws::Api::Sts.new(
         | 
| 475 552 | 
             
                          :aws_access_key_id => creds[:aws_access_key_id],
         | 
| 476 553 | 
             
                          :aws_secret_access_key => creds[:aws_secret_access_key],
         | 
| 477 554 | 
             
                          :aws_region => creds.fetch(:aws_sts_region, 'us-east-1'),
         | 
| 478 | 
            -
                          :aws_credentials_file => creds.fetch( | 
| 555 | 
            +
                          :aws_credentials_file => creds.fetch(
         | 
| 556 | 
            +
                            :aws_credentials_file, aws_credentials_file
         | 
| 557 | 
            +
                          ),
         | 
| 479 558 | 
             
                          :aws_config_file => creds.fetch(:aws_config_file, aws_config_file),
         | 
| 480 559 | 
             
                          :aws_profile_name => creds[:aws_profile_name],
         | 
| 481 | 
            -
                          :aws_host => creds[:aws_sts_host] | 
| 560 | 
            +
                          :aws_host => creds[:aws_sts_host]
         | 
| 482 561 | 
             
                        )
         | 
| 483 562 | 
             
                        creds.merge!(
         | 
| 484 563 | 
             
                          sts.mfa_session(
         | 
| @@ -500,7 +579,9 @@ module Miasma | |
| 500 579 | 
             
                          :aws_access_key_id => get_credential(:access_key_id, creds),
         | 
| 501 580 | 
             
                          :aws_secret_access_key => get_credential(:secret_access_key, creds),
         | 
| 502 581 | 
             
                          :aws_region => creds.fetch(:aws_sts_region, 'us-east-1'),
         | 
| 503 | 
            -
                          :aws_credentials_file => creds.fetch( | 
| 582 | 
            +
                          :aws_credentials_file => creds.fetch(
         | 
| 583 | 
            +
                            :aws_credentials_file, aws_credentials_file
         | 
| 584 | 
            +
                          ),
         | 
| 504 585 | 
             
                          :aws_config_file => creds.fetch(:aws_config_file, aws_config_file),
         | 
| 505 586 | 
             
                          :aws_host => creds[:aws_sts_host],
         | 
| 506 587 | 
             
                          :aws_sts_token => creds[:aws_sts_session_token]
         | 
| @@ -521,7 +602,7 @@ module Miasma | |
| 521 602 | 
             
                    # @param profile [String] name of profile to load
         | 
| 522 603 | 
             
                    # @return [Smash]
         | 
| 523 604 | 
             
                    def load_aws_file(file_path, profile)
         | 
| 524 | 
            -
                      if(File. | 
| 605 | 
            +
                      if(File.exist?(file_path))
         | 
| 525 606 | 
             
                        l_config = Smash.new.tap do |creds|
         | 
| 526 607 | 
             
                          key = nil
         | 
| 527 608 | 
             
                          File.readlines(file_path).each_with_index do |line, idx|
         | 
| @@ -529,13 +610,18 @@ module Miasma | |
| 529 610 | 
             
                            next if line.empty? || line.start_with?('#')
         | 
| 530 611 | 
             
                            if(line.start_with?('['))
         | 
| 531 612 | 
             
                              unless(line.end_with?(']'))
         | 
| 532 | 
            -
                                raise ArgumentError.new( | 
| 613 | 
            +
                                raise ArgumentError.new(
         | 
| 614 | 
            +
                                  "Failed to parse aws file! (#{file_path} line #{idx + 1})"
         | 
| 615 | 
            +
                                )
         | 
| 533 616 | 
             
                              end
         | 
| 534 617 | 
             
                              key = line.tr('[]', '').strip.sub(/^profile /, '')
         | 
| 535 618 | 
             
                              creds[key] = Smash.new
         | 
| 536 619 | 
             
                            else
         | 
| 537 620 | 
             
                              unless(key)
         | 
| 538 | 
            -
                                raise ArgumentError.new( | 
| 621 | 
            +
                                raise ArgumentError.new(
         | 
| 622 | 
            +
                                  "Failed to parse aws file! (#{file_path} line #{idx + 1}) " \
         | 
| 623 | 
            +
                                    '- No section defined!'
         | 
| 624 | 
            +
                                )
         | 
| 539 625 | 
             
                              end
         | 
| 540 626 | 
             
                              line_args = line.split('=', 2).map(&:strip)
         | 
| 541 627 | 
             
                              line_args.first.replace(
         | 
| @@ -545,14 +631,18 @@ module Miasma | |
| 545 631 | 
             
                              )
         | 
| 546 632 | 
             
                              if(line_args.last.start_with?('"'))
         | 
| 547 633 | 
             
                                unless(line_args.last.end_with?('"'))
         | 
| 548 | 
            -
                                  raise ArgumentError.new( | 
| 634 | 
            +
                                  raise ArgumentError.new(
         | 
| 635 | 
            +
                                    "Failed to parse aws file! (#{file_path} line #{idx + 1})"
         | 
| 636 | 
            +
                                  )
         | 
| 549 637 | 
             
                                end
         | 
| 550 638 | 
             
                                line_args.last.replace(line_args.last[1..-2]) # NOTE: strip quoted values
         | 
| 551 639 | 
             
                              end
         | 
| 552 640 | 
             
                              begin
         | 
| 553 641 | 
             
                                creds[key].merge!(Smash[*line_args])
         | 
| 554 642 | 
             
                              rescue => e
         | 
| 555 | 
            -
                                raise ArgumentError.new( | 
| 643 | 
            +
                                raise ArgumentError.new(
         | 
| 644 | 
            +
                                  "Failed to parse aws file! (#{file_path} line #{idx + 1})"
         | 
| 645 | 
            +
                                )
         | 
| 556 646 | 
             
                              end
         | 
| 557 647 | 
             
                            end
         | 
| 558 648 | 
             
                          end
         | 
| @@ -659,7 +749,8 @@ module Miasma | |
| 659 749 | 
             
                      dest, options = request_args
         | 
| 660 750 | 
             
                      path = URI.parse(dest).path
         | 
| 661 751 | 
             
                      options = options ? options.to_smash : Smash.new
         | 
| 662 | 
            -
                      options[:headers] = Smash[connection.default_options.headers.to_a]. | 
| 752 | 
            +
                      options[:headers] = Smash[connection.default_options.headers.to_a].
         | 
| 753 | 
            +
                        merge(options.fetch(:headers, Smash.new))
         | 
| 663 754 | 
             
                      if(self.class::API_VERSION)
         | 
| 664 755 | 
             
                        if(options[:form])
         | 
| 665 756 | 
             
                          options.set(:form, 'Version', self.class::API_VERSION)
         | 
| @@ -687,7 +778,7 @@ module Miasma | |
| 687 778 | 
             
                      end
         | 
| 688 779 | 
             
                      signature = signer.generate(http_method, path, options)
         | 
| 689 780 | 
             
                      update_request(connection, options)
         | 
| 690 | 
            -
                      options = Hash[options.map{|k,v|[k.to_sym,v]}]
         | 
| 781 | 
            +
                      options = Hash[options.map{|k, v| [k.to_sym, v] }]
         | 
| 691 782 | 
             
                      connection.auth(signature).send(http_method, dest, options)
         | 
| 692 783 | 
             
                    end
         | 
| 693 784 |  | 
| @@ -705,8 +796,12 @@ module Miasma | |
| 705 796 | 
             
                    # @return [TrueClass, FalseClass]
         | 
| 706 797 | 
             
                    # @note update check only applied if assuming role
         | 
| 707 798 | 
             
                    def sts_mfa_session_update_required?(args={})
         | 
| 708 | 
            -
                      if(args.fetch(:aws_sts_session_token_code, | 
| 709 | 
            -
                         | 
| 799 | 
            +
                      if(args.fetch(:aws_sts_session_token_code,
         | 
| 800 | 
            +
                        attributes[:aws_sts_session_token_code]))
         | 
| 801 | 
            +
                        expiry = args.fetch(
         | 
| 802 | 
            +
                          :aws_sts_session_token_expires,
         | 
| 803 | 
            +
                          attributes[:aws_sts_session_token_expires]
         | 
| 804 | 
            +
                        )
         | 
| 710 805 | 
             
                        expiry.nil? || expiry <= Time.now - 15
         | 
| 711 806 | 
             
                      else
         | 
| 712 807 | 
             
                        false
         | 
| @@ -3,12 +3,13 @@ require 'miasma' | |
| 3 3 | 
             
            module Miasma
         | 
| 4 4 | 
             
              module Models
         | 
| 5 5 | 
             
                class AutoScale
         | 
| 6 | 
            +
                  # AWS autoscaling API
         | 
| 6 7 | 
             
                  class Aws < AutoScale
         | 
| 7 8 |  | 
| 8 9 | 
             
                    # Service name of the API
         | 
| 9 | 
            -
                    API_SERVICE = 'autoscaling'
         | 
| 10 | 
            +
                    API_SERVICE = 'autoscaling'.freeze
         | 
| 10 11 | 
             
                    # Supported version of the AutoScaling API
         | 
| 11 | 
            -
                    API_VERSION = '2011-01-01'
         | 
| 12 | 
            +
                    API_VERSION = '2011-01-01'.freeze
         | 
| 12 13 |  | 
| 13 14 | 
             
                    include Contrib::AwsApiCore::ApiCommon
         | 
| 14 15 | 
             
                    include Contrib::AwsApiCore::RequestUtils
         | 
| @@ -49,7 +50,10 @@ module Miasma | |
| 49 50 | 
             
                      if(group)
         | 
| 50 51 | 
             
                        params.merge('AutoScalingGroupNames.member.1' => group.id || group.name)
         | 
| 51 52 | 
             
                      end
         | 
| 52 | 
            -
                      result = all_result_pages(nil, :body, | 
| 53 | 
            +
                      result = all_result_pages(nil, :body,
         | 
| 54 | 
            +
                        'DescribeAutoScalingGroupsResponse', 'DescribeAutoScalingGroupsResult',
         | 
| 55 | 
            +
                        'AutoScalingGroups', 'member'
         | 
| 56 | 
            +
                      ) do |options|
         | 
| 53 57 | 
             
                        request(
         | 
| 54 58 | 
             
                          :method => :post,
         | 
| 55 59 | 
             
                          :path => '/',
         | 
| @@ -65,14 +69,15 @@ module Miasma | |
| 65 69 | 
             
                          :minimum_size => grp['MinSize'],
         | 
| 66 70 | 
             
                          :maximum_size => grp['MaxSize'],
         | 
| 67 71 | 
             
                          :status => grp['Status'],
         | 
| 68 | 
            -
                          :load_balancers => [ | 
| 72 | 
            +
                          :load_balancers => [
         | 
| 73 | 
            +
                            grp.get('LoadBalancerNames', 'member')
         | 
| 74 | 
            +
                          ].flatten(1).compact.map{|i|
         | 
| 69 75 | 
             
                            Group::Balancer.new(self, :id => i, :name => i)
         | 
| 70 76 | 
             
                          }
         | 
| 71 77 | 
             
                        ).valid_state
         | 
| 72 78 | 
             
                      end
         | 
| 73 79 | 
             
                    end
         | 
| 74 80 |  | 
| 75 | 
            -
             | 
| 76 81 | 
             
                    # Return all auto scale groups
         | 
| 77 82 | 
             
                    #
         | 
| 78 83 | 
             
                    # @param options [Hash] filter
         | 
| @@ -8,9 +8,9 @@ module Miasma | |
| 8 8 | 
             
                  class Aws < Compute
         | 
| 9 9 |  | 
| 10 10 | 
             
                    # Service name of the API
         | 
| 11 | 
            -
                    API_SERVICE = 'ec2'
         | 
| 11 | 
            +
                    API_SERVICE = 'ec2'.freeze
         | 
| 12 12 | 
             
                    # Supported version of the EC2 API
         | 
| 13 | 
            -
                    API_VERSION = '2014-06-15'
         | 
| 13 | 
            +
                    API_VERSION = '2014-06-15'.freeze
         | 
| 14 14 |  | 
| 15 15 | 
             
                    include Contrib::AwsApiCore::ApiCommon
         | 
| 16 16 | 
             
                    include Contrib::AwsApiCore::RequestUtils
         | 
| @@ -23,7 +23,7 @@ module Miasma | |
| 23 23 | 
             
                      'terminated' => :terminated,
         | 
| 24 24 | 
             
                      'stopping' => :pending,
         | 
| 25 25 | 
             
                      'stopped' => :stopped
         | 
| 26 | 
            -
                    )
         | 
| 26 | 
            +
                    ).to_smash(:freeze)
         | 
| 27 27 |  | 
| 28 28 | 
             
                    # @todo catch bad lookup and clear model
         | 
| 29 29 | 
             
                    def server_reload(server)
         | 
| @@ -35,15 +35,24 @@ module Miasma | |
| 35 35 | 
             
                          'InstanceId.1' => server.id
         | 
| 36 36 | 
             
                        }
         | 
| 37 37 | 
             
                      )
         | 
| 38 | 
            -
                      srv = result.get(:body, | 
| 38 | 
            +
                      srv = result.get(:body,
         | 
| 39 | 
            +
                        'DescribeInstancesResponse', 'reservationSet',
         | 
| 40 | 
            +
                        'item', 'instancesSet', 'item'
         | 
| 41 | 
            +
                      )
         | 
| 39 42 | 
             
                      server.load_data(
         | 
| 40 43 | 
             
                        :id => srv[:instanceId],
         | 
| 41 | 
            -
                        :name => [srv.fetch(:tagSet, :item, [])].flatten.map{|tag| | 
| 44 | 
            +
                        :name => [srv.fetch(:tagSet, :item, [])].flatten.map{|tag|
         | 
| 45 | 
            +
                          tag[:value] if tag.is_a?(Hash) && tag[:key] == 'Name'
         | 
| 46 | 
            +
                        }.compact.first,
         | 
| 42 47 | 
             
                        :image_id => srv[:imageId],
         | 
| 43 48 | 
             
                        :flavor_id => srv[:instanceType],
         | 
| 44 49 | 
             
                        :state => SERVER_STATE_MAP.fetch(srv.get(:instanceState, :name), :pending),
         | 
| 45 | 
            -
                        :addresses_private => [ | 
| 46 | 
            -
             | 
| 50 | 
            +
                        :addresses_private => [
         | 
| 51 | 
            +
                          Server::Address.new(:version => 4, :address => srv[:privateIpAddress])
         | 
| 52 | 
            +
                        ],
         | 
| 53 | 
            +
                        :addresses_public => [
         | 
| 54 | 
            +
                          Server::Address.new(:version => 4, :address => srv[:ipAddress])
         | 
| 55 | 
            +
                        ],
         | 
| 47 56 | 
             
                        :status => srv.get(:instanceState, :name),
         | 
| 48 57 | 
             
                        :key_name => srv[:keyName]
         | 
| 49 58 | 
             
                      )
         | 
| @@ -80,7 +89,9 @@ module Miasma | |
| 80 89 | 
             
                            'MaxCount' => 1
         | 
| 81 90 | 
             
                          }
         | 
| 82 91 | 
             
                        )
         | 
| 83 | 
            -
                        server.id = result.get(:body, | 
| 92 | 
            +
                        server.id = result.get(:body,
         | 
| 93 | 
            +
                          'RunInstancesResponse', 'instancesSet', 'item', 'instanceId'
         | 
| 94 | 
            +
                        )
         | 
| 84 95 | 
             
                        server.valid_state
         | 
| 85 96 | 
             
                        request(
         | 
| 86 97 | 
             
                          :method => :post,
         | 
| @@ -99,7 +110,9 @@ module Miasma | |
| 99 110 |  | 
| 100 111 | 
             
                    # @todo need to add auto pagination helper (as common util)
         | 
| 101 112 | 
             
                    def server_all
         | 
| 102 | 
            -
                      results = all_result_pages(nil, :body, | 
| 113 | 
            +
                      results = all_result_pages(nil, :body,
         | 
| 114 | 
            +
                        'DescribeInstancesResponse', 'reservationSet', 'item'
         | 
| 115 | 
            +
                      ) do |options|
         | 
| 103 116 | 
             
                        request(
         | 
| 104 117 | 
             
                          :method => :post,
         | 
| 105 118 | 
             
                          :path => '/',
         | 
| @@ -108,17 +121,23 @@ module Miasma | |
| 108 121 | 
             
                          )
         | 
| 109 122 | 
             
                        )
         | 
| 110 123 | 
             
                      end
         | 
| 111 | 
            -
                      results.map do | | 
| 112 | 
            -
                        [ | 
| 124 | 
            +
                      results.map do |server|
         | 
| 125 | 
            +
                        [server[:instancesSet][:item]].flatten.compact.map do |srv|
         | 
| 113 126 | 
             
                          Server.new(
         | 
| 114 127 | 
             
                            self,
         | 
| 115 128 | 
             
                            :id => srv[:instanceId],
         | 
| 116 | 
            -
                            :name => srv.fetch(:tagSet, :item, []).map{|tag| | 
| 129 | 
            +
                            :name => srv.fetch(:tagSet, :item, []).map{|tag|
         | 
| 130 | 
            +
                              tag[:value] if tag.is_a?(Hash) && tag[:key] == 'Name'
         | 
| 131 | 
            +
                            }.compact.first,
         | 
| 117 132 | 
             
                            :image_id => srv[:imageId],
         | 
| 118 133 | 
             
                            :flavor_id => srv[:instanceType],
         | 
| 119 134 | 
             
                            :state => SERVER_STATE_MAP.fetch(srv.get(:instanceState, :name), :pending),
         | 
| 120 | 
            -
                            :addresses_private => [ | 
| 121 | 
            -
             | 
| 135 | 
            +
                            :addresses_private => [
         | 
| 136 | 
            +
                              Server::Address.new(:version => 4, :address => srv[:privateIpAddress])
         | 
| 137 | 
            +
                            ],
         | 
| 138 | 
            +
                            :addresses_public => [
         | 
| 139 | 
            +
                              Server::Address.new(:version => 4, :address => srv[:ipAddress])
         | 
| 140 | 
            +
                            ],
         | 
| 122 141 | 
             
                            :status => srv.get(:instanceState, :name),
         | 
| 123 142 | 
             
                            :key_name => srv[:keyName]
         | 
| 124 143 | 
             
                          ).valid_state
         | 
| @@ -129,5 +148,4 @@ module Miasma | |
| 129 148 | 
             
                  end
         | 
| 130 149 | 
             
                end
         | 
| 131 150 | 
             
              end
         | 
| 132 | 
            -
             | 
| 133 151 | 
             
            end
         | 
| @@ -3,15 +3,16 @@ require 'miasma' | |
| 3 3 | 
             
            module Miasma
         | 
| 4 4 | 
             
              module Models
         | 
| 5 5 | 
             
                class LoadBalancer
         | 
| 6 | 
            +
                  # AWS load balancer API
         | 
| 6 7 | 
             
                  class Aws < LoadBalancer
         | 
| 7 8 |  | 
| 8 9 | 
             
                    include Contrib::AwsApiCore::ApiCommon
         | 
| 9 10 | 
             
                    include Contrib::AwsApiCore::RequestUtils
         | 
| 10 11 |  | 
| 11 12 | 
             
                    # Service name of API
         | 
| 12 | 
            -
                    API_SERVICE = 'elasticloadbalancing'
         | 
| 13 | 
            +
                    API_SERVICE = 'elasticloadbalancing'.freeze
         | 
| 13 14 | 
             
                    # Supported version of the ELB API
         | 
| 14 | 
            -
                    API_VERSION = '2012-06-01'
         | 
| 15 | 
            +
                    API_VERSION = '2012-06-01'.freeze
         | 
| 15 16 |  | 
| 16 17 | 
             
                    # Save load balancer
         | 
| 17 18 | 
             
                    #
         | 
| @@ -23,7 +24,7 @@ module Miasma | |
| 23 24 | 
             
                          'LoadBalancerName' => balancer.name
         | 
| 24 25 | 
             
                        )
         | 
| 25 26 | 
             
                        availability_zones.each_with_index do |az, i|
         | 
| 26 | 
            -
                          params["AvailabilityZones.member.#{i+1}"] = az
         | 
| 27 | 
            +
                          params["AvailabilityZones.member.#{i + 1}"] = az
         | 
| 27 28 | 
             
                        end
         | 
| 28 29 | 
             
                        if(balancer.listeners)
         | 
| 29 30 | 
             
                          balancer.listeners.each_with_index do |listener, i|
         | 
| @@ -47,7 +48,9 @@ module Miasma | |
| 47 48 | 
             
                          )
         | 
| 48 49 | 
             
                        )
         | 
| 49 50 | 
             
                        balancer.public_addresses = [
         | 
| 50 | 
            -
                          :address => result.get(:body, | 
| 51 | 
            +
                          :address => result.get(:body,
         | 
| 52 | 
            +
                            'CreateLoadBalancerResponse', 'CreateLoadBalancerResult', 'DNSName'
         | 
| 53 | 
            +
                          )
         | 
| 51 54 | 
             
                        ]
         | 
| 52 55 | 
             
                        balancer.load_data(:id => balancer.name).valid_state
         | 
| 53 56 | 
             
                        if(balancer.health_check)
         | 
| @@ -114,9 +117,12 @@ module Miasma | |
| 114 117 | 
             
                    def load_balancer_data(balancer=nil)
         | 
| 115 118 | 
             
                      params = Smash.new('Action' => 'DescribeLoadBalancers')
         | 
| 116 119 | 
             
                      if(balancer)
         | 
| 117 | 
            -
                        params | 
| 120 | 
            +
                        params['LoadBalancerNames.member.1'] = balancer.id || balancer.name
         | 
| 118 121 | 
             
                      end
         | 
| 119 | 
            -
                      result = all_result_pages(nil, :body, | 
| 122 | 
            +
                      result = all_result_pages(nil, :body,
         | 
| 123 | 
            +
                        'DescribeLoadBalancersResponse', 'DescribeLoadBalancersResult',
         | 
| 124 | 
            +
                        'LoadBalancerDescriptions', 'member'
         | 
| 125 | 
            +
                      ) do |options|
         | 
| 120 126 | 
             
                        request(
         | 
| 121 127 | 
             
                          :method => :post,
         | 
| 122 128 | 
             
                          :path => '/',
         | 
| @@ -124,7 +130,10 @@ module Miasma | |
| 124 130 | 
             
                        )
         | 
| 125 131 | 
             
                      end
         | 
| 126 132 | 
             
                      if(balancer)
         | 
| 127 | 
            -
                        health_result = all_result_pages(nil, :body, | 
| 133 | 
            +
                        health_result = all_result_pages(nil, :body,
         | 
| 134 | 
            +
                          'DescribeInstanceHealthResponse', 'DescribeInstanceHealthResult',
         | 
| 135 | 
            +
                          'InstanceStates', 'member'
         | 
| 136 | 
            +
                        ) do |options|
         | 
| 128 137 | 
             
                          request(
         | 
| 129 138 | 
             
                            :method => :post,
         | 
| 130 139 | 
             
                            :path => '/',
         | 
| @@ -156,7 +165,13 @@ module Miasma | |
| 156 165 | 
             
                          ).merge(
         | 
| 157 166 | 
             
                            health_result.nil? ? {} : Smash.new(
         | 
| 158 167 | 
             
                              :server_states => health_result.nil? ? nil : health_result.map{|i|
         | 
| 159 | 
            -
                                Balancer::ServerState.new( | 
| 168 | 
            +
                                Balancer::ServerState.new(
         | 
| 169 | 
            +
                                  self.api_for(:compute),
         | 
| 170 | 
            +
                                  :id => i['InstanceId'],
         | 
| 171 | 
            +
                                  :status => i['State'],
         | 
| 172 | 
            +
                                  :reason => i['ReasonCode'],
         | 
| 173 | 
            +
                                  :state => i['State'] == 'InService' ? :up : :down
         | 
| 174 | 
            +
                                )
         | 
| 160 175 | 
             
                              }
         | 
| 161 176 | 
             
                            )
         | 
| 162 177 | 
             
                          )
         | 
| @@ -206,7 +221,9 @@ module Miasma | |
| 206 221 | 
             
                          :form => Smash.new(
         | 
| 207 222 | 
             
                            'Action' => 'DescribeAvailabilityZones'
         | 
| 208 223 | 
             
                          )
         | 
| 209 | 
            -
                        ).fetch(:body, | 
| 224 | 
            +
                        ).fetch(:body,
         | 
| 225 | 
            +
                          'DescribeAvailabilityZonesResponse', 'availabilityZoneInfo', 'item', []
         | 
| 226 | 
            +
                        )
         | 
| 210 227 | 
             
                        [res].flatten.compact.map do |item|
         | 
| 211 228 | 
             
                          if(item['zoneState'] == 'available')
         | 
| 212 229 | 
             
                            item['zoneName']
         | 
| @@ -3,6 +3,7 @@ require 'miasma' | |
| 3 3 | 
             
            module Miasma
         | 
| 4 4 | 
             
              module Models
         | 
| 5 5 | 
             
                class Orchestration
         | 
| 6 | 
            +
                  # AWS Orchestration API
         | 
| 6 7 | 
             
                  class Aws < Orchestration
         | 
| 7 8 |  | 
| 8 9 | 
             
                    # Extended stack model to provide AWS specific stack options
         | 
| @@ -12,20 +13,20 @@ module Miasma | |
| 12 13 | 
             
                    end
         | 
| 13 14 |  | 
| 14 15 | 
             
                    # Service name of the API
         | 
| 15 | 
            -
                    API_SERVICE = 'cloudformation'
         | 
| 16 | 
            +
                    API_SERVICE = 'cloudformation'.freeze
         | 
| 16 17 | 
             
                    # Service name of the eucalyptus API
         | 
| 17 | 
            -
                    EUCA_API_SERVICE = 'CloudFormation'
         | 
| 18 | 
            +
                    EUCA_API_SERVICE = 'CloudFormation'.freeze
         | 
| 18 19 | 
             
                    # Supported version of the AutoScaling API
         | 
| 19 | 
            -
                    API_VERSION = '2010-05-15'
         | 
| 20 | 
            +
                    API_VERSION = '2010-05-15'.freeze
         | 
| 20 21 |  | 
| 21 22 | 
             
                    # Valid stack lookup states
         | 
| 22 23 | 
             
                    STACK_STATES = [
         | 
| 23 | 
            -
                       | 
| 24 | 
            -
                       | 
| 25 | 
            -
                       | 
| 26 | 
            -
                       | 
| 27 | 
            -
                       | 
| 28 | 
            -
                    ]
         | 
| 24 | 
            +
                      'CREATE_COMPLETE', 'CREATE_FAILED', 'CREATE_IN_PROGRESS', 'DELETE_FAILED',
         | 
| 25 | 
            +
                      'DELETE_IN_PROGRESS', 'ROLLBACK_COMPLETE', 'ROLLBACK_FAILED', 'ROLLBACK_IN_PROGRESS',
         | 
| 26 | 
            +
                      'UPDATE_COMPLETE', 'UPDATE_COMPLETE_CLEANUP_IN_PROGRESS', 'UPDATE_IN_PROGRESS',
         | 
| 27 | 
            +
                      'UPDATE_ROLLBACK_COMPLETE', 'UPDATE_ROLLBACK_COMPLETE_CLEANUP_IN_PROGRESS', 'UPDATE_ROLLBACK_FAILED',
         | 
| 28 | 
            +
                      'UPDATE_ROLLBACK_IN_PROGRESS'
         | 
| 29 | 
            +
                    ].map(&:freeze).freeze
         | 
| 29 30 |  | 
| 30 31 | 
             
                    include Contrib::AwsApiCore::ApiCommon
         | 
| 31 32 | 
             
                    include Contrib::AwsApiCore::RequestUtils
         | 
| @@ -48,7 +49,7 @@ module Miasma | |
| 48 49 | 
             
                        :api => :orchestration,
         | 
| 49 50 | 
             
                        :collection => :stacks
         | 
| 50 51 | 
             
                      )
         | 
| 51 | 
            -
                    )
         | 
| 52 | 
            +
                    ).to_smash(:freeze)
         | 
| 52 53 |  | 
| 53 54 | 
             
                    # Fetch stacks or update provided stack data
         | 
| 54 55 | 
             
                    #
         | 
| @@ -62,7 +63,10 @@ module Miasma | |
| 62 63 | 
             
                      end
         | 
| 63 64 | 
             
                      if(stack)
         | 
| 64 65 | 
             
                        d_params['StackName'] = stack.id
         | 
| 65 | 
            -
                        descriptions = all_result_pages(nil, :body, | 
| 66 | 
            +
                        descriptions = all_result_pages(nil, :body,
         | 
| 67 | 
            +
                          'DescribeStacksResponse', 'DescribeStacksResult',
         | 
| 68 | 
            +
                          'Stacks', 'member'
         | 
| 69 | 
            +
                        ) do |options|
         | 
| 66 70 | 
             
                          request(
         | 
| 67 71 | 
             
                            :method => :post,
         | 
| 68 72 | 
             
                            :path => '/',
         | 
| @@ -70,7 +74,10 @@ module Miasma | |
| 70 74 | 
             
                          )
         | 
| 71 75 | 
             
                        end
         | 
| 72 76 | 
             
                      else
         | 
| 73 | 
            -
                        lists = all_result_pages(nil, :body, | 
| 77 | 
            +
                        lists = all_result_pages(nil, :body,
         | 
| 78 | 
            +
                          'ListStacksResponse', 'ListStacksResult',
         | 
| 79 | 
            +
                          'StackSummaries', 'member'
         | 
| 80 | 
            +
                        ) do |options|
         | 
| 74 81 | 
             
                          request(
         | 
| 75 82 | 
             
                            :method => :post,
         | 
| 76 83 | 
             
                            :path => '/',
         | 
| @@ -143,9 +150,18 @@ module Miasma | |
| 143 150 | 
             
                    # @return [Models::Orchestration::Stack]
         | 
| 144 151 | 
             
                    def stack_save(stack)
         | 
| 145 152 | 
             
                      params = Smash.new('StackName' => stack.name)
         | 
| 153 | 
            +
                      if(stack.dirty?(:parameters))
         | 
| 154 | 
            +
                        initial_parameters = stack.data[:parameters] || {}
         | 
| 155 | 
            +
                      else
         | 
| 156 | 
            +
                        initial_parameters = {}
         | 
| 157 | 
            +
                      end
         | 
| 146 158 | 
             
                      (stack.parameters || {}).each_with_index do |pair, idx|
         | 
| 147 159 | 
             
                        params["Parameters.member.#{idx + 1}.ParameterKey"] = pair.first
         | 
| 148 | 
            -
                         | 
| 160 | 
            +
                        if(initial_parameters[pair.first] == pair.last)
         | 
| 161 | 
            +
                          params["Parameters.member.#{idx + 1}.UsePreviousValue"] = true
         | 
| 162 | 
            +
                        else
         | 
| 163 | 
            +
                          params["Parameters.member.#{idx + 1}.ParameterValue"] = pair.last
         | 
| 164 | 
            +
                        end
         | 
| 149 165 | 
             
                      end
         | 
| 150 166 | 
             
                      (stack.capabilities || []).each_with_index do |cap, idx|
         | 
| 151 167 | 
             
                        params["Capabilities.member.#{idx + 1}"] = cap
         | 
| @@ -171,12 +187,10 @@ module Miasma | |
| 171 187 | 
             
                      end
         | 
| 172 188 | 
             
                      if(stack.template_url)
         | 
| 173 189 | 
             
                        params['TemplateURL'] = stack.template_url
         | 
| 190 | 
            +
                      elsif(!stack.dirty?(:template) && stack.persisted?)
         | 
| 191 | 
            +
                        params['UsePreviousTemplate'] = true
         | 
| 174 192 | 
             
                      else
         | 
| 175 | 
            -
                         | 
| 176 | 
            -
                          params['UsePreviousTemplate'] = true
         | 
| 177 | 
            -
                        else
         | 
| 178 | 
            -
                          params['TemplateBody'] = MultiJson.dump(stack.template)
         | 
| 179 | 
            -
                        end
         | 
| 193 | 
            +
                        params['TemplateBody'] = MultiJson.dump(stack.template)
         | 
| 180 194 | 
             
                      end
         | 
| 181 195 | 
             
                      if(stack.persisted?)
         | 
| 182 196 | 
             
                        result = request(
         | 
| @@ -316,7 +330,10 @@ module Miasma | |
| 316 330 | 
             
                    # @param stack [Models::Orchestration::Stack]
         | 
| 317 331 | 
             
                    # @return [Array<Models::Orchestration::Stack::Resource>]
         | 
| 318 332 | 
             
                    def resource_all(stack)
         | 
| 319 | 
            -
                      results = all_result_pages(nil, :body, | 
| 333 | 
            +
                      results = all_result_pages(nil, :body,
         | 
| 334 | 
            +
                        'ListStackResourcesResponse', 'ListStackResourcesResult',
         | 
| 335 | 
            +
                        'StackResourceSummaries', 'member'
         | 
| 336 | 
            +
                      ) do |options|
         | 
| 320 337 | 
             
                        request(
         | 
| 321 338 | 
             
                          :method => :post,
         | 
| 322 339 | 
             
                          :path => '/',
         | 
| @@ -353,7 +370,10 @@ module Miasma | |
| 353 370 | 
             
                          'LogicalResourceId' => resource.logical_id,
         | 
| 354 371 | 
             
                          'StackName' => resource.stack.name
         | 
| 355 372 | 
             
                        )
         | 
| 356 | 
            -
                      ).get(:body, | 
| 373 | 
            +
                      ).get(:body,
         | 
| 374 | 
            +
                        'DescribeStackResourceResponse', 'DescribeStackResourceResult',
         | 
| 375 | 
            +
                        'StackResourceDetail'
         | 
| 376 | 
            +
                      )
         | 
| 357 377 | 
             
                      resource.updated = result['LastUpdatedTimestamp']
         | 
| 358 378 | 
             
                      resource.type = result['ResourceType']
         | 
| 359 379 | 
             
                      resource.state = result['ResourceStatus'].downcase.to_sym
         | 
| @@ -368,7 +388,11 @@ module Miasma | |
| 368 388 | 
             
                    # @param stack [Models::Orchestration::Stack]
         | 
| 369 389 | 
             
                    # @return [Array<Models::Orchestration::Stack::Event>]
         | 
| 370 390 | 
             
                    def event_all(stack, evt_id=nil)
         | 
| 371 | 
            -
                       | 
| 391 | 
            +
                      evt_id = stack.custom[:last_event_token] if evt_id == true
         | 
| 392 | 
            +
                      results = all_result_pages(evt_id, :body,
         | 
| 393 | 
            +
                        'DescribeStackEventsResponse', 'DescribeStackEventsResult',
         | 
| 394 | 
            +
                        'StackEvents', 'member'
         | 
| 395 | 
            +
                      ) do |options|
         | 
| 372 396 | 
             
                        request(
         | 
| 373 397 | 
             
                          :method => :post,
         | 
| 374 398 | 
             
                          :path => '/',
         | 
| @@ -379,6 +403,7 @@ module Miasma | |
| 379 403 | 
             
                        )
         | 
| 380 404 | 
             
                      end
         | 
| 381 405 | 
             
                      events = results.map do |event|
         | 
| 406 | 
            +
                        stack.custom[:last_event_token] = event['NextToken'] if event['NextToken']
         | 
| 382 407 | 
             
                        Stack::Event.new(
         | 
| 383 408 | 
             
                          stack,
         | 
| 384 409 | 
             
                          :id => event['EventId'],
         | 
| @@ -393,8 +418,7 @@ module Miasma | |
| 393 418 | 
             
                      end
         | 
| 394 419 | 
             
                      if(evt_id)
         | 
| 395 420 | 
             
                        idx = events.index{|d| d.id == evt_id}
         | 
| 396 | 
            -
                        idx ? idx  | 
| 397 | 
            -
                        events.slice(0, idx)
         | 
| 421 | 
            +
                        idx ? events.slice(0, idx) : events
         | 
| 398 422 | 
             
                      else
         | 
| 399 423 | 
             
                        events
         | 
| 400 424 | 
             
                      end
         | 
| @@ -5,14 +5,15 @@ require 'miasma' | |
| 5 5 | 
             
            module Miasma
         | 
| 6 6 | 
             
              module Models
         | 
| 7 7 | 
             
                class Storage
         | 
| 8 | 
            +
                  # AWS storage API
         | 
| 8 9 | 
             
                  class Aws < Storage
         | 
| 9 10 |  | 
| 10 11 | 
             
                    # Service name of the API
         | 
| 11 | 
            -
                    API_SERVICE = 's3'
         | 
| 12 | 
            +
                    API_SERVICE = 's3'.freeze
         | 
| 12 13 | 
             
                    # Service name of the API in eucalyptus
         | 
| 13 | 
            -
                    EUCA_API_SERVICE = 'objectstorage'
         | 
| 14 | 
            +
                    EUCA_API_SERVICE = 'objectstorage'.freeze
         | 
| 14 15 | 
             
                    # Supported version of the Storage API
         | 
| 15 | 
            -
                    API_VERSION = '2006-03-01'
         | 
| 16 | 
            +
                    API_VERSION = '2006-03-01'.freeze
         | 
| 16 17 |  | 
| 17 18 | 
             
                    include Contrib::AwsApiCore::ApiCommon
         | 
| 18 19 | 
             
                    include Contrib::AwsApiCore::RequestUtils
         | 
| @@ -39,7 +40,9 @@ module Miasma | |
| 39 40 | 
             
                      end
         | 
| 40 41 | 
             
                      set = result.get(*result_key.slice(0, 2))
         | 
| 41 42 | 
             
                      if(set.is_a?(Hash) && set['IsTruncated'] && set['Contents'])
         | 
| 42 | 
            -
                        content_key = ( | 
| 43 | 
            +
                        content_key = (
         | 
| 44 | 
            +
                          set['Contents'].respond_to?(:last) ? set['Contents'].last : set['Contents']
         | 
| 45 | 
            +
                        )['Key']
         | 
| 43 46 | 
             
                        list += all_result_pages(content_key, *result_key, &block)
         | 
| 44 47 | 
             
                      end
         | 
| 45 48 | 
             
                      list.compact
         | 
| @@ -256,7 +259,8 @@ module Miasma | |
| 256 259 | 
             
                        unless(headers.empty?)
         | 
| 257 260 | 
             
                          args[:headers] = headers
         | 
| 258 261 | 
             
                        end
         | 
| 259 | 
            -
                        if(file.attributes[:body].respond_to?(:read) && | 
| 262 | 
            +
                        if(file.attributes[:body].respond_to?(:read) &&
         | 
| 263 | 
            +
                          file.attributes[:body].size >= Storage::MAX_BODY_SIZE_FOR_STRINGIFY)
         | 
| 260 264 | 
             
                          upload_id = request(
         | 
| 261 265 | 
             
                            args.merge(
         | 
| 262 266 | 
             
                              Smash.new(
         | 
    
        data/miasma-aws.gemspec
    CHANGED
    
    | @@ -11,12 +11,12 @@ Gem::Specification.new do |s| | |
| 11 11 | 
             
              s.license = 'Apache 2.0'
         | 
| 12 12 | 
             
              s.require_path = 'lib'
         | 
| 13 13 | 
             
              s.add_runtime_dependency 'miasma', '>= 0.3.1', '< 0.5'
         | 
| 14 | 
            -
              s.add_development_dependency 'rake'
         | 
| 15 | 
            -
              s.add_development_dependency 'rubocop'
         | 
| 14 | 
            +
              s.add_development_dependency 'rake', '~> 10.5.0'
         | 
| 15 | 
            +
              s.add_development_dependency 'rubocop', '0.37.2'
         | 
| 16 16 | 
             
              s.add_development_dependency 'pry'
         | 
| 17 17 | 
             
              s.add_development_dependency 'vcr'
         | 
| 18 18 | 
             
              s.add_development_dependency 'mocha'
         | 
| 19 | 
            -
              s.add_development_dependency 'webmock'
         | 
| 19 | 
            +
              s.add_development_dependency 'webmock', '~> 1.23.0'
         | 
| 20 20 | 
             
              s.add_development_dependency 'minitest'
         | 
| 21 21 | 
             
              s.add_development_dependency 'minitest-vcr'
         | 
| 22 22 | 
             
              s.files = Dir['lib/**/*'] + %w(miasma-aws.gemspec README.md CHANGELOG.md LICENSE)
         | 
    
        metadata
    CHANGED
    
    | @@ -1,14 +1,14 @@ | |
| 1 1 | 
             
            --- !ruby/object:Gem::Specification
         | 
| 2 2 | 
             
            name: miasma-aws
         | 
| 3 3 | 
             
            version: !ruby/object:Gem::Version
         | 
| 4 | 
            -
              version: 0.3. | 
| 4 | 
            +
              version: 0.3.6
         | 
| 5 5 | 
             
            platform: ruby
         | 
| 6 6 | 
             
            authors:
         | 
| 7 7 | 
             
            - Chris Roberts
         | 
| 8 8 | 
             
            autorequire: 
         | 
| 9 9 | 
             
            bindir: bin
         | 
| 10 10 | 
             
            cert_chain: []
         | 
| 11 | 
            -
            date: 2016- | 
| 11 | 
            +
            date: 2016-10-01 00:00:00.000000000 Z
         | 
| 12 12 | 
             
            dependencies:
         | 
| 13 13 | 
             
            - !ruby/object:Gem::Dependency
         | 
| 14 14 | 
             
              name: miasma
         | 
| @@ -34,30 +34,30 @@ dependencies: | |
| 34 34 | 
             
              name: rake
         | 
| 35 35 | 
             
              requirement: !ruby/object:Gem::Requirement
         | 
| 36 36 | 
             
                requirements:
         | 
| 37 | 
            -
                - - " | 
| 37 | 
            +
                - - "~>"
         | 
| 38 38 | 
             
                  - !ruby/object:Gem::Version
         | 
| 39 | 
            -
                    version:  | 
| 39 | 
            +
                    version: 10.5.0
         | 
| 40 40 | 
             
              type: :development
         | 
| 41 41 | 
             
              prerelease: false
         | 
| 42 42 | 
             
              version_requirements: !ruby/object:Gem::Requirement
         | 
| 43 43 | 
             
                requirements:
         | 
| 44 | 
            -
                - - " | 
| 44 | 
            +
                - - "~>"
         | 
| 45 45 | 
             
                  - !ruby/object:Gem::Version
         | 
| 46 | 
            -
                    version:  | 
| 46 | 
            +
                    version: 10.5.0
         | 
| 47 47 | 
             
            - !ruby/object:Gem::Dependency
         | 
| 48 48 | 
             
              name: rubocop
         | 
| 49 49 | 
             
              requirement: !ruby/object:Gem::Requirement
         | 
| 50 50 | 
             
                requirements:
         | 
| 51 | 
            -
                - -  | 
| 51 | 
            +
                - - '='
         | 
| 52 52 | 
             
                  - !ruby/object:Gem::Version
         | 
| 53 | 
            -
                    version:  | 
| 53 | 
            +
                    version: 0.37.2
         | 
| 54 54 | 
             
              type: :development
         | 
| 55 55 | 
             
              prerelease: false
         | 
| 56 56 | 
             
              version_requirements: !ruby/object:Gem::Requirement
         | 
| 57 57 | 
             
                requirements:
         | 
| 58 | 
            -
                - -  | 
| 58 | 
            +
                - - '='
         | 
| 59 59 | 
             
                  - !ruby/object:Gem::Version
         | 
| 60 | 
            -
                    version:  | 
| 60 | 
            +
                    version: 0.37.2
         | 
| 61 61 | 
             
            - !ruby/object:Gem::Dependency
         | 
| 62 62 | 
             
              name: pry
         | 
| 63 63 | 
             
              requirement: !ruby/object:Gem::Requirement
         | 
| @@ -104,16 +104,16 @@ dependencies: | |
| 104 104 | 
             
              name: webmock
         | 
| 105 105 | 
             
              requirement: !ruby/object:Gem::Requirement
         | 
| 106 106 | 
             
                requirements:
         | 
| 107 | 
            -
                - - " | 
| 107 | 
            +
                - - "~>"
         | 
| 108 108 | 
             
                  - !ruby/object:Gem::Version
         | 
| 109 | 
            -
                    version:  | 
| 109 | 
            +
                    version: 1.23.0
         | 
| 110 110 | 
             
              type: :development
         | 
| 111 111 | 
             
              prerelease: false
         | 
| 112 112 | 
             
              version_requirements: !ruby/object:Gem::Requirement
         | 
| 113 113 | 
             
                requirements:
         | 
| 114 | 
            -
                - - " | 
| 114 | 
            +
                - - "~>"
         | 
| 115 115 | 
             
                  - !ruby/object:Gem::Version
         | 
| 116 | 
            -
                    version:  | 
| 116 | 
            +
                    version: 1.23.0
         | 
| 117 117 | 
             
            - !ruby/object:Gem::Dependency
         | 
| 118 118 | 
             
              name: minitest
         | 
| 119 119 | 
             
              requirement: !ruby/object:Gem::Requirement
         | 
| @@ -183,9 +183,8 @@ required_rubygems_version: !ruby/object:Gem::Requirement | |
| 183 183 | 
             
                  version: '0'
         | 
| 184 184 | 
             
            requirements: []
         | 
| 185 185 | 
             
            rubyforge_project: 
         | 
| 186 | 
            -
            rubygems_version: 2. | 
| 186 | 
            +
            rubygems_version: 2.5.1
         | 
| 187 187 | 
             
            signing_key: 
         | 
| 188 188 | 
             
            specification_version: 4
         | 
| 189 189 | 
             
            summary: Smoggy AWS API
         | 
| 190 190 | 
             
            test_files: []
         | 
| 191 | 
            -
            has_rdoc: 
         |