uipath-langchain 0.0.132__py3-none-any.whl → 0.0.133__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.
File without changes
@@ -0,0 +1,191 @@
1
+ import json
2
+ import logging
3
+ from typing import Iterable, Optional, Type
4
+
5
+ import httpx
6
+ from jsonschema_pydantic import jsonschema_to_pydantic as create_model # type: ignore
7
+ from langchain_core.caches import BaseCache
8
+ from langchain_core.runnables.utils import Output
9
+ from langchain_core.tools import BaseTool, StructuredTool
10
+ from langgraph.types import interrupt
11
+ from pydantic import BaseModel
12
+ from uipath import UiPath
13
+ from uipath.agent.models.agent import (
14
+ AgentDefinition,
15
+ AgentEscalationChannel,
16
+ AgentEscalationResourceConfig,
17
+ AgentIntegrationToolResourceConfig,
18
+ AgentProcessToolResourceConfig,
19
+ AgentResourceConfig,
20
+ )
21
+ from uipath.models import CreateAction, InvokeProcess
22
+
23
+ logger = logging.getLogger(__name__)
24
+
25
+
26
+ def create_process_tool(resource: AgentProcessToolResourceConfig) -> Iterable[BaseTool]:
27
+ async def process(**kwargs) -> BaseModel:
28
+ return interrupt(
29
+ InvokeProcess(
30
+ name=resource.name,
31
+ input_arguments=kwargs,
32
+ process_folder_path=resource.properties.folder_path,
33
+ )
34
+ )
35
+
36
+ input_schema = create_model(resource.input_schema)
37
+
38
+ class ProcessTool(StructuredTool):
39
+ @property
40
+ def OutputType(self) -> type[Output]:
41
+ return create_model(resource.output_schema)
42
+
43
+ yield ProcessTool(
44
+ name=resource.name,
45
+ args_schema=input_schema,
46
+ description=resource.description,
47
+ coroutine=process,
48
+ )
49
+
50
+
51
+ def create_escalation_tool_from_channel(channel: AgentEscalationChannel) -> BaseTool:
52
+ async def escalate(**kwargs) -> BaseModel:
53
+ recipients = channel.recipients
54
+ if len(recipients) > 1:
55
+ logger.warning(
56
+ "Received more than one recipient. Defaulting to first recipient."
57
+ )
58
+ assignee = recipients[0].value if recipients else None
59
+ return interrupt(
60
+ CreateAction(
61
+ title=channel.description,
62
+ data=kwargs,
63
+ assignee=assignee,
64
+ app_name=channel.properties.app_name,
65
+ app_folder_path=None, # Channels specify folder name but not folder path.
66
+ app_folder_key=channel.properties.resource_key,
67
+ app_key=channel.properties.resource_key,
68
+ app_version=channel.properties.app_version,
69
+ )
70
+ )
71
+
72
+ input_schema = create_model(channel.input_schema)
73
+
74
+ class EscalationTool(StructuredTool):
75
+ @property
76
+ def OutputType(self) -> type[Output]:
77
+ return create_model(channel.output_schema)
78
+
79
+ return EscalationTool(
80
+ name=channel.name,
81
+ args_schema=input_schema,
82
+ description=channel.description,
83
+ coroutine=escalate,
84
+ )
85
+
86
+
87
+ def create_escalation_tool(
88
+ resource: AgentEscalationResourceConfig,
89
+ ) -> Iterable[BaseTool]:
90
+ for channel in resource.channels:
91
+ yield create_escalation_tool_from_channel(channel)
92
+
93
+
94
+ def create_integration_tool(
95
+ resource: AgentIntegrationToolResourceConfig,
96
+ ) -> Iterable[BaseTool]:
97
+ async def integration(**kwargs) -> BaseModel:
98
+ uipath = UiPath()
99
+ remote_connection = await uipath.connections.retrieve_async(
100
+ resource.properties.connection.id
101
+ )
102
+ token = await uipath.connections.retrieve_token_async(
103
+ resource.properties.connection.id
104
+ )
105
+ tool_url = f"{remote_connection.api_base_uri}/v3/element/instances/{remote_connection.element_instance_id}{resource.properties.tool_path}"
106
+
107
+ response = await httpx.AsyncClient().request(
108
+ resource.properties.method,
109
+ tool_url,
110
+ headers={"Authorization": f"Bearer {token.access_token}"},
111
+ content=json.dumps(kwargs),
112
+ )
113
+ return response.json()
114
+
115
+ input_schema = create_model(resource.input_schema)
116
+
117
+ class IntegrationTool(StructuredTool):
118
+ @property
119
+ def OutputType(self) -> type[Output]:
120
+ return create_model({})
121
+
122
+ yield IntegrationTool(
123
+ name=resource.name,
124
+ args_schema=input_schema,
125
+ description=resource.description,
126
+ coroutine=integration,
127
+ )
128
+
129
+
130
+ def create_cached_wrapper_from_tool(
131
+ wrapped: BaseTool, cache: Optional[BaseCache]
132
+ ) -> BaseTool:
133
+ if cache is None:
134
+ return wrapped
135
+ else:
136
+
137
+ async def cached_invocation(**kwargs) -> BaseModel:
138
+ namespace = f"{wrapped.name}.tool_invoke"
139
+ key = str(kwargs)
140
+ cached = cache.lookup(namespace, key)
141
+ if cached:
142
+ return cached[0]
143
+ response = await wrapped.ainvoke(input=kwargs)
144
+ cache.update(namespace, key, [response])
145
+ return response
146
+
147
+ input_schema = wrapped.args_schema
148
+
149
+ class CachedTool(StructuredTool):
150
+ OutputType: Type[BaseModel] = wrapped.OutputType
151
+
152
+ return CachedTool(
153
+ name=wrapped.name,
154
+ args_schema=input_schema,
155
+ description=wrapped.description,
156
+ coroutine=cached_invocation,
157
+ )
158
+
159
+
160
+ def create_cached_wrapper(
161
+ tools: Iterable[BaseTool], cache: Optional[BaseCache]
162
+ ) -> Iterable[BaseTool]:
163
+ for wrapped in tools:
164
+ yield create_cached_wrapper_from_tool(wrapped, cache)
165
+
166
+
167
+ def create_resource_tool(
168
+ resource: AgentResourceConfig, cache: Optional[BaseCache] = None
169
+ ) -> Iterable[BaseTool]:
170
+ match resource:
171
+ case AgentProcessToolResourceConfig():
172
+ return create_cached_wrapper(create_process_tool(resource), cache)
173
+ case AgentIntegrationToolResourceConfig():
174
+ return create_cached_wrapper(create_integration_tool(resource), cache)
175
+ case AgentEscalationResourceConfig():
176
+ return create_cached_wrapper(create_escalation_tool(resource), cache)
177
+ case _:
178
+ raise NotImplementedError()
179
+
180
+
181
+ def safe_extract_tools(
182
+ agent_definition: AgentDefinition, cache: Optional[BaseCache] = None
183
+ ) -> list[BaseTool]:
184
+ tools = []
185
+ for resource in agent_definition.resources:
186
+ try:
187
+ for structured_tool in create_resource_tool(resource, cache):
188
+ tools.append(structured_tool)
189
+ except NotImplementedError:
190
+ logger.warning(f"Unable to convert {resource.name} into a tool.")
191
+ return tools
@@ -1,6 +1,6 @@
1
1
  Metadata-Version: 2.4
2
2
  Name: uipath-langchain
3
- Version: 0.0.132
3
+ Version: 0.0.133
4
4
  Summary: UiPath Langchain
5
5
  Project-URL: Homepage, https://uipath.com
6
6
  Project-URL: Repository, https://github.com/UiPath/uipath-langchain-python
@@ -15,6 +15,7 @@ Classifier: Programming Language :: Python :: 3.12
15
15
  Classifier: Topic :: Software Development :: Build Tools
16
16
  Requires-Python: >=3.10
17
17
  Requires-Dist: httpx>=0.27.0
18
+ Requires-Dist: jsonschema-pydantic>=0.6
18
19
  Requires-Dist: langchain-community>=0.3.21
19
20
  Requires-Dist: langchain-core>=0.3.34
20
21
  Requires-Dist: langchain-openai>=0.3.3
@@ -25,7 +26,7 @@ Requires-Dist: openai>=1.65.5
25
26
  Requires-Dist: openinference-instrumentation-langchain>=0.1.50
26
27
  Requires-Dist: pydantic-settings>=2.6.0
27
28
  Requires-Dist: python-dotenv>=1.0.1
28
- Requires-Dist: uipath<2.2.0,>=2.1.56
29
+ Requires-Dist: uipath<2.2.0,>=2.1.60
29
30
  Provides-Extra: langchain
30
31
  Description-Content-Type: text/markdown
31
32
 
@@ -30,10 +30,12 @@ uipath_langchain/embeddings/__init__.py,sha256=QICtYB58ZyqFfDQrEaO8lTEgAU5NuEKlR
30
30
  uipath_langchain/embeddings/embeddings.py,sha256=45gKyb6HVKigwE-0CXeZcAk33c0mteaEdPGa8hviqcw,4339
31
31
  uipath_langchain/retrievers/__init__.py,sha256=rOn7PyyHgZ4pMnXWPkGqmuBmx8eGuo-Oyndo7Wm9IUU,108
32
32
  uipath_langchain/retrievers/context_grounding_retriever.py,sha256=YLCIwy89LhLnNqcM0YJ5mZoeNyCs5UiKD3Wly8gnW1E,2239
33
+ uipath_langchain/tools/__init__.py,sha256=47DEQpj8HBSa-_TImW-5JCeuQeRkm5NMpJWZG3hSuFU,0
34
+ uipath_langchain/tools/preconfigured.py,sha256=Ucl9mta_su66j-5uOC-vZzyc3V2-Em6q9WO1qsmQv48,6291
33
35
  uipath_langchain/vectorstores/__init__.py,sha256=w8qs1P548ud1aIcVA_QhBgf_jZDrRMK5Lono78yA8cs,114
34
36
  uipath_langchain/vectorstores/context_grounding_vectorstore.py,sha256=TncIXG-YsUlO0R5ZYzWsM-Dj1SVCZbzmo2LraVxXelc,9559
35
- uipath_langchain-0.0.132.dist-info/METADATA,sha256=ta5yg3ODDXTdagrZVgeevO7wzwjwDeIIR3dAC_wmctM,4235
36
- uipath_langchain-0.0.132.dist-info/WHEEL,sha256=qtCwoSJWgHk21S1Kb4ihdzI2rlJ1ZKaIurTj_ngOhyQ,87
37
- uipath_langchain-0.0.132.dist-info/entry_points.txt,sha256=FUtzqGOEntlJKMJIXhQUfT7ZTbQmGhke1iCmDWZaQZI,81
38
- uipath_langchain-0.0.132.dist-info/licenses/LICENSE,sha256=JDpt-uotAkHFmxpwxi6gwx6HQ25e-lG4U_Gzcvgp7JY,1063
39
- uipath_langchain-0.0.132.dist-info/RECORD,,
37
+ uipath_langchain-0.0.133.dist-info/METADATA,sha256=zcSdl4I5iJsBi0jOf7IUUprBv39KfQZyrM1EzVH1OCQ,4275
38
+ uipath_langchain-0.0.133.dist-info/WHEEL,sha256=qtCwoSJWgHk21S1Kb4ihdzI2rlJ1ZKaIurTj_ngOhyQ,87
39
+ uipath_langchain-0.0.133.dist-info/entry_points.txt,sha256=FUtzqGOEntlJKMJIXhQUfT7ZTbQmGhke1iCmDWZaQZI,81
40
+ uipath_langchain-0.0.133.dist-info/licenses/LICENSE,sha256=JDpt-uotAkHFmxpwxi6gwx6HQ25e-lG4U_Gzcvgp7JY,1063
41
+ uipath_langchain-0.0.133.dist-info/RECORD,,