Skip to content

Commit be4acde

Browse files
committed
Merge KAB-871 to KAB-916
2 parents 9de97cc + bafa091 commit be4acde

File tree

2 files changed

+43
-30
lines changed

2 files changed

+43
-30
lines changed

src/keboola_mcp_server/component_tools.py

+24-21
Original file line numberDiff line numberDiff line change
@@ -254,8 +254,8 @@ def add_component_tools(mcp: FastMCP) -> None:
254254
"""Add tools to the MCP server."""
255255

256256
component_tools = [
257-
list_components,
258-
list_component_configurations,
257+
retrieve_core_components,
258+
retrieve_component_configurations,
259259
get_component_configuration_details,
260260
get_core_component_details,
261261
list_all_component_configurations,
@@ -319,15 +319,17 @@ async def list_all_component_configurations(
319319
types = handle_component_types(types)
320320

321321
# retrieve all core components
322-
components = await list_components(ctx, types)
322+
components = await retrieve_core_components(ctx, types)
323323
logger.info(f"Found {len(components)} core components for given types {types}.")
324324
# iterate over all core components and retrieve their configurations
325325
component_configs: List[ComponentConfigurationsList] = []
326326
for component in components:
327327
# check if the component matches the types filter
328328
if conform_types(component, types):
329329
# retrieve all configurations for the component
330-
cur_component_configs = await list_component_configurations(component.component_id, ctx)
330+
cur_component_configs = await retrieve_component_configurations(
331+
component.component_id, ctx
332+
)
331333
component_configs.append(
332334
ComponentConfigurationsList(
333335
component=component,
@@ -337,7 +339,7 @@ async def list_all_component_configurations(
337339
return sorted(component_configs, key=lambda x: x.component.component_type)
338340

339341

340-
async def list_components(
342+
async def retrieve_core_components(
341343
ctx: Context,
342344
types: Annotated[
343345
List[ComponentType],
@@ -380,18 +382,17 @@ async def list_components(
380382
-> set types to ["all"] because we need to find all components from which we can derive postgresql configurations
381383
-> returns all core components
382384
"""
383-
client = ctx.session.state["sapi_client"]
384-
assert isinstance(client, KeboolaClient)
385+
client = KeboolaClient.from_state(ctx.session.state)
385386

386387
types = handle_component_types(types)
387388

388-
r_components = client.storage_client.components.list()
389-
logger.info(f"Found {len(r_components)} components for given types {types}.")
390-
components = [ComponentListItem.model_validate(r_comp) for r_comp in r_components]
389+
raw_components = client.storage_client.components.list()
390+
logger.info(f"Found {len(raw_components)} components for given types {types}.")
391+
components = [ComponentListItem.model_validate(r_comp) for r_comp in raw_components]
391392
return list(filter(lambda x: conform_types(x, types), components))
392393

393394

394-
async def list_component_configurations(
395+
async def retrieve_component_configurations(
395396
component_id: Annotated[
396397
str,
397398
Field(
@@ -416,15 +417,17 @@ async def list_component_configurations(
416417
client = KeboolaClient.from_state(ctx.session.state)
417418

418419
component = await get_core_component_details(component_id, ctx)
419-
r_configs = client.storage_client.configurations.list(component_id)
420-
logger.info(f"Found {len(r_configs)} component configurations for component {component_id}.")
420+
raw_configurations = client.storage_client.configurations.list(component_id)
421+
logger.info(
422+
f"Found {len(raw_configurations)} component configurations for component {component_id}."
423+
)
421424
return ComponentConfigurationsList(
422425
component=component.to_list_item(),
423426
configurations=[
424427
ComponentConfigurationListItem.model_validate(
425-
{**r_config, "component_id": component_id}
428+
{**raw_config, "component_id": component_id}
426429
)
427-
for r_config in r_configs
430+
for raw_config in raw_configurations
428431
],
429432
)
430433

@@ -450,10 +453,10 @@ async def get_core_component_details(
450453
"""
451454
client = KeboolaClient.from_state(ctx.session.state)
452455

453-
endpoint = "branch/{}/components/{}".format(client.storage_client._branch_id, component_id)
454-
r_component = await client.get(endpoint)
456+
endpoint = f"branch/{client.storage_client._branch_id}/components/{component_id}"
457+
raw_component = await client.get(endpoint)
455458
logger.info(f"Retrieved component details for component {component_id}.")
456-
return ComponentDetail.model_validate(r_component)
459+
return ComponentDetail.model_validate(raw_component)
457460

458461

459462
async def get_component_configuration_details(
@@ -470,7 +473,7 @@ async def get_component_configuration_details(
470473
ctx: Context,
471474
) -> ComponentConfigurationDetail:
472475
"""
473-
Retrieve detailed information about a specific Keboola component configuration given component ID and configuration
476+
Get detailed information about a specific Keboola component configuration given component ID and configuration
474477
ID. Those IDs can be retrieved from fully_qualified_id of the component configuration which are seperated by `::`.
475478
Use to get the configuration details, metadata and the core Keboola component.
476479
PARAMETERS:
@@ -489,7 +492,7 @@ async def get_component_configuration_details(
489492
client = KeboolaClient.from_state(ctx.session.state)
490493

491494
component = await get_core_component_details(component_id, ctx)
492-
r_config = client.storage_client.configurations.detail(component_id, configuration_id)
495+
raw_configuration = client.storage_client.configurations.detail(component_id, configuration_id)
493496
logger.info(
494497
f"Retrieved configuration details for component configuration {component_id}::{configuration_id}."
495498
)
@@ -508,7 +511,7 @@ async def get_component_configuration_details(
508511
)
509512

510513
return ComponentConfigurationDetail.model_validate(
511-
{**r_config, "component": component, "component_id": component_id, "metadata": r_metadata}
514+
{**raw_configuration, "component": component, "component_id": component_id, "metadata": r_metadata}
512515
)
513516

514517

tests/test_component_tools.py

+19-9
Original file line numberDiff line numberDiff line change
@@ -14,9 +14,9 @@
1414
get_component_configuration_details,
1515
get_core_component_details,
1616
handle_component_types,
17-
list_component_configurations,
17+
retrieve_component_configurations,
1818
list_all_component_configurations,
19-
list_components,
19+
retrieve_core_components,
2020
)
2121

2222

@@ -141,18 +141,28 @@ async def test_list_core_components(mcp_context_components_configs, mock_compone
141141
keboola_client = context.session.state["sapi_client"]
142142
keboola_client.storage_client.components.list = MagicMock(return_value=mock_components)
143143

144-
result = await list_components(context)
144+
result = await retrieve_core_components(context)
145145

146146
assert len(result) == 2
147147

148148
assert all(isinstance(component, ComponentListItem) for component in result)
149-
assert all(c.component_id == item["id"] for c, item in zip(result, mock_components))
150-
assert all(c.component_name == item["name"] for c, item in zip(result, mock_components))
151-
assert all(c.component_type == item["type"] for c, item in zip(result, mock_components))
152149
assert all(
153-
c.component_description == item["description"] for c, item in zip(result, mock_components)
150+
component.component_id == expected["id"]
151+
for component, expected in zip(result, mock_components)
154152
)
155-
assert all(not hasattr(c, "version") for c in result)
153+
assert all(
154+
component.component_name == expected["name"]
155+
for component, expected in zip(result, mock_components)
156+
)
157+
assert all(
158+
component.component_type == expected["type"]
159+
for component, expected in zip(result, mock_components)
160+
)
161+
assert all(
162+
component.component_description == expected["description"]
163+
for component, expected in zip(result, mock_components)
164+
)
165+
assert all(not hasattr(component, "version") for component in result)
156166

157167
keboola_client.storage_client.components.list.assert_called_once()
158168

@@ -264,7 +274,7 @@ async def test_list_component_configurations(
264274
keboola_client.storage_client.configurations.list = MagicMock(return_value=mock_configurations)
265275
keboola_client.get = AsyncMock(return_value=mock_component)
266276

267-
result = await list_component_configurations("keboola.ex-aws-s3", context)
277+
result = await retrieve_component_configurations("keboola.ex-aws-s3", context)
268278

269279
# assert basics
270280
assert isinstance(result, ComponentConfigurationsList)

0 commit comments

Comments
 (0)