Skip to content

Commit

Permalink
Merge pull request #28 from PADAS/gundi-3648-ui-schema
Browse files Browse the repository at this point in the history
GUNDI-3648: Support ui:schema in action configs
  • Loading branch information
marianobrc authored Oct 29, 2024
2 parents 51ca63b + 633621e commit 4da2ad1
Show file tree
Hide file tree
Showing 8 changed files with 507 additions and 83 deletions.
54 changes: 54 additions & 0 deletions README.md
Original file line number Diff line number Diff line change
Expand Up @@ -403,3 +403,57 @@ Sample configuration in Gundi:
"""
```
Notice: This can also be combined with Dynamic Schema and JSON Transformations. In that case the hex string will be parsed first, adn then the JQ filter can be applied to the extracted data.

### Custom UI for configurations (ui schema)
It's possible to customize how the forms for configurations are displayed in the Gundi portal.
To do that, use `FieldWithUIOptions` in your models. The `UIOptions` and `GlobalUISchemaOptions` will allow you to customize the appearance of the fields in the portal by setting any of the ["ui schema"](https://rjsf-team.github.io/react-jsonschema-form/docs/api-reference/uiSchema) supported options.

```python
# Example
import pydantic
from app.services.utils import FieldWithUIOptions, GlobalUISchemaOptions, UIOptions
from .core import AuthActionConfiguration, PullActionConfiguration


class AuthenticateConfig(AuthActionConfiguration):
email: str # This will be rendered with default widget and settings
password: pydantic.SecretStr = FieldWithUIOptions(
...,
format="password",
title="Password",
description="Password for the Global Forest Watch account.",
ui_options=UIOptions(
widget="password", # This will be rendered as a password input hiding the input
)
)
ui_global_options = GlobalUISchemaOptions(
order=["email", "password"], # This will set the order of the fields in the form
)


class MyPullActionConfiguration(PullActionConfiguration):
lookback_days: int = FieldWithUIOptions(
10,
le=30,
ge=1,
title="Data lookback days",
description="Number of days to look back for data.",
ui_options=UIOptions(
widget="range", # This will be rendered ad a range slider
)
)
force_fetch: bool = FieldWithUIOptions(
False,
title="Force fetch",
description="Force fetch even if in a quiet period.",
ui_options=UIOptions(
widget="radio", # This will be rendered as a radio button
)
)
ui_global_options = GlobalUISchemaOptions(
order=[
"lookback_days",
"force_fetch",
],
)
```
5 changes: 4 additions & 1 deletion app/actions/core.py
Original file line number Diff line number Diff line change
@@ -1,9 +1,12 @@
import importlib
import inspect
from typing import Optional

from pydantic import BaseModel
from app.services.utils import UISchemaModelMixin


class ActionConfiguration(BaseModel):
class ActionConfiguration(UISchemaModelMixin, BaseModel):
pass


Expand Down
70 changes: 61 additions & 9 deletions app/conftest.py
Original file line number Diff line number Diff line change
@@ -1,15 +1,13 @@
import asyncio
import datetime
import json

import pydantic
import pytest
from unittest.mock import MagicMock
from app import settings
from gcloud.aio import pubsub
from gundi_core.schemas.v2 import Integration, IntegrationActionConfiguration, IntegrationActionSummary
from gundi_core.schemas.v2 import Integration
from gundi_core.events import (
SystemEventBaseModel,
IntegrationActionCustomLog,
CustomActivityLog,
IntegrationActionStarted,
Expand All @@ -28,9 +26,9 @@
CustomWebhookLog,
LogLevel
)

from app.actions import PullActionConfiguration
from app.webhooks import GenericJsonTransformConfig, GenericJsonPayload, WebhookPayload
from app.services.utils import GlobalUISchemaOptions, FieldWithUIOptions, UIOptions
from app.webhooks import GenericJsonTransformConfig, GenericJsonPayload, WebhookPayload, WebhookConfiguration


class AsyncMock(MagicMock):
Expand Down Expand Up @@ -139,6 +137,14 @@ def integration_v2_with_webhook():
"allowed_devices_list": {"title": "Allowed Devices List", "type": "array", "items": {}},
"deduplication_enabled": {"title": "Deduplication Enabled", "type": "boolean"}},
"required": ["allowed_devices_list", "deduplication_enabled"]
},
"ui_schema": {
"allowed_devices_list": {
"ui:widget": "select"
},
"deduplication_enabled": {
"ui:widget": "radio"
}
}
}
},
Expand Down Expand Up @@ -218,6 +224,17 @@ def integration_v2_with_webhook_generic():
"description": "Output type for the transformed data: 'obv' or 'event'"
}
}
},
"ui_schema": {
"jq_filter": {
"ui:widget": "textarea"
},
"json_schema": {
"ui:widget": "textarea"
},
"output_type": {
"ui:widget": "text"
}
}
}
},
Expand Down Expand Up @@ -898,7 +915,30 @@ def mock_publish_event(gcp_pubsub_publish_response):


class MockPullActionConfiguration(PullActionConfiguration):
lookback_days: int = 10
lookback_days: int = FieldWithUIOptions(
30,
le=30,
ge=1,
title="Data lookback days",
description="Number of days to look back for data.",
ui_options=UIOptions(
widget="range",
)
)
force_fetch: bool = FieldWithUIOptions(
False,
title="Force fetch",
description="Force fetch even if in a quiet period.",
ui_options=UIOptions(
widget="select",
)
)
ui_global_options = GlobalUISchemaOptions(
order=[
"lookback_days",
"force_fetch",
],
)


@pytest.fixture
Expand Down Expand Up @@ -1172,9 +1212,21 @@ class MockWebhookPayloadModel(WebhookPayload):
lon: float


class MockWebhookConfigModel(pydantic.BaseModel):
allowed_devices_list: list
deduplication_enabled: bool
class MockWebhookConfigModel(WebhookConfiguration):
allowed_devices_list: list = FieldWithUIOptions(
...,
title="Allowed Devices List",
ui_options=UIOptions(
widget="list",
)
)
deduplication_enabled: bool = FieldWithUIOptions(
...,
title="Deduplication Enabled",
ui_options=UIOptions(
widget="radio",
)
)


@pytest.fixture
Expand Down
5 changes: 3 additions & 2 deletions app/services/self_registration.py
Original file line number Diff line number Diff line change
Expand Up @@ -14,8 +14,6 @@


async def register_integration_in_gundi(gundi_client, type_slug=None, service_url=None):
#from ..webhooks.configurations import LiquidTechPayload
#print(GenericJsonTransformConfig.schema_json())
# Prepare the integration name and value
integration_type_slug = type_slug or INTEGRATION_TYPE_SLUG
if not integration_type_slug:
Expand All @@ -38,6 +36,7 @@ async def register_integration_in_gundi(gundi_client, type_slug=None, service_ur
_, config_model = handler
action_name = action_id.replace("_", " ").title()
action_schema = json.loads(config_model.schema_json())
action_ui_schema = config_model.ui_schema()
if issubclass(config_model, AuthActionConfiguration):
action_type = ActionTypeEnum.AUTHENTICATION.value
elif issubclass(config_model, PullActionConfiguration):
Expand All @@ -53,6 +52,7 @@ async def register_integration_in_gundi(gundi_client, type_slug=None, service_ur
"value": action_id,
"description": f"{integration_type_name} {action_name} action",
"schema": action_schema,
"ui_schema": action_ui_schema,
"is_periodic_action": True if issubclass(config_model, PullActionConfiguration) else False,
}
)
Expand All @@ -70,6 +70,7 @@ async def register_integration_in_gundi(gundi_client, type_slug=None, service_ur
"value": f"{integration_type_slug}_webhook",
"description": f"Webhook Integration with {integration_type_name}",
"schema": json.loads(config_model.schema_json()),
"ui_schema": config_model.ui_schema(),
}

logger.info(f"Registering '{integration_type_slug}' with actions: '{actions}'")
Expand Down
Loading

0 comments on commit 4da2ad1

Please sign in to comment.