aws-sdk-core 3.189.0 → 3.199.0
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 +183 -0
- data/VERSION +1 -1
- data/lib/aws-sdk-core/binary/decode_handler.rb +3 -9
- data/lib/aws-sdk-core/binary/encode_handler.rb +1 -1
- data/lib/aws-sdk-core/binary/event_builder.rb +34 -37
- data/lib/aws-sdk-core/binary/event_stream_decoder.rb +1 -0
- data/lib/aws-sdk-core/binary/event_stream_encoder.rb +4 -3
- data/lib/aws-sdk-core/cbor/cbor_engine.rb +19 -0
- data/lib/aws-sdk-core/cbor/decoder.rb +310 -0
- data/lib/aws-sdk-core/cbor/encoder.rb +243 -0
- data/lib/aws-sdk-core/cbor.rb +106 -0
- data/lib/aws-sdk-core/client_stubs.rb +18 -14
- data/lib/aws-sdk-core/credential_provider.rb +1 -1
- data/lib/aws-sdk-core/ec2_metadata.rb +1 -1
- data/lib/aws-sdk-core/ecs_credentials.rb +2 -1
- data/lib/aws-sdk-core/endpoints/matchers.rb +5 -1
- data/lib/aws-sdk-core/error_handler.rb +41 -0
- data/lib/aws-sdk-core/errors.rb +2 -2
- data/lib/aws-sdk-core/event_emitter.rb +0 -16
- data/lib/aws-sdk-core/instance_profile_credentials.rb +3 -2
- data/lib/aws-sdk-core/json/builder.rb +8 -1
- data/lib/aws-sdk-core/json/error_handler.rb +15 -10
- data/lib/aws-sdk-core/json/handler.rb +5 -6
- data/lib/aws-sdk-core/json/json_engine.rb +3 -1
- data/lib/aws-sdk-core/json/oj_engine.rb +7 -1
- data/lib/aws-sdk-core/json/parser.rb +6 -1
- data/lib/aws-sdk-core/json.rb +43 -14
- data/lib/aws-sdk-core/lru_cache.rb +75 -0
- data/lib/aws-sdk-core/pageable_response.rb +1 -1
- data/lib/aws-sdk-core/param_validator.rb +7 -2
- data/lib/aws-sdk-core/plugins/client_metrics_plugin.rb +1 -0
- data/lib/aws-sdk-core/plugins/client_metrics_send_plugin.rb +14 -2
- data/lib/aws-sdk-core/plugins/credentials_configuration.rb +2 -0
- data/lib/aws-sdk-core/plugins/global_configuration.rb +8 -9
- data/lib/aws-sdk-core/plugins/invocation_id.rb +1 -11
- data/lib/aws-sdk-core/plugins/logging.rb +2 -0
- data/lib/aws-sdk-core/plugins/protocols/api_gateway.rb +3 -1
- data/lib/aws-sdk-core/plugins/protocols/ec2.rb +2 -24
- data/lib/aws-sdk-core/plugins/protocols/json_rpc.rb +6 -8
- data/lib/aws-sdk-core/plugins/protocols/query.rb +4 -2
- data/lib/aws-sdk-core/plugins/protocols/rest_json.rb +3 -15
- data/lib/aws-sdk-core/plugins/protocols/rest_xml.rb +3 -0
- data/lib/aws-sdk-core/plugins/protocols/rpc_v2.rb +17 -0
- data/lib/aws-sdk-core/plugins/request_compression.rb +11 -2
- data/lib/aws-sdk-core/plugins/retry_errors.rb +12 -3
- data/lib/aws-sdk-core/plugins/sign.rb +8 -3
- data/lib/aws-sdk-core/plugins/stub_responses.rb +1 -0
- data/lib/aws-sdk-core/plugins/user_agent.rb +61 -26
- data/lib/aws-sdk-core/process_credentials.rb +45 -27
- data/lib/aws-sdk-core/query/ec2_handler.rb +27 -0
- data/lib/aws-sdk-core/query/ec2_param_builder.rb +5 -7
- data/lib/aws-sdk-core/query/handler.rb +4 -4
- data/lib/aws-sdk-core/query/param_builder.rb +2 -2
- data/lib/aws-sdk-core/query.rb +2 -1
- data/lib/aws-sdk-core/rest/content_type_handler.rb +60 -0
- data/lib/aws-sdk-core/rest/handler.rb +3 -4
- data/lib/aws-sdk-core/rest/request/body.rb +32 -5
- data/lib/aws-sdk-core/rest/request/endpoint.rb +24 -4
- data/lib/aws-sdk-core/rest/request/headers.rb +15 -7
- data/lib/aws-sdk-core/rest/request/querystring_builder.rb +23 -11
- data/lib/aws-sdk-core/rest/response/body.rb +15 -1
- data/lib/aws-sdk-core/rest/response/header_list_parser.rb +79 -0
- data/lib/aws-sdk-core/rest/response/headers.rb +8 -3
- data/lib/aws-sdk-core/rest.rb +1 -0
- data/lib/aws-sdk-core/rpc_v2/builder.rb +62 -0
- data/lib/aws-sdk-core/rpc_v2/content_type_handler.rb +45 -0
- data/lib/aws-sdk-core/rpc_v2/error_handler.rb +84 -0
- data/lib/aws-sdk-core/rpc_v2/handler.rb +74 -0
- data/lib/aws-sdk-core/rpc_v2/parser.rb +90 -0
- data/lib/aws-sdk-core/rpc_v2.rb +6 -0
- data/lib/aws-sdk-core/stubbing/protocols/rpc_v2.rb +41 -0
- data/lib/aws-sdk-core/util.rb +39 -0
- data/lib/aws-sdk-core/waiters/poller.rb +2 -2
- data/lib/aws-sdk-core/xml/builder.rb +17 -9
- data/lib/aws-sdk-core/xml/error_handler.rb +32 -42
- data/lib/aws-sdk-core/xml/parser/frame.rb +4 -20
- data/lib/aws-sdk-core/xml/parser/stack.rb +2 -0
- data/lib/aws-sdk-core/xml/parser.rb +2 -6
- data/lib/aws-sdk-core.rb +7 -2
- data/lib/aws-sdk-sso/client.rb +77 -49
- data/lib/aws-sdk-sso/plugins/endpoints.rb +1 -0
- data/lib/aws-sdk-sso.rb +1 -1
- data/lib/aws-sdk-ssooidc/client.rb +127 -51
- data/lib/aws-sdk-ssooidc/client_api.rb +22 -0
- data/lib/aws-sdk-ssooidc/errors.rb +21 -0
- data/lib/aws-sdk-ssooidc/plugins/endpoints.rb +1 -0
- data/lib/aws-sdk-ssooidc/types.rb +77 -9
- data/lib/aws-sdk-ssooidc.rb +1 -1
- data/lib/aws-sdk-sts/client.rb +78 -50
- data/lib/aws-sdk-sts/client_api.rb +8 -8
- data/lib/aws-sdk-sts/plugins/endpoints.rb +1 -0
- data/lib/aws-sdk-sts/types.rb +1 -1
- data/lib/aws-sdk-sts.rb +1 -1
- data/lib/seahorse/client/async_base.rb +1 -1
- data/lib/seahorse/client/async_response.rb +19 -0
- data/lib/seahorse/client/base.rb +18 -7
- data/lib/seahorse/client/h2/handler.rb +1 -0
- data/lib/seahorse/client/handler.rb +1 -1
- data/lib/seahorse/client/net_http/connection_pool.rb +3 -9
- data/lib/seahorse/client/plugin.rb +9 -0
- data/lib/seahorse/client/plugins/endpoint.rb +0 -1
- data/lib/seahorse/client/plugins/net_http.rb +48 -16
- data/lib/seahorse/model/shapes.rb +2 -2
- data/sig/aws-sdk-core/client_stubs.rbs +10 -0
- data/sig/aws-sdk-core/errors.rbs +22 -0
- data/sig/aws-sdk-core/resources/collection.rbs +21 -0
- data/sig/aws-sdk-core/structure.rbs +4 -0
- data/sig/aws-sdk-core/waiters/errors.rbs +20 -0
- data/sig/aws-sdk-core.rbs +7 -0
- data/sig/seahorse/client/base.rbs +25 -0
- data/sig/seahorse/client/handler_builder.rbs +16 -0
- data/sig/seahorse/client/response.rbs +61 -0
- metadata +38 -12
- /data/lib/aws-sdk-core/xml/parser/{engines/libxml.rb → libxml_engine.rb} +0 -0
- /data/lib/aws-sdk-core/xml/parser/{engines/nokogiri.rb → nokogiri_engine.rb} +0 -0
- /data/lib/aws-sdk-core/xml/parser/{engines/oga.rb → oga_engine.rb} +0 -0
- /data/lib/aws-sdk-core/xml/parser/{engines/ox.rb → ox_engine.rb} +0 -0
- /data/lib/aws-sdk-core/xml/parser/{engines/rexml.rb → rexml_engine.rb} +0 -0
data/lib/aws-sdk-core/query.rb
CHANGED
|
@@ -1,7 +1,8 @@
|
|
|
1
1
|
# frozen_string_literal: true
|
|
2
2
|
|
|
3
|
-
require_relative 'query/ec2_param_builder'
|
|
4
3
|
require_relative 'query/handler'
|
|
4
|
+
require_relative 'query/ec2_handler'
|
|
5
5
|
require_relative 'query/param'
|
|
6
6
|
require_relative 'query/param_builder'
|
|
7
|
+
require_relative 'query/ec2_param_builder'
|
|
7
8
|
require_relative 'query/param_list'
|
|
@@ -0,0 +1,60 @@
|
|
|
1
|
+
# frozen_string_literal: true
|
|
2
|
+
|
|
3
|
+
module Aws
|
|
4
|
+
module Rest
|
|
5
|
+
# NOTE: headers could be already populated if specified on input shape
|
|
6
|
+
class ContentTypeHandler < Seahorse::Client::Handler
|
|
7
|
+
def call(context)
|
|
8
|
+
if eventstream?(context)
|
|
9
|
+
context.http_request.headers['Content-Type'] ||=
|
|
10
|
+
'application/vnd.amazon.eventstream'
|
|
11
|
+
elsif (payload = context.operation.input[:payload_member])
|
|
12
|
+
case payload.shape
|
|
13
|
+
when Seahorse::Model::Shapes::BlobShape
|
|
14
|
+
context.http_request.headers['Content-Type'] ||=
|
|
15
|
+
'application/octet-stream'
|
|
16
|
+
when Seahorse::Model::Shapes::StringShape
|
|
17
|
+
context.http_request.headers['Content-Type'] ||=
|
|
18
|
+
'text/plain'
|
|
19
|
+
else
|
|
20
|
+
apply_default_content_type(context)
|
|
21
|
+
end
|
|
22
|
+
elsif (body = context.http_request.body) &&
|
|
23
|
+
(!body.respond_to?(:size) || non_empty_body?(body))
|
|
24
|
+
apply_default_content_type(context)
|
|
25
|
+
end
|
|
26
|
+
|
|
27
|
+
@handler.call(context)
|
|
28
|
+
end
|
|
29
|
+
|
|
30
|
+
private
|
|
31
|
+
|
|
32
|
+
def non_empty_body?(body)
|
|
33
|
+
body.respond_to?(:size) && body.size.positive?
|
|
34
|
+
end
|
|
35
|
+
|
|
36
|
+
def eventstream?(context)
|
|
37
|
+
context.operation.input.shape.members.each do |_, ref|
|
|
38
|
+
return true if ref.eventstream
|
|
39
|
+
end
|
|
40
|
+
false
|
|
41
|
+
end
|
|
42
|
+
|
|
43
|
+
# content-type defaults as noted here:
|
|
44
|
+
# rest-json: https://smithy.io/2.0/aws/protocols/aws-restxml-protocol.html#content-type
|
|
45
|
+
# rest-xml: https://smithy.io/2.0/aws/protocols/aws-restxml-protocol.html#content-type
|
|
46
|
+
def apply_default_content_type(context)
|
|
47
|
+
protocol = context.config.api.metadata['protocol']
|
|
48
|
+
case protocol
|
|
49
|
+
when 'rest-json'
|
|
50
|
+
context.http_request.headers['Content-Type'] ||=
|
|
51
|
+
'application/json'
|
|
52
|
+
when 'rest-xml'
|
|
53
|
+
context.http_request.headers['Content-Type'] ||=
|
|
54
|
+
'application/xml'
|
|
55
|
+
else raise "Unsupported protocol #{protocol}"
|
|
56
|
+
end
|
|
57
|
+
end
|
|
58
|
+
end
|
|
59
|
+
end
|
|
60
|
+
end
|
|
@@ -7,10 +7,9 @@ module Aws
|
|
|
7
7
|
|
|
8
8
|
def call(context)
|
|
9
9
|
Rest::Request::Builder.new.apply(context)
|
|
10
|
-
|
|
11
|
-
|
|
12
|
-
|
|
13
|
-
resp
|
|
10
|
+
response = @handler.call(context)
|
|
11
|
+
response.on(200..299) { |resp| Response::Parser.new.apply(resp) }
|
|
12
|
+
response.on(200..599) { |_resp| apply_request_id(context) }
|
|
14
13
|
end
|
|
15
14
|
|
|
16
15
|
private
|
|
@@ -18,10 +18,13 @@ module Aws
|
|
|
18
18
|
# @param [Hash] params
|
|
19
19
|
def apply(http_req, params)
|
|
20
20
|
body = build_body(params)
|
|
21
|
+
|
|
21
22
|
# for rest-json, ensure we send at least an empty object
|
|
22
23
|
# don't send an empty object for streaming? case.
|
|
23
|
-
if body.nil? &&
|
|
24
|
-
|
|
24
|
+
if body.nil? &&
|
|
25
|
+
json_builder? &&
|
|
26
|
+
modeled_body? &&
|
|
27
|
+
!streaming?
|
|
25
28
|
body = '{}'
|
|
26
29
|
end
|
|
27
30
|
http_req.body = body
|
|
@@ -45,13 +48,29 @@ module Aws
|
|
|
45
48
|
params[@rules[:payload]]
|
|
46
49
|
elsif @rules[:payload]
|
|
47
50
|
params = params[@rules[:payload]]
|
|
48
|
-
|
|
51
|
+
if params
|
|
52
|
+
if xml_builder? &&
|
|
53
|
+
@rules.shape.member?(@rules[:payload_member].location_name)
|
|
54
|
+
# serializing payload member name for rest-xml is as follows:
|
|
55
|
+
# 1. Use the member locationName if the member value doesn't match the member's name (default)
|
|
56
|
+
# 2. Use the value of the locationName on the member's target if present
|
|
57
|
+
# 3. Use the shape name of the member's target
|
|
58
|
+
serialize(@rules[:payload_member], params, location_name: payload_location_name)
|
|
59
|
+
else
|
|
60
|
+
serialize(@rules[:payload_member], params)
|
|
61
|
+
end
|
|
62
|
+
end
|
|
49
63
|
else
|
|
50
64
|
params = body_params(params)
|
|
51
65
|
serialize(@rules, params) unless params.empty?
|
|
52
66
|
end
|
|
53
67
|
end
|
|
54
68
|
|
|
69
|
+
def payload_location_name
|
|
70
|
+
@rules[:payload_member].shape['locationName'] ||
|
|
71
|
+
@rules[:payload_member].shape.name
|
|
72
|
+
end
|
|
73
|
+
|
|
55
74
|
def streaming?
|
|
56
75
|
@rules[:payload] && (
|
|
57
76
|
BlobShape === @rules[:payload_member].shape ||
|
|
@@ -59,8 +78,16 @@ module Aws
|
|
|
59
78
|
)
|
|
60
79
|
end
|
|
61
80
|
|
|
62
|
-
def
|
|
63
|
-
@serializer_class
|
|
81
|
+
def xml_builder?
|
|
82
|
+
@serializer_class == Xml::Builder
|
|
83
|
+
end
|
|
84
|
+
|
|
85
|
+
def json_builder?
|
|
86
|
+
@serializer_class == Json::Builder
|
|
87
|
+
end
|
|
88
|
+
|
|
89
|
+
def serialize(rules, params, location_name: nil)
|
|
90
|
+
@serializer_class.new(rules, location_name: location_name).serialize(params)
|
|
64
91
|
end
|
|
65
92
|
|
|
66
93
|
def body_params(params)
|
|
@@ -30,7 +30,9 @@ module Aws
|
|
|
30
30
|
private
|
|
31
31
|
|
|
32
32
|
def apply_path_params(uri, params)
|
|
33
|
-
path = uri.path.sub(
|
|
33
|
+
path = uri.path.sub(%r{/$}, '')
|
|
34
|
+
# handle trailing slash
|
|
35
|
+
path += @path_pattern.split('?')[0] if path.empty? || @path_pattern != '/'
|
|
34
36
|
uri.path = path.gsub(/{.+?}/) do |placeholder|
|
|
35
37
|
param_value_for_placeholder(placeholder, params)
|
|
36
38
|
end
|
|
@@ -38,22 +40,40 @@ module Aws
|
|
|
38
40
|
|
|
39
41
|
def param_value_for_placeholder(placeholder, params)
|
|
40
42
|
name = param_name(placeholder)
|
|
41
|
-
|
|
43
|
+
param_shape = @rules.shape.member(name).shape
|
|
44
|
+
value =
|
|
45
|
+
case param_shape
|
|
46
|
+
when Seahorse::Model::Shapes::TimestampShape
|
|
47
|
+
timestamp(param_shape, params[name]).to_s
|
|
48
|
+
else
|
|
49
|
+
params[name].to_s
|
|
50
|
+
end
|
|
51
|
+
|
|
42
52
|
raise ArgumentError, ":#{name} must not be blank" if value.empty?
|
|
43
53
|
|
|
44
54
|
if placeholder.include?('+')
|
|
45
|
-
value.gsub(
|
|
55
|
+
value.gsub(%r{[^/]+}) { |v| escape(v) }
|
|
46
56
|
else
|
|
47
57
|
escape(value)
|
|
48
58
|
end
|
|
49
59
|
end
|
|
50
60
|
|
|
51
61
|
def param_name(placeholder)
|
|
52
|
-
location_name = placeholder.gsub(/[{}+]/,'')
|
|
62
|
+
location_name = placeholder.gsub(/[{}+]/, '')
|
|
53
63
|
param_name, _ = @rules.shape.member_by_location_name(location_name)
|
|
54
64
|
param_name
|
|
55
65
|
end
|
|
56
66
|
|
|
67
|
+
def timestamp(ref, value)
|
|
68
|
+
case ref['timestampFormat']
|
|
69
|
+
when 'unixTimestamp' then value.to_i
|
|
70
|
+
when 'rfc822' then value.utc.httpdate
|
|
71
|
+
else
|
|
72
|
+
# serializing as RFC 3399 date-time is the default
|
|
73
|
+
value.utc.iso8601
|
|
74
|
+
end
|
|
75
|
+
end
|
|
76
|
+
|
|
57
77
|
def apply_querystring_params(uri, params)
|
|
58
78
|
# collect params that are supposed to be part of the query string
|
|
59
79
|
parts = @rules.shape.members.inject([]) do |prts, (member_name, member_ref)|
|
|
@@ -20,7 +20,8 @@ module Aws
|
|
|
20
20
|
def apply(http_req, params)
|
|
21
21
|
@rules.shape.members.each do |name, ref|
|
|
22
22
|
value = params[name]
|
|
23
|
-
next if value.nil?
|
|
23
|
+
next if value.nil? || ((ref.shape).is_a?(StringShape) && value.empty?)
|
|
24
|
+
|
|
24
25
|
case ref.location
|
|
25
26
|
when 'header' then apply_header_value(http_req.headers, ref, value)
|
|
26
27
|
when 'headers' then apply_header_map(http_req.headers, ref, value)
|
|
@@ -49,12 +50,19 @@ module Aws
|
|
|
49
50
|
end
|
|
50
51
|
end
|
|
51
52
|
|
|
52
|
-
def list(headers, ref,
|
|
53
|
-
return if !
|
|
54
|
-
|
|
55
|
-
|
|
56
|
-
|
|
57
|
-
.
|
|
53
|
+
def list(headers, ref, values)
|
|
54
|
+
return if !values || values.empty?
|
|
55
|
+
|
|
56
|
+
member_ref = ref.shape.member
|
|
57
|
+
values = values.collect do |value|
|
|
58
|
+
case member_ref.shape
|
|
59
|
+
when TimestampShape
|
|
60
|
+
timestamp(member_ref, value).to_s
|
|
61
|
+
else
|
|
62
|
+
Seahorse::Util.escape_header_list_string(value.to_s)
|
|
63
|
+
end
|
|
64
|
+
end
|
|
65
|
+
headers[ref.location_name] = values.compact.join(', ')
|
|
58
66
|
end
|
|
59
67
|
|
|
60
68
|
def apply_header_map(headers, ref, values)
|
|
@@ -30,20 +30,30 @@ module Aws
|
|
|
30
30
|
#
|
|
31
31
|
# @return [String] Returns a built querystring
|
|
32
32
|
def build(params)
|
|
33
|
+
# keys in query maps must NOT override other keys
|
|
34
|
+
query_keys = query_keys(params)
|
|
33
35
|
params.map do |(shape_ref, param_value)|
|
|
34
|
-
build_part(shape_ref, param_value)
|
|
35
|
-
end.join('&')
|
|
36
|
+
build_part(shape_ref, param_value, query_keys)
|
|
37
|
+
end.reject { |p| p.nil? || p.empty? }.join('&')
|
|
36
38
|
end
|
|
37
39
|
|
|
38
40
|
private
|
|
39
41
|
|
|
40
|
-
def
|
|
42
|
+
def query_keys(params)
|
|
43
|
+
keys = Set.new
|
|
44
|
+
params.each do |(shape_ref, _)|
|
|
45
|
+
keys << shape_ref.location_name unless shape_ref.shape.is_a?(MapShape)
|
|
46
|
+
end
|
|
47
|
+
keys
|
|
48
|
+
end
|
|
49
|
+
|
|
50
|
+
def build_part(shape_ref, param_value, query_keys)
|
|
41
51
|
case shape_ref.shape
|
|
42
52
|
# supported scalar types
|
|
43
53
|
when *SUPPORTED_TYPES
|
|
44
54
|
"#{shape_ref.location_name}=#{query_value(shape_ref, param_value)}"
|
|
45
55
|
when MapShape
|
|
46
|
-
generate_query_map(shape_ref, param_value)
|
|
56
|
+
generate_query_map(shape_ref, param_value, query_keys)
|
|
47
57
|
when ListShape
|
|
48
58
|
generate_query_list(shape_ref, param_value)
|
|
49
59
|
else
|
|
@@ -80,31 +90,33 @@ module Aws
|
|
|
80
90
|
end
|
|
81
91
|
end
|
|
82
92
|
|
|
83
|
-
def generate_query_map(ref, value)
|
|
93
|
+
def generate_query_map(ref, value, query_keys)
|
|
84
94
|
case ref.shape.value.shape
|
|
85
95
|
when StringShape
|
|
86
|
-
query_map_of_string(value)
|
|
96
|
+
query_map_of_string(value, query_keys)
|
|
87
97
|
when ListShape
|
|
88
|
-
query_map_of_string_list(value)
|
|
98
|
+
query_map_of_string_list(value, query_keys)
|
|
89
99
|
else
|
|
90
100
|
msg = 'Only map of string and string list supported'
|
|
91
101
|
raise NotImplementedError, msg
|
|
92
102
|
end
|
|
93
103
|
end
|
|
94
104
|
|
|
95
|
-
def query_map_of_string(hash)
|
|
105
|
+
def query_map_of_string(hash, query_keys)
|
|
96
106
|
list = []
|
|
97
107
|
hash.each_pair do |key, value|
|
|
98
|
-
|
|
108
|
+
key = escape(key)
|
|
109
|
+
list << "#{key}=#{escape(value)}" unless query_keys.include?(key)
|
|
99
110
|
end
|
|
100
111
|
list
|
|
101
112
|
end
|
|
102
113
|
|
|
103
|
-
def query_map_of_string_list(hash)
|
|
114
|
+
def query_map_of_string_list(hash, query_keys)
|
|
104
115
|
list = []
|
|
105
116
|
hash.each_pair do |key, values|
|
|
117
|
+
key = escape(key)
|
|
106
118
|
values.each do |value|
|
|
107
|
-
list << "#{
|
|
119
|
+
list << "#{key}=#{escape(value)}" unless query_keys.include?(key)
|
|
108
120
|
end
|
|
109
121
|
end
|
|
110
122
|
list
|
|
@@ -20,7 +20,8 @@ module Aws
|
|
|
20
20
|
if event_stream?
|
|
21
21
|
data[@rules[:payload]] = parse_eventstream(body)
|
|
22
22
|
elsif streaming?
|
|
23
|
-
|
|
23
|
+
# empty blob payloads are omitted
|
|
24
|
+
data[@rules[:payload]] = body unless empty_blob_payload?(body)
|
|
24
25
|
elsif @rules[:payload]
|
|
25
26
|
data[@rules[:payload]] = parse(body.read, @rules[:payload_member])
|
|
26
27
|
elsif !@rules.shape.member_names.empty?
|
|
@@ -30,6 +31,19 @@ module Aws
|
|
|
30
31
|
|
|
31
32
|
private
|
|
32
33
|
|
|
34
|
+
def empty_blob_payload?(body)
|
|
35
|
+
true if non_streaming_blob_payload? && empty_body?(body)
|
|
36
|
+
end
|
|
37
|
+
|
|
38
|
+
def non_streaming_blob_payload?
|
|
39
|
+
@rules[:payload_member].shape.is_a?(BlobShape) &&
|
|
40
|
+
!@rules[:payload_member]['streaming']
|
|
41
|
+
end
|
|
42
|
+
|
|
43
|
+
def empty_body?(body)
|
|
44
|
+
body.respond_to?(:size) && body.size.zero?
|
|
45
|
+
end
|
|
46
|
+
|
|
33
47
|
def event_stream?
|
|
34
48
|
@rules[:payload] && @rules[:payload_member].eventstream
|
|
35
49
|
end
|
|
@@ -0,0 +1,79 @@
|
|
|
1
|
+
# frozen_string_literal: true
|
|
2
|
+
|
|
3
|
+
require 'strscan'
|
|
4
|
+
|
|
5
|
+
module Aws
|
|
6
|
+
module Rest
|
|
7
|
+
module Response
|
|
8
|
+
# @api private
|
|
9
|
+
module HeaderListParser
|
|
10
|
+
|
|
11
|
+
class << self
|
|
12
|
+
# parse a list of possibly quoted and escaped string values
|
|
13
|
+
# Follows:
|
|
14
|
+
# # [RFC-7230's specification of header values](https://datatracker.ietf.org/doc/html/rfc7230#section-3.2.6).
|
|
15
|
+
def parse_string_list(value)
|
|
16
|
+
buffer = StringScanner.new(value)
|
|
17
|
+
parsed = []
|
|
18
|
+
|
|
19
|
+
parsed << read_value(buffer) until buffer.eos?
|
|
20
|
+
|
|
21
|
+
parsed
|
|
22
|
+
end
|
|
23
|
+
|
|
24
|
+
def parse_timestamp_list(value, ref)
|
|
25
|
+
# timestamp lists use an http-date by default and are unescaped
|
|
26
|
+
# eg: Mon, 16 Dec 2019 23:48:18 GMT, Mon, 16 Dec 2019 23:48:18 GMT
|
|
27
|
+
case ref['timestampFormat'] || ref.shape['timestampFormat']
|
|
28
|
+
when 'unixTimestamp'
|
|
29
|
+
value.split(', ').map { |v| Time.at(v.to_f) }
|
|
30
|
+
when 'iso8601' then value.split(', ').map { |v| Time.parse(v) }
|
|
31
|
+
else
|
|
32
|
+
# header default to rfc822/http-date, which has a comma after day
|
|
33
|
+
value.split(',').each_slice(2).map { |v| Time.parse(v[0] + v[1])}
|
|
34
|
+
end
|
|
35
|
+
end
|
|
36
|
+
|
|
37
|
+
private
|
|
38
|
+
|
|
39
|
+
def read_value(buffer)
|
|
40
|
+
until buffer.eos?
|
|
41
|
+
case buffer.peek(1)
|
|
42
|
+
when ' ', "\t"
|
|
43
|
+
# drop leading whitespace
|
|
44
|
+
buffer.getch
|
|
45
|
+
next
|
|
46
|
+
when '"'
|
|
47
|
+
buffer.getch # drop the quote and advance
|
|
48
|
+
return read_quoted_value(buffer)
|
|
49
|
+
else
|
|
50
|
+
return read_unquoted_value(buffer)
|
|
51
|
+
end
|
|
52
|
+
end
|
|
53
|
+
# buffer is only whitespace
|
|
54
|
+
nil
|
|
55
|
+
end
|
|
56
|
+
|
|
57
|
+
def read_unquoted_value(buffer)
|
|
58
|
+
# there cannot be any escaped values
|
|
59
|
+
value = buffer.scan_until(/,|$/)
|
|
60
|
+
# drop the comma if we matched it
|
|
61
|
+
buffer.matched == ',' ? value.chop : value
|
|
62
|
+
end
|
|
63
|
+
|
|
64
|
+
def read_quoted_value(buffer)
|
|
65
|
+
# scan until we have an unescaped double quote
|
|
66
|
+
value = buffer.scan_until(/[^\\]"/)
|
|
67
|
+
raise ArgumentError, 'Invalid String list: No closing quote found' unless value
|
|
68
|
+
|
|
69
|
+
# drop any remaining whitespace/commas
|
|
70
|
+
buffer.scan_until(/[\s,]*/)
|
|
71
|
+
# the last character will always be the closing quote.
|
|
72
|
+
# Add a starting quote and then unescape (undump)
|
|
73
|
+
"\"#{value}".undump
|
|
74
|
+
end
|
|
75
|
+
end
|
|
76
|
+
end
|
|
77
|
+
end
|
|
78
|
+
end
|
|
79
|
+
end
|
|
@@ -2,6 +2,7 @@
|
|
|
2
2
|
|
|
3
3
|
require 'time'
|
|
4
4
|
require 'base64'
|
|
5
|
+
require_relative 'header_list_parser'
|
|
5
6
|
|
|
6
7
|
module Aws
|
|
7
8
|
module Rest
|
|
@@ -36,12 +37,16 @@ module Aws
|
|
|
36
37
|
def cast_value(ref, value)
|
|
37
38
|
value = extract_json_trait(value) if ref['jsonvalue']
|
|
38
39
|
case ref.shape
|
|
39
|
-
when StringShape then value
|
|
40
|
+
when StringShape then value.to_s
|
|
40
41
|
when IntegerShape then value.to_i
|
|
41
|
-
when FloatShape then value
|
|
42
|
+
when FloatShape then Util.deserialize_number(value)
|
|
42
43
|
when BooleanShape then value == 'true'
|
|
43
44
|
when ListShape then
|
|
44
|
-
|
|
45
|
+
case ref.shape.member.shape
|
|
46
|
+
when StringShape then HeaderListParser.parse_string_list(value)
|
|
47
|
+
when TimestampShape then HeaderListParser.parse_timestamp_list(value, ref.shape.member)
|
|
48
|
+
else value.split(', ').map { |v| cast_value(ref.shape.member, v) }
|
|
49
|
+
end
|
|
45
50
|
when TimestampShape
|
|
46
51
|
if value =~ /^\d+(\.\d*)/
|
|
47
52
|
Time.at(value.to_f)
|
data/lib/aws-sdk-core/rest.rb
CHANGED
|
@@ -0,0 +1,62 @@
|
|
|
1
|
+
# frozen_string_literal: true
|
|
2
|
+
|
|
3
|
+
require 'base64'
|
|
4
|
+
|
|
5
|
+
module Aws
|
|
6
|
+
module RpcV2
|
|
7
|
+
class Builder
|
|
8
|
+
include Seahorse::Model::Shapes
|
|
9
|
+
|
|
10
|
+
def initialize(rules, _options = {})
|
|
11
|
+
@rules = rules
|
|
12
|
+
end
|
|
13
|
+
|
|
14
|
+
def serialize(params)
|
|
15
|
+
# If the input shape is empty, do not set a body. This is
|
|
16
|
+
# different than if the input shape is a structure with no members.
|
|
17
|
+
return nil if @rules.shape.struct_class == EmptyStructure
|
|
18
|
+
|
|
19
|
+
Cbor.encode(format(@rules, params))
|
|
20
|
+
end
|
|
21
|
+
|
|
22
|
+
private
|
|
23
|
+
|
|
24
|
+
def structure(ref, values)
|
|
25
|
+
shape = ref.shape
|
|
26
|
+
values.each_pair.with_object({}) do |(key, value), data|
|
|
27
|
+
if shape.member?(key) && !value.nil?
|
|
28
|
+
member_ref = shape.member(key)
|
|
29
|
+
member_name = member_ref.location_name || key
|
|
30
|
+
data[member_name] = format(member_ref, value)
|
|
31
|
+
end
|
|
32
|
+
end
|
|
33
|
+
end
|
|
34
|
+
|
|
35
|
+
def list(ref, values)
|
|
36
|
+
member_ref = ref.shape.member
|
|
37
|
+
values.collect { |value| format(member_ref, value) }
|
|
38
|
+
end
|
|
39
|
+
|
|
40
|
+
def map(ref, values)
|
|
41
|
+
value_ref = ref.shape.value
|
|
42
|
+
values.each.with_object({}) do |(key, value), data|
|
|
43
|
+
data[key] = format(value_ref, value)
|
|
44
|
+
end
|
|
45
|
+
end
|
|
46
|
+
|
|
47
|
+
def blob(value)
|
|
48
|
+
(String === value ? value : value.read).force_encoding(Encoding::BINARY)
|
|
49
|
+
end
|
|
50
|
+
|
|
51
|
+
def format(ref, value)
|
|
52
|
+
case ref.shape
|
|
53
|
+
when StructureShape then structure(ref, value)
|
|
54
|
+
when ListShape then list(ref, value)
|
|
55
|
+
when MapShape then map(ref, value)
|
|
56
|
+
when BlobShape then blob(value)
|
|
57
|
+
else value
|
|
58
|
+
end
|
|
59
|
+
end
|
|
60
|
+
end
|
|
61
|
+
end
|
|
62
|
+
end
|
|
@@ -0,0 +1,45 @@
|
|
|
1
|
+
# frozen_string_literal: true
|
|
2
|
+
|
|
3
|
+
module Aws
|
|
4
|
+
module RpcV2
|
|
5
|
+
class ContentTypeHandler < Seahorse::Client::Handler
|
|
6
|
+
def call(context)
|
|
7
|
+
content_type =
|
|
8
|
+
if eventstream_input?(context)
|
|
9
|
+
'application/vnd.amazon.eventstream'
|
|
10
|
+
elsif !empty_input_structure?(context)
|
|
11
|
+
'application/cbor'
|
|
12
|
+
end
|
|
13
|
+
accept =
|
|
14
|
+
if eventstream_output?(context)
|
|
15
|
+
'application/vnd.amazon.eventstream'
|
|
16
|
+
end
|
|
17
|
+
|
|
18
|
+
headers = context.http_request.headers
|
|
19
|
+
headers['Content-Type'] ||= content_type if content_type
|
|
20
|
+
headers['Accept'] ||= accept if accept
|
|
21
|
+
@handler.call(context)
|
|
22
|
+
end
|
|
23
|
+
|
|
24
|
+
private
|
|
25
|
+
|
|
26
|
+
def eventstream_input?(context)
|
|
27
|
+
context.operation.input.shape.members.each do |_, ref|
|
|
28
|
+
return true if ref.eventstream
|
|
29
|
+
end
|
|
30
|
+
false
|
|
31
|
+
end
|
|
32
|
+
|
|
33
|
+
def eventstream_output?(context)
|
|
34
|
+
context.operation.output.shape.members.each do |_, ref|
|
|
35
|
+
return true if ref.eventstream
|
|
36
|
+
end
|
|
37
|
+
false
|
|
38
|
+
end
|
|
39
|
+
|
|
40
|
+
def empty_input_structure?(context)
|
|
41
|
+
context.operation.input.shape.struct_class == EmptyStructure
|
|
42
|
+
end
|
|
43
|
+
end
|
|
44
|
+
end
|
|
45
|
+
end
|
|
@@ -0,0 +1,84 @@
|
|
|
1
|
+
# frozen_string_literal: true
|
|
2
|
+
|
|
3
|
+
module Aws
|
|
4
|
+
module RpcV2
|
|
5
|
+
class ErrorHandler < Aws::ErrorHandler
|
|
6
|
+
|
|
7
|
+
def call(context)
|
|
8
|
+
# Malformed responses should throw an http based error, so we check
|
|
9
|
+
# 200 range for error handling only for this case.
|
|
10
|
+
@handler.call(context).on(200..599) do |response|
|
|
11
|
+
if !valid_response?(context)
|
|
12
|
+
code, message, data = http_status_error(context)
|
|
13
|
+
response.error = build_error(context, code, message, data)
|
|
14
|
+
elsif (300..599).cover?(context.http_response.status_code)
|
|
15
|
+
response.error = error(context)
|
|
16
|
+
end
|
|
17
|
+
response.data = nil
|
|
18
|
+
end
|
|
19
|
+
end
|
|
20
|
+
|
|
21
|
+
private
|
|
22
|
+
|
|
23
|
+
def valid_response?(context)
|
|
24
|
+
req_header = context.http_request.headers['smithy-protocol']
|
|
25
|
+
resp_header = context.http_response.headers['smithy-protocol']
|
|
26
|
+
req_header == resp_header
|
|
27
|
+
end
|
|
28
|
+
|
|
29
|
+
def extract_error(body, context)
|
|
30
|
+
data = Cbor.decode(body)
|
|
31
|
+
code = error_code(data, context)
|
|
32
|
+
message = data['message']
|
|
33
|
+
data = parse_error_data(context, body, code)
|
|
34
|
+
[code, message, data]
|
|
35
|
+
rescue Cbor::Error
|
|
36
|
+
[http_status_error_code(context), '', EmptyStructure.new]
|
|
37
|
+
end
|
|
38
|
+
|
|
39
|
+
def error_code(data, context)
|
|
40
|
+
code =
|
|
41
|
+
if aws_query_error?(context)
|
|
42
|
+
error = context.http_response.headers['x-amzn-query-error'].split(';')[0]
|
|
43
|
+
remove_prefix(error, context)
|
|
44
|
+
else
|
|
45
|
+
data['__type']
|
|
46
|
+
end
|
|
47
|
+
if code
|
|
48
|
+
code.split('#').last
|
|
49
|
+
else
|
|
50
|
+
http_status_error_code(context)
|
|
51
|
+
end
|
|
52
|
+
end
|
|
53
|
+
|
|
54
|
+
def parse_error_data(context, body, code)
|
|
55
|
+
data = EmptyStructure.new
|
|
56
|
+
if (error_rules = context.operation.errors)
|
|
57
|
+
error_rules.each do |rule|
|
|
58
|
+
# match modeled shape name with the type(code) only
|
|
59
|
+
# some type(code) might contains invalid characters
|
|
60
|
+
# such as ':' (efs) etc
|
|
61
|
+
match = rule.shape.name == code.gsub(/[^^a-zA-Z0-9]/, '')
|
|
62
|
+
next unless match && rule.shape.members.any?
|
|
63
|
+
|
|
64
|
+
data = Parser.new(rule).parse(body)
|
|
65
|
+
end
|
|
66
|
+
end
|
|
67
|
+
data
|
|
68
|
+
end
|
|
69
|
+
|
|
70
|
+
def aws_query_error?(context)
|
|
71
|
+
context.config.api.metadata['awsQueryCompatible'] &&
|
|
72
|
+
context.http_response.headers['x-amzn-query-error']
|
|
73
|
+
end
|
|
74
|
+
|
|
75
|
+
def remove_prefix(error_code, context)
|
|
76
|
+
if (prefix = context.config.api.metadata['errorPrefix'])
|
|
77
|
+
error_code.sub(/^#{prefix}/, '')
|
|
78
|
+
else
|
|
79
|
+
error_code
|
|
80
|
+
end
|
|
81
|
+
end
|
|
82
|
+
end
|
|
83
|
+
end
|
|
84
|
+
end
|