2023-05-20 20:47:08 +00:00
|
|
|
import datetime
|
2021-08-04 16:24:26 +00:00
|
|
|
from collections import OrderedDict
|
2023-08-26 07:14:40 +00:00
|
|
|
from gzip import compress as gzip_compress
|
2023-12-14 21:35:36 +00:00
|
|
|
from typing import Any, Dict
|
2023-11-30 15:55:51 +00:00
|
|
|
from unittest import SkipTest, mock
|
2021-07-16 07:01:14 +00:00
|
|
|
|
2023-12-14 21:35:36 +00:00
|
|
|
from botocore.awsrequest import AWSPreparedRequest, HTTPHeaders
|
2023-11-30 15:55:51 +00:00
|
|
|
from freezegun import freeze_time
|
2019-05-25 17:34:47 +00:00
|
|
|
|
2022-10-10 13:05:28 +00:00
|
|
|
from moto import settings
|
2023-11-30 15:55:51 +00:00
|
|
|
from moto.core.responses import AWSServiceSpec, BaseResponse, flatten_json_request_body
|
|
|
|
from moto.s3.responses import S3Response
|
2016-09-22 03:59:19 +00:00
|
|
|
|
|
|
|
|
2023-12-14 21:35:36 +00:00
|
|
|
def test_flatten_json_request_body() -> None:
|
2017-02-24 02:37:43 +00:00
|
|
|
spec = AWSServiceSpec("data/emr/2009-03-31/service-2.json").input_spec("RunJobFlow")
|
2016-09-22 03:59:19 +00:00
|
|
|
|
2023-12-14 21:35:36 +00:00
|
|
|
body: Dict[str, Any] = {
|
2016-09-22 03:59:19 +00:00
|
|
|
"Name": "cluster",
|
|
|
|
"Instances": {
|
|
|
|
"Ec2KeyName": "ec2key",
|
|
|
|
"InstanceGroups": [
|
|
|
|
{"InstanceRole": "MASTER", "InstanceType": "m1.small"},
|
|
|
|
{"InstanceRole": "CORE", "InstanceType": "m1.medium"},
|
|
|
|
],
|
|
|
|
"Placement": {"AvailabilityZone": "us-east-1"},
|
|
|
|
},
|
|
|
|
"Steps": [
|
|
|
|
{
|
|
|
|
"HadoopJarStep": {
|
|
|
|
"Properties": [
|
|
|
|
{"Key": "k1", "Value": "v1"},
|
|
|
|
{"Key": "k2", "Value": "v2"},
|
|
|
|
],
|
|
|
|
"Args": ["arg1", "arg2"],
|
2019-10-31 15:44:26 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
],
|
2016-09-22 03:59:19 +00:00
|
|
|
"Configurations": [
|
2019-10-31 15:44:26 +00:00
|
|
|
{
|
2016-09-22 03:59:19 +00:00
|
|
|
"Classification": "class",
|
|
|
|
"Properties": {"propkey1": "propkey1", "propkey2": "propkey2"},
|
2019-10-31 15:44:26 +00:00
|
|
|
},
|
2016-09-22 03:59:19 +00:00
|
|
|
{"Classification": "anotherclass", "Properties": {"propkey3": "propkey3"}},
|
|
|
|
],
|
|
|
|
}
|
|
|
|
|
|
|
|
flat = flatten_json_request_body("", body, spec)
|
2023-07-10 21:04:31 +00:00
|
|
|
assert flat["Name"] == body["Name"]
|
|
|
|
assert flat["Instances.Ec2KeyName"] == body["Instances"]["Ec2KeyName"]
|
2016-09-22 03:59:19 +00:00
|
|
|
for idx in range(2):
|
2023-07-10 21:04:31 +00:00
|
|
|
inst = body["Instances"]["InstanceGroups"][idx]
|
|
|
|
assert (
|
|
|
|
flat[f"Instances.InstanceGroups.member.{(idx + 1)}.InstanceRole"]
|
|
|
|
== inst["InstanceRole"]
|
|
|
|
)
|
|
|
|
assert (
|
|
|
|
flat[f"Instances.InstanceGroups.member.{(idx + 1)}.InstanceType"]
|
|
|
|
== inst["InstanceType"]
|
|
|
|
)
|
|
|
|
assert (
|
|
|
|
flat["Instances.Placement.AvailabilityZone"]
|
|
|
|
== body["Instances"]["Placement"]["AvailabilityZone"]
|
2017-02-24 02:37:43 +00:00
|
|
|
)
|
2016-09-22 03:59:19 +00:00
|
|
|
|
|
|
|
for idx in range(1):
|
|
|
|
prefix = "Steps.member." + str(idx + 1) + ".HadoopJarStep"
|
|
|
|
step = body["Steps"][idx]["HadoopJarStep"]
|
|
|
|
i = 0
|
|
|
|
while prefix + ".Properties.member." + str(i + 1) + ".Key" in flat:
|
2023-07-10 21:04:31 +00:00
|
|
|
prop = step["Properties"][i]
|
|
|
|
assert flat[f"{prefix}.Properties.member.{(i + 1)}.Key"] == prop["Key"]
|
|
|
|
assert flat[f"{prefix}.Properties.member.{(i + 1)}.Value"] == prop["Value"]
|
2016-09-22 03:59:19 +00:00
|
|
|
i += 1
|
|
|
|
i = 0
|
|
|
|
while prefix + ".Args.member." + str(i + 1) in flat:
|
2023-07-10 21:04:31 +00:00
|
|
|
assert flat[f"{prefix}.Args.member.{(i + 1)}"] == step["Args"][i]
|
2016-09-22 03:59:19 +00:00
|
|
|
i += 1
|
|
|
|
|
|
|
|
for idx in range(2):
|
2023-07-10 21:04:31 +00:00
|
|
|
assert (
|
|
|
|
flat["Configurations.member." + str(idx + 1) + ".Classification"]
|
|
|
|
== body["Configurations"][idx]["Classification"]
|
2017-02-24 02:37:43 +00:00
|
|
|
)
|
2016-09-22 03:59:19 +00:00
|
|
|
|
|
|
|
props = {}
|
|
|
|
i = 1
|
|
|
|
keyfmt = "Configurations.member.{0}.Properties.entry.{1}"
|
|
|
|
key = keyfmt.format(idx + 1, i)
|
|
|
|
while key + ".key" in flat:
|
|
|
|
props[flat[key + ".key"]] = flat[key + ".value"]
|
|
|
|
i += 1
|
|
|
|
key = keyfmt.format(idx + 1, i)
|
2023-07-10 21:04:31 +00:00
|
|
|
assert props == body["Configurations"][idx]["Properties"]
|
2019-05-25 17:34:47 +00:00
|
|
|
|
|
|
|
|
2023-12-14 21:35:36 +00:00
|
|
|
def test_parse_qs_unicode_decode_error() -> None:
|
2019-05-25 17:34:47 +00:00
|
|
|
body = b'{"key": "%D0"}, "C": "#0 = :0"}'
|
2023-12-14 21:35:36 +00:00
|
|
|
headers = HTTPHeaders()
|
|
|
|
headers["foo"] = "bar"
|
|
|
|
request = AWSPreparedRequest("GET", "http://request", headers, body, False)
|
2019-05-25 17:34:47 +00:00
|
|
|
BaseResponse().setup_class(request, request.url, request.headers)
|
2021-07-16 07:01:14 +00:00
|
|
|
|
|
|
|
|
2023-12-14 21:35:36 +00:00
|
|
|
def test_get_params() -> None:
|
2021-07-16 07:01:14 +00:00
|
|
|
subject = BaseResponse()
|
|
|
|
subject.querystring = OrderedDict(
|
|
|
|
[
|
|
|
|
("Action", ["CreateRule"]),
|
|
|
|
("Version", ["2015-12-01"]),
|
|
|
|
(
|
|
|
|
"ListenerArn",
|
|
|
|
[
|
|
|
|
"arn:aws:elasticloadbalancing:us-east-1:1:listener/my-lb/50dc6c495c0c9188/80139731473870416"
|
|
|
|
],
|
|
|
|
),
|
|
|
|
("Priority", ["100"]),
|
|
|
|
("Conditions.member.1.Field", ["http-header"]),
|
|
|
|
("Conditions.member.1.HttpHeaderConfig.HttpHeaderName", ["User-Agent"]),
|
|
|
|
("Conditions.member.1.HttpHeaderConfig.Values.member.2", ["curl"]),
|
|
|
|
("Conditions.member.1.HttpHeaderConfig.Values.member.1", ["Mozilla"]),
|
|
|
|
("Actions.member.1.FixedResponseConfig.StatusCode", ["200"]),
|
|
|
|
("Actions.member.1.FixedResponseConfig.ContentType", ["text/plain"]),
|
|
|
|
("Actions.member.1.Type", ["fixed-response"]),
|
|
|
|
]
|
|
|
|
)
|
|
|
|
|
|
|
|
result = subject._get_params()
|
|
|
|
|
2023-07-10 21:04:31 +00:00
|
|
|
assert result == {
|
|
|
|
"Action": "CreateRule",
|
|
|
|
"Version": "2015-12-01",
|
|
|
|
"ListenerArn": "arn:aws:elasticloadbalancing:us-east-1:1:listener/my-lb/50dc6c495c0c9188/80139731473870416",
|
|
|
|
"Priority": "100",
|
|
|
|
"Conditions": [
|
|
|
|
{
|
|
|
|
"Field": "http-header",
|
|
|
|
"HttpHeaderConfig": {
|
|
|
|
"HttpHeaderName": "User-Agent",
|
|
|
|
"Values": ["Mozilla", "curl"],
|
|
|
|
},
|
|
|
|
}
|
|
|
|
],
|
|
|
|
"Actions": [
|
|
|
|
{
|
|
|
|
"Type": "fixed-response",
|
|
|
|
"FixedResponseConfig": {
|
|
|
|
"StatusCode": "200",
|
|
|
|
"ContentType": "text/plain",
|
|
|
|
},
|
|
|
|
}
|
|
|
|
],
|
|
|
|
}
|
2021-07-30 05:19:38 +00:00
|
|
|
|
|
|
|
|
2023-12-14 21:35:36 +00:00
|
|
|
def test_get_dict_list_params() -> None:
|
2021-07-30 05:19:38 +00:00
|
|
|
subject = BaseResponse()
|
|
|
|
subject.querystring = OrderedDict(
|
|
|
|
[
|
|
|
|
("Action", ["CreateDBCluster"]),
|
|
|
|
("Version", ["2014-10-31"]),
|
|
|
|
("VpcSecurityGroupIds.VpcSecurityGroupId.1", ["sg-123"]),
|
|
|
|
("VpcSecurityGroupIds.VpcSecurityGroupId.2", ["sg-456"]),
|
|
|
|
("VpcSecurityGroupIds.VpcSecurityGroupId.3", ["sg-789"]),
|
|
|
|
]
|
|
|
|
)
|
|
|
|
|
|
|
|
# TODO: extend test and logic such that we can call subject._get_params() directly here
|
|
|
|
result = subject._get_multi_param_dict("VpcSecurityGroupIds")
|
|
|
|
|
2023-07-10 21:04:31 +00:00
|
|
|
assert result == {"VpcSecurityGroupId": ["sg-123", "sg-456", "sg-789"]}
|
2022-07-26 00:29:18 +00:00
|
|
|
|
|
|
|
|
2023-12-14 21:35:36 +00:00
|
|
|
def test_response_environment_preserved_by_type() -> None:
|
2022-07-26 00:29:18 +00:00
|
|
|
"""Ensure Jinja environment is cached by response type."""
|
|
|
|
|
|
|
|
class ResponseA(BaseResponse):
|
|
|
|
pass
|
|
|
|
|
|
|
|
class ResponseB(BaseResponse):
|
|
|
|
pass
|
|
|
|
|
|
|
|
resp_a = ResponseA()
|
|
|
|
another_resp_a = ResponseA()
|
|
|
|
resp_b = ResponseB()
|
|
|
|
|
|
|
|
assert resp_a.environment is another_resp_a.environment
|
|
|
|
assert resp_b.environment is not resp_a.environment
|
|
|
|
|
|
|
|
source_1 = "template"
|
|
|
|
source_2 = "amother template"
|
|
|
|
|
2022-07-29 17:41:14 +00:00
|
|
|
assert not resp_a.contains_template(BaseResponse._make_template_id(source_1))
|
2022-07-26 00:29:18 +00:00
|
|
|
resp_a.response_template(source_1)
|
2022-07-29 17:41:14 +00:00
|
|
|
assert resp_a.contains_template(BaseResponse._make_template_id(source_1))
|
2022-07-26 00:29:18 +00:00
|
|
|
|
2022-07-29 17:41:14 +00:00
|
|
|
assert not resp_a.contains_template(BaseResponse._make_template_id(source_2))
|
2022-07-26 00:29:18 +00:00
|
|
|
resp_a.response_template(source_2)
|
2022-07-29 17:41:14 +00:00
|
|
|
assert resp_a.contains_template(BaseResponse._make_template_id(source_2))
|
2022-07-26 00:29:18 +00:00
|
|
|
|
2022-07-29 17:41:14 +00:00
|
|
|
assert not resp_b.contains_template(BaseResponse._make_template_id(source_1))
|
|
|
|
assert not resp_b.contains_template(BaseResponse._make_template_id(source_2))
|
2022-07-26 00:29:18 +00:00
|
|
|
|
2022-07-29 17:41:14 +00:00
|
|
|
assert another_resp_a.contains_template(BaseResponse._make_template_id(source_1))
|
|
|
|
assert another_resp_a.contains_template(BaseResponse._make_template_id(source_2))
|
2022-07-26 00:29:18 +00:00
|
|
|
|
|
|
|
resp_a_new_instance = ResponseA()
|
2022-07-29 17:41:14 +00:00
|
|
|
assert resp_a_new_instance.contains_template(
|
|
|
|
BaseResponse._make_template_id(source_1)
|
|
|
|
)
|
|
|
|
assert resp_a_new_instance.contains_template(
|
|
|
|
BaseResponse._make_template_id(source_2)
|
|
|
|
)
|
2022-10-10 13:05:28 +00:00
|
|
|
|
|
|
|
|
|
|
|
@mock.patch(
|
|
|
|
"moto.core.responses.settings.PRETTIFY_RESPONSES",
|
|
|
|
new_callable=mock.PropertyMock(return_value=True),
|
|
|
|
)
|
2023-12-14 21:35:36 +00:00
|
|
|
def test_jinja_render_prettify(m_env_var: Any) -> None: # type: ignore[misc]
|
2022-10-10 13:05:28 +00:00
|
|
|
if settings.TEST_SERVER_MODE:
|
|
|
|
raise SkipTest(
|
|
|
|
"It is not possible to set the environment variable in server mode"
|
|
|
|
)
|
|
|
|
response = BaseResponse()
|
|
|
|
TEMPLATE = """<TestTemplate><ResponseText>Test text</ResponseText></TestTemplate>"""
|
|
|
|
expected_output = '<?xml version="1.0" ?>\n<TestTemplate>\n\t<ResponseText>Test text</ResponseText>\n</TestTemplate>'
|
|
|
|
template = response.response_template(TEMPLATE)
|
|
|
|
xml_string = template.render()
|
|
|
|
assert xml_string == expected_output
|
|
|
|
assert m_env_var
|
2023-05-20 20:47:08 +00:00
|
|
|
|
|
|
|
|
2023-12-14 21:35:36 +00:00
|
|
|
def test_response_metadata() -> None:
|
2023-05-20 20:47:08 +00:00
|
|
|
# Setup
|
|
|
|
frozen_time = datetime.datetime(
|
|
|
|
2023, 5, 20, 10, 20, 30, tzinfo=datetime.timezone.utc
|
|
|
|
)
|
2023-12-14 21:35:36 +00:00
|
|
|
request = AWSPreparedRequest("GET", "http://request", HTTPHeaders(), None, False)
|
2023-05-20 20:47:08 +00:00
|
|
|
|
|
|
|
# Execute
|
|
|
|
with freeze_time(frozen_time):
|
|
|
|
bc = BaseResponse()
|
|
|
|
bc.setup_class(request, request.url, request.headers)
|
|
|
|
|
|
|
|
# Verify
|
|
|
|
assert "date" in bc.response_headers
|
|
|
|
if not settings.TEST_SERVER_MODE:
|
|
|
|
assert bc.response_headers["date"] == "Sat, 20 May 2023 10:20:30 GMT"
|
2023-08-26 07:14:40 +00:00
|
|
|
|
|
|
|
|
2023-12-14 21:35:36 +00:00
|
|
|
def test_compression_gzip() -> None:
|
2023-08-26 07:14:40 +00:00
|
|
|
body = '{"key": "%D0"}, "C": "#0 = :0"}'
|
2023-12-14 21:35:36 +00:00
|
|
|
headers = HTTPHeaders()
|
|
|
|
headers["Content-Encoding"] = "gzip"
|
2023-08-26 07:14:40 +00:00
|
|
|
request = AWSPreparedRequest(
|
|
|
|
"GET",
|
|
|
|
url="http://request",
|
2023-12-14 21:35:36 +00:00
|
|
|
headers=headers,
|
2023-08-26 07:14:40 +00:00
|
|
|
body=_gzip_compress_body(body),
|
|
|
|
stream_output=False,
|
|
|
|
)
|
|
|
|
response = BaseResponse()
|
|
|
|
response.setup_class(request, request.url, request.headers)
|
|
|
|
|
|
|
|
assert body == response.body
|
|
|
|
|
|
|
|
|
2023-12-14 21:35:36 +00:00
|
|
|
def test_compression_gzip_in_s3() -> None:
|
2023-08-26 07:14:40 +00:00
|
|
|
body = b"some random data"
|
2023-12-14 21:35:36 +00:00
|
|
|
headers = HTTPHeaders()
|
|
|
|
headers["Content-Encoding"] = "gzip"
|
2023-08-26 07:14:40 +00:00
|
|
|
request = AWSPreparedRequest(
|
|
|
|
"GET",
|
|
|
|
url="http://request",
|
2023-12-14 21:35:36 +00:00
|
|
|
headers=headers,
|
2023-08-26 07:14:40 +00:00
|
|
|
body=body,
|
|
|
|
stream_output=False,
|
|
|
|
)
|
|
|
|
response = S3Response()
|
|
|
|
response.setup_class(request, request.url, request.headers)
|
|
|
|
|
|
|
|
assert body == response.body.encode("utf-8")
|
|
|
|
|
|
|
|
|
2023-12-14 21:35:36 +00:00
|
|
|
def _gzip_compress_body(body: str) -> bytes:
|
2023-08-26 07:14:40 +00:00
|
|
|
assert isinstance(body, str)
|
|
|
|
return gzip_compress(data=body.encode("utf-8"))
|