diff --git a/benefits/enrollment/templates/enrollment/index.html b/benefits/enrollment/templates/enrollment/index.html
index bc231a9f3..c48cf9129 100644
--- a/benefits/enrollment/templates/enrollment/index.html
+++ b/benefits/enrollment/templates/enrollment/index.html
@@ -41,6 +41,12 @@
$.ajax({ dataType: "script", attrs: { nonce: "{{ request.csp_nonce }}"}, url: "{{ card_tokenize_url }}" })
.done(function() {
$.get("{{ access_token_url }}", function(data) {
+ if (data.redirect) {
+ // https://stackoverflow.com/a/42469170
+ // use 'assign' because 'replace' was giving strange Back button behavior
+ window.location.assign(data.redirect);
+ }
+
$(".loading").remove();
// remove invisible and add back visible, so we aren't left with
// a div with an empty class attribute
diff --git a/benefits/enrollment/urls.py b/benefits/enrollment/urls.py
index 22398bc98..446cf6ae4 100644
--- a/benefits/enrollment/urls.py
+++ b/benefits/enrollment/urls.py
@@ -14,4 +14,5 @@
path("token", views.token, name="token"),
path("retry", views.retry, name="retry"),
path("success", views.success, name="success"),
+ path("system_error", views.system_error, name="system_error"),
]
diff --git a/benefits/enrollment/views.py b/benefits/enrollment/views.py
index 4c23fa783..00831ffae 100644
--- a/benefits/enrollment/views.py
+++ b/benefits/enrollment/views.py
@@ -10,6 +10,7 @@
from django.utils.decorators import decorator_from_middleware
from littlepay.api.client import Client
from requests.exceptions import HTTPError
+import sentry_sdk
from benefits.core import session
from benefits.core.middleware import (
@@ -24,6 +25,7 @@
ROUTE_INDEX = "enrollment:index"
ROUTE_RETRY = "enrollment:retry"
ROUTE_SUCCESS = "enrollment:success"
+ROUTE_SYSTEM_ERROR = "enrollment:system_error"
ROUTE_TOKEN = "enrollment:token"
TEMPLATE_RETRY = "enrollment/retry.html"
@@ -47,8 +49,20 @@ def token(request):
audience=payment_processor.audience,
)
client.oauth.ensure_active_token(client.token)
- response = client.request_card_tokenization_access()
- session.update(request, enrollment_token=response.get("access_token"), enrollment_token_exp=response.get("expires_at"))
+
+ try:
+ response = client.request_card_tokenization_access()
+ except Exception as e:
+ if isinstance(e, HTTPError) and e.response.status_code >= 500:
+ sentry_sdk.capture_exception(e)
+ data = {"redirect": reverse(ROUTE_SYSTEM_ERROR)}
+ return JsonResponse(data)
+ else:
+ raise e
+ else:
+ session.update(
+ request, enrollment_token=response.get("access_token"), enrollment_token_exp=response.get("expires_at")
+ )
data = {"token": session.enrollment_token(request)}
diff --git a/tests/pytest/enrollment/test_views.py b/tests/pytest/enrollment/test_views.py
index f853ecbfc..39d2170d9 100644
--- a/tests/pytest/enrollment/test_views.py
+++ b/tests/pytest/enrollment/test_views.py
@@ -12,6 +12,7 @@
ROUTE_INDEX,
ROUTE_TOKEN,
ROUTE_SUCCESS,
+ ROUTE_SYSTEM_ERROR,
ROUTE_RETRY,
TEMPLATE_SUCCESS,
TEMPLATE_SYSTEM_ERROR,
@@ -99,6 +100,31 @@ def test_token_valid(mocker, client):
assert data["token"] == "enrollment_token"
+@pytest.mark.django_db
+@pytest.mark.usefixtures("mocked_session_agency", "mocked_session_eligibility")
+def test_token_http_error_500(mocker, client):
+ mocker.patch("benefits.core.session.enrollment_token_valid", return_value=False)
+
+ mock_client_cls = mocker.patch("benefits.enrollment.views.Client")
+ mock_client = mock_client_cls.return_value
+
+ mock_error = {"message": "Mock error message"}
+ mock_error_response = mocker.Mock(status_code=500, **mock_error)
+ mock_error_response.json.return_value = mock_error
+ mock_client.request_card_tokenization_access.side_effect = HTTPError(
+ response=mock_error_response,
+ )
+
+ path = reverse(ROUTE_TOKEN)
+ response = client.get(path)
+
+ assert response.status_code == 200
+ data = response.json()
+ assert "token" not in data
+ assert "redirect" in data
+ assert data["redirect"] == reverse(ROUTE_SYSTEM_ERROR)
+
+
@pytest.mark.django_db
@pytest.mark.usefixtures("mocked_session_agency", "mocked_session_verifier", "mocked_session_eligibility")
def test_index_eligible_get(client, model_EligibilityType):