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.
@@ -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,4 @@
1
+
2
+ [dev]
3
+ pytest
4
+ pytest-asyncio
@@ -0,0 +1,3 @@
1
+ librefang
2
+ librefang_client
3
+ librefang_sdk
@@ -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,4 @@
1
+ [egg_info]
2
+ tag_build =
3
+ tag_date = 0
4
+
@@ -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
+ )