Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

refactor: Remove deprecated action key #686

Closed
wants to merge 2 commits into from
Closed
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
12 changes: 2 additions & 10 deletions frontend/src/client/schemas.gen.ts
Original file line number Diff line number Diff line change
Expand Up @@ -105,16 +105,12 @@ export const $ActionRead = {
type: 'object',
title: 'Inputs'
},
key: {
type: 'string',
title: 'Key'
},
control_flow: {
'$ref': '#/components/schemas/ActionControlFlow'
}
},
type: 'object',
required: ['id', 'type', 'title', 'description', 'status', 'inputs', 'key'],
required: ['id', 'type', 'title', 'description', 'status', 'inputs'],
title: 'ActionRead'
} as const;

Expand Down Expand Up @@ -143,14 +139,10 @@ export const $ActionReadMinimal = {
status: {
type: 'string',
title: 'Status'
},
key: {
type: 'string',
title: 'Key'
}
},
type: 'object',
required: ['id', 'workflow_id', 'type', 'title', 'description', 'status', 'key'],
required: ['id', 'workflow_id', 'type', 'title', 'description', 'status'],
title: 'ActionReadMinimal'
} as const;

Expand Down
2 changes: 0 additions & 2 deletions frontend/src/client/types.gen.ts
Original file line number Diff line number Diff line change
Expand Up @@ -31,7 +31,6 @@ export type ActionRead = {
inputs: {
[key: string]: unknown;
};
key: string;
control_flow?: ActionControlFlow;
};

Expand All @@ -42,7 +41,6 @@ export type ActionReadMinimal = {
title: string;
description: string;
status: string;
key: string;
};

export type ActionRetryPolicy = {
Expand Down
125 changes: 125 additions & 0 deletions tests/unit/test_client.py
Original file line number Diff line number Diff line change
@@ -0,0 +1,125 @@
import uuid

import pytest
from fastapi import APIRouter, FastAPI
from fastapi.testclient import TestClient

from tracecat.auth.credentials import RoleACL
from tracecat.logger import logger
from tracecat.types.auth import AccessLevel, Role

WORKSPACE_ID = uuid.uuid4()
USER_ID = uuid.uuid4()
SERVICE_ID = "tracecat-api"


router = APIRouter()


@router.get("/test")
async def test(role: Role = RoleACL(allow_user=True, allow_service=True)):
return {"test": "test", "role": role.model_dump()}


@pytest.fixture
def test_client(test_role):
"""Create a FastAPI test client with our router and mocked RoleACL."""
app = FastAPI()

app.include_router(router)
return TestClient(app)


def test_endpoint_can_be_hit(test_client: TestClient):
"""
Test that the /test endpoint is accessible and returns the expected response.

Args:
test_client: Pytest fixture that provides a configured TestClient
"""
response = test_client.get("/test")

assert response.status_code == 200
assert response.json() == {
"test": "test",
"role": {
"type": "user",
"access_level": 0,
"workspace_id": str(WORKSPACE_ID),
"user_id": str(USER_ID),
"service_id": SERVICE_ID,
},
}


@pytest.fixture
def override_test_role():
"""
Fixture that creates a different test role for dependency override testing.


Role: A test role with modified values
"""

async def _override_test_role(*args, **kwargs):
return Role(
type="service",
workspace_id=WORKSPACE_ID,
user_id=USER_ID,
service_id=SERVICE_ID,
access_level=AccessLevel.ADMIN,
)

return _override_test_role


@pytest.fixture
def test_client_with_override(
monkeypatch: pytest.MonkeyPatch, override_test_role: Role
):
"""
Create a FastAPI test client with our router and overridden role dependency.

Args:
override_test_role: Pytest fixture providing the override role

Returns:
TestClient: Configured test client with dependency override
"""

app = FastAPI()
app.include_router(router)

# Override the role dependency with our custom role
monkeypatch.setattr(
"tracecat.auth.credentials._role_dependency",
override_test_role,
)

return TestClient(app)


def test_endpoint_access_with_override(test_client_with_override: TestClient):
"""
Test that the /test endpoint uses the overridden role dependency.

Args:
test_client_with_override: Pytest fixture that provides a TestClient with overridden dependencies
"""
response = test_client_with_override.get(
"/test", params={"workspace_id": str(WORKSPACE_ID)}
)
details = response.json()
logger.info("RESPONSE", response=response, details=details)

assert response.status_code == 200
assert response.json() == {
"test": "test",
"role": {
"type": "service",
"access_level": AccessLevel.ADMIN,
"workspace_id": str(WORKSPACE_ID),
"user_id": str(USER_ID),
"service_id": SERVICE_ID,
},
}
121 changes: 121 additions & 0 deletions tests/unit/test_dependencies.py
Original file line number Diff line number Diff line change
@@ -0,0 +1,121 @@
import uuid
from collections.abc import AsyncGenerator

import pytest
from fastapi import FastAPI
from httpx import ASGITransport, AsyncClient

from tracecat.contexts import RunContext
from tracecat.dsl.common import create_default_execution_context
from tracecat.dsl.models import ActionStatement, RunActionInput
from tracecat.executor.router import router
from tracecat.logger import logger
from tracecat.registry.actions.models import RegistryActionErrorInfo
from tracecat.types.auth import Role
from tracecat.types.exceptions import WrappedExecutionError

# Unit tests


@pytest.fixture
def mock_workspace_id():
return uuid.uuid4()


@pytest.fixture
def mock_role(mock_org_id: uuid.UUID, mock_workspace_id: uuid.UUID):
return Role(
type="service",
user_id=mock_org_id,
workspace_id=mock_workspace_id,
service_id="tracecat-runner",
)


@pytest.fixture
def override_role_dependency(mock_role: Role):
async def dep(*args, **kwargs):
return mock_role

return dep


@pytest.fixture
async def test_client_noauth(
monkeypatch: pytest.MonkeyPatch, override_role_dependency: Role
) -> AsyncGenerator[AsyncClient, None]:
app = FastAPI()
app.include_router(router)
# Override the role dependency with our custom role
monkeypatch.setattr(
"tracecat.auth.credentials._role_dependency", override_role_dependency
)

host = "localhost"
port = 8000
async with AsyncClient(
transport=ASGITransport(app=app, client=(host, port)), # type: ignore
base_url="http://test",
) as client:
yield client


async def mock_dispatch_error(*args, **kwargs):
"""Mock dispatch that raises a WrappedExecutionError"""
error_info = RegistryActionErrorInfo(
type="ValueError",
message="Test error message",
action_name="test.action",
filename="test_file.py",
function="test_function",
)
raise WrappedExecutionError(error=error_info)


@pytest.mark.anyio
async def test_run_action_endpoint_error_handling(
test_client_noauth: AsyncClient,
mock_run_context: RunContext,
mock_role: Role,
monkeypatch: pytest.MonkeyPatch,
):
"""Test that the run_action endpoint properly handles wrapped execution errors."""

# Mock the dispatch function to raise our error
if not mock_role.workspace_id:
return pytest.fail("Workspace ID is not set in test role")

monkeypatch.setattr(
"tracecat.executor.service.dispatch_action_on_cluster", mock_dispatch_error
)

# Create test input
input_data = RunActionInput(
task=ActionStatement(
ref="test",
action="test.action",
args={},
run_if=None,
for_each=None,
),
exec_context=create_default_execution_context(),
run_context=mock_run_context,
).model_dump(mode="json")

logger.warning("BASE URL", base_url=test_client_noauth.base_url)
response = await test_client_noauth.post(
"/run/test.action",
params={"workspace_id": str(mock_role.workspace_id)},
json=input_data,
)

# Verify response
logger.info("RESPONSE", response=response)
assert response.status_code == 500
error_detail = response.json()
err_info = RegistryActionErrorInfo.model_validate(error_detail["detail"])
assert err_info.type == "ValueError"
assert err_info.message == "Test error message"
assert err_info.action_name == "test.action"
assert err_info.filename == "test_file.py"
assert err_info.function == "mock_dispatch_error"
6 changes: 0 additions & 6 deletions tracecat/db/schemas.py
Original file line number Diff line number Diff line change
Expand Up @@ -440,12 +440,6 @@ class Action(Resource, table=True):
back_populates="actions", sa_relationship_kwargs=DEFAULT_SA_RELATIONSHIP_KWARGS
)

@computed_field
@property
def key(self) -> str:
"""Workflow-relative key for an Action."""
return action.key(self.workflow_id, self.id)

@property
def ref(self) -> str:
"""Slugified title of the action. Used for references."""
Expand Down
2 changes: 0 additions & 2 deletions tracecat/workflow/actions/models.py
Original file line number Diff line number Diff line change
Expand Up @@ -23,7 +23,6 @@ class ActionRead(BaseModel):
description: str
status: str
inputs: dict[str, Any]
key: str # Computed field
control_flow: ActionControlFlow = Field(default_factory=ActionControlFlow)


Expand All @@ -34,7 +33,6 @@ class ActionReadMinimal(BaseModel):
title: str
description: str
status: str
key: str


class ActionCreate(BaseModel):
Expand Down
4 changes: 0 additions & 4 deletions tracecat/workflow/actions/router.py
Original file line number Diff line number Diff line change
Expand Up @@ -37,7 +37,6 @@ async def list_actions(
title=action.title,
description=action.description,
status=action.status,
key=action.key,
)
for action in actions
]
Expand Down Expand Up @@ -82,7 +81,6 @@ async def create_action(
title=action.title,
description=action.description,
status=action.status,
key=action.key,
)
return action_metadata

Expand Down Expand Up @@ -115,7 +113,6 @@ async def get_action(
description=action.description,
status=action.status,
inputs=action.inputs,
key=action.key,
control_flow=ActionControlFlow(**action.control_flow),
)

Expand Down Expand Up @@ -163,7 +160,6 @@ async def update_action(
description=action.description,
status=action.status,
inputs=action.inputs,
key=action.key,
)


Expand Down
Loading