mirror of
https://github.com/GSA/notifications-api.git
synced 2026-05-05 16:48:31 -04:00
Made changes for zap scans
This commit is contained in:
@@ -301,15 +301,44 @@ def init_app(app):
|
|||||||
g.start = monotonic()
|
g.start = monotonic()
|
||||||
g.endpoint = request.endpoint
|
g.endpoint = request.endpoint
|
||||||
|
|
||||||
|
@app.before_request
|
||||||
|
def handle_options():
|
||||||
|
if request.method == "OPTIONS":
|
||||||
|
response = make_response("", 204)
|
||||||
|
response.headers["Access-Control-Allow-Origin"] = "*"
|
||||||
|
response.headers["Access-Control-Allow-Methods"] = (
|
||||||
|
"GET, POST, PUT, DELETE, OPTIONS"
|
||||||
|
)
|
||||||
|
response.headers["Access-Control-Allow-Headers"] = (
|
||||||
|
"Content-Type, Authorization"
|
||||||
|
)
|
||||||
|
response.headers["Access-Control-Max-Age"] = "3600"
|
||||||
|
return response
|
||||||
|
|
||||||
@app.after_request
|
@app.after_request
|
||||||
def after_request(response):
|
def after_request(response):
|
||||||
|
# Security headers for government compliance
|
||||||
response.headers.add("X-Content-Type-Options", "nosniff")
|
response.headers.add("X-Content-Type-Options", "nosniff")
|
||||||
|
response.headers.add("X-Frame-Options", "DENY")
|
||||||
|
response.headers.add("X-XSS-Protection", "1; mode=block")
|
||||||
|
response.headers.add("Referrer-Policy", "strict-origin-when-cross-origin")
|
||||||
|
response.headers.add(
|
||||||
|
"Permissions-Policy", "geolocation=(), microphone=(), camera=()"
|
||||||
|
)
|
||||||
|
|
||||||
# Some dynamic scan findings
|
# CORS-related security headers
|
||||||
response.headers.add("Cross-Origin-Opener-Policy", "same-origin")
|
response.headers.add("Cross-Origin-Opener-Policy", "same-origin")
|
||||||
response.headers.add("Cross-Origin-Embedder-Policy", "require-corp")
|
response.headers.add("Cross-Origin-Embedder-Policy", "require-corp")
|
||||||
response.headers.add("Cross-Origin-Resource-Policy", "same-origin")
|
response.headers.add("Cross-Origin-Resource-Policy", "same-origin")
|
||||||
response.headers.add("Cross-Origin-Opener-Policy", "same-origin")
|
|
||||||
|
if not request.path.startswith("/docs"):
|
||||||
|
response.headers.add(
|
||||||
|
"Content-Security-Policy", "default-src 'none'; frame-ancestors 'none';"
|
||||||
|
)
|
||||||
|
|
||||||
|
response.headers.add(
|
||||||
|
"Strict-Transport-Security", "max-age=31536000; includeSubDomains"
|
||||||
|
)
|
||||||
|
|
||||||
return response
|
return response
|
||||||
|
|
||||||
|
|||||||
@@ -14,17 +14,20 @@ status = Blueprint("status", __name__)
|
|||||||
def show_status():
|
def show_status():
|
||||||
try:
|
try:
|
||||||
if request.args.get("simple", None):
|
if request.args.get("simple", None):
|
||||||
return jsonify(status="ok"), 200
|
response = jsonify(status="ok")
|
||||||
else:
|
else:
|
||||||
return (
|
response = jsonify(
|
||||||
jsonify(
|
status="ok", # This should be considered part of the public API
|
||||||
status="ok", # This should be considered part of the public API
|
git_commit=version.__git_commit__,
|
||||||
git_commit=version.__git_commit__,
|
build_time=version.__time__,
|
||||||
build_time=version.__time__,
|
db_version=get_db_version(),
|
||||||
db_version=get_db_version(),
|
|
||||||
),
|
|
||||||
200,
|
|
||||||
)
|
)
|
||||||
|
|
||||||
|
response.headers["Cache-Control"] = "no-cache, no-store, must-revalidate"
|
||||||
|
response.headers["Pragma"] = "no-cache"
|
||||||
|
response.headers["Expires"] = "0"
|
||||||
|
|
||||||
|
return response, 200
|
||||||
except Exception as e:
|
except Exception as e:
|
||||||
current_app.logger.error(
|
current_app.logger.error(
|
||||||
f"Unexpected error in show_status: {str(e)}", exc_info=True
|
f"Unexpected error in show_status: {str(e)}", exc_info=True
|
||||||
@@ -36,13 +39,16 @@ def show_status():
|
|||||||
@status.route("/_status/live-service-and-organization-counts")
|
@status.route("/_status/live-service-and-organization-counts")
|
||||||
def live_service_and_organization_counts():
|
def live_service_and_organization_counts():
|
||||||
try:
|
try:
|
||||||
return (
|
response = jsonify(
|
||||||
jsonify(
|
organizations=dao_count_organizations_with_live_services(),
|
||||||
organizations=dao_count_organizations_with_live_services(),
|
services=dao_count_live_services(),
|
||||||
services=dao_count_live_services(),
|
|
||||||
),
|
|
||||||
200,
|
|
||||||
)
|
)
|
||||||
|
|
||||||
|
response.headers["Cache-Control"] = "no-cache, no-store, must-revalidate"
|
||||||
|
response.headers["Pragma"] = "no-cache"
|
||||||
|
response.headers["Expires"] = "0"
|
||||||
|
|
||||||
|
return response, 200
|
||||||
except Exception as e:
|
except Exception as e:
|
||||||
current_app.logger.error(
|
current_app.logger.error(
|
||||||
f"Unexpected error in live_service_and_organization_counts: {str(e)}",
|
f"Unexpected error in live_service_and_organization_counts: {str(e)}",
|
||||||
|
|||||||
@@ -23,9 +23,7 @@ class BaseAPIClient:
|
|||||||
This class is not thread-safe.
|
This class is not thread-safe.
|
||||||
"""
|
"""
|
||||||
|
|
||||||
def __init__(
|
def __init__(self, api_key, base_url=API_HOST_NAME, timeout=30):
|
||||||
self, api_key, base_url=API_HOST_NAME, timeout=30
|
|
||||||
):
|
|
||||||
"""
|
"""
|
||||||
Initialise the client
|
Initialise the client
|
||||||
Error if either of base_url or secret missing
|
Error if either of base_url or secret missing
|
||||||
|
|||||||
30
tests/app/test_security_headers.py
Normal file
30
tests/app/test_security_headers.py
Normal file
@@ -0,0 +1,30 @@
|
|||||||
|
import pytest
|
||||||
|
|
||||||
|
|
||||||
|
@pytest.mark.usefixtures('notify_db_session')
|
||||||
|
class TestSecurityHeaders:
|
||||||
|
"""Test security headers for ZAP scan compliance."""
|
||||||
|
|
||||||
|
def test_options_request_returns_204_with_cors_headers(self, client):
|
||||||
|
"""Test that OPTIONS requests return 204 with proper CORS headers."""
|
||||||
|
response = client.options('/')
|
||||||
|
|
||||||
|
assert response.status_code == 204
|
||||||
|
assert response.headers.get('Access-Control-Allow-Origin') == '*'
|
||||||
|
assert response.headers.get('Access-Control-Allow-Methods') == 'GET, POST, PUT, DELETE, OPTIONS'
|
||||||
|
assert response.headers.get('Access-Control-Allow-Headers') == 'Content-Type, Authorization'
|
||||||
|
assert response.headers.get('Access-Control-Max-Age') == '3600'
|
||||||
|
|
||||||
|
@pytest.mark.parametrize("endpoint", [
|
||||||
|
'/_status',
|
||||||
|
'/_status?simple=1',
|
||||||
|
'/_status/live-service-and-organization-counts'
|
||||||
|
])
|
||||||
|
def test_status_endpoints_have_cache_control_headers(self, client, endpoint):
|
||||||
|
"""Test that all status endpoints have proper cache-control headers."""
|
||||||
|
response = client.get(endpoint)
|
||||||
|
|
||||||
|
assert response.status_code == 200
|
||||||
|
assert response.headers.get('Cache-Control') == 'no-cache, no-store, must-revalidate'
|
||||||
|
assert response.headers.get('Pragma') == 'no-cache'
|
||||||
|
assert response.headers.get('Expires') == '0'
|
||||||
Reference in New Issue
Block a user