bohr-agent-sdk 0.1.110__py3-none-any.whl → 0.1.111__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.
@@ -1,6 +1,6 @@
1
1
  Metadata-Version: 2.4
2
2
  Name: bohr-agent-sdk
3
- Version: 0.1.110
3
+ Version: 0.1.111
4
4
  Summary: SDK for scientific agents
5
5
  Home-page: https://github.com/dptech-corp/bohr-agent-sdk/
6
6
  Author: DP Technology
@@ -51,7 +51,7 @@ dp/agent/cli/templates/ui/server/session_manager.py,sha256=ZbNHGCFvswa-LKWn6c6RM
51
51
  dp/agent/cli/templates/ui/server/user_files.py,sha256=khkiyY2UOOysHqO6JgCPUDqtrInp83G1M62i3Lj-0aY,2995
52
52
  dp/agent/cli/templates/ui/server/utils.py,sha256=f4NfwFBq_RdZyFn_KCW6ZThYW8TvQyVruK7PJZ-DA80,1530
53
53
  dp/agent/client/__init__.py,sha256=yu7HYZwAkD7g5dL9JttLkGmspBcyOf-6OoCjci4oPDA,59
54
- dp/agent/client/mcp_client.py,sha256=xp8GSM5effjjp34yGp5-HwldalyqT-CWb6PxlqkclCA,6313
54
+ dp/agent/client/mcp_client.py,sha256=glbQa-fv2aOBhv_GC2ldwyWHOVcSIEwtVLAjzLZvp0c,6289
55
55
  dp/agent/cloud/__init__.py,sha256=e16ymCZX2f-S8DyGB5jSK8gnQqVObRIsvtLXLALIKxQ,441
56
56
  dp/agent/cloud/main.py,sha256=5QIEjpZ1RxWnR8wyLf-vlgz1bn9oOnxCYn158LBaLN4,727
57
57
  dp/agent/cloud/mcp.py,sha256=tsAwC3doVMLYr6Oh8PxVqF-qCygYkDZJTIhoF_h8eGQ,4537
@@ -62,12 +62,12 @@ dp/agent/device/device/__init__.py,sha256=w7_1S16S1vWUq0RGl0GFgjq2vFkc5oNvy8cQTn
62
62
  dp/agent/device/device/device.py,sha256=9ZRIJth-4qMO-i-u_b_cO3d6a4eTbTQjPaxFsV_zEkc,9643
63
63
  dp/agent/device/device/types.py,sha256=JuxB-hjf1CjjvfBxCLwRAXVFlYS-nPEdiJpBWLFVCzo,1924
64
64
  dp/agent/server/__init__.py,sha256=rckaYd8pbYyB4ENEhgjXKeGMXjdnrgcJpdM1gu5u1Wc,508
65
- dp/agent/server/calculation_mcp_server.py,sha256=hsTxuguyqgq_4HSeJYS7VnW-jTvqf9DI0mpvRj0wK3w,13566
65
+ dp/agent/server/calculation_mcp_server.py,sha256=vvsf58aKbBtH0AqrG5_qhGqg5g2nEhmXgabDzZKpa6o,18534
66
66
  dp/agent/server/preprocessor.py,sha256=XUWu7QOwo_sIDMYS2b1OTrM33EXEVH_73vk-ju1Ok8A,1264
67
67
  dp/agent/server/utils.py,sha256=ui3lca9EagcGqmYf8BKLsPARIzXxJ3jgN98yuEO3OSQ,1668
68
68
  dp/agent/server/executor/__init__.py,sha256=s95M5qKQk39Yi9qaVJZhk_nfj54quSf7EDghR3OCFUA,248
69
69
  dp/agent/server/executor/base_executor.py,sha256=nR2jI-wFvKoOk8QaK11pnSAkHj2MsE6uyzPWDx-vgJA,3018
70
- dp/agent/server/executor/dispatcher_executor.py,sha256=-P1HAuHDbfLi-3mkT29I84x2aMvmXiM2h-qfZGEobjI,10872
70
+ dp/agent/server/executor/dispatcher_executor.py,sha256=CZRxbVkLaDvStXhNaMKrKcx2Z0tPPVzIxkU1ufqWgYc,12081
71
71
  dp/agent/server/executor/local_executor.py,sha256=pGXlDOrfjfP40hSMzbF-Wls3OnOTsH8PdkQjcDjP6_w,6580
72
72
  dp/agent/server/storage/__init__.py,sha256=Sgsyp5hb0_hhIGugAPfQFzBHt_854rS_MuMuE3sn8Gs,389
73
73
  dp/agent/server/storage/base_storage.py,sha256=728-oNG6N8isV95gZVnyi4vTznJPJhSjxw9Gl5Y_y5o,2356
@@ -75,8 +75,8 @@ dp/agent/server/storage/bohrium_storage.py,sha256=EsKX4dWWvZTn2TEhZv4zsvihfDK0mm
75
75
  dp/agent/server/storage/http_storage.py,sha256=KiySq7g9-iJr12XQCKKyJLn8wJoDnSRpQAR5_qPJ1ZU,1471
76
76
  dp/agent/server/storage/local_storage.py,sha256=t1wfjByjXew9ws3PuUxWxmZQ0-Wt1a6t4wmj3fW62GI,1352
77
77
  dp/agent/server/storage/oss_storage.py,sha256=pgjmi7Gir3Y5wkMDCvU4fvSls15fXT7Ax-h9MYHFPK0,3359
78
- bohr_agent_sdk-0.1.110.dist-info/METADATA,sha256=zqodSHxsxgJ5AYkjncOWOL4qA5hWhiY8Sx1AvPOfMO0,11070
79
- bohr_agent_sdk-0.1.110.dist-info/WHEEL,sha256=_zCd3N1l69ArxyTb8rzEoP9TpbYXkqRFSNOD5OuxnTs,91
80
- bohr_agent_sdk-0.1.110.dist-info/entry_points.txt,sha256=5n5kneF5IbDQtoQ2WfF-QuBjDtsimJte9Rv9baSGgc0,86
81
- bohr_agent_sdk-0.1.110.dist-info/top_level.txt,sha256=87xLUDhu_1nQHoGLwlhJ6XlO7OsjILh6i1nX6ljFzDo,3
82
- bohr_agent_sdk-0.1.110.dist-info/RECORD,,
78
+ bohr_agent_sdk-0.1.111.dist-info/METADATA,sha256=savZekEjjj6ToO6eqTYjuaEqgq7jJF-KH4Xzf_XSukM,11070
79
+ bohr_agent_sdk-0.1.111.dist-info/WHEEL,sha256=_zCd3N1l69ArxyTb8rzEoP9TpbYXkqRFSNOD5OuxnTs,91
80
+ bohr_agent_sdk-0.1.111.dist-info/entry_points.txt,sha256=5n5kneF5IbDQtoQ2WfF-QuBjDtsimJte9Rv9baSGgc0,86
81
+ bohr_agent_sdk-0.1.111.dist-info/top_level.txt,sha256=87xLUDhu_1nQHoGLwlhJ6XlO7OsjILh6i1nX6ljFzDo,3
82
+ bohr_agent_sdk-0.1.111.dist-info/RECORD,,
@@ -125,7 +125,7 @@ class MCPClient:
125
125
 
126
126
  executor = arguments.get("executor")
127
127
  storage = arguments.get("storage")
128
- res = await self.session.call_tool("submit_" + tool_name, arguments)
128
+ res = await self.call_tool("submit_" + tool_name, arguments)
129
129
  if res.isError:
130
130
  logger.error("Failed to submit %s: %s" % (
131
131
  tool_name, res.content[0].text))
@@ -137,7 +137,7 @@ class MCPClient:
137
137
  logger.info(job_info["extra_info"])
138
138
 
139
139
  while True:
140
- res = await self.session.call_tool("query_job_status", {
140
+ res = await self.call_tool("query_job_status", {
141
141
  "job_id": job_id, "executor": executor})
142
142
  if res.isError:
143
143
  logger.error(res.content[0].text)
@@ -148,7 +148,7 @@ class MCPClient:
148
148
  break
149
149
  await asyncio.sleep(self.query_interval)
150
150
 
151
- res = await self.session.call_tool("get_job_results", {
151
+ res = await self.call_tool("get_job_results", {
152
152
  "job_id": job_id, "executor": executor, "storage": storage})
153
153
  if res.isError:
154
154
  logger.error("Job %s failed: %s" % (job_id, res.content[0].text))
@@ -7,7 +7,7 @@ from copy import deepcopy
7
7
  from datetime import datetime
8
8
  from pathlib import Path
9
9
  from urllib.parse import urlparse
10
- from typing import Any, Literal, Optional, TypedDict
10
+ from typing import Any, Literal, Optional, TypedDict, List, Dict
11
11
 
12
12
  import mcp
13
13
  from mcp.server.fastmcp import FastMCP
@@ -133,6 +133,69 @@ def handle_input_artifacts(fn, kwargs, storage):
133
133
  "storage_type": scheme,
134
134
  "uri": uri,
135
135
  }
136
+ elif param.annotation is List[Path] or (
137
+ param.annotation is Optional[List[Path]] and
138
+ kwargs.get(name) is not None):
139
+ uris = kwargs[name]
140
+ new_paths = []
141
+ for uri in uris:
142
+ scheme, key = parse_uri(uri)
143
+ if scheme == storage_type:
144
+ s = storage
145
+ else:
146
+ s = storage_dict[scheme]()
147
+ path = s.download(key, "inputs/%s" % name)
148
+ new_paths.append(Path(path))
149
+ logger.info("Artifact %s downloaded to %s" % (
150
+ uri, path))
151
+ kwargs[name] = new_paths
152
+ input_artifacts[name] = {
153
+ "storage_type": storage_type,
154
+ "uri": uris,
155
+ }
156
+ elif param.annotation is Dict[str, Path] or (
157
+ param.annotation is Optional[Dict[str, Path]] and
158
+ kwargs.get(name) is not None):
159
+ uris_dict = kwargs[name]
160
+ new_paths_dict = {}
161
+ for key_name, uri in uris_dict.items():
162
+ scheme, key = parse_uri(uri)
163
+ if scheme == storage_type:
164
+ s = storage
165
+ else:
166
+ s = storage_dict[scheme]()
167
+ path = s.download(key, f"inputs/{name}/{key_name}")
168
+ new_paths_dict[key_name] = Path(path)
169
+ logger.info("Artifact %s (key=%s) downloaded to %s" % (
170
+ uri, key_name, path))
171
+ kwargs[name] = new_paths_dict
172
+ input_artifacts[name] = {
173
+ "storage_type": storage_type,
174
+ "uri": uris_dict,
175
+ }
176
+ elif param.annotation is Dict[str, List[Path]] or (
177
+ param.annotation is Optional[Dict[str, List[Path]]] and
178
+ kwargs.get(name) is not None):
179
+ uris_dict = kwargs[name]
180
+ new_paths_dict = {}
181
+ for key_name, uris in uris_dict.items():
182
+ new_paths = []
183
+ for uri in uris:
184
+ scheme, key = parse_uri(uri)
185
+ if scheme == storage_type:
186
+ s = storage
187
+ else:
188
+ s = storage_dict[scheme]()
189
+ path = s.download(key, f"inputs/{name}/{key_name}")
190
+ new_paths.append(Path(path))
191
+ logger.info("Artifact %s (key=%s) downloaded to %s" % (
192
+ uri, key_name, path))
193
+ new_paths_dict[key_name] = new_paths
194
+ kwargs[name] = new_paths_dict
195
+ input_artifacts[name] = {
196
+ "storage_type": storage_type,
197
+ "uri": uris_dict,
198
+ }
136
199
  return kwargs, input_artifacts
137
200
 
138
201
 
@@ -152,6 +215,20 @@ def handle_output_artifacts(results, exec_id, storage):
152
215
  "storage_type": storage_type,
153
216
  "uri": uri,
154
217
  }
218
+ elif isinstance(results[name], list) and all(isinstance(item, Path) for item in results[name]):
219
+ new_uris = []
220
+ for item in results[name]:
221
+ key = storage.upload("%s/outputs/%s" % (exec_id, name),
222
+ item)
223
+ uri = storage_type + "://" + key
224
+ logger.info("Artifact %s uploaded to %s" % (
225
+ item, uri))
226
+ new_uris.append(uri)
227
+ results[name] = new_uris
228
+ output_artifacts[name] = {
229
+ "storage_type": storage_type,
230
+ "uri": new_uris,
231
+ }
155
232
  return results, output_artifacts
156
233
 
157
234
 
@@ -211,6 +288,30 @@ class CalculationMCPServer:
211
288
  parameters.append(inspect.Parameter(
212
289
  name=param.name, default=param.default,
213
290
  annotation=Optional[str], kind=param.kind))
291
+ elif param.annotation is List[Path]:
292
+ parameters.append(inspect.Parameter(
293
+ name=param.name, default=param.default,
294
+ annotation=List[str], kind=param.kind))
295
+ elif param.annotation is Optional[List[Path]]:
296
+ parameters.append(inspect.Parameter(
297
+ name=param.name, default=param.default,
298
+ annotation=Optional[List[str]], kind=param.kind))
299
+ elif param.annotation is Dict[str, Path]:
300
+ parameters.append(inspect.Parameter(
301
+ name=param.name, default=param.default,
302
+ annotation=Dict[str, str], kind=param.kind))
303
+ elif param.annotation is Optional[Dict[str, Path]]:
304
+ parameters.append(inspect.Parameter(
305
+ name=param.name, default=param.default,
306
+ annotation=Optional[Dict[str, str]], kind=param.kind))
307
+ elif param.annotation is Dict[str, List[Path]]:
308
+ parameters.append(inspect.Parameter(
309
+ name=param.name, default=param.default,
310
+ annotation=Dict[str, List[str]], kind=param.kind))
311
+ elif param.annotation is Optional[Dict[str, List[Path]]]:
312
+ parameters.append(inspect.Parameter(
313
+ name=param.name, default=param.default,
314
+ annotation=Optional[Dict[str, List[str]]], kind=param.kind))
214
315
  else:
215
316
  parameters.append(param)
216
317
  for param in new_typed_signature.parameters.values():
@@ -135,7 +135,7 @@ class DispatcherExecutor(BaseExecutor):
135
135
  func_def_script, packages = get_func_def_script(fn)
136
136
  self.python_packages.extend(packages)
137
137
 
138
- script += "import asyncio, jsonpickle, os\n"
138
+ script += "import asyncio, jsonpickle, os, shutil\n"
139
139
  script += "from pathlib import Path\n\n"
140
140
  script += "if __name__ == \"__main__\":\n"
141
141
  script += " cwd = os.getcwd()\n"
@@ -149,11 +149,24 @@ class DispatcherExecutor(BaseExecutor):
149
149
  script += " results = asyncio.run(%s(**kwargs))\n" % fn_name
150
150
  else:
151
151
  script += " results = %s(**kwargs)\n" % fn_name
152
+ script += " result_dir = None\n"
153
+ script += " import uuid\n"
152
154
  script += " if isinstance(results, dict):\n"
153
155
  script += " for name in results:\n"
154
156
  script += " if isinstance(results[name], Path):\n"
155
- script += " results[name] = " \
156
- "results[name].absolute().relative_to(cwd)\n"
157
+ script += " if not results[name].absolute().is_relative_to(cwd):\n"
158
+ script += " if result_dir is None:\n"
159
+ script += " result_dir = Path('result_files_dir_' + str(uuid.uuid4()))\n"
160
+ script += " result_dir.mkdir(parents=True, exist_ok=True)\n"
161
+ script += " dest_path = result_dir / results[name].absolute().relative_to('/')\n"
162
+ script += " dest_path.parent.mkdir(parents=True, exist_ok=True)\n"
163
+ script += " if results[name].is_file():\n"
164
+ script += " shutil.copy2(results[name], dest_path)\n"
165
+ script += " elif results[name].is_dir():\n"
166
+ script += " shutil.copytree(results[name], dest_path, dirs_exist_ok=True)\n"
167
+ script += " results[name] = dest_path.absolute().relative_to(cwd)\n"
168
+ script += " else:\n"
169
+ script += " results[name] = results[name].absolute().relative_to(cwd)\n"
157
170
  script += " except Exception as e:\n"
158
171
  script += " os.chdir(cwd)\n"
159
172
  script += " with open('err', 'w') as f:\n"