fix(OAuth2): Support OAuth2 exception with legacy endpoint (#39897)

This commit is contained in:
Vitor Avila
2026-05-05 21:21:48 -03:00
committed by GitHub
parent 4b17ac2629
commit 3745e37182
6 changed files with 358 additions and 3 deletions

View File

@@ -21,7 +21,12 @@ from flask_babel import lazy_gettext as _
from superset.commands.chart.exceptions import ChartDataQueryFailedError
from superset.errors import ErrorLevel, SupersetError, SupersetErrorType
from superset.exceptions import SupersetErrorException, SupersetErrorsException
from superset.exceptions import (
OAuth2RedirectError,
SupersetErrorException,
SupersetErrorsException,
SupersetVizException,
)
@mock.patch("superset.tasks.async_queries.security_manager")
@@ -149,3 +154,170 @@ def test_load_chart_data_into_cache_with_superset_errors_exception(
assert errors[1]["message"] == "Table not found"
assert errors[1]["error_type"] == SupersetErrorType.TABLE_DOES_NOT_EXIST_ERROR
assert errors[1]["level"] == ErrorLevel.WARNING
@mock.patch("superset.tasks.async_queries.security_manager")
@mock.patch("superset.tasks.async_queries.async_query_manager")
@mock.patch("superset.tasks.async_queries.get_viz")
@mock.patch("superset.tasks.async_queries.get_datasource_info")
def test_load_explore_json_into_cache_preserves_oauth2_redirect_error(
mock_get_datasource_info,
mock_get_viz,
mock_async_query_manager,
mock_security_manager,
):
"""
OAuth2RedirectError raised by ``viz_obj.get_payload`` must reach the async
job's errors list as a structured SIP-40 envelope so the frontend can
render the OAuth2 banner identically to the sync legacy path.
"""
from superset.tasks.async_queries import load_explore_json_into_cache
job_metadata = {"user_id": 1}
form_data: dict = {}
mock_get_datasource_info.return_value = (1, "table")
mock_security_manager.get_user_by_id.return_value = mock.MagicMock()
mock_async_query_manager.STATUS_ERROR = "error"
viz_obj = mock.MagicMock()
viz_obj.get_payload.side_effect = OAuth2RedirectError(
url="https://accounts.example.com/o/oauth2/v2/auth?...",
tab_id="tab-123",
redirect_uri="https://superset.example.com/oauth2/redirect",
)
mock_get_viz.return_value = viz_obj
with pytest.raises(OAuth2RedirectError):
load_explore_json_into_cache(job_metadata, form_data)
call_args = mock_async_query_manager.update_job.call_args
assert call_args[0] == (job_metadata, "error")
errors = call_args[1]["errors"]
assert len(errors) == 1
assert errors[0]["error_type"] == SupersetErrorType.OAUTH2_REDIRECT
assert errors[0]["extra"] == {
"url": "https://accounts.example.com/o/oauth2/v2/auth?...",
"tab_id": "tab-123",
"redirect_uri": "https://superset.example.com/oauth2/redirect",
}
@mock.patch("superset.tasks.async_queries.security_manager")
@mock.patch("superset.tasks.async_queries.async_query_manager")
@mock.patch("superset.tasks.async_queries.get_viz")
@mock.patch("superset.tasks.async_queries.get_datasource_info")
def test_load_explore_json_into_cache_preserves_superset_errors_exception(
mock_get_datasource_info,
mock_get_viz,
mock_async_query_manager,
mock_security_manager,
):
"""SupersetErrorsException must be preserved as a list of SIP-40 dicts."""
from superset.tasks.async_queries import load_explore_json_into_cache
job_metadata = {"user_id": 1}
form_data: dict = {}
mock_get_datasource_info.return_value = (1, "table")
mock_security_manager.get_user_by_id.return_value = mock.MagicMock()
mock_async_query_manager.STATUS_ERROR = "error"
viz_obj = mock.MagicMock()
viz_obj.get_payload.side_effect = SupersetErrorsException(
[
SupersetError(
message="Column not found",
error_type=SupersetErrorType.COLUMN_DOES_NOT_EXIST_ERROR,
level=ErrorLevel.ERROR,
),
SupersetError(
message="Table not found",
error_type=SupersetErrorType.TABLE_DOES_NOT_EXIST_ERROR,
level=ErrorLevel.WARNING,
),
]
)
mock_get_viz.return_value = viz_obj
with pytest.raises(SupersetErrorsException):
load_explore_json_into_cache(job_metadata, form_data)
errors = mock_async_query_manager.update_job.call_args[1]["errors"]
assert len(errors) == 2
assert errors[0]["error_type"] == SupersetErrorType.COLUMN_DOES_NOT_EXIST_ERROR
assert errors[1]["error_type"] == SupersetErrorType.TABLE_DOES_NOT_EXIST_ERROR
@mock.patch("superset.tasks.async_queries.security_manager")
@mock.patch("superset.tasks.async_queries.async_query_manager")
@mock.patch("superset.tasks.async_queries.get_viz")
@mock.patch("superset.tasks.async_queries.get_datasource_info")
def test_load_explore_json_into_cache_preserves_superset_viz_exception(
mock_get_datasource_info,
mock_get_viz,
mock_async_query_manager,
mock_security_manager,
):
"""
Test that SupersetVizException passes ``ex.errors`` straight through.
"""
from superset.tasks.async_queries import load_explore_json_into_cache
job_metadata = {"user_id": 1}
form_data: dict = {}
mock_get_datasource_info.return_value = (1, "table")
mock_security_manager.get_user_by_id.return_value = mock.MagicMock()
mock_async_query_manager.STATUS_ERROR = "error"
payload_errors = [
{
"message": "Bad column",
"error_type": SupersetErrorType.VIZ_GET_DF_ERROR,
"level": ErrorLevel.ERROR,
}
]
viz_obj = mock.MagicMock()
viz_obj.get_payload.return_value = {"errors": payload_errors}
viz_obj.has_error.return_value = True
mock_get_viz.return_value = viz_obj
with pytest.raises(SupersetVizException):
load_explore_json_into_cache(job_metadata, form_data)
errors = mock_async_query_manager.update_job.call_args[1]["errors"]
assert errors == payload_errors
@mock.patch("superset.tasks.async_queries.security_manager")
@mock.patch("superset.tasks.async_queries.async_query_manager")
@mock.patch("superset.tasks.async_queries.get_viz")
@mock.patch("superset.tasks.async_queries.get_datasource_info")
def test_load_explore_json_into_cache_falls_back_to_string_for_generic_exception(
mock_get_datasource_info,
mock_get_viz,
mock_async_query_manager,
mock_security_manager,
):
"""
Test that Non-Superset exception are passed as plain-string error.
"""
from superset.tasks.async_queries import load_explore_json_into_cache
job_metadata = {"user_id": 1}
form_data: dict = {}
mock_get_datasource_info.return_value = (1, "table")
mock_security_manager.get_user_by_id.return_value = mock.MagicMock()
mock_async_query_manager.STATUS_ERROR = "error"
viz_obj = mock.MagicMock()
viz_obj.get_payload.side_effect = RuntimeError("boom")
mock_get_viz.return_value = viz_obj
with pytest.raises(RuntimeError):
load_explore_json_into_cache(job_metadata, form_data)
errors = mock_async_query_manager.update_job.call_args[1]["errors"]
assert errors == ["boom"]