datarobot-genai 0.2.0__tar.gz → 0.2.2__tar.gz

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (102) hide show
  1. {datarobot_genai-0.2.0 → datarobot_genai-0.2.2}/PKG-INFO +2 -2
  2. {datarobot_genai-0.2.0 → datarobot_genai-0.2.2}/pyproject.toml +2 -2
  3. {datarobot_genai-0.2.0 → datarobot_genai-0.2.2}/src/datarobot_genai/drmcp/core/dr_mcp_server.py +34 -0
  4. datarobot_genai-0.2.2/src/datarobot_genai/drmcp/core/dynamic_prompts/dr_lib.py +70 -0
  5. {datarobot_genai-0.2.0 → datarobot_genai-0.2.2}/src/datarobot_genai/drmcp/core/dynamic_prompts/register.py +4 -5
  6. {datarobot_genai-0.2.0 → datarobot_genai-0.2.2}/src/datarobot_genai/drmcp/core/mcp_instance.py +41 -2
  7. {datarobot_genai-0.2.0 → datarobot_genai-0.2.2}/src/datarobot_genai/drmcp/core/routes.py +4 -1
  8. {datarobot_genai-0.2.0 → datarobot_genai-0.2.2}/src/datarobot_genai/drmcp/test_utils/mcp_utils_ete.py +29 -0
  9. datarobot_genai-0.2.0/src/datarobot_genai/drmcp/core/dynamic_prompts/dr_lib.py +0 -128
  10. {datarobot_genai-0.2.0 → datarobot_genai-0.2.2}/.gitignore +0 -0
  11. {datarobot_genai-0.2.0 → datarobot_genai-0.2.2}/AUTHORS +0 -0
  12. {datarobot_genai-0.2.0 → datarobot_genai-0.2.2}/LICENSE +0 -0
  13. {datarobot_genai-0.2.0 → datarobot_genai-0.2.2}/README.md +0 -0
  14. {datarobot_genai-0.2.0 → datarobot_genai-0.2.2}/src/datarobot_genai/__init__.py +0 -0
  15. {datarobot_genai-0.2.0 → datarobot_genai-0.2.2}/src/datarobot_genai/core/__init__.py +0 -0
  16. {datarobot_genai-0.2.0 → datarobot_genai-0.2.2}/src/datarobot_genai/core/agents/__init__.py +0 -0
  17. {datarobot_genai-0.2.0 → datarobot_genai-0.2.2}/src/datarobot_genai/core/agents/base.py +0 -0
  18. {datarobot_genai-0.2.0 → datarobot_genai-0.2.2}/src/datarobot_genai/core/chat/__init__.py +0 -0
  19. {datarobot_genai-0.2.0 → datarobot_genai-0.2.2}/src/datarobot_genai/core/chat/auth.py +0 -0
  20. {datarobot_genai-0.2.0 → datarobot_genai-0.2.2}/src/datarobot_genai/core/chat/client.py +0 -0
  21. {datarobot_genai-0.2.0 → datarobot_genai-0.2.2}/src/datarobot_genai/core/chat/responses.py +0 -0
  22. {datarobot_genai-0.2.0 → datarobot_genai-0.2.2}/src/datarobot_genai/core/cli/__init__.py +0 -0
  23. {datarobot_genai-0.2.0 → datarobot_genai-0.2.2}/src/datarobot_genai/core/cli/agent_environment.py +0 -0
  24. {datarobot_genai-0.2.0 → datarobot_genai-0.2.2}/src/datarobot_genai/core/cli/agent_kernel.py +0 -0
  25. {datarobot_genai-0.2.0 → datarobot_genai-0.2.2}/src/datarobot_genai/core/custom_model.py +0 -0
  26. {datarobot_genai-0.2.0 → datarobot_genai-0.2.2}/src/datarobot_genai/core/mcp/__init__.py +0 -0
  27. {datarobot_genai-0.2.0 → datarobot_genai-0.2.2}/src/datarobot_genai/core/mcp/common.py +0 -0
  28. {datarobot_genai-0.2.0 → datarobot_genai-0.2.2}/src/datarobot_genai/core/telemetry_agent.py +0 -0
  29. {datarobot_genai-0.2.0 → datarobot_genai-0.2.2}/src/datarobot_genai/core/utils/__init__.py +0 -0
  30. {datarobot_genai-0.2.0 → datarobot_genai-0.2.2}/src/datarobot_genai/core/utils/auth.py +0 -0
  31. {datarobot_genai-0.2.0 → datarobot_genai-0.2.2}/src/datarobot_genai/core/utils/urls.py +0 -0
  32. {datarobot_genai-0.2.0 → datarobot_genai-0.2.2}/src/datarobot_genai/crewai/__init__.py +0 -0
  33. {datarobot_genai-0.2.0 → datarobot_genai-0.2.2}/src/datarobot_genai/crewai/agent.py +0 -0
  34. {datarobot_genai-0.2.0 → datarobot_genai-0.2.2}/src/datarobot_genai/crewai/base.py +0 -0
  35. {datarobot_genai-0.2.0 → datarobot_genai-0.2.2}/src/datarobot_genai/crewai/events.py +0 -0
  36. {datarobot_genai-0.2.0 → datarobot_genai-0.2.2}/src/datarobot_genai/crewai/mcp.py +0 -0
  37. {datarobot_genai-0.2.0 → datarobot_genai-0.2.2}/src/datarobot_genai/drmcp/__init__.py +0 -0
  38. {datarobot_genai-0.2.0 → datarobot_genai-0.2.2}/src/datarobot_genai/drmcp/core/__init__.py +0 -0
  39. {datarobot_genai-0.2.0 → datarobot_genai-0.2.2}/src/datarobot_genai/drmcp/core/auth.py +0 -0
  40. {datarobot_genai-0.2.0 → datarobot_genai-0.2.2}/src/datarobot_genai/drmcp/core/clients.py +0 -0
  41. {datarobot_genai-0.2.0 → datarobot_genai-0.2.2}/src/datarobot_genai/drmcp/core/config.py +0 -0
  42. {datarobot_genai-0.2.0 → datarobot_genai-0.2.2}/src/datarobot_genai/drmcp/core/config_utils.py +0 -0
  43. {datarobot_genai-0.2.0 → datarobot_genai-0.2.2}/src/datarobot_genai/drmcp/core/constants.py +0 -0
  44. {datarobot_genai-0.2.0 → datarobot_genai-0.2.2}/src/datarobot_genai/drmcp/core/credentials.py +0 -0
  45. {datarobot_genai-0.2.0 → datarobot_genai-0.2.2}/src/datarobot_genai/drmcp/core/dr_mcp_server_logo.py +0 -0
  46. {datarobot_genai-0.2.0 → datarobot_genai-0.2.2}/src/datarobot_genai/drmcp/core/dynamic_prompts/__init__.py +0 -0
  47. {datarobot_genai-0.2.0 → datarobot_genai-0.2.2}/src/datarobot_genai/drmcp/core/dynamic_prompts/controllers.py +0 -0
  48. {datarobot_genai-0.2.0 → datarobot_genai-0.2.2}/src/datarobot_genai/drmcp/core/dynamic_prompts/utils.py +0 -0
  49. {datarobot_genai-0.2.0 → datarobot_genai-0.2.2}/src/datarobot_genai/drmcp/core/dynamic_tools/__init__.py +0 -0
  50. {datarobot_genai-0.2.0 → datarobot_genai-0.2.2}/src/datarobot_genai/drmcp/core/dynamic_tools/deployment/__init__.py +0 -0
  51. {datarobot_genai-0.2.0 → datarobot_genai-0.2.2}/src/datarobot_genai/drmcp/core/dynamic_tools/deployment/adapters/__init__.py +0 -0
  52. {datarobot_genai-0.2.0 → datarobot_genai-0.2.2}/src/datarobot_genai/drmcp/core/dynamic_tools/deployment/adapters/base.py +0 -0
  53. {datarobot_genai-0.2.0 → datarobot_genai-0.2.2}/src/datarobot_genai/drmcp/core/dynamic_tools/deployment/adapters/default.py +0 -0
  54. {datarobot_genai-0.2.0 → datarobot_genai-0.2.2}/src/datarobot_genai/drmcp/core/dynamic_tools/deployment/adapters/drum.py +0 -0
  55. {datarobot_genai-0.2.0 → datarobot_genai-0.2.2}/src/datarobot_genai/drmcp/core/dynamic_tools/deployment/config.py +0 -0
  56. {datarobot_genai-0.2.0 → datarobot_genai-0.2.2}/src/datarobot_genai/drmcp/core/dynamic_tools/deployment/controllers.py +0 -0
  57. {datarobot_genai-0.2.0 → datarobot_genai-0.2.2}/src/datarobot_genai/drmcp/core/dynamic_tools/deployment/metadata.py +0 -0
  58. {datarobot_genai-0.2.0 → datarobot_genai-0.2.2}/src/datarobot_genai/drmcp/core/dynamic_tools/deployment/register.py +0 -0
  59. {datarobot_genai-0.2.0 → datarobot_genai-0.2.2}/src/datarobot_genai/drmcp/core/dynamic_tools/deployment/schemas/drum_agentic_fallback_schema.json +0 -0
  60. {datarobot_genai-0.2.0 → datarobot_genai-0.2.2}/src/datarobot_genai/drmcp/core/dynamic_tools/deployment/schemas/drum_prediction_fallback_schema.json +0 -0
  61. {datarobot_genai-0.2.0 → datarobot_genai-0.2.2}/src/datarobot_genai/drmcp/core/dynamic_tools/register.py +0 -0
  62. {datarobot_genai-0.2.0 → datarobot_genai-0.2.2}/src/datarobot_genai/drmcp/core/dynamic_tools/schema.py +0 -0
  63. {datarobot_genai-0.2.0 → datarobot_genai-0.2.2}/src/datarobot_genai/drmcp/core/exceptions.py +0 -0
  64. {datarobot_genai-0.2.0 → datarobot_genai-0.2.2}/src/datarobot_genai/drmcp/core/logging.py +0 -0
  65. {datarobot_genai-0.2.0 → datarobot_genai-0.2.2}/src/datarobot_genai/drmcp/core/mcp_server_tools.py +0 -0
  66. {datarobot_genai-0.2.0 → datarobot_genai-0.2.2}/src/datarobot_genai/drmcp/core/memory_management/__init__.py +0 -0
  67. {datarobot_genai-0.2.0 → datarobot_genai-0.2.2}/src/datarobot_genai/drmcp/core/memory_management/manager.py +0 -0
  68. {datarobot_genai-0.2.0 → datarobot_genai-0.2.2}/src/datarobot_genai/drmcp/core/memory_management/memory_tools.py +0 -0
  69. {datarobot_genai-0.2.0 → datarobot_genai-0.2.2}/src/datarobot_genai/drmcp/core/routes_utils.py +0 -0
  70. {datarobot_genai-0.2.0 → datarobot_genai-0.2.2}/src/datarobot_genai/drmcp/core/server_life_cycle.py +0 -0
  71. {datarobot_genai-0.2.0 → datarobot_genai-0.2.2}/src/datarobot_genai/drmcp/core/telemetry.py +0 -0
  72. {datarobot_genai-0.2.0 → datarobot_genai-0.2.2}/src/datarobot_genai/drmcp/core/tool_filter.py +0 -0
  73. {datarobot_genai-0.2.0 → datarobot_genai-0.2.2}/src/datarobot_genai/drmcp/core/utils.py +0 -0
  74. {datarobot_genai-0.2.0 → datarobot_genai-0.2.2}/src/datarobot_genai/drmcp/server.py +0 -0
  75. {datarobot_genai-0.2.0 → datarobot_genai-0.2.2}/src/datarobot_genai/drmcp/test_utils/__init__.py +0 -0
  76. {datarobot_genai-0.2.0 → datarobot_genai-0.2.2}/src/datarobot_genai/drmcp/test_utils/integration_mcp_server.py +0 -0
  77. {datarobot_genai-0.2.0 → datarobot_genai-0.2.2}/src/datarobot_genai/drmcp/test_utils/mcp_utils_integration.py +0 -0
  78. {datarobot_genai-0.2.0 → datarobot_genai-0.2.2}/src/datarobot_genai/drmcp/test_utils/openai_llm_mcp_client.py +0 -0
  79. {datarobot_genai-0.2.0 → datarobot_genai-0.2.2}/src/datarobot_genai/drmcp/test_utils/tool_base_ete.py +0 -0
  80. {datarobot_genai-0.2.0 → datarobot_genai-0.2.2}/src/datarobot_genai/drmcp/test_utils/utils.py +0 -0
  81. {datarobot_genai-0.2.0 → datarobot_genai-0.2.2}/src/datarobot_genai/drmcp/tools/__init__.py +0 -0
  82. {datarobot_genai-0.2.0 → datarobot_genai-0.2.2}/src/datarobot_genai/drmcp/tools/predictive/__init__.py +0 -0
  83. {datarobot_genai-0.2.0 → datarobot_genai-0.2.2}/src/datarobot_genai/drmcp/tools/predictive/data.py +0 -0
  84. {datarobot_genai-0.2.0 → datarobot_genai-0.2.2}/src/datarobot_genai/drmcp/tools/predictive/deployment.py +0 -0
  85. {datarobot_genai-0.2.0 → datarobot_genai-0.2.2}/src/datarobot_genai/drmcp/tools/predictive/deployment_info.py +0 -0
  86. {datarobot_genai-0.2.0 → datarobot_genai-0.2.2}/src/datarobot_genai/drmcp/tools/predictive/model.py +0 -0
  87. {datarobot_genai-0.2.0 → datarobot_genai-0.2.2}/src/datarobot_genai/drmcp/tools/predictive/predict.py +0 -0
  88. {datarobot_genai-0.2.0 → datarobot_genai-0.2.2}/src/datarobot_genai/drmcp/tools/predictive/predict_realtime.py +0 -0
  89. {datarobot_genai-0.2.0 → datarobot_genai-0.2.2}/src/datarobot_genai/drmcp/tools/predictive/project.py +0 -0
  90. {datarobot_genai-0.2.0 → datarobot_genai-0.2.2}/src/datarobot_genai/drmcp/tools/predictive/training.py +0 -0
  91. {datarobot_genai-0.2.0 → datarobot_genai-0.2.2}/src/datarobot_genai/langgraph/__init__.py +0 -0
  92. {datarobot_genai-0.2.0 → datarobot_genai-0.2.2}/src/datarobot_genai/langgraph/agent.py +0 -0
  93. {datarobot_genai-0.2.0 → datarobot_genai-0.2.2}/src/datarobot_genai/langgraph/mcp.py +0 -0
  94. {datarobot_genai-0.2.0 → datarobot_genai-0.2.2}/src/datarobot_genai/llama_index/__init__.py +0 -0
  95. {datarobot_genai-0.2.0 → datarobot_genai-0.2.2}/src/datarobot_genai/llama_index/agent.py +0 -0
  96. {datarobot_genai-0.2.0 → datarobot_genai-0.2.2}/src/datarobot_genai/llama_index/base.py +0 -0
  97. {datarobot_genai-0.2.0 → datarobot_genai-0.2.2}/src/datarobot_genai/llama_index/mcp.py +0 -0
  98. {datarobot_genai-0.2.0 → datarobot_genai-0.2.2}/src/datarobot_genai/nat/__init__.py +0 -0
  99. {datarobot_genai-0.2.0 → datarobot_genai-0.2.2}/src/datarobot_genai/nat/agent.py +0 -0
  100. {datarobot_genai-0.2.0 → datarobot_genai-0.2.2}/src/datarobot_genai/nat/datarobot_llm_clients.py +0 -0
  101. {datarobot_genai-0.2.0 → datarobot_genai-0.2.2}/src/datarobot_genai/nat/datarobot_llm_providers.py +0 -0
  102. {datarobot_genai-0.2.0 → datarobot_genai-0.2.2}/src/datarobot_genai/py.typed +0 -0
@@ -1,6 +1,6 @@
1
1
  Metadata-Version: 2.4
2
2
  Name: datarobot-genai
3
- Version: 0.2.0
3
+ Version: 0.2.2
4
4
  Summary: Generic helpers for GenAI
5
5
  Project-URL: Homepage, https://github.com/datarobot-oss/datarobot-genai
6
6
  Author: DataRobot, Inc.
@@ -11,7 +11,7 @@ Requires-Python: <3.13,>=3.10
11
11
  Requires-Dist: ag-ui-protocol<0.2.0,>=0.1.9
12
12
  Requires-Dist: datarobot-drum<2.0.0,>=1.17.5
13
13
  Requires-Dist: datarobot-predict<2.0.0,>=1.13.2
14
- Requires-Dist: datarobot<4.0.0,>=3.9.1
14
+ Requires-Dist: datarobot<4.0.0,>=3.10.0
15
15
  Requires-Dist: openai<2.0.0,>=1.76.2
16
16
  Requires-Dist: opentelemetry-instrumentation-aiohttp-client<1.0.0,>=0.43b0
17
17
  Requires-Dist: opentelemetry-instrumentation-httpx<1.0.0,>=0.43b0
@@ -4,7 +4,7 @@ build-backend = "hatchling.build"
4
4
 
5
5
  [project]
6
6
  name = "datarobot-genai"
7
- version = "0.2.0"
7
+ version = "0.2.2"
8
8
  description = "Generic helpers for GenAI"
9
9
  readme = "README.md"
10
10
  requires-python = ">=3.10, <3.13"
@@ -12,7 +12,7 @@ license = { text = "Apache-2.0" }
12
12
  authors = [{ name = "DataRobot, Inc." }]
13
13
  dependencies = [
14
14
  "requests>=2.32.4,<3.0.0",
15
- "datarobot>=3.9.1,<4.0.0",
15
+ "datarobot>=3.10.0,<4.0.0",
16
16
  "datarobot-predict>=1.13.2,<2.0.0",
17
17
  "openai>=1.76.2,<2.0.0",
18
18
  "pandas>=2.2.3,<3.0.0",
@@ -115,6 +115,9 @@ class DataRobotMCPServer:
115
115
  self._mcp = mcp
116
116
  self._mcp_transport = transport
117
117
 
118
+ # Configure MCP server capabilities
119
+ self._configure_mcp_capabilities()
120
+
118
121
  # Initialize telemetry
119
122
  initialize_telemetry(mcp)
120
123
 
@@ -163,6 +166,37 @@ class DataRobotMCPServer:
163
166
  if transport == "streamable-http":
164
167
  register_routes(self._mcp)
165
168
 
169
+ def _configure_mcp_capabilities(self) -> None:
170
+ """Configure MCP capabilities that FastMCP doesn't expose directly.
171
+
172
+ See: https://github.com/modelcontextprotocol/python-sdk/issues/1126
173
+ """
174
+ server = self._mcp._mcp_server
175
+
176
+ # Declare prompts_changed capability (capabilities.prompts.listChanged: true)
177
+ server.notification_options.prompts_changed = True
178
+
179
+ # Declare experimental capabilities ( experimental.dynamic_prompts: true)
180
+ server.experimental_capabilities = {"dynamic_prompts": {"enabled": True}}
181
+
182
+ # Patch to include experimental_capabilities (FastMCP doesn't expose this)
183
+ original = server.create_initialization_options
184
+
185
+ def patched(
186
+ notification_options: Any = None,
187
+ experimental_capabilities: dict[str, dict[str, Any]] | None = None,
188
+ **kwargs: Any,
189
+ ) -> Any:
190
+ if experimental_capabilities is None:
191
+ experimental_capabilities = getattr(server, "experimental_capabilities", None)
192
+ return original(
193
+ notification_options=notification_options,
194
+ experimental_capabilities=experimental_capabilities,
195
+ **kwargs,
196
+ )
197
+
198
+ server.create_initialization_options = patched
199
+
166
200
  def run(self, show_banner: bool = False) -> None:
167
201
  """Run the DataRobot MCP server synchronously."""
168
202
  try:
@@ -0,0 +1,70 @@
1
+ # Copyright 2025 DataRobot, Inc.
2
+ #
3
+ # Licensed under the Apache License, Version 2.0 (the "License");
4
+ # you may not use this file except in compliance with the License.
5
+ # You may obtain a copy of the License at
6
+ #
7
+ # http://www.apache.org/licenses/LICENSE-2.0
8
+ #
9
+ # Unless required by applicable law or agreed to in writing, software
10
+ # distributed under the License is distributed on an "AS IS" BASIS,
11
+ # WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
12
+ # See the License for the specific language governing permissions and
13
+ # limitations under the License.
14
+ from collections import defaultdict
15
+
16
+ import datarobot as dr
17
+
18
+ from datarobot_genai.drmcp.core.clients import get_api_client
19
+
20
+
21
+ def get_datarobot_prompt_templates() -> list[dr.genai.PromptTemplate]:
22
+ try:
23
+ return dr.genai.PromptTemplate.list()
24
+ except Exception:
25
+ return []
26
+
27
+
28
+ def get_datarobot_prompt_template_versions(
29
+ prompt_template_ids: list[str],
30
+ ) -> dict[str, list[dr.genai.PromptTemplateVersion]]:
31
+ # Still missing in SDK
32
+ prompt_template_versions_data = dr.utils.pagination.unpaginate(
33
+ initial_url="genai/promptTemplates/versions/",
34
+ initial_params={
35
+ "promptTemplateIds": prompt_template_ids,
36
+ },
37
+ client=get_api_client(),
38
+ )
39
+ prompt_template_versions = defaultdict(list)
40
+ for prompt_template_version in prompt_template_versions_data:
41
+ prompt_template_versions[prompt_template_version["promptTemplateId"]].append(
42
+ dr.genai.PromptTemplateVersion(
43
+ id=prompt_template_version["id"],
44
+ prompt_template_id=prompt_template_version["promptTemplateId"],
45
+ prompt_text=prompt_template_version["promptText"],
46
+ commit_comment=prompt_template_version["commitComment"],
47
+ version=prompt_template_version["version"],
48
+ variables=prompt_template_version["variables"],
49
+ creation_date=prompt_template_version["creationDate"],
50
+ creation_user_id=prompt_template_version["creationUserId"],
51
+ user_name=prompt_template_version["userName"],
52
+ )
53
+ )
54
+ return prompt_template_versions
55
+
56
+
57
+ def get_datarobot_prompt_template(prompt_template_id: str) -> dr.genai.PromptTemplate | None:
58
+ try:
59
+ return dr.genai.PromptTemplate.get(prompt_template_id)
60
+ except Exception:
61
+ return None
62
+
63
+
64
+ def get_datarobot_prompt_template_version(
65
+ prompt_template_id: str, prompt_template_version_id: str
66
+ ) -> dr.genai.PromptTemplateVersion | None:
67
+ try:
68
+ return dr.genai.PromptTemplateVersion.get(prompt_template_id, prompt_template_version_id)
69
+ except Exception:
70
+ return None
@@ -18,15 +18,13 @@ from collections.abc import Callable
18
18
  from inspect import Parameter
19
19
  from inspect import Signature
20
20
 
21
+ import datarobot as dr
21
22
  from fastmcp.prompts.prompt import Prompt
22
23
  from pydantic import Field
23
24
 
24
25
  from datarobot_genai.drmcp.core.exceptions import DynamicPromptRegistrationError
25
26
  from datarobot_genai.drmcp.core.mcp_instance import register_prompt
26
27
 
27
- from .dr_lib import DrPrompt
28
- from .dr_lib import DrPromptVersion
29
- from .dr_lib import DrVariable
30
28
  from .dr_lib import get_datarobot_prompt_template_versions
31
29
  from .dr_lib import get_datarobot_prompt_templates
32
30
 
@@ -57,7 +55,8 @@ async def register_prompts_from_datarobot_prompt_management() -> None:
57
55
 
58
56
 
59
57
  async def register_prompt_from_datarobot_prompt_management(
60
- prompt_template: DrPrompt, prompt_template_version: DrPromptVersion | None = None
58
+ prompt_template: dr.genai.PromptTemplate,
59
+ prompt_template_version: dr.genai.PromptTemplateVersion | None = None,
61
60
  ) -> Prompt:
62
61
  """Register a single prompt.
63
62
 
@@ -173,7 +172,7 @@ def to_valid_mcp_prompt_name(s: str) -> str:
173
172
 
174
173
 
175
174
  def make_prompt_function(
176
- name: str, description: str, prompt_text: str, variables: list[DrVariable]
175
+ name: str, description: str, prompt_text: str, variables: list[dr.genai.Variable]
177
176
  ) -> Callable:
178
177
  params = []
179
178
  for v in variables:
@@ -22,6 +22,7 @@ from fastmcp import Context
22
22
  from fastmcp import FastMCP
23
23
  from fastmcp.exceptions import NotFoundError
24
24
  from fastmcp.prompts.prompt import Prompt
25
+ from fastmcp.server.dependencies import get_context
25
26
  from fastmcp.tools import FunctionTool
26
27
  from fastmcp.tools import Tool
27
28
  from fastmcp.utilities.types import NotSet
@@ -91,6 +92,34 @@ class TaggedFastMCP(FastMCP):
91
92
  self._deployments_map: dict[str, str] = {}
92
93
  self._prompts_map: dict[str, tuple[str, str]] = {}
93
94
 
95
+ async def notify_prompts_changed(self) -> None:
96
+ """
97
+ Notify connected clients that the prompt list has changed.
98
+
99
+ This method attempts to send a prompts/list_changed notification to inform
100
+ clients that they should refresh their prompt list.
101
+
102
+ Note: In stateless HTTP mode (default for this server), notifications may not
103
+ reach clients since each request is independent. This method still logs the
104
+ change for auditing purposes and will work if the server is configured for
105
+ stateful connections.
106
+
107
+ See: https://github.com/modelcontextprotocol/python-sdk/issues/710
108
+ """
109
+ logger.info("Prompt list changed - attempting to notify connected clients")
110
+
111
+ # Try to use FastMCP's built-in notification mechanism if in an MCP context
112
+ try:
113
+ context = get_context()
114
+ context._queue_prompt_list_changed()
115
+ logger.debug("Queued prompts_changed notification via MCP context")
116
+ except RuntimeError:
117
+ # No active MCP context - this is expected when called from REST API
118
+ logger.debug(
119
+ "No active MCP context for notification. "
120
+ "In stateless mode, clients will see changes on next request."
121
+ )
122
+
94
123
  @overload
95
124
  def tool(
96
125
  self,
@@ -286,6 +315,9 @@ class TaggedFastMCP(FastMCP):
286
315
  f"already mapped to {existing_prompt_template_version_id}. "
287
316
  f"Updating to version id = {prompt_template_version_id} and name = {prompt_name}"
288
317
  )
318
+ await self.remove_prompt_mapping(
319
+ prompt_template_id, existing_prompt_template_version_id
320
+ )
289
321
 
290
322
  self._prompts_map[prompt_template_id] = (prompt_template_version_id, prompt_name)
291
323
 
@@ -308,7 +340,7 @@ class TaggedFastMCP(FastMCP):
308
340
  f"skipping removal."
309
341
  )
310
342
  else:
311
- prompts_d = await mcp.get_prompts()
343
+ prompts_d = await self.get_prompts()
312
344
  for prompt in prompts_d.values():
313
345
  if (
314
346
  prompt.meta is not None
@@ -319,6 +351,9 @@ class TaggedFastMCP(FastMCP):
319
351
  prompt.disable()
320
352
 
321
353
  self._prompts_map.pop(prompt_template_id, None)
354
+
355
+ # Notify clients that the prompt list has changed
356
+ await self.notify_prompts_changed()
322
357
  else:
323
358
  logger.debug(
324
359
  f"Do not found prompt template with id = {prompt_template_id} in registry, "
@@ -526,17 +561,21 @@ async def register_prompt(
526
561
  )
527
562
 
528
563
  # Register the prompt
529
- registered_prompt = mcp.add_prompt(prompt)
530
564
  if prompt_template:
531
565
  prompt_template_id, prompt_template_version_id = prompt_template
532
566
  await mcp.set_prompt_mapping(
533
567
  prompt_template_id, prompt_template_version_id, prompt_name_no_duplicate
534
568
  )
535
569
 
570
+ registered_prompt = mcp.add_prompt(prompt)
571
+
536
572
  # Verify prompt is registered
537
573
  prompts = await mcp.get_prompts()
538
574
  if not any(prompt.name == prompt_name_no_duplicate for prompt in prompts.values()):
539
575
  raise RuntimeError(f"Prompt {prompt_name_no_duplicate} was not registered successfully")
540
576
  logger.info(f"Registered prompts: {len(prompts)}")
541
577
 
578
+ # Notify clients that the prompt list has changed
579
+ await mcp.notify_prompts_changed()
580
+
542
581
  return registered_prompt
@@ -428,7 +428,10 @@ def register_routes(mcp: TaggedFastMCP) -> None:
428
428
  """Refresh prompt templates."""
429
429
  try:
430
430
  await refresh_registered_prompt_template()
431
- return JSONResponse(status_code=HTTPStatus.NO_CONTENT, content=None)
431
+ return JSONResponse(
432
+ status_code=HTTPStatus.OK,
433
+ content={"message": "Prompts refreshed successfully"},
434
+ )
432
435
  except Exception as e:
433
436
  return JSONResponse(
434
437
  status_code=HTTPStatus.INTERNAL_SERVER_ERROR,
@@ -16,6 +16,8 @@ import os
16
16
  from collections.abc import AsyncGenerator
17
17
  from contextlib import asynccontextmanager
18
18
 
19
+ import aiohttp
20
+ from aiohttp import ClientSession as HttpClientSession
19
21
  from mcp import ClientSession
20
22
  from mcp.client.streamable_http import streamablehttp_client
21
23
 
@@ -29,6 +31,11 @@ def get_dr_mcp_server_url() -> str | None:
29
31
  return os.environ.get("DR_MCP_SERVER_URL")
30
32
 
31
33
 
34
+ def get_dr_mcp_server_http_url() -> str | None:
35
+ """Get DataRobot MCP server http URL."""
36
+ return os.environ.get("DR_MCP_SERVER_HTTP_URL")
37
+
38
+
32
39
  def get_openai_llm_client_config() -> dict[str, str]:
33
40
  """Get OpenAI LLM client configuration."""
34
41
  openai_api_key = os.environ.get("OPENAI_API_KEY")
@@ -94,3 +101,25 @@ async def ete_test_mcp_session(
94
101
  yield session
95
102
  except asyncio.TimeoutError:
96
103
  raise TimeoutError(f"Check if the MCP server is running at {get_dr_mcp_server_url()}")
104
+
105
+
106
+ @asynccontextmanager
107
+ async def ete_test_http_session(
108
+ additional_headers: dict[str, str] | None = None,
109
+ ) -> AsyncGenerator[HttpClientSession, None]:
110
+ """Create an HTTP session for each test that can connect to MCP custom http routes.
111
+
112
+ Parameters
113
+ ----------
114
+ additional_headers : dict[str, str], optional
115
+ Additional headers to include in the HTTP session (e.g., auth headers for testing).
116
+ """
117
+ headers = get_headers()
118
+ if additional_headers:
119
+ headers.update(additional_headers)
120
+
121
+ async with ete_test_mcp_session(additional_headers=additional_headers):
122
+ async with aiohttp.ClientSession(
123
+ base_url=get_dr_mcp_server_http_url(), headers=headers
124
+ ) as client:
125
+ yield client
@@ -1,128 +0,0 @@
1
- # Copyright 2025 DataRobot, Inc.
2
- #
3
- # Licensed under the Apache License, Version 2.0 (the "License");
4
- # you may not use this file except in compliance with the License.
5
- # You may obtain a copy of the License at
6
- #
7
- # http://www.apache.org/licenses/LICENSE-2.0
8
- #
9
- # Unless required by applicable law or agreed to in writing, software
10
- # distributed under the License is distributed on an "AS IS" BASIS,
11
- # WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
12
- # See the License for the specific language governing permissions and
13
- # limitations under the License.
14
- from collections import defaultdict
15
- from dataclasses import dataclass
16
-
17
- import datarobot as dr
18
-
19
- from datarobot_genai.drmcp.core.clients import get_api_client
20
-
21
- # Needed SDK version (3.10.0) is not published yet. We'll reimplement simplified version of it.
22
- # get_datarobot_prompt_templates = dr.genai.PromptTemplate.list()
23
- # DrPrompt = dr.genai.PromptTemplate
24
- # DrPromptVersion = dr.genai.PromptTemplateVersion
25
- # DrVariable = dr.genai.Variable
26
-
27
-
28
- @dataclass
29
- class DrVariable:
30
- name: str
31
- description: str
32
-
33
-
34
- @dataclass
35
- class DrPromptVersion:
36
- id: str
37
- prompt_template_id: str
38
- version: int
39
- prompt_text: str
40
- variables: list[DrVariable]
41
-
42
- @classmethod
43
- def from_dict(cls, d: dict) -> "DrPromptVersion":
44
- variables = [
45
- DrVariable(name=v["name"], description=v["description"]) for v in d["variables"]
46
- ]
47
- return cls(
48
- id=d["id"],
49
- prompt_template_id=d["promptTemplateId"],
50
- version=d["version"],
51
- prompt_text=d["promptText"],
52
- variables=variables,
53
- )
54
-
55
-
56
- @dataclass
57
- class DrPrompt:
58
- id: str
59
- name: str
60
- description: str
61
-
62
- def get_latest_version(self) -> DrPromptVersion | None:
63
- all_prompt_template_versions = get_datarobot_prompt_template_versions([self.id])
64
- prompt_template_versions = all_prompt_template_versions.get(self.id)
65
-
66
- if not prompt_template_versions:
67
- return None
68
- latest_version = max(prompt_template_versions, key=lambda v: v.version)
69
- return latest_version
70
-
71
- @classmethod
72
- def from_dict(cls, d: dict) -> "DrPrompt":
73
- return cls(id=d["id"], name=d["name"], description=d["description"])
74
-
75
-
76
- def get_datarobot_prompt_templates() -> list[DrPrompt]:
77
- prompt_templates_data = dr.utils.pagination.unpaginate(
78
- initial_url="genai/promptTemplates/", initial_params={}, client=get_api_client()
79
- )
80
-
81
- return [DrPrompt.from_dict(prompt_template) for prompt_template in prompt_templates_data]
82
-
83
-
84
- def get_datarobot_prompt_template_versions(
85
- prompt_template_ids: list[str],
86
- ) -> dict[str, list[DrPromptVersion]]:
87
- prompt_template_versions_data = dr.utils.pagination.unpaginate(
88
- initial_url="genai/promptTemplates/versions/",
89
- initial_params={
90
- "promptTemplateIds": prompt_template_ids,
91
- },
92
- client=get_api_client(),
93
- )
94
- prompt_template_versions = defaultdict(list)
95
- for prompt_template_version in prompt_template_versions_data:
96
- prompt_template_versions[prompt_template_version["promptTemplateId"]].append(
97
- DrPromptVersion.from_dict(prompt_template_version)
98
- )
99
- return prompt_template_versions
100
-
101
-
102
- def get_datarobot_prompt_template(prompt_template_id: str) -> DrPrompt | None:
103
- api_client = get_api_client()
104
- try:
105
- prompt_template_response = api_client.get(
106
- f"genai/promptTemplates/{prompt_template_id}/", join_endpoint=True
107
- )
108
- prompt_template_json = prompt_template_response.json()
109
- except Exception:
110
- return None
111
-
112
- return DrPrompt.from_dict(prompt_template_json)
113
-
114
-
115
- def get_datarobot_prompt_template_version(
116
- prompt_template_id: str, prompt_template_version_id: str
117
- ) -> DrPromptVersion | None:
118
- api_client = get_api_client()
119
- try:
120
- prompt_template_version_response = api_client.get(
121
- f"genai/promptTemplates/{prompt_template_id}/versions/{prompt_template_version_id}/",
122
- join_endpoint=True,
123
- )
124
- prompt_template_version_json = prompt_template_version_response.json()
125
- except Exception:
126
- return None
127
-
128
- return DrPromptVersion.from_dict(prompt_template_version_json)
File without changes
File without changes