Skip to content

Commit

Permalink
Add CORS preflight OPTIONS request
Browse files Browse the repository at this point in the history
  • Loading branch information
reweeden committed Mar 19, 2024
1 parent 10d61b2 commit fa6c9cf
Show file tree
Hide file tree
Showing 2 changed files with 112 additions and 11 deletions.
63 changes: 59 additions & 4 deletions tests_e2e/test_cors.py
Original file line number Diff line number Diff line change
Expand Up @@ -5,16 +5,71 @@ def test_cors(urls, auth_cookies):
origin_host = "https://something.asf.alaska.edu"

url = urls.join(urls.METADATA_FILE_CH)
origin_headers = {"origin": origin_host}
request_headers = {"origin": origin_host}

r = requests.get(url, cookies=auth_cookies, headers=origin_headers, allow_redirects=False)
r = requests.get(
url,
cookies=auth_cookies,
headers=request_headers,
allow_redirects=False,
)
headers = dict(r.headers)

assert headers.get("Access-Control-Allow-Origin") == origin_host
assert headers.get("Access-Control-Allow-Credentials") == "true"

headers = {"origin": "null"}
r = requests.get(url, cookies=auth_cookies, headers=headers, allow_redirects=False)

def test_cors_origin_null(urls, auth_cookies):
url = urls.join(urls.METADATA_FILE_CH)
request_headers = {"origin": "null"}
r = requests.get(
url,
cookies=auth_cookies,
headers=request_headers,
allow_redirects=False,
)
headers = dict(r.headers)

assert headers.get("Access-Control-Allow-Origin") == "null"


def test_cors_preflight_options(urls, auth_cookies):
origin_host = "https://something.asf.alaska.edu"

url = urls.join(urls.METADATA_FILE_CH)
request_headers = {
"Origin": origin_host,
"Access-Control-Request-Method": "GET"
}

r = requests.options(
url,
cookies=auth_cookies,
headers=request_headers,
allow_redirects=False,
)
headers = dict(r.headers)

assert r.status_code == 204
assert headers.get("Access-Control-Allow-Origin") == origin_host
assert "GET" in headers.get("Access-Control-Allow-Methods")


def test_cors_preflight_options_origin_null(urls, auth_cookies):
url = urls.join(urls.METADATA_FILE_CH)
request_headers = {
"Origin": "null",
"Access-Control-Request-Method": "GET"
}

r = requests.get(
url,
cookies=auth_cookies,
headers=request_headers,
allow_redirects=False,
)
headers = dict(r.headers)

assert r.status_code == 204
assert headers.get("Access-Control-Allow-Origin") == "null"
assert "GET" in headers.get("Access-Control-Allow-Methods")
60 changes: 53 additions & 7 deletions thin_egress_app/app.py
Original file line number Diff line number Diff line change
Expand Up @@ -438,14 +438,34 @@ def add_cors_headers(headers):
assert app.current_request is not None

# send CORS headers if we're configured to use them
origin_header = app.current_request.headers.get('origin')
if origin_header is not None:
origin_header = app.current_request.headers.get("origin")
if is_cors_allowed():
headers["Access-Control-Allow-Origin"] = origin_header
headers["Access-Control-Allow-Credentials"] = "true"
else:
cors_origin = os.getenv("CORS_ORIGIN")
if cors_origin and (origin_header.endswith(cors_origin) or origin_header.lower() == 'null'):
headers['Access-Control-Allow-Origin'] = origin_header
headers['Access-Control-Allow-Credentials'] = 'true'
else:
log.warning(f'Origin {origin_header} is not an approved CORS host: {cors_origin}')
log.warning(
"Origin %s is not an approved CORS host: %s",
origin_header,
cors_origin,
)


def is_cors_allowed():
assert app.current_request is not None

# send CORS headers if we're configured to use them
origin_header = app.current_request.headers.get("origin")
cors_origin = os.getenv("CORS_ORIGIN")

return (
origin_header
and cors_origin
and (
origin_header.endswith(cors_origin)
or origin_header.lower() == "null"
)
)


@with_trace()
Expand Down Expand Up @@ -849,6 +869,32 @@ def try_download_head(bucket, filename):
return make_redirect(presigned_url, {}, 303)


@app.route('/{proxy+}', methods=['OPTIONS'])
@with_trace(context={})
def dynamic_url_options():
allowed_methods = [

Check warning on line 875 in thin_egress_app/app.py

View check run for this annotation

Codecov / codecov/patch

thin_egress_app/app.py#L875

Added line #L875 was not covered by tests
"GET",
"HEAD",
"OPTIONS",
]
request_method = app.request.headers.get("Access-Control-Request-Method")

Check warning on line 880 in thin_egress_app/app.py

View check run for this annotation

Codecov / codecov/patch

thin_egress_app/app.py#L880

Added line #L880 was not covered by tests
if is_cors_allowed() and request_method in allowed_methods:
headers = {

Check warning on line 882 in thin_egress_app/app.py

View check run for this annotation

Codecov / codecov/patch

thin_egress_app/app.py#L882

Added line #L882 was not covered by tests
"Access-Control-Allow-Methods": ", ".join(allowed_methods)
}
add_cors_headers(headers)
return Response(

Check warning on line 886 in thin_egress_app/app.py

View check run for this annotation

Codecov / codecov/patch

thin_egress_app/app.py#L885-L886

Added lines #L885 - L886 were not covered by tests
body="",
headers=headers,
status_code=204,
)

return Response(

Check warning on line 892 in thin_egress_app/app.py

View check run for this annotation

Codecov / codecov/patch

thin_egress_app/app.py#L892

Added line #L892 was not covered by tests
body="Method Not Allowed",
status_code=405,
)


# Attempt to validate HEAD request
@app.route('/{proxy+}', methods=['HEAD'])
@with_trace(context={})
Expand Down

0 comments on commit fa6c9cf

Please sign in to comment.