datarobot-genai 0.2.0__py3-none-any.whl → 0.2.1__py3-none-any.whl

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.
@@ -12,78 +12,23 @@
12
12
  # See the License for the specific language governing permissions and
13
13
  # limitations under the License.
14
14
  from collections import defaultdict
15
- from dataclasses import dataclass
16
15
 
17
16
  import datarobot as dr
18
17
 
19
18
  from datarobot_genai.drmcp.core.clients import get_api_client
20
19
 
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
20
 
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]
21
+ def get_datarobot_prompt_templates() -> list[dr.genai.PromptTemplate]:
22
+ try:
23
+ return dr.genai.PromptTemplate.list()
24
+ except Exception:
25
+ return []
82
26
 
83
27
 
84
28
  def get_datarobot_prompt_template_versions(
85
29
  prompt_template_ids: list[str],
86
- ) -> dict[str, list[DrPromptVersion]]:
30
+ ) -> dict[str, list[dr.genai.PromptTemplateVersion]]:
31
+ # Still missing in SDK
87
32
  prompt_template_versions_data = dr.utils.pagination.unpaginate(
88
33
  initial_url="genai/promptTemplates/versions/",
89
34
  initial_params={
@@ -94,35 +39,32 @@ def get_datarobot_prompt_template_versions(
94
39
  prompt_template_versions = defaultdict(list)
95
40
  for prompt_template_version in prompt_template_versions_data:
96
41
  prompt_template_versions[prompt_template_version["promptTemplateId"]].append(
97
- DrPromptVersion.from_dict(prompt_template_version)
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
+ )
98
53
  )
99
54
  return prompt_template_versions
100
55
 
101
56
 
102
- def get_datarobot_prompt_template(prompt_template_id: str) -> DrPrompt | None:
103
- api_client = get_api_client()
57
+ def get_datarobot_prompt_template(prompt_template_id: str) -> dr.genai.PromptTemplate | None:
104
58
  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()
59
+ return dr.genai.PromptTemplate.get(prompt_template_id)
109
60
  except Exception:
110
61
  return None
111
62
 
112
- return DrPrompt.from_dict(prompt_template_json)
113
-
114
63
 
115
64
  def get_datarobot_prompt_template_version(
116
65
  prompt_template_id: str, prompt_template_version_id: str
117
- ) -> DrPromptVersion | None:
118
- api_client = get_api_client()
66
+ ) -> dr.genai.PromptTemplateVersion | None:
119
67
  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()
68
+ return dr.genai.PromptTemplateVersion.get(prompt_template_id, prompt_template_version_id)
125
69
  except Exception:
126
70
  return None
127
-
128
- return DrPromptVersion.from_dict(prompt_template_version_json)
@@ -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:
@@ -286,6 +286,9 @@ class TaggedFastMCP(FastMCP):
286
286
  f"already mapped to {existing_prompt_template_version_id}. "
287
287
  f"Updating to version id = {prompt_template_version_id} and name = {prompt_name}"
288
288
  )
289
+ await self.remove_prompt_mapping(
290
+ prompt_template_id, existing_prompt_template_version_id
291
+ )
289
292
 
290
293
  self._prompts_map[prompt_template_id] = (prompt_template_version_id, prompt_name)
291
294
 
@@ -526,13 +529,14 @@ async def register_prompt(
526
529
  )
527
530
 
528
531
  # Register the prompt
529
- registered_prompt = mcp.add_prompt(prompt)
530
532
  if prompt_template:
531
533
  prompt_template_id, prompt_template_version_id = prompt_template
532
534
  await mcp.set_prompt_mapping(
533
535
  prompt_template_id, prompt_template_version_id, prompt_name_no_duplicate
534
536
  )
535
537
 
538
+ registered_prompt = mcp.add_prompt(prompt)
539
+
536
540
  # Verify prompt is registered
537
541
  prompts = await mcp.get_prompts()
538
542
  if not any(prompt.name == prompt_name_no_duplicate for prompt in prompts.values()):
@@ -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,6 +1,6 @@
1
1
  Metadata-Version: 2.4
2
2
  Name: datarobot-genai
3
- Version: 0.2.0
3
+ Version: 0.2.1
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
@@ -35,9 +35,9 @@ datarobot_genai/drmcp/core/dr_mcp_server.py,sha256=7mu5UXHQmKNbIpNoQE0lPJaUI7AZa
35
35
  datarobot_genai/drmcp/core/dr_mcp_server_logo.py,sha256=hib-nfR1SNTW6CnpFsFCkL9H_OMwa4YYyinV7VNOuLk,4708
36
36
  datarobot_genai/drmcp/core/exceptions.py,sha256=eqsGI-lxybgvWL5w4BFhbm3XzH1eU5tetwjnhJxelpc,905
37
37
  datarobot_genai/drmcp/core/logging.py,sha256=Y_hig4eBWiXGaVV7B_3wBcaYVRNH4ydptbEQhrP9-mY,3414
38
- datarobot_genai/drmcp/core/mcp_instance.py,sha256=wMsP39xqTmNBYqd49olEQb5UHTSsxj6BOIoIElorRB0,19235
38
+ datarobot_genai/drmcp/core/mcp_instance.py,sha256=9hw5GrAltYxlOoc-l6S0Eb9ppJVhqx-RvD7OAkdgHjc,19368
39
39
  datarobot_genai/drmcp/core/mcp_server_tools.py,sha256=odNZKozfx0VV38SLZHw9lY0C0JM_JnRI06W3BBXnyE4,4278
40
- datarobot_genai/drmcp/core/routes.py,sha256=nrgzYkiWWcwSy0BUJ-k5AuXdUHNrnGRudCo4AhtYvlY,17856
40
+ datarobot_genai/drmcp/core/routes.py,sha256=dqE2M0UzAyyN9vQjlyTjYW4rpju3LT039po5weuO__I,17936
41
41
  datarobot_genai/drmcp/core/routes_utils.py,sha256=vSseXWlplMSnRgoJgtP_rHxWSAVYcx_tpTv4lyTpQoc,944
42
42
  datarobot_genai/drmcp/core/server_life_cycle.py,sha256=WKGJWGxalvqxupzJ2y67Kklc_9PgpZT0uyjlv_sr5wc,3419
43
43
  datarobot_genai/drmcp/core/telemetry.py,sha256=NEkSTC1w6uQgtukLHI-sWvR4EMgInysgATcvfQ5CplM,15378
@@ -45,8 +45,8 @@ datarobot_genai/drmcp/core/tool_filter.py,sha256=tLOcG50QBvS48cOVHM6OqoODYiiS6Ke
45
45
  datarobot_genai/drmcp/core/utils.py,sha256=dSjrayWVcnC5GxQcvOIOSHaoEymPIVtG_s2ZBMlmSOw,4336
46
46
  datarobot_genai/drmcp/core/dynamic_prompts/__init__.py,sha256=y4yapzp3KnFMzSR6HlNDS4uSuyNT7I1iPBvaCLsS0sU,577
47
47
  datarobot_genai/drmcp/core/dynamic_prompts/controllers.py,sha256=AGJlKqgHRO0Kd7Gl-Ulw9KYBgzjTTFXWBvOUF-SuKUI,5454
48
- datarobot_genai/drmcp/core/dynamic_prompts/dr_lib.py,sha256=IEdD2Gqm4SfUdiXJB99RiWxkN6frGaxJ2SfATetMM3c,4243
49
- datarobot_genai/drmcp/core/dynamic_prompts/register.py,sha256=5AEh1m8GX-gPZHUdiE1VATt7IKJQk-eThcxh01sWn0I,7204
48
+ datarobot_genai/drmcp/core/dynamic_prompts/dr_lib.py,sha256=4j33AKmq7kQX_EE2_RWAbP8-K5KPVEvpUginTWn_MHs,2701
49
+ datarobot_genai/drmcp/core/dynamic_prompts/register.py,sha256=2c-vBaTfu3mq_8tSFfDAzG5hG06uS9CghIC1sJxHRNw,7173
50
50
  datarobot_genai/drmcp/core/dynamic_prompts/utils.py,sha256=BZ3792AgfvYlwL0_J0MzQfGecyEA5_OKUMynEZYzCds,1136
51
51
  datarobot_genai/drmcp/core/dynamic_tools/__init__.py,sha256=0kq9vMkF7EBsS6lkEdiLibmUrghTQqosHbZ5k-V9a5g,578
52
52
  datarobot_genai/drmcp/core/dynamic_tools/register.py,sha256=3M5-F0mhUYTZJWmFDmqzsj3QAd7ut7b0kPv-JZyaTzg,9204
@@ -67,7 +67,7 @@ datarobot_genai/drmcp/core/memory_management/manager.py,sha256=gmc_SQs12YQFMWl2U
67
67
  datarobot_genai/drmcp/core/memory_management/memory_tools.py,sha256=AxzpwOlldmhhDfKZcAxaGs7Xih2SCe0XbQuXX5nQczI,6397
68
68
  datarobot_genai/drmcp/test_utils/__init__.py,sha256=y4yapzp3KnFMzSR6HlNDS4uSuyNT7I1iPBvaCLsS0sU,577
69
69
  datarobot_genai/drmcp/test_utils/integration_mcp_server.py,sha256=MdoR7r3m9uT7crodyhY69yhkrM7Thpe__BBD9lB_2oA,3328
70
- datarobot_genai/drmcp/test_utils/mcp_utils_ete.py,sha256=rvLOePXF9epIEksJzTOqdL9-fGn21bRLbK3yzpvi-6E,3430
70
+ datarobot_genai/drmcp/test_utils/mcp_utils_ete.py,sha256=rgZkPF26YCHX2FGppWE4v22l_NQ3kLSPSUimO0tD4nM,4402
71
71
  datarobot_genai/drmcp/test_utils/mcp_utils_integration.py,sha256=0sU29Khal0CelnHBDInyTRiuPKrFFbTbIomOoUbyMhs,3271
72
72
  datarobot_genai/drmcp/test_utils/openai_llm_mcp_client.py,sha256=Va3_5c2ToZyfIsEjK2ef5d3z-FA5SE51voikvjKPt8Q,8837
73
73
  datarobot_genai/drmcp/test_utils/tool_base_ete.py,sha256=-mKHBkGkyOKQCVS2LHFhSnRofIqJBbeAPRkwizBDtTg,6104
@@ -93,9 +93,9 @@ datarobot_genai/nat/__init__.py,sha256=47DEQpj8HBSa-_TImW-5JCeuQeRkm5NMpJWZG3hSu
93
93
  datarobot_genai/nat/agent.py,sha256=siBLDWAff2-JwZ8Q3iNpM_e4_IoSwG9IvY0hyEjNenw,10292
94
94
  datarobot_genai/nat/datarobot_llm_clients.py,sha256=STzAZ4OF8U-Y_cUTywxmKBGVotwsnbGP6vTojnu6q0g,9921
95
95
  datarobot_genai/nat/datarobot_llm_providers.py,sha256=aDoQcTeGI-odqydPXEX9OGGNFbzAtpqzTvHHEkmJuEQ,4963
96
- datarobot_genai-0.2.0.dist-info/METADATA,sha256=Qfh60pUsGehZwD7PkQNDLxItoJIkH12bZBZx95F_Ujw,5941
97
- datarobot_genai-0.2.0.dist-info/WHEEL,sha256=WLgqFyCfm_KASv4WHyYy0P3pM_m7J5L9k2skdKLirC8,87
98
- datarobot_genai-0.2.0.dist-info/entry_points.txt,sha256=CZhmZcSyt_RBltgLN_b9xasJD6J5SaDc_z7K0wuOY9Y,150
99
- datarobot_genai-0.2.0.dist-info/licenses/AUTHORS,sha256=isJGUXdjq1U7XZ_B_9AH8Qf0u4eX0XyQifJZ_Sxm4sA,80
100
- datarobot_genai-0.2.0.dist-info/licenses/LICENSE,sha256=U2_VkLIktQoa60Nf6Tbt7E4RMlfhFSjWjcJJfVC-YCE,11341
101
- datarobot_genai-0.2.0.dist-info/RECORD,,
96
+ datarobot_genai-0.2.1.dist-info/METADATA,sha256=iucgLchfePrXrkSZslI9AAxEwQjYxvDpeSnG-_ZtyrQ,5942
97
+ datarobot_genai-0.2.1.dist-info/WHEEL,sha256=WLgqFyCfm_KASv4WHyYy0P3pM_m7J5L9k2skdKLirC8,87
98
+ datarobot_genai-0.2.1.dist-info/entry_points.txt,sha256=CZhmZcSyt_RBltgLN_b9xasJD6J5SaDc_z7K0wuOY9Y,150
99
+ datarobot_genai-0.2.1.dist-info/licenses/AUTHORS,sha256=isJGUXdjq1U7XZ_B_9AH8Qf0u4eX0XyQifJZ_Sxm4sA,80
100
+ datarobot_genai-0.2.1.dist-info/licenses/LICENSE,sha256=U2_VkLIktQoa60Nf6Tbt7E4RMlfhFSjWjcJJfVC-YCE,11341
101
+ datarobot_genai-0.2.1.dist-info/RECORD,,