2023-08-07 16:48:48 +00:00
|
|
|
"""Test different server responses."""
|
2021-02-10 09:06:03 +00:00
|
|
|
import io
|
2023-08-07 16:48:48 +00:00
|
|
|
from unittest.mock import patch
|
2023-11-30 15:55:51 +00:00
|
|
|
from urllib.parse import parse_qs, urlparse
|
2013-03-05 13:14:43 +00:00
|
|
|
|
2023-08-07 16:48:48 +00:00
|
|
|
import pytest
|
|
|
|
import requests
|
2023-11-30 15:55:51 +00:00
|
|
|
import xmltodict
|
|
|
|
from flask.testing import FlaskClient
|
2023-08-07 16:48:48 +00:00
|
|
|
|
2013-03-05 13:14:43 +00:00
|
|
|
import moto.server as server
|
2023-03-03 22:40:55 +00:00
|
|
|
from moto.moto_server.threaded_moto_server import ThreadedMotoServer
|
2013-03-05 13:14:43 +00:00
|
|
|
|
|
|
|
|
2017-09-16 13:38:40 +00:00
|
|
|
class AuthenticatedClient(FlaskClient):
|
|
|
|
def open(self, *args, **kwargs):
|
|
|
|
kwargs["headers"] = kwargs.get("headers", {})
|
|
|
|
kwargs["headers"]["Authorization"] = "Any authorization header"
|
2019-03-25 20:19:01 +00:00
|
|
|
kwargs["content_length"] = 0 # Fixes content-length complaints.
|
2021-12-01 23:06:58 +00:00
|
|
|
return super().open(*args, **kwargs)
|
2017-09-16 13:38:40 +00:00
|
|
|
|
|
|
|
|
|
|
|
def authenticated_client():
|
2013-12-29 01:15:37 +00:00
|
|
|
backend = server.create_backend_app("s3")
|
2017-09-16 13:38:40 +00:00
|
|
|
backend.test_client_class = AuthenticatedClient
|
|
|
|
return backend.test_client()
|
2013-12-29 01:15:37 +00:00
|
|
|
|
2017-09-16 13:38:40 +00:00
|
|
|
|
|
|
|
def test_s3_server_get():
|
|
|
|
test_client = authenticated_client()
|
2013-03-05 13:14:43 +00:00
|
|
|
res = test_client.get("/")
|
|
|
|
|
2023-08-07 16:48:48 +00:00
|
|
|
assert b"ListAllMyBucketsResult" in res.data
|
2013-03-05 13:14:43 +00:00
|
|
|
|
|
|
|
|
2023-04-28 14:38:04 +00:00
|
|
|
@pytest.mark.parametrize("key_name", ["bar_baz", "bar+baz", "baz bar"])
|
2022-11-22 23:41:02 +00:00
|
|
|
def test_s3_server_bucket_create(key_name):
|
2017-09-16 13:38:40 +00:00
|
|
|
test_client = authenticated_client()
|
2013-12-29 01:15:37 +00:00
|
|
|
|
2014-03-17 01:25:14 +00:00
|
|
|
res = test_client.put("/", "http://foobaz.localhost:5000/")
|
2023-08-07 16:48:48 +00:00
|
|
|
assert res.status_code == 200
|
2013-03-05 13:14:43 +00:00
|
|
|
|
|
|
|
res = test_client.get("/")
|
2023-08-07 16:48:48 +00:00
|
|
|
assert b"<Name>foobaz</Name>" in res.data
|
2013-03-05 13:14:43 +00:00
|
|
|
|
2014-03-17 01:25:14 +00:00
|
|
|
res = test_client.get("/", "http://foobaz.localhost:5000/")
|
2023-08-07 16:48:48 +00:00
|
|
|
assert res.status_code == 200
|
|
|
|
assert b"ListBucketResult" in res.data
|
2013-03-05 13:14:43 +00:00
|
|
|
|
2022-11-22 23:41:02 +00:00
|
|
|
res = test_client.put(
|
|
|
|
f"/{key_name}", "http://foobaz.localhost:5000/", data="test value"
|
|
|
|
)
|
2023-08-07 16:48:48 +00:00
|
|
|
assert res.status_code == 200
|
2022-11-22 23:41:02 +00:00
|
|
|
assert "ETag" in dict(res.headers)
|
2013-03-05 13:14:43 +00:00
|
|
|
|
2023-04-01 20:20:29 +00:00
|
|
|
# ListBuckets
|
2022-11-22 23:41:02 +00:00
|
|
|
res = test_client.get(
|
|
|
|
"/", "http://foobaz.localhost:5000/", query_string={"prefix": key_name}
|
|
|
|
)
|
2023-08-07 16:48:48 +00:00
|
|
|
assert res.status_code == 200
|
2022-11-22 23:41:02 +00:00
|
|
|
content = xmltodict.parse(res.data)["ListBucketResult"]["Contents"]
|
|
|
|
# If we receive a dict, we only received one result
|
|
|
|
# If content is of type list, our call returned multiple results - which is not correct
|
2023-08-07 16:48:48 +00:00
|
|
|
assert isinstance(content, dict)
|
|
|
|
assert content["Key"] == key_name
|
2022-05-05 11:06:31 +00:00
|
|
|
|
2023-04-01 20:20:29 +00:00
|
|
|
# GetBucket
|
2023-03-23 10:13:51 +00:00
|
|
|
res = test_client.head("http://foobaz.localhost:5000")
|
|
|
|
assert res.status_code == 200
|
|
|
|
assert res.headers.get("x-amz-bucket-region") == "us-east-1"
|
|
|
|
|
2023-04-01 20:20:29 +00:00
|
|
|
# HeadObject
|
|
|
|
res = test_client.head(f"/{key_name}", "http://foobaz.localhost:5000/")
|
2023-08-07 16:48:48 +00:00
|
|
|
assert res.status_code == 200
|
2023-05-15 22:45:34 +00:00
|
|
|
assert res.headers.get("Accept-Ranges") == "bytes"
|
2023-04-01 20:20:29 +00:00
|
|
|
|
|
|
|
# GetObject
|
2022-11-22 23:41:02 +00:00
|
|
|
res = test_client.get(f"/{key_name}", "http://foobaz.localhost:5000/")
|
2023-08-07 16:48:48 +00:00
|
|
|
assert res.status_code == 200
|
|
|
|
assert res.data == b"test value"
|
2023-05-15 22:45:34 +00:00
|
|
|
assert res.headers.get("Accept-Ranges") == "bytes"
|
2013-05-17 23:41:39 +00:00
|
|
|
|
|
|
|
|
2021-03-26 16:51:19 +00:00
|
|
|
def test_s3_server_ignore_subdomain_for_bucketnames():
|
2021-11-17 21:02:14 +00:00
|
|
|
with patch("moto.settings.S3_IGNORE_SUBDOMAIN_BUCKETNAME", True):
|
2021-03-26 16:51:19 +00:00
|
|
|
test_client = authenticated_client()
|
|
|
|
|
|
|
|
res = test_client.put("/mybucket", "http://foobaz.localhost:5000/")
|
2023-08-07 16:48:48 +00:00
|
|
|
assert res.status_code == 200
|
|
|
|
assert b"mybucket" in res.data
|
2021-03-26 16:51:19 +00:00
|
|
|
|
|
|
|
|
2015-02-22 15:13:34 +00:00
|
|
|
def test_s3_server_bucket_versioning():
|
2017-09-16 13:38:40 +00:00
|
|
|
test_client = authenticated_client()
|
2015-02-22 15:13:34 +00:00
|
|
|
|
2022-06-09 17:40:22 +00:00
|
|
|
res = test_client.put("/", "http://foobaz.localhost:5000/")
|
2023-08-07 16:48:48 +00:00
|
|
|
assert res.status_code == 200
|
2022-06-09 17:40:22 +00:00
|
|
|
|
2015-02-22 15:13:34 +00:00
|
|
|
# Just enough XML to enable versioning
|
|
|
|
body = "<Status>Enabled</Status>"
|
2017-02-24 02:37:43 +00:00
|
|
|
res = test_client.put("/?versioning", "http://foobaz.localhost:5000", data=body)
|
2023-08-07 16:48:48 +00:00
|
|
|
assert res.status_code == 200
|
2015-02-22 15:13:34 +00:00
|
|
|
|
|
|
|
|
2013-05-17 23:41:39 +00:00
|
|
|
def test_s3_server_post_to_bucket():
|
2017-09-16 13:38:40 +00:00
|
|
|
test_client = authenticated_client()
|
2013-12-29 01:15:37 +00:00
|
|
|
|
2014-03-17 01:25:14 +00:00
|
|
|
res = test_client.put("/", "http://tester.localhost:5000/")
|
2023-08-07 16:48:48 +00:00
|
|
|
assert res.status_code == 200
|
2013-05-17 23:41:39 +00:00
|
|
|
|
2014-03-17 01:25:14 +00:00
|
|
|
test_client.post(
|
|
|
|
"/",
|
|
|
|
"https://tester.localhost:5000/",
|
2013-05-17 23:41:39 +00:00
|
|
|
data={"key": "the-key", "file": "nothing"},
|
|
|
|
)
|
|
|
|
|
2014-03-17 01:25:14 +00:00
|
|
|
res = test_client.get("/the-key", "http://tester.localhost:5000/")
|
2023-08-07 16:48:48 +00:00
|
|
|
assert res.status_code == 200
|
|
|
|
assert res.data == b"nothing"
|
2017-05-01 19:13:12 +00:00
|
|
|
|
|
|
|
|
2021-02-10 09:06:03 +00:00
|
|
|
def test_s3_server_post_to_bucket_redirect():
|
|
|
|
test_client = authenticated_client()
|
|
|
|
|
|
|
|
res = test_client.put("/", "http://tester.localhost:5000/")
|
2023-08-07 16:48:48 +00:00
|
|
|
assert res.status_code == 200
|
2021-02-10 09:06:03 +00:00
|
|
|
|
|
|
|
redirect_base = "https://redirect.com/success/"
|
|
|
|
filecontent = "nothing"
|
|
|
|
filename = "test_filename.txt"
|
|
|
|
res = test_client.post(
|
|
|
|
"/",
|
|
|
|
"https://tester.localhost:5000/",
|
|
|
|
data={
|
|
|
|
"key": "asdf/the-key/${filename}",
|
|
|
|
"file": (io.BytesIO(filecontent.encode("utf8")), filename),
|
|
|
|
"success_action_redirect": redirect_base,
|
|
|
|
},
|
|
|
|
)
|
2022-11-17 22:41:08 +00:00
|
|
|
real_key = f"asdf/the-key/{filename}"
|
2023-08-07 16:48:48 +00:00
|
|
|
assert res.status_code == 303
|
2021-02-10 09:06:03 +00:00
|
|
|
redirect = res.headers["location"]
|
|
|
|
assert redirect.startswith(redirect_base)
|
|
|
|
|
|
|
|
parts = urlparse(redirect)
|
|
|
|
args = parse_qs(parts.query)
|
|
|
|
assert args["key"][0] == real_key
|
|
|
|
assert args["bucket"][0] == "tester"
|
|
|
|
|
2022-11-17 22:41:08 +00:00
|
|
|
res = test_client.get(f"/{real_key}", "http://tester.localhost:5000/")
|
2023-08-07 16:48:48 +00:00
|
|
|
assert res.status_code == 200
|
|
|
|
assert res.data == filecontent.encode("utf8")
|
2021-02-10 09:06:03 +00:00
|
|
|
|
|
|
|
|
2017-05-01 19:13:12 +00:00
|
|
|
def test_s3_server_post_without_content_length():
|
2017-09-16 13:38:40 +00:00
|
|
|
test_client = authenticated_client()
|
2017-05-01 19:13:12 +00:00
|
|
|
|
2023-04-09 10:05:43 +00:00
|
|
|
# You can create a bucket without specifying Content-Length
|
2017-05-01 19:13:12 +00:00
|
|
|
res = test_client.put(
|
|
|
|
"/", "http://tester.localhost:5000/", environ_overrides={"CONTENT_LENGTH": ""}
|
|
|
|
)
|
2023-08-07 16:48:48 +00:00
|
|
|
assert res.status_code == 200
|
2023-04-09 10:05:43 +00:00
|
|
|
|
|
|
|
# You can specify a bucket in another region without specifying Content-Length
|
|
|
|
# (The body is just ignored..)
|
|
|
|
res = test_client.put(
|
|
|
|
"/",
|
|
|
|
"http://tester.localhost:5000/",
|
|
|
|
environ_overrides={"CONTENT_LENGTH": ""},
|
2023-08-07 16:48:48 +00:00
|
|
|
data=(
|
|
|
|
"<CreateBucketConfiguration>"
|
|
|
|
"<LocationConstraint>us-west-2</LocationConstraint>"
|
|
|
|
"</CreateBucketConfiguration>"
|
|
|
|
),
|
2023-04-09 10:05:43 +00:00
|
|
|
)
|
2023-08-07 16:48:48 +00:00
|
|
|
assert res.status_code == 200
|
2023-04-09 10:05:43 +00:00
|
|
|
|
|
|
|
# You cannot make any other bucket-related requests without specifying Content-Length
|
|
|
|
for path in ["/?versioning", "/?policy"]:
|
|
|
|
res = test_client.put(
|
|
|
|
path, "http://t.localhost:5000", environ_overrides={"CONTENT_LENGTH": ""}
|
|
|
|
)
|
2023-08-07 16:48:48 +00:00
|
|
|
assert res.status_code == 411
|
2017-05-01 19:13:12 +00:00
|
|
|
|
2023-04-09 10:05:43 +00:00
|
|
|
# You cannot make any POST-request
|
2017-05-01 19:13:12 +00:00
|
|
|
res = test_client.post(
|
|
|
|
"/", "https://tester.localhost:5000/", environ_overrides={"CONTENT_LENGTH": ""}
|
|
|
|
)
|
2023-08-07 16:48:48 +00:00
|
|
|
assert res.status_code == 411
|
2017-08-05 10:29:40 +00:00
|
|
|
|
|
|
|
|
|
|
|
def test_s3_server_post_unicode_bucket_key():
|
2023-08-07 16:48:48 +00:00
|
|
|
"""Verify non-ascii characters in request URLs (e.g., S3 object names)."""
|
2017-08-05 10:29:40 +00:00
|
|
|
dispatcher = server.DomainDispatcherApplication(server.create_backend_app)
|
|
|
|
backend_app = dispatcher.get_application(
|
|
|
|
{"HTTP_HOST": "s3.amazonaws.com", "PATH_INFO": "/test-bucket/test-object-てすと"}
|
|
|
|
)
|
|
|
|
assert backend_app
|
|
|
|
backend_app = dispatcher.get_application(
|
|
|
|
{
|
|
|
|
"HTTP_HOST": "s3.amazonaws.com",
|
|
|
|
"PATH_INFO": "/test-bucket/test-object-てすと".encode("utf-8"),
|
|
|
|
}
|
|
|
|
)
|
|
|
|
assert backend_app
|
2020-09-19 09:07:17 +00:00
|
|
|
|
|
|
|
|
|
|
|
def test_s3_server_post_cors():
|
2021-10-30 10:02:30 +00:00
|
|
|
"""Test default CORS headers set by flask-cors plugin"""
|
2020-09-19 09:07:17 +00:00
|
|
|
test_client = authenticated_client()
|
2021-10-30 10:02:30 +00:00
|
|
|
# Create the bucket
|
|
|
|
test_client.put("/", "http://tester.localhost:5000/")
|
2020-09-19 09:07:17 +00:00
|
|
|
|
|
|
|
preflight_headers = {
|
|
|
|
"Access-Control-Request-Method": "POST",
|
|
|
|
"Access-Control-Request-Headers": "origin, x-requested-with",
|
|
|
|
"Origin": "https://localhost:9000",
|
|
|
|
}
|
|
|
|
|
|
|
|
res = test_client.options(
|
|
|
|
"/", "http://tester.localhost:5000/", headers=preflight_headers
|
|
|
|
)
|
|
|
|
assert res.status_code in [200, 204]
|
|
|
|
|
|
|
|
expected_methods = set(["DELETE", "PATCH", "PUT", "GET", "HEAD", "POST", "OPTIONS"])
|
|
|
|
assert (
|
|
|
|
set(res.headers["Access-Control-Allow-Methods"].split(", ")) == expected_methods
|
|
|
|
)
|
|
|
|
|
2023-08-07 16:48:48 +00:00
|
|
|
assert res.headers["Access-Control-Allow-Origin"] == "https://localhost:9000"
|
|
|
|
assert res.headers["Access-Control-Allow-Headers"] == "origin, x-requested-with"
|
2021-10-30 10:02:30 +00:00
|
|
|
|
|
|
|
|
|
|
|
def test_s3_server_post_cors_exposed_header():
|
2023-08-07 16:48:48 +00:00
|
|
|
"""Test overriding default CORS headers with custom bucket rules"""
|
2023-01-07 11:35:14 +00:00
|
|
|
# github.com/getmoto/moto/issues/4220
|
2021-10-30 10:02:30 +00:00
|
|
|
|
|
|
|
cors_config_payload = """<CORSConfiguration xmlns="http://s3.amazonaws.com/doc/2006-03-01/">
|
|
|
|
<CORSRule>
|
|
|
|
<AllowedOrigin>https://example.org</AllowedOrigin>
|
|
|
|
<AllowedMethod>HEAD</AllowedMethod>
|
|
|
|
<AllowedMethod>GET</AllowedMethod>
|
|
|
|
<AllowedMethod>PUT</AllowedMethod>
|
|
|
|
<AllowedMethod>POST</AllowedMethod>
|
|
|
|
<AllowedMethod>DELETE</AllowedMethod>
|
|
|
|
<AllowedHeader>*</AllowedHeader>
|
|
|
|
<ExposeHeader>ETag</ExposeHeader>
|
|
|
|
<MaxAgeSeconds>3000</MaxAgeSeconds>
|
|
|
|
</CORSRule>
|
|
|
|
</CORSConfiguration>
|
|
|
|
"""
|
|
|
|
|
|
|
|
test_client = authenticated_client()
|
2023-06-07 22:28:40 +00:00
|
|
|
valid_origin = "https://example.org"
|
2021-10-30 10:02:30 +00:00
|
|
|
preflight_headers = {
|
|
|
|
"Access-Control-Request-Method": "POST",
|
|
|
|
"Access-Control-Request-Headers": "origin, x-requested-with",
|
2023-06-07 22:28:40 +00:00
|
|
|
"Origin": valid_origin,
|
2021-10-30 10:02:30 +00:00
|
|
|
}
|
|
|
|
# Returns 403 on non existing bucket
|
|
|
|
preflight_response = test_client.options(
|
|
|
|
"/", "http://testcors.localhost:5000/", headers=preflight_headers
|
|
|
|
)
|
|
|
|
assert preflight_response.status_code == 403
|
|
|
|
|
2023-06-07 22:28:40 +00:00
|
|
|
# Create the bucket & file
|
2021-10-30 10:02:30 +00:00
|
|
|
test_client.put("/", "http://testcors.localhost:5000/")
|
2023-06-07 22:28:40 +00:00
|
|
|
test_client.put("/test", "http://testcors.localhost:5000/")
|
2021-10-30 10:02:30 +00:00
|
|
|
res = test_client.put(
|
|
|
|
"/?cors", "http://testcors.localhost:5000", data=cors_config_payload
|
|
|
|
)
|
|
|
|
assert res.status_code == 200
|
|
|
|
|
|
|
|
cors_res = test_client.get("/?cors", "http://testcors.localhost:5000")
|
|
|
|
assert b"<ExposedHeader>ETag</ExposedHeader>" in cors_res.data
|
|
|
|
|
2021-11-05 13:19:06 +00:00
|
|
|
# Test OPTIONS bucket response and key response
|
|
|
|
for key_name in ("/", "/test"):
|
|
|
|
preflight_response = test_client.options(
|
|
|
|
key_name, "http://testcors.localhost:5000/", headers=preflight_headers
|
|
|
|
)
|
|
|
|
assert preflight_response.status_code == 200
|
|
|
|
expected_cors_headers = {
|
|
|
|
"Access-Control-Allow-Methods": "HEAD, GET, PUT, POST, DELETE",
|
|
|
|
"Access-Control-Allow-Origin": "https://example.org",
|
|
|
|
"Access-Control-Allow-Headers": "*",
|
|
|
|
"Access-Control-Expose-Headers": "ETag",
|
|
|
|
"Access-Control-Max-Age": "3000",
|
|
|
|
}
|
|
|
|
for header_name, header_value in expected_cors_headers.items():
|
|
|
|
assert header_name in preflight_response.headers
|
|
|
|
assert preflight_response.headers[header_name] == header_value
|
2023-03-03 22:40:55 +00:00
|
|
|
|
2023-06-07 22:28:40 +00:00
|
|
|
# Test GET key response
|
|
|
|
# A regular GET should not receive any CORS headers
|
|
|
|
resp = test_client.get("/test", "http://testcors.localhost:5000/")
|
|
|
|
assert "Access-Control-Allow-Methods" not in resp.headers
|
|
|
|
assert "Access-Control-Expose-Headers" not in resp.headers
|
|
|
|
|
|
|
|
# A GET with mismatched Origin-header should not receive any CORS headers
|
|
|
|
resp = test_client.get(
|
|
|
|
"/test", "http://testcors.localhost:5000/", headers={"Origin": "something.com"}
|
|
|
|
)
|
|
|
|
assert "Access-Control-Allow-Methods" not in resp.headers
|
|
|
|
assert "Access-Control-Expose-Headers" not in resp.headers
|
|
|
|
|
|
|
|
# Only a GET with matching Origin-header should receive CORS headers
|
|
|
|
resp = test_client.get(
|
|
|
|
"/test", "http://testcors.localhost:5000/", headers={"Origin": valid_origin}
|
|
|
|
)
|
|
|
|
assert (
|
|
|
|
resp.headers["Access-Control-Allow-Methods"] == "HEAD, GET, PUT, POST, DELETE"
|
|
|
|
)
|
|
|
|
assert resp.headers["Access-Control-Expose-Headers"] == "ETag"
|
|
|
|
|
|
|
|
# Test PUT key response
|
|
|
|
# A regular PUT should not receive any CORS headers
|
|
|
|
resp = test_client.put("/test", "http://testcors.localhost:5000/")
|
|
|
|
assert "Access-Control-Allow-Methods" not in resp.headers
|
|
|
|
assert "Access-Control-Expose-Headers" not in resp.headers
|
|
|
|
|
|
|
|
# A PUT with mismatched Origin-header should not receive any CORS headers
|
|
|
|
resp = test_client.put(
|
|
|
|
"/test", "http://testcors.localhost:5000/", headers={"Origin": "something.com"}
|
|
|
|
)
|
|
|
|
assert "Access-Control-Allow-Methods" not in resp.headers
|
|
|
|
assert "Access-Control-Expose-Headers" not in resp.headers
|
|
|
|
|
|
|
|
# Only a PUT with matching Origin-header should receive CORS headers
|
|
|
|
resp = test_client.put(
|
|
|
|
"/test", "http://testcors.localhost:5000/", headers={"Origin": valid_origin}
|
|
|
|
)
|
|
|
|
assert (
|
|
|
|
resp.headers["Access-Control-Allow-Methods"] == "HEAD, GET, PUT, POST, DELETE"
|
|
|
|
)
|
|
|
|
assert resp.headers["Access-Control-Expose-Headers"] == "ETag"
|
|
|
|
|
2023-03-03 22:40:55 +00:00
|
|
|
|
|
|
|
def test_s3_server_post_cors_multiple_origins():
|
|
|
|
"""Test that Moto only responds with the Origin that we that hosts the server"""
|
|
|
|
# github.com/getmoto/moto/issues/6003
|
|
|
|
|
|
|
|
cors_config_payload = """<CORSConfiguration xmlns="http://s3.amazonaws.com/doc/2006-03-01/">
|
|
|
|
<CORSRule>
|
|
|
|
<AllowedOrigin>https://example.org</AllowedOrigin>
|
|
|
|
<AllowedOrigin>https://localhost:6789</AllowedOrigin>
|
|
|
|
<AllowedMethod>POST</AllowedMethod>
|
|
|
|
</CORSRule>
|
|
|
|
</CORSConfiguration>
|
|
|
|
"""
|
|
|
|
|
|
|
|
thread = ThreadedMotoServer(port="6789", verbose=False)
|
|
|
|
thread.start()
|
|
|
|
|
|
|
|
# Create the bucket
|
|
|
|
requests.put("http://testcors.localhost:6789/")
|
|
|
|
requests.put("http://testcors.localhost:6789/?cors", data=cors_config_payload)
|
|
|
|
|
|
|
|
# Test only our requested origin is returned
|
|
|
|
preflight_response = requests.options(
|
2023-06-07 22:28:40 +00:00
|
|
|
"http://testcors.localhost:6789/test2",
|
2023-03-03 22:40:55 +00:00
|
|
|
headers={
|
|
|
|
"Access-Control-Request-Method": "POST",
|
|
|
|
"Origin": "https://localhost:6789",
|
|
|
|
},
|
|
|
|
)
|
|
|
|
assert preflight_response.status_code == 200
|
|
|
|
assert (
|
|
|
|
preflight_response.headers["Access-Control-Allow-Origin"]
|
|
|
|
== "https://localhost:6789"
|
|
|
|
)
|
|
|
|
assert preflight_response.content == b""
|
|
|
|
|
|
|
|
# Verify a request with unknown origin fails
|
|
|
|
preflight_response = requests.options(
|
2023-06-07 22:28:40 +00:00
|
|
|
"http://testcors.localhost:6789/test2",
|
2023-03-03 22:40:55 +00:00
|
|
|
headers={
|
|
|
|
"Access-Control-Request-Method": "POST",
|
|
|
|
"Origin": "https://unknown.host",
|
|
|
|
},
|
|
|
|
)
|
|
|
|
assert preflight_response.status_code == 403
|
|
|
|
assert b"<Code>AccessForbidden</Code>" in preflight_response.content
|
|
|
|
|
|
|
|
# Verify we can use a wildcard anywhere in the origin
|
2023-08-07 16:48:48 +00:00
|
|
|
cors_config_payload = (
|
|
|
|
'<CORSConfiguration xmlns="http://s3.amazonaws.com/doc/2006-03-01/"><CORSRule>'
|
|
|
|
"<AllowedOrigin>https://*.google.com</AllowedOrigin>"
|
|
|
|
"<AllowedMethod>POST</AllowedMethod>"
|
|
|
|
"</CORSRule></CORSConfiguration>"
|
|
|
|
)
|
2023-03-03 22:40:55 +00:00
|
|
|
requests.put("http://testcors.localhost:6789/?cors", data=cors_config_payload)
|
|
|
|
for origin in ["https://sth.google.com", "https://a.google.com"]:
|
|
|
|
preflight_response = requests.options(
|
2023-06-07 22:28:40 +00:00
|
|
|
"http://testcors.localhost:6789/test2",
|
2023-03-03 22:40:55 +00:00
|
|
|
headers={"Access-Control-Request-Method": "POST", "Origin": origin},
|
|
|
|
)
|
|
|
|
assert preflight_response.status_code == 200
|
|
|
|
assert preflight_response.headers["Access-Control-Allow-Origin"] == origin
|
|
|
|
|
|
|
|
# Non-matching requests throw an error though - it does not act as a full wildcard
|
|
|
|
preflight_response = requests.options(
|
2023-06-07 22:28:40 +00:00
|
|
|
"http://testcors.localhost:6789/test2",
|
2023-03-03 22:40:55 +00:00
|
|
|
headers={
|
|
|
|
"Access-Control-Request-Method": "POST",
|
|
|
|
"Origin": "sth.microsoft.com",
|
|
|
|
},
|
|
|
|
)
|
|
|
|
assert preflight_response.status_code == 403
|
|
|
|
assert b"<Code>AccessForbidden</Code>" in preflight_response.content
|
|
|
|
|
|
|
|
# Verify we can use a wildcard as the origin
|
2023-08-07 16:48:48 +00:00
|
|
|
cors_config_payload = (
|
|
|
|
'<CORSConfiguration xmlns="http://s3.amazonaws.com/doc/2006-03-01/"><CORSRule>'
|
|
|
|
"<AllowedOrigin>*</AllowedOrigin>"
|
|
|
|
"<AllowedMethod>POST</AllowedMethod>"
|
|
|
|
"</CORSRule></CORSConfiguration>"
|
|
|
|
)
|
2023-03-03 22:40:55 +00:00
|
|
|
requests.put("http://testcors.localhost:6789/?cors", data=cors_config_payload)
|
|
|
|
for origin in ["https://a.google.com", "http://b.microsoft.com", "any"]:
|
|
|
|
preflight_response = requests.options(
|
2023-06-07 22:28:40 +00:00
|
|
|
"http://testcors.localhost:6789/test2",
|
2023-03-03 22:40:55 +00:00
|
|
|
headers={"Access-Control-Request-Method": "POST", "Origin": origin},
|
|
|
|
)
|
|
|
|
assert preflight_response.status_code == 200
|
|
|
|
assert preflight_response.headers["Access-Control-Allow-Origin"] == origin
|
|
|
|
|
|
|
|
thread.stop()
|