librefang-sdk 0.5.4__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.
- librefang_sdk-0.5.4/LICENSE +21 -0
- librefang_sdk-0.5.4/PKG-INFO +110 -0
- librefang_sdk-0.5.4/README.md +81 -0
- librefang_sdk-0.5.4/librefang/__init__.py +14 -0
- librefang_sdk-0.5.4/librefang/librefang_client.py +367 -0
- librefang_sdk-0.5.4/librefang/librefang_sdk.py +147 -0
- librefang_sdk-0.5.4/librefang_sdk.egg-info/PKG-INFO +110 -0
- librefang_sdk-0.5.4/librefang_sdk.egg-info/SOURCES.txt +12 -0
- librefang_sdk-0.5.4/librefang_sdk.egg-info/dependency_links.txt +1 -0
- librefang_sdk-0.5.4/librefang_sdk.egg-info/requires.txt +4 -0
- librefang_sdk-0.5.4/librefang_sdk.egg-info/top_level.txt +3 -0
- librefang_sdk-0.5.4/pyproject.toml +53 -0
- librefang_sdk-0.5.4/setup.cfg +4 -0
- librefang_sdk-0.5.4/setup.py +14 -0
|
@@ -0,0 +1,21 @@
|
|
|
1
|
+
MIT License
|
|
2
|
+
|
|
3
|
+
Copyright (c) 2026 LibreFang Contributors
|
|
4
|
+
|
|
5
|
+
Permission is hereby granted, free of charge, to any person obtaining a copy
|
|
6
|
+
of this software and associated documentation files (the "Software"), to deal
|
|
7
|
+
in the Software without restriction, including without limitation the rights
|
|
8
|
+
to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
|
9
|
+
copies of the Software, and to permit persons to whom the Software is
|
|
10
|
+
furnished to do so, subject to the following conditions:
|
|
11
|
+
|
|
12
|
+
The above copyright notice and this permission notice shall be included in all
|
|
13
|
+
copies or substantial portions of the Software.
|
|
14
|
+
|
|
15
|
+
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
|
16
|
+
IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
|
17
|
+
FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
|
18
|
+
AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
|
19
|
+
LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
|
20
|
+
OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
|
|
21
|
+
SOFTWARE.
|
|
@@ -0,0 +1,110 @@
|
|
|
1
|
+
Metadata-Version: 2.4
|
|
2
|
+
Name: librefang-sdk
|
|
3
|
+
Version: 0.5.4
|
|
4
|
+
Summary: Official Python client and SDK for the LibreFang Agent OS
|
|
5
|
+
Author-email: LibreFang Team <team@librefang.ai>
|
|
6
|
+
License-Expression: MIT
|
|
7
|
+
Project-URL: Homepage, https://librefang.ai
|
|
8
|
+
Project-URL: Repository, https://github.com/librefang/librefang
|
|
9
|
+
Project-URL: Issues, https://github.com/librefang/librefang/issues
|
|
10
|
+
Keywords: librefang,agent,ai,llm,autonomous-agent,sdk
|
|
11
|
+
Classifier: Development Status :: 4 - Beta
|
|
12
|
+
Classifier: Intended Audience :: Developers
|
|
13
|
+
Classifier: Operating System :: OS Independent
|
|
14
|
+
Classifier: Programming Language :: Python :: 3
|
|
15
|
+
Classifier: Programming Language :: Python :: 3.8
|
|
16
|
+
Classifier: Programming Language :: Python :: 3.9
|
|
17
|
+
Classifier: Programming Language :: Python :: 3.10
|
|
18
|
+
Classifier: Programming Language :: Python :: 3.11
|
|
19
|
+
Classifier: Programming Language :: Python :: 3.12
|
|
20
|
+
Classifier: Topic :: Software Development :: Libraries :: Python Modules
|
|
21
|
+
Requires-Python: >=3.8
|
|
22
|
+
Description-Content-Type: text/markdown
|
|
23
|
+
License-File: LICENSE
|
|
24
|
+
Provides-Extra: dev
|
|
25
|
+
Requires-Dist: pytest; extra == "dev"
|
|
26
|
+
Requires-Dist: pytest-asyncio; extra == "dev"
|
|
27
|
+
Dynamic: license-file
|
|
28
|
+
Dynamic: requires-python
|
|
29
|
+
|
|
30
|
+
# LibreFang Python SDK
|
|
31
|
+
|
|
32
|
+
Official Python client and SDK for the LibreFang Agent OS.
|
|
33
|
+
|
|
34
|
+
## Installation
|
|
35
|
+
|
|
36
|
+
```bash
|
|
37
|
+
pip install librefang
|
|
38
|
+
```
|
|
39
|
+
|
|
40
|
+
## Two Packages
|
|
41
|
+
|
|
42
|
+
This package provides two different interfaces:
|
|
43
|
+
|
|
44
|
+
### 1. REST API Client (`librefang.client`)
|
|
45
|
+
|
|
46
|
+
Control LibreFang remotely via its REST API.
|
|
47
|
+
|
|
48
|
+
```python
|
|
49
|
+
from librefang import Client
|
|
50
|
+
|
|
51
|
+
client = Client("http://localhost:4545")
|
|
52
|
+
|
|
53
|
+
# Create an agent
|
|
54
|
+
agent = client.agents.create(template="assistant")
|
|
55
|
+
print(f"Agent created: {agent['id']}")
|
|
56
|
+
|
|
57
|
+
# Send a message
|
|
58
|
+
reply = client.agents.message(agent["id"], "Hello!")
|
|
59
|
+
print(reply)
|
|
60
|
+
|
|
61
|
+
# Stream a response
|
|
62
|
+
for event in client.agents.stream(agent["id"], "Tell me a story"):
|
|
63
|
+
if event.get("type") == "text_delta":
|
|
64
|
+
print(event["delta"], end="", flush=True)
|
|
65
|
+
```
|
|
66
|
+
|
|
67
|
+
### 2. Agent SDK (`librefang.sdk`)
|
|
68
|
+
|
|
69
|
+
Write Python agents that run inside LibreFang.
|
|
70
|
+
|
|
71
|
+
```python
|
|
72
|
+
from librefang import Agent
|
|
73
|
+
|
|
74
|
+
agent = Agent()
|
|
75
|
+
|
|
76
|
+
@agent.on_message
|
|
77
|
+
def handle(message: str, context: dict) -> str:
|
|
78
|
+
return f"You said: {message}"
|
|
79
|
+
|
|
80
|
+
agent.run()
|
|
81
|
+
```
|
|
82
|
+
|
|
83
|
+
Or use the simple input/output functions:
|
|
84
|
+
|
|
85
|
+
```python
|
|
86
|
+
from librefang import read_input, respond
|
|
87
|
+
|
|
88
|
+
data = read_input()
|
|
89
|
+
result = f"Echo: {data['message']}"
|
|
90
|
+
respond(result)
|
|
91
|
+
```
|
|
92
|
+
|
|
93
|
+
## Examples
|
|
94
|
+
|
|
95
|
+
See the `examples/` directory for more examples:
|
|
96
|
+
|
|
97
|
+
### Client Examples
|
|
98
|
+
- `client_basic.py` - Basic REST API usage
|
|
99
|
+
- `client_streaming.py` - Streaming responses
|
|
100
|
+
|
|
101
|
+
### SDK Examples
|
|
102
|
+
- `echo_agent.py` - Simple echo agent
|
|
103
|
+
|
|
104
|
+
## Requirements
|
|
105
|
+
|
|
106
|
+
- Python 3.8+
|
|
107
|
+
|
|
108
|
+
## License
|
|
109
|
+
|
|
110
|
+
MIT
|
|
@@ -0,0 +1,81 @@
|
|
|
1
|
+
# LibreFang Python SDK
|
|
2
|
+
|
|
3
|
+
Official Python client and SDK for the LibreFang Agent OS.
|
|
4
|
+
|
|
5
|
+
## Installation
|
|
6
|
+
|
|
7
|
+
```bash
|
|
8
|
+
pip install librefang
|
|
9
|
+
```
|
|
10
|
+
|
|
11
|
+
## Two Packages
|
|
12
|
+
|
|
13
|
+
This package provides two different interfaces:
|
|
14
|
+
|
|
15
|
+
### 1. REST API Client (`librefang.client`)
|
|
16
|
+
|
|
17
|
+
Control LibreFang remotely via its REST API.
|
|
18
|
+
|
|
19
|
+
```python
|
|
20
|
+
from librefang import Client
|
|
21
|
+
|
|
22
|
+
client = Client("http://localhost:4545")
|
|
23
|
+
|
|
24
|
+
# Create an agent
|
|
25
|
+
agent = client.agents.create(template="assistant")
|
|
26
|
+
print(f"Agent created: {agent['id']}")
|
|
27
|
+
|
|
28
|
+
# Send a message
|
|
29
|
+
reply = client.agents.message(agent["id"], "Hello!")
|
|
30
|
+
print(reply)
|
|
31
|
+
|
|
32
|
+
# Stream a response
|
|
33
|
+
for event in client.agents.stream(agent["id"], "Tell me a story"):
|
|
34
|
+
if event.get("type") == "text_delta":
|
|
35
|
+
print(event["delta"], end="", flush=True)
|
|
36
|
+
```
|
|
37
|
+
|
|
38
|
+
### 2. Agent SDK (`librefang.sdk`)
|
|
39
|
+
|
|
40
|
+
Write Python agents that run inside LibreFang.
|
|
41
|
+
|
|
42
|
+
```python
|
|
43
|
+
from librefang import Agent
|
|
44
|
+
|
|
45
|
+
agent = Agent()
|
|
46
|
+
|
|
47
|
+
@agent.on_message
|
|
48
|
+
def handle(message: str, context: dict) -> str:
|
|
49
|
+
return f"You said: {message}"
|
|
50
|
+
|
|
51
|
+
agent.run()
|
|
52
|
+
```
|
|
53
|
+
|
|
54
|
+
Or use the simple input/output functions:
|
|
55
|
+
|
|
56
|
+
```python
|
|
57
|
+
from librefang import read_input, respond
|
|
58
|
+
|
|
59
|
+
data = read_input()
|
|
60
|
+
result = f"Echo: {data['message']}"
|
|
61
|
+
respond(result)
|
|
62
|
+
```
|
|
63
|
+
|
|
64
|
+
## Examples
|
|
65
|
+
|
|
66
|
+
See the `examples/` directory for more examples:
|
|
67
|
+
|
|
68
|
+
### Client Examples
|
|
69
|
+
- `client_basic.py` - Basic REST API usage
|
|
70
|
+
- `client_streaming.py` - Streaming responses
|
|
71
|
+
|
|
72
|
+
### SDK Examples
|
|
73
|
+
- `echo_agent.py` - Simple echo agent
|
|
74
|
+
|
|
75
|
+
## Requirements
|
|
76
|
+
|
|
77
|
+
- Python 3.8+
|
|
78
|
+
|
|
79
|
+
## License
|
|
80
|
+
|
|
81
|
+
MIT
|
|
@@ -0,0 +1,14 @@
|
|
|
1
|
+
"""
|
|
2
|
+
LibreFang Python SDK and Client.
|
|
3
|
+
|
|
4
|
+
Two packages:
|
|
5
|
+
- librefang.client: REST API client for controlling LibreFang remotely
|
|
6
|
+
- librefang.sdk: Helper library for writing Python agents that run inside LibreFang
|
|
7
|
+
"""
|
|
8
|
+
|
|
9
|
+
from librefang.librefang_client import LibreFang as Client
|
|
10
|
+
from librefang.librefang_sdk import Agent, read_input, respond, log
|
|
11
|
+
|
|
12
|
+
__version__ = "0.5.2"
|
|
13
|
+
|
|
14
|
+
__all__ = ["Client", "Agent", "read_input", "respond", "log"]
|
|
@@ -0,0 +1,367 @@
|
|
|
1
|
+
"""
|
|
2
|
+
LibreFang Python Client — REST API client for controlling LibreFang remotely.
|
|
3
|
+
|
|
4
|
+
Usage:
|
|
5
|
+
|
|
6
|
+
from librefang_client import LibreFang
|
|
7
|
+
|
|
8
|
+
client = LibreFang("http://localhost:4545")
|
|
9
|
+
|
|
10
|
+
# Create an agent
|
|
11
|
+
agent = client.agents.create(template="assistant")
|
|
12
|
+
print(agent["id"])
|
|
13
|
+
|
|
14
|
+
# Send a message
|
|
15
|
+
reply = client.agents.message(agent["id"], "Hello!")
|
|
16
|
+
print(reply)
|
|
17
|
+
|
|
18
|
+
# Stream a response
|
|
19
|
+
for event in client.agents.stream(agent["id"], "Tell me a joke"):
|
|
20
|
+
if event.get("type") == "text_delta":
|
|
21
|
+
print(event["delta"], end="", flush=True)
|
|
22
|
+
|
|
23
|
+
Note: This is the REST API *client* library.
|
|
24
|
+
For writing Python agents that run inside LibreFang, see librefang_sdk.py instead.
|
|
25
|
+
"""
|
|
26
|
+
|
|
27
|
+
import json
|
|
28
|
+
from typing import Any, Dict, Generator, Optional
|
|
29
|
+
from urllib.request import urlopen, Request
|
|
30
|
+
from urllib.error import HTTPError
|
|
31
|
+
from urllib.parse import urlencode, quote
|
|
32
|
+
|
|
33
|
+
|
|
34
|
+
class LibreFangError(Exception):
|
|
35
|
+
def __init__(self, message: str, status: int = 0, body: str = ""):
|
|
36
|
+
super().__init__(message)
|
|
37
|
+
self.status = status
|
|
38
|
+
self.body = body
|
|
39
|
+
|
|
40
|
+
|
|
41
|
+
class _Resource:
|
|
42
|
+
def __init__(self, client: "LibreFang"):
|
|
43
|
+
self._c = client
|
|
44
|
+
|
|
45
|
+
|
|
46
|
+
class LibreFang:
|
|
47
|
+
"""LibreFang REST API client. Zero dependencies — uses only stdlib urllib."""
|
|
48
|
+
|
|
49
|
+
def __init__(self, base_url: str, headers: Optional[Dict[str, str]] = None):
|
|
50
|
+
self.base_url = base_url.rstrip("/")
|
|
51
|
+
self._headers = {"Content-Type": "application/json"}
|
|
52
|
+
if headers:
|
|
53
|
+
self._headers.update(headers)
|
|
54
|
+
|
|
55
|
+
self.agents = _AgentResource(self)
|
|
56
|
+
self.sessions = _SessionResource(self)
|
|
57
|
+
self.workflows = _WorkflowResource(self)
|
|
58
|
+
self.skills = _SkillResource(self)
|
|
59
|
+
self.channels = _ChannelResource(self)
|
|
60
|
+
self.tools = _ToolResource(self)
|
|
61
|
+
self.models = _ModelResource(self)
|
|
62
|
+
self.providers = _ProviderResource(self)
|
|
63
|
+
self.memory = _MemoryResource(self)
|
|
64
|
+
self.triggers = _TriggerResource(self)
|
|
65
|
+
self.schedules = _ScheduleResource(self)
|
|
66
|
+
|
|
67
|
+
def _request(self, method: str, path: str, body: Any = None) -> Any:
|
|
68
|
+
url = self.base_url + path
|
|
69
|
+
data = json.dumps(body).encode() if body is not None else None
|
|
70
|
+
req = Request(url, data=data, headers=self._headers, method=method)
|
|
71
|
+
try:
|
|
72
|
+
with urlopen(req) as resp:
|
|
73
|
+
ct = resp.headers.get("content-type", "")
|
|
74
|
+
text = resp.read().decode()
|
|
75
|
+
if "application/json" in ct:
|
|
76
|
+
return json.loads(text)
|
|
77
|
+
return text
|
|
78
|
+
except HTTPError as e:
|
|
79
|
+
body_text = e.read().decode() if e.fp else ""
|
|
80
|
+
raise LibreFangError(f"HTTP {e.code}: {body_text}", e.code, body_text) from e
|
|
81
|
+
|
|
82
|
+
def _stream(self, method: str, path: str, body: Any = None) -> Generator[Dict, None, None]:
|
|
83
|
+
"""SSE streaming. Yields parsed JSON events."""
|
|
84
|
+
url = self.base_url + path
|
|
85
|
+
data = json.dumps(body).encode() if body is not None else None
|
|
86
|
+
headers = dict(self._headers)
|
|
87
|
+
headers["Accept"] = "text/event-stream"
|
|
88
|
+
req = Request(url, data=data, headers=headers, method=method)
|
|
89
|
+
try:
|
|
90
|
+
resp = urlopen(req)
|
|
91
|
+
except HTTPError as e:
|
|
92
|
+
body_text = e.read().decode() if e.fp else ""
|
|
93
|
+
raise LibreFangError(f"HTTP {e.code}: {body_text}", e.code, body_text) from e
|
|
94
|
+
|
|
95
|
+
buffer = ""
|
|
96
|
+
while True:
|
|
97
|
+
chunk = resp.read(4096)
|
|
98
|
+
if not chunk:
|
|
99
|
+
break
|
|
100
|
+
buffer += chunk.decode()
|
|
101
|
+
lines = buffer.split("\n")
|
|
102
|
+
buffer = lines.pop()
|
|
103
|
+
for line in lines:
|
|
104
|
+
line = line.strip()
|
|
105
|
+
if line.startswith("data: "):
|
|
106
|
+
data_str = line[6:]
|
|
107
|
+
if data_str == "[DONE]":
|
|
108
|
+
return
|
|
109
|
+
try:
|
|
110
|
+
yield json.loads(data_str)
|
|
111
|
+
except json.JSONDecodeError:
|
|
112
|
+
yield {"raw": data_str}
|
|
113
|
+
resp.close()
|
|
114
|
+
|
|
115
|
+
def health(self) -> Any:
|
|
116
|
+
return self._request("GET", "/api/health")
|
|
117
|
+
|
|
118
|
+
def health_detail(self) -> Any:
|
|
119
|
+
return self._request("GET", "/api/health/detail")
|
|
120
|
+
|
|
121
|
+
def status(self) -> Any:
|
|
122
|
+
return self._request("GET", "/api/status")
|
|
123
|
+
|
|
124
|
+
def version(self) -> Any:
|
|
125
|
+
return self._request("GET", "/api/version")
|
|
126
|
+
|
|
127
|
+
def metrics(self) -> str:
|
|
128
|
+
return self._request("GET", "/api/metrics")
|
|
129
|
+
|
|
130
|
+
def usage(self) -> Any:
|
|
131
|
+
return self._request("GET", "/api/usage")
|
|
132
|
+
|
|
133
|
+
def config(self) -> Any:
|
|
134
|
+
return self._request("GET", "/api/config")
|
|
135
|
+
|
|
136
|
+
|
|
137
|
+
# ── Agent Resource ──────────────────────────────────────────────
|
|
138
|
+
|
|
139
|
+
class _AgentResource(_Resource):
|
|
140
|
+
|
|
141
|
+
def list(self):
|
|
142
|
+
return self._c._request("GET", "/api/agents")
|
|
143
|
+
|
|
144
|
+
def get(self, agent_id: str):
|
|
145
|
+
return self._c._request("GET", f"/api/agents/{agent_id}")
|
|
146
|
+
|
|
147
|
+
def create(self, **kwargs):
|
|
148
|
+
return self._c._request("POST", "/api/agents", kwargs)
|
|
149
|
+
|
|
150
|
+
def delete(self, agent_id: str):
|
|
151
|
+
return self._c._request("DELETE", f"/api/agents/{agent_id}")
|
|
152
|
+
|
|
153
|
+
def stop(self, agent_id: str):
|
|
154
|
+
return self._c._request("POST", f"/api/agents/{agent_id}/stop")
|
|
155
|
+
|
|
156
|
+
def clone(self, agent_id: str):
|
|
157
|
+
return self._c._request("POST", f"/api/agents/{agent_id}/clone")
|
|
158
|
+
|
|
159
|
+
def update(self, agent_id: str, **data):
|
|
160
|
+
return self._c._request("PUT", f"/api/agents/{agent_id}/update", data)
|
|
161
|
+
|
|
162
|
+
def set_mode(self, agent_id: str, mode: str):
|
|
163
|
+
return self._c._request("PUT", f"/api/agents/{agent_id}/mode", {"mode": mode})
|
|
164
|
+
|
|
165
|
+
def set_model(self, agent_id: str, model: str):
|
|
166
|
+
return self._c._request("PUT", f"/api/agents/{agent_id}/model", {"model": model})
|
|
167
|
+
|
|
168
|
+
def message(self, agent_id: str, text: str, **opts):
|
|
169
|
+
body = {"message": text, **opts}
|
|
170
|
+
return self._c._request("POST", f"/api/agents/{agent_id}/message", body)
|
|
171
|
+
|
|
172
|
+
def stream(self, agent_id: str, text: str, **opts) -> Generator[Dict, None, None]:
|
|
173
|
+
"""Stream response events. Usage:
|
|
174
|
+
for event in client.agents.stream(id, "Hello"):
|
|
175
|
+
if event.get("type") == "text_delta":
|
|
176
|
+
print(event["delta"], end="")
|
|
177
|
+
"""
|
|
178
|
+
body = {"message": text, **opts}
|
|
179
|
+
return self._c._stream("POST", f"/api/agents/{agent_id}/message/stream", body)
|
|
180
|
+
|
|
181
|
+
def session(self, agent_id: str):
|
|
182
|
+
return self._c._request("GET", f"/api/agents/{agent_id}/session")
|
|
183
|
+
|
|
184
|
+
def reset_session(self, agent_id: str):
|
|
185
|
+
return self._c._request("POST", f"/api/agents/{agent_id}/session/reset")
|
|
186
|
+
|
|
187
|
+
def compact_session(self, agent_id: str):
|
|
188
|
+
return self._c._request("POST", f"/api/agents/{agent_id}/session/compact")
|
|
189
|
+
|
|
190
|
+
def list_sessions(self, agent_id: str):
|
|
191
|
+
return self._c._request("GET", f"/api/agents/{agent_id}/sessions")
|
|
192
|
+
|
|
193
|
+
def create_session(self, agent_id: str, label: Optional[str] = None):
|
|
194
|
+
return self._c._request("POST", f"/api/agents/{agent_id}/sessions", {"label": label})
|
|
195
|
+
|
|
196
|
+
def switch_session(self, agent_id: str, session_id: str):
|
|
197
|
+
return self._c._request("POST", f"/api/agents/{agent_id}/sessions/{session_id}/switch")
|
|
198
|
+
|
|
199
|
+
def get_skills(self, agent_id: str):
|
|
200
|
+
return self._c._request("GET", f"/api/agents/{agent_id}/skills")
|
|
201
|
+
|
|
202
|
+
def set_skills(self, agent_id: str, skills):
|
|
203
|
+
return self._c._request("PUT", f"/api/agents/{agent_id}/skills", skills)
|
|
204
|
+
|
|
205
|
+
def set_identity(self, agent_id: str, **identity):
|
|
206
|
+
return self._c._request("PATCH", f"/api/agents/{agent_id}/identity", identity)
|
|
207
|
+
|
|
208
|
+
def patch_config(self, agent_id: str, **config):
|
|
209
|
+
return self._c._request("PATCH", f"/api/agents/{agent_id}/config", config)
|
|
210
|
+
|
|
211
|
+
|
|
212
|
+
# ── Session Resource ────────────────────────────────────────────
|
|
213
|
+
|
|
214
|
+
class _SessionResource(_Resource):
|
|
215
|
+
|
|
216
|
+
def list(self):
|
|
217
|
+
return self._c._request("GET", "/api/sessions")
|
|
218
|
+
|
|
219
|
+
def delete(self, session_id: str):
|
|
220
|
+
return self._c._request("DELETE", f"/api/sessions/{session_id}")
|
|
221
|
+
|
|
222
|
+
def set_label(self, session_id: str, label: str):
|
|
223
|
+
return self._c._request("PUT", f"/api/sessions/{session_id}/label", {"label": label})
|
|
224
|
+
|
|
225
|
+
|
|
226
|
+
# ── Workflow Resource ───────────────────────────────────────────
|
|
227
|
+
|
|
228
|
+
class _WorkflowResource(_Resource):
|
|
229
|
+
|
|
230
|
+
def list(self):
|
|
231
|
+
return self._c._request("GET", "/api/workflows")
|
|
232
|
+
|
|
233
|
+
def create(self, **workflow):
|
|
234
|
+
return self._c._request("POST", "/api/workflows", workflow)
|
|
235
|
+
|
|
236
|
+
def run(self, workflow_id: str, input_data=None):
|
|
237
|
+
return self._c._request("POST", f"/api/workflows/{workflow_id}/run", input_data)
|
|
238
|
+
|
|
239
|
+
def runs(self, workflow_id: str):
|
|
240
|
+
return self._c._request("GET", f"/api/workflows/{workflow_id}/runs")
|
|
241
|
+
|
|
242
|
+
|
|
243
|
+
# ── Skill Resource ──────────────────────────────────────────────
|
|
244
|
+
|
|
245
|
+
class _SkillResource(_Resource):
|
|
246
|
+
|
|
247
|
+
def list(self):
|
|
248
|
+
return self._c._request("GET", "/api/skills")
|
|
249
|
+
|
|
250
|
+
def install(self, **skill):
|
|
251
|
+
return self._c._request("POST", "/api/skills/install", skill)
|
|
252
|
+
|
|
253
|
+
def uninstall(self, **skill):
|
|
254
|
+
return self._c._request("POST", "/api/skills/uninstall", skill)
|
|
255
|
+
|
|
256
|
+
def search(self, query: str):
|
|
257
|
+
return self._c._request("GET", f"/api/marketplace/search?q={quote(query)}")
|
|
258
|
+
|
|
259
|
+
|
|
260
|
+
# ── Channel Resource ────────────────────────────────────────────
|
|
261
|
+
|
|
262
|
+
class _ChannelResource(_Resource):
|
|
263
|
+
|
|
264
|
+
def list(self):
|
|
265
|
+
return self._c._request("GET", "/api/channels")
|
|
266
|
+
|
|
267
|
+
def configure(self, name: str, **config):
|
|
268
|
+
return self._c._request("POST", f"/api/channels/{name}/configure", config)
|
|
269
|
+
|
|
270
|
+
def remove(self, name: str):
|
|
271
|
+
return self._c._request("DELETE", f"/api/channels/{name}/configure")
|
|
272
|
+
|
|
273
|
+
def test(self, name: str):
|
|
274
|
+
return self._c._request("POST", f"/api/channels/{name}/test")
|
|
275
|
+
|
|
276
|
+
|
|
277
|
+
# ── Tool Resource ───────────────────────────────────────────────
|
|
278
|
+
|
|
279
|
+
class _ToolResource(_Resource):
|
|
280
|
+
|
|
281
|
+
def list(self):
|
|
282
|
+
return self._c._request("GET", "/api/tools")
|
|
283
|
+
|
|
284
|
+
|
|
285
|
+
# ── Model Resource ──────────────────────────────────────────────
|
|
286
|
+
|
|
287
|
+
class _ModelResource(_Resource):
|
|
288
|
+
|
|
289
|
+
def list(self):
|
|
290
|
+
return self._c._request("GET", "/api/models")
|
|
291
|
+
|
|
292
|
+
def get(self, model_id: str):
|
|
293
|
+
return self._c._request("GET", f"/api/models/{model_id}")
|
|
294
|
+
|
|
295
|
+
def aliases(self):
|
|
296
|
+
return self._c._request("GET", "/api/models/aliases")
|
|
297
|
+
|
|
298
|
+
|
|
299
|
+
# ── Provider Resource ───────────────────────────────────────────
|
|
300
|
+
|
|
301
|
+
class _ProviderResource(_Resource):
|
|
302
|
+
|
|
303
|
+
def list(self):
|
|
304
|
+
return self._c._request("GET", "/api/providers")
|
|
305
|
+
|
|
306
|
+
def set_key(self, name: str, key: str):
|
|
307
|
+
return self._c._request("POST", f"/api/providers/{name}/key", {"key": key})
|
|
308
|
+
|
|
309
|
+
def delete_key(self, name: str):
|
|
310
|
+
return self._c._request("DELETE", f"/api/providers/{name}/key")
|
|
311
|
+
|
|
312
|
+
def test(self, name: str):
|
|
313
|
+
return self._c._request("POST", f"/api/providers/{name}/test")
|
|
314
|
+
|
|
315
|
+
|
|
316
|
+
# ── Memory Resource ─────────────────────────────────────────────
|
|
317
|
+
|
|
318
|
+
class _MemoryResource(_Resource):
|
|
319
|
+
|
|
320
|
+
def get_all(self, agent_id: str):
|
|
321
|
+
return self._c._request("GET", f"/api/memory/agents/{agent_id}/kv")
|
|
322
|
+
|
|
323
|
+
def get(self, agent_id: str, key: str):
|
|
324
|
+
return self._c._request("GET", f"/api/memory/agents/{agent_id}/kv/{key}")
|
|
325
|
+
|
|
326
|
+
def set(self, agent_id: str, key: str, value):
|
|
327
|
+
return self._c._request("PUT", f"/api/memory/agents/{agent_id}/kv/{key}", {"value": value})
|
|
328
|
+
|
|
329
|
+
def delete(self, agent_id: str, key: str):
|
|
330
|
+
return self._c._request("DELETE", f"/api/memory/agents/{agent_id}/kv/{key}")
|
|
331
|
+
|
|
332
|
+
|
|
333
|
+
# ── Trigger Resource ────────────────────────────────────────────
|
|
334
|
+
|
|
335
|
+
class _TriggerResource(_Resource):
|
|
336
|
+
|
|
337
|
+
def list(self):
|
|
338
|
+
return self._c._request("GET", "/api/triggers")
|
|
339
|
+
|
|
340
|
+
def create(self, **trigger):
|
|
341
|
+
return self._c._request("POST", "/api/triggers", trigger)
|
|
342
|
+
|
|
343
|
+
def update(self, trigger_id: str, **trigger):
|
|
344
|
+
return self._c._request("PUT", f"/api/triggers/{trigger_id}", trigger)
|
|
345
|
+
|
|
346
|
+
def delete(self, trigger_id: str):
|
|
347
|
+
return self._c._request("DELETE", f"/api/triggers/{trigger_id}")
|
|
348
|
+
|
|
349
|
+
|
|
350
|
+
# ── Schedule Resource ───────────────────────────────────────────
|
|
351
|
+
|
|
352
|
+
class _ScheduleResource(_Resource):
|
|
353
|
+
|
|
354
|
+
def list(self):
|
|
355
|
+
return self._c._request("GET", "/api/schedules")
|
|
356
|
+
|
|
357
|
+
def create(self, **schedule):
|
|
358
|
+
return self._c._request("POST", "/api/schedules", schedule)
|
|
359
|
+
|
|
360
|
+
def update(self, schedule_id: str, **schedule):
|
|
361
|
+
return self._c._request("PUT", f"/api/schedules/{schedule_id}", schedule)
|
|
362
|
+
|
|
363
|
+
def delete(self, schedule_id: str):
|
|
364
|
+
return self._c._request("DELETE", f"/api/schedules/{schedule_id}")
|
|
365
|
+
|
|
366
|
+
def run(self, schedule_id: str):
|
|
367
|
+
return self._c._request("POST", f"/api/schedules/{schedule_id}/run")
|
|
@@ -0,0 +1,147 @@
|
|
|
1
|
+
"""
|
|
2
|
+
LibreFang Python SDK — helper library for writing Python agents.
|
|
3
|
+
|
|
4
|
+
Usage:
|
|
5
|
+
|
|
6
|
+
from librefang_sdk import Agent
|
|
7
|
+
|
|
8
|
+
agent = Agent()
|
|
9
|
+
|
|
10
|
+
@agent.on_message
|
|
11
|
+
def handle(message: str, context: dict) -> str:
|
|
12
|
+
return f"You said: {message}"
|
|
13
|
+
|
|
14
|
+
agent.run()
|
|
15
|
+
|
|
16
|
+
Or for simple scripts without the decorator pattern:
|
|
17
|
+
|
|
18
|
+
from librefang_sdk import read_input, respond
|
|
19
|
+
|
|
20
|
+
data = read_input()
|
|
21
|
+
result = f"Echo: {data['message']}"
|
|
22
|
+
respond(result)
|
|
23
|
+
"""
|
|
24
|
+
|
|
25
|
+
import json
|
|
26
|
+
import os
|
|
27
|
+
import sys
|
|
28
|
+
from typing import Callable, Optional, Dict, Any
|
|
29
|
+
|
|
30
|
+
|
|
31
|
+
def read_input() -> Dict[str, Any]:
|
|
32
|
+
"""Read the input JSON from stdin (sent by the LibreFang kernel)."""
|
|
33
|
+
line = sys.stdin.readline().strip()
|
|
34
|
+
if not line:
|
|
35
|
+
# Fallback: check environment variables
|
|
36
|
+
agent_id = os.environ.get("LIBREFANG_AGENT_ID", "")
|
|
37
|
+
message = os.environ.get("LIBREFANG_MESSAGE", "")
|
|
38
|
+
return {
|
|
39
|
+
"type": "message",
|
|
40
|
+
"agent_id": agent_id,
|
|
41
|
+
"message": message,
|
|
42
|
+
"context": {},
|
|
43
|
+
}
|
|
44
|
+
return json.loads(line)
|
|
45
|
+
|
|
46
|
+
|
|
47
|
+
def respond(text: str, metadata: Optional[Dict[str, Any]] = None) -> None:
|
|
48
|
+
"""Send a response back to the LibreFang kernel via stdout."""
|
|
49
|
+
response = {"type": "response", "text": text}
|
|
50
|
+
if metadata:
|
|
51
|
+
response["metadata"] = metadata
|
|
52
|
+
print(json.dumps(response), flush=True)
|
|
53
|
+
|
|
54
|
+
|
|
55
|
+
def log(message: str, level: str = "info") -> None:
|
|
56
|
+
"""Log a message to stderr (visible in LibreFang daemon logs)."""
|
|
57
|
+
print(f"[{level.upper()}] {message}", file=sys.stderr, flush=True)
|
|
58
|
+
|
|
59
|
+
|
|
60
|
+
class Agent:
|
|
61
|
+
"""Decorator-based Python agent framework.
|
|
62
|
+
|
|
63
|
+
Example:
|
|
64
|
+
|
|
65
|
+
agent = Agent()
|
|
66
|
+
|
|
67
|
+
@agent.on_message
|
|
68
|
+
def handle(message: str, context: dict) -> str:
|
|
69
|
+
return f"Hello! You said: {message}"
|
|
70
|
+
|
|
71
|
+
agent.run()
|
|
72
|
+
"""
|
|
73
|
+
|
|
74
|
+
def __init__(self):
|
|
75
|
+
self._handler: Optional[Callable] = None
|
|
76
|
+
self._setup: Optional[Callable] = None
|
|
77
|
+
self._teardown: Optional[Callable] = None
|
|
78
|
+
|
|
79
|
+
def on_message(self, func: Callable) -> Callable:
|
|
80
|
+
"""Register a message handler function.
|
|
81
|
+
|
|
82
|
+
The function should accept (message: str, context: dict) and return str.
|
|
83
|
+
"""
|
|
84
|
+
self._handler = func
|
|
85
|
+
return func
|
|
86
|
+
|
|
87
|
+
def on_setup(self, func: Callable) -> Callable:
|
|
88
|
+
"""Register a setup function called once before message handling."""
|
|
89
|
+
self._setup = func
|
|
90
|
+
return func
|
|
91
|
+
|
|
92
|
+
def on_teardown(self, func: Callable) -> Callable:
|
|
93
|
+
"""Register a teardown function called once after message handling."""
|
|
94
|
+
self._teardown = func
|
|
95
|
+
return func
|
|
96
|
+
|
|
97
|
+
def run(self) -> None:
|
|
98
|
+
"""Run the agent, reading input and producing output."""
|
|
99
|
+
if self._handler is None:
|
|
100
|
+
log("No message handler registered", "error")
|
|
101
|
+
sys.exit(1)
|
|
102
|
+
|
|
103
|
+
try:
|
|
104
|
+
if self._setup:
|
|
105
|
+
self._setup()
|
|
106
|
+
|
|
107
|
+
data = read_input()
|
|
108
|
+
message = data.get("message", "")
|
|
109
|
+
context = data.get("context", {})
|
|
110
|
+
|
|
111
|
+
result = self._handler(message, context)
|
|
112
|
+
|
|
113
|
+
if isinstance(result, str):
|
|
114
|
+
respond(result)
|
|
115
|
+
elif isinstance(result, dict):
|
|
116
|
+
respond(result.get("text", str(result)), result.get("metadata"))
|
|
117
|
+
else:
|
|
118
|
+
respond(str(result))
|
|
119
|
+
|
|
120
|
+
except Exception as e:
|
|
121
|
+
log(f"Agent error: {e}", "error")
|
|
122
|
+
respond(f"Error: {e}")
|
|
123
|
+
sys.exit(1)
|
|
124
|
+
finally:
|
|
125
|
+
if self._teardown:
|
|
126
|
+
try:
|
|
127
|
+
self._teardown()
|
|
128
|
+
except Exception as e:
|
|
129
|
+
log(f"Teardown error: {e}", "error")
|
|
130
|
+
|
|
131
|
+
|
|
132
|
+
# Convenience: if this file is run directly, show usage
|
|
133
|
+
if __name__ == "__main__":
|
|
134
|
+
print("LibreFang Python SDK")
|
|
135
|
+
print("====================")
|
|
136
|
+
print()
|
|
137
|
+
print("Import this module in your agent scripts:")
|
|
138
|
+
print()
|
|
139
|
+
print(" from librefang_sdk import Agent")
|
|
140
|
+
print()
|
|
141
|
+
print(" agent = Agent()")
|
|
142
|
+
print()
|
|
143
|
+
print(" @agent.on_message")
|
|
144
|
+
print(" def handle(message, context):")
|
|
145
|
+
print(" return f'You said: {message}'")
|
|
146
|
+
print()
|
|
147
|
+
print(" agent.run()")
|
|
@@ -0,0 +1,110 @@
|
|
|
1
|
+
Metadata-Version: 2.4
|
|
2
|
+
Name: librefang-sdk
|
|
3
|
+
Version: 0.5.4
|
|
4
|
+
Summary: Official Python client and SDK for the LibreFang Agent OS
|
|
5
|
+
Author-email: LibreFang Team <team@librefang.ai>
|
|
6
|
+
License-Expression: MIT
|
|
7
|
+
Project-URL: Homepage, https://librefang.ai
|
|
8
|
+
Project-URL: Repository, https://github.com/librefang/librefang
|
|
9
|
+
Project-URL: Issues, https://github.com/librefang/librefang/issues
|
|
10
|
+
Keywords: librefang,agent,ai,llm,autonomous-agent,sdk
|
|
11
|
+
Classifier: Development Status :: 4 - Beta
|
|
12
|
+
Classifier: Intended Audience :: Developers
|
|
13
|
+
Classifier: Operating System :: OS Independent
|
|
14
|
+
Classifier: Programming Language :: Python :: 3
|
|
15
|
+
Classifier: Programming Language :: Python :: 3.8
|
|
16
|
+
Classifier: Programming Language :: Python :: 3.9
|
|
17
|
+
Classifier: Programming Language :: Python :: 3.10
|
|
18
|
+
Classifier: Programming Language :: Python :: 3.11
|
|
19
|
+
Classifier: Programming Language :: Python :: 3.12
|
|
20
|
+
Classifier: Topic :: Software Development :: Libraries :: Python Modules
|
|
21
|
+
Requires-Python: >=3.8
|
|
22
|
+
Description-Content-Type: text/markdown
|
|
23
|
+
License-File: LICENSE
|
|
24
|
+
Provides-Extra: dev
|
|
25
|
+
Requires-Dist: pytest; extra == "dev"
|
|
26
|
+
Requires-Dist: pytest-asyncio; extra == "dev"
|
|
27
|
+
Dynamic: license-file
|
|
28
|
+
Dynamic: requires-python
|
|
29
|
+
|
|
30
|
+
# LibreFang Python SDK
|
|
31
|
+
|
|
32
|
+
Official Python client and SDK for the LibreFang Agent OS.
|
|
33
|
+
|
|
34
|
+
## Installation
|
|
35
|
+
|
|
36
|
+
```bash
|
|
37
|
+
pip install librefang
|
|
38
|
+
```
|
|
39
|
+
|
|
40
|
+
## Two Packages
|
|
41
|
+
|
|
42
|
+
This package provides two different interfaces:
|
|
43
|
+
|
|
44
|
+
### 1. REST API Client (`librefang.client`)
|
|
45
|
+
|
|
46
|
+
Control LibreFang remotely via its REST API.
|
|
47
|
+
|
|
48
|
+
```python
|
|
49
|
+
from librefang import Client
|
|
50
|
+
|
|
51
|
+
client = Client("http://localhost:4545")
|
|
52
|
+
|
|
53
|
+
# Create an agent
|
|
54
|
+
agent = client.agents.create(template="assistant")
|
|
55
|
+
print(f"Agent created: {agent['id']}")
|
|
56
|
+
|
|
57
|
+
# Send a message
|
|
58
|
+
reply = client.agents.message(agent["id"], "Hello!")
|
|
59
|
+
print(reply)
|
|
60
|
+
|
|
61
|
+
# Stream a response
|
|
62
|
+
for event in client.agents.stream(agent["id"], "Tell me a story"):
|
|
63
|
+
if event.get("type") == "text_delta":
|
|
64
|
+
print(event["delta"], end="", flush=True)
|
|
65
|
+
```
|
|
66
|
+
|
|
67
|
+
### 2. Agent SDK (`librefang.sdk`)
|
|
68
|
+
|
|
69
|
+
Write Python agents that run inside LibreFang.
|
|
70
|
+
|
|
71
|
+
```python
|
|
72
|
+
from librefang import Agent
|
|
73
|
+
|
|
74
|
+
agent = Agent()
|
|
75
|
+
|
|
76
|
+
@agent.on_message
|
|
77
|
+
def handle(message: str, context: dict) -> str:
|
|
78
|
+
return f"You said: {message}"
|
|
79
|
+
|
|
80
|
+
agent.run()
|
|
81
|
+
```
|
|
82
|
+
|
|
83
|
+
Or use the simple input/output functions:
|
|
84
|
+
|
|
85
|
+
```python
|
|
86
|
+
from librefang import read_input, respond
|
|
87
|
+
|
|
88
|
+
data = read_input()
|
|
89
|
+
result = f"Echo: {data['message']}"
|
|
90
|
+
respond(result)
|
|
91
|
+
```
|
|
92
|
+
|
|
93
|
+
## Examples
|
|
94
|
+
|
|
95
|
+
See the `examples/` directory for more examples:
|
|
96
|
+
|
|
97
|
+
### Client Examples
|
|
98
|
+
- `client_basic.py` - Basic REST API usage
|
|
99
|
+
- `client_streaming.py` - Streaming responses
|
|
100
|
+
|
|
101
|
+
### SDK Examples
|
|
102
|
+
- `echo_agent.py` - Simple echo agent
|
|
103
|
+
|
|
104
|
+
## Requirements
|
|
105
|
+
|
|
106
|
+
- Python 3.8+
|
|
107
|
+
|
|
108
|
+
## License
|
|
109
|
+
|
|
110
|
+
MIT
|
|
@@ -0,0 +1,12 @@
|
|
|
1
|
+
LICENSE
|
|
2
|
+
README.md
|
|
3
|
+
pyproject.toml
|
|
4
|
+
setup.py
|
|
5
|
+
librefang/__init__.py
|
|
6
|
+
librefang/librefang_client.py
|
|
7
|
+
librefang/librefang_sdk.py
|
|
8
|
+
librefang_sdk.egg-info/PKG-INFO
|
|
9
|
+
librefang_sdk.egg-info/SOURCES.txt
|
|
10
|
+
librefang_sdk.egg-info/dependency_links.txt
|
|
11
|
+
librefang_sdk.egg-info/requires.txt
|
|
12
|
+
librefang_sdk.egg-info/top_level.txt
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
|
|
@@ -0,0 +1,53 @@
|
|
|
1
|
+
[build-system]
|
|
2
|
+
requires = ["setuptools>=61.0", "wheel"]
|
|
3
|
+
build-backend = "setuptools.build_meta"
|
|
4
|
+
|
|
5
|
+
[project]
|
|
6
|
+
name = "librefang-sdk"
|
|
7
|
+
version = "0.5.4"
|
|
8
|
+
description = "Official Python client and SDK for the LibreFang Agent OS"
|
|
9
|
+
readme = "README.md"
|
|
10
|
+
license = "MIT"
|
|
11
|
+
authors = [
|
|
12
|
+
{ name = "LibreFang Team", email = "team@librefang.ai" }
|
|
13
|
+
]
|
|
14
|
+
keywords = ["librefang", "agent", "ai", "llm", "autonomous-agent", "sdk"]
|
|
15
|
+
classifiers = [
|
|
16
|
+
"Development Status :: 4 - Beta",
|
|
17
|
+
"Intended Audience :: Developers",
|
|
18
|
+
"Operating System :: OS Independent",
|
|
19
|
+
"Programming Language :: Python :: 3",
|
|
20
|
+
"Programming Language :: Python :: 3.8",
|
|
21
|
+
"Programming Language :: Python :: 3.9",
|
|
22
|
+
"Programming Language :: Python :: 3.10",
|
|
23
|
+
"Programming Language :: Python :: 3.11",
|
|
24
|
+
"Programming Language :: Python :: 3.12",
|
|
25
|
+
"Topic :: Software Development :: Libraries :: Python Modules",
|
|
26
|
+
]
|
|
27
|
+
requires-python = ">=3.8"
|
|
28
|
+
dependencies = []
|
|
29
|
+
|
|
30
|
+
[project.optional-dependencies]
|
|
31
|
+
dev = [
|
|
32
|
+
"pytest",
|
|
33
|
+
"pytest-asyncio",
|
|
34
|
+
]
|
|
35
|
+
|
|
36
|
+
[project.urls]
|
|
37
|
+
Homepage = "https://librefang.ai"
|
|
38
|
+
Repository = "https://github.com/librefang/librefang"
|
|
39
|
+
Issues = "https://github.com/librefang/librefang/issues"
|
|
40
|
+
|
|
41
|
+
[tool.setuptools.packages.find]
|
|
42
|
+
where = ["."]
|
|
43
|
+
include = ["librefang*"]
|
|
44
|
+
exclude = ["examples*"]
|
|
45
|
+
|
|
46
|
+
[tool.setuptools.package-data]
|
|
47
|
+
librefang = ["py.typed"]
|
|
48
|
+
|
|
49
|
+
[tool.pytest.ini_options]
|
|
50
|
+
testpaths = ["examples"]
|
|
51
|
+
python_files = ["*.py"]
|
|
52
|
+
python_classes = ["Test*"]
|
|
53
|
+
python_functions = ["test_*"]
|
|
@@ -0,0 +1,14 @@
|
|
|
1
|
+
from setuptools import setup
|
|
2
|
+
|
|
3
|
+
setup(
|
|
4
|
+
name="librefang",
|
|
5
|
+
version="0.5.4-20260317",
|
|
6
|
+
description="Official Python client for the LibreFang Agent OS REST API",
|
|
7
|
+
py_modules=["librefang_sdk", "librefang_client"],
|
|
8
|
+
python_requires=">=3.8",
|
|
9
|
+
classifiers=[
|
|
10
|
+
"Programming Language :: Python :: 3",
|
|
11
|
+
"License :: OSI Approved :: MIT License",
|
|
12
|
+
"Operating System :: OS Independent",
|
|
13
|
+
],
|
|
14
|
+
)
|