Skip to content

Commit ffa5bab

Browse files
Merge pull request #28 from keboola/KAB-923-fix-server-start
Kab 923 fix server start
2 parents bf84b17 + f06ef80 commit ffa5bab

File tree

5 files changed

+48
-25
lines changed

5 files changed

+48
-25
lines changed

pyproject.toml

+11-13
Original file line numberDiff line numberDiff line change
@@ -5,7 +5,7 @@ build-backend = "hatchling.build"
55
[project]
66
name = "keboola-mcp-server"
77

8-
version = "0.2.2"
8+
version = "0.2.3"
99
description = "MCP server for interacting with Keboola Connection"
1010
readme = "README.md"
1111
requires-python = ">=3.10"
@@ -14,22 +14,20 @@ authors = [
1414
{ name = "Your Name", email = "[email protected]" }
1515
]
1616
dependencies = [
17-
"mcp[cli]",
18-
"kbcstorage",
19-
"httpx",
20-
"pandas"
17+
"mcp[cli] ~= 1.6",
18+
"kbcstorage ~= 0.9",
19+
"httpx ~= 0.28",
2120
]
2221

2322
[project.optional-dependencies]
2423
dev = [
25-
"pytest",
26-
"pytest-asyncio",
27-
"pytest-cov",
28-
"pytest-mock",
29-
"black",
30-
"isort",
31-
"mypy",
32-
"pandas-stubs"
24+
"pytest ~= 8.3",
25+
"pytest-asyncio ~= 0.25",
26+
"pytest-cov ~= 6.0",
27+
"pytest-mock ~= 3.14",
28+
"black ~= 25.1",
29+
"isort ~= 6.0",
30+
"mypy ~= 1.5",
3331
]
3432

3533
[project.scripts]

src/keboola_mcp_server/cli.py

+1-1
Original file line numberDiff line numberDiff line change
@@ -65,7 +65,7 @@ def main(args: Optional[List[str]] = None) -> None:
6565
mcp = create_server(config)
6666
mcp.run(transport=parsed_args.transport)
6767
except Exception as e:
68-
logger.error(f"Server failed: {e}")
68+
logger.exception(f"Server failed: {e}")
6969
sys.exit(1)
7070

7171

src/keboola_mcp_server/server.py

+1-8
Original file line numberDiff line numberDiff line change
@@ -64,14 +64,7 @@ def create_server(config: Optional[Config] = None) -> FastMCP:
6464
"""
6565
# Initialize FastMCP server with system instructions
6666
mcp = KeboolaMcpServer(
67-
"Keboola Explorer",
68-
session_state_factory=_create_session_state_factory(config),
69-
dependencies=[
70-
"keboola.storage-api-client",
71-
"httpx",
72-
"pandas",
73-
"snowflake-connector-python",
74-
],
67+
"Keboola Explorer", session_state_factory=_create_session_state_factory(config)
7568
)
7669

7770
add_storage_tools(mcp)

src/keboola_mcp_server/sql_tools.py

+3-3
Original file line numberDiff line numberDiff line change
@@ -9,7 +9,7 @@
99

1010
from keboola_mcp_server.client import KeboolaClient
1111

12-
LOG = logging.getLogger(__name__)
12+
logger = logging.getLogger(__name__)
1313

1414

1515
def add_sql_tools(mcp: FastMCP) -> None:
@@ -120,7 +120,7 @@ async def get_table_fqn(self, table: dict[str, Any]) -> Optional[TableFqn]:
120120
if result.is_ok and result.data and result.data.rows:
121121
db_name = result.data.rows[0]["DATABASE_NAME"]
122122
else:
123-
LOG.error(f"Failed to run SQL: {sql}, SAPI response: {result}")
123+
logger.error(f"Failed to run SQL: {sql}, SAPI response: {result}")
124124

125125
else:
126126
sql = f'select CURRENT_DATABASE() as "current_database", CURRENT_SCHEMA() as "current_schema";'
@@ -138,7 +138,7 @@ async def get_table_fqn(self, table: dict[str, Any]) -> Optional[TableFqn]:
138138
schema_name = row["current_schema"]
139139
table_name = table["name"]
140140
else:
141-
LOG.error(f"Failed to run SQL: {sql}, SAPI response: {result}")
141+
logger.error(f"Failed to run SQL: {sql}, SAPI response: {result}")
142142

143143
if db_name and schema_name and table_name:
144144
fqn = TableFqn(db_name, schema_name, table_name)

tests/test_server.py

+32
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,32 @@
1+
import pytest
2+
3+
from keboola_mcp_server.server import create_server
4+
5+
6+
class TestServer:
7+
@pytest.mark.asyncio
8+
async def test_list_tools(self):
9+
server = create_server()
10+
tools = await server.list_tools()
11+
assert sorted(t.name for t in tools) == [
12+
"get_bucket_metadata",
13+
"get_table_metadata",
14+
"list_bucket_info",
15+
"list_bucket_tables",
16+
"list_component_configs",
17+
"list_components",
18+
"query_table",
19+
]
20+
21+
@pytest.mark.asyncio
22+
async def test_tools_have_descriptions(self):
23+
server = create_server()
24+
tools = await server.list_tools()
25+
26+
missing_descriptions: list[str] = []
27+
for t in tools:
28+
if not t.description:
29+
missing_descriptions.append(t.name)
30+
31+
missing_descriptions.sort()
32+
assert not missing_descriptions, f"These tools have no description: {missing_descriptions}"

0 commit comments

Comments
 (0)