@@ -34,55 +34,34 @@ npm i @pompeii-labs/magma
34
34
``` ts
35
35
import { MagmaAgent } from " @pompeii-labs/magma" ;
36
36
37
- // That's it! You've got a working agent
38
- const agent = new MagmaAgent ();
39
-
40
- // Want to give it some personality? Add system prompts:
41
- agent .fetchSystemPrompts = () => [{
42
- role: " system" ,
43
- content: " You are a friendly assistant who loves dad jokes"
44
- }];
45
-
46
- // Need the agent to do something? Add tools:
47
- agent .fetchTools = () => [{
48
- name: " tell_joke" ,
49
- description: " Tell a dad joke" ,
50
- target : async () => {
51
- return " Why don't eggs tell jokes? They'd crack up! 🥚" ;
37
+ // Magma Agents are class based, so you can extend them with your own methods
38
+ class MyAgent extends MagmaAgent {
39
+
40
+ // Want to give it some personality? Add system prompts:
41
+ getSystemPrompts() {
42
+ return [{
43
+ role: " system" ,
44
+ content: " You are a friendly assistant who loves dad jokes"
45
+ }];
52
46
}
53
- }];
47
+ }
48
+
49
+ // That's it! You've got a working agent
50
+ const myAgent = new MyAgent ();
54
51
55
52
// Run it:
56
- const reply = await agent .main ();
53
+ const reply = await myAgent .main ();
57
54
console .log (reply .content );
58
55
```
59
56
60
57
## 🔥 Key Features
61
58
62
59
- ** Simple** : Build agents in minutes with minimal code
63
60
- ** Flexible** : Use any AI provider (OpenAI, Anthropic, Groq)
64
- - ** Hosted** : Deploy your agents in seconds with the MagmaDeploy platform (coming soon )
61
+ - ** Hosted** : Deploy your agents in seconds with the [ MagmaDeploy platform] ( https://magmadeploy.com )
65
62
- ** Powerful** : Add tools and middleware when you need them
66
63
- ** Observable** : See exactly what your agent is doing
67
64
68
- ## 🚀 MagmaFlow
69
-
70
- Want even more power? MagmaFlow gives you instant access to:
71
- - Voice input/output
72
- - Streaming responses
73
- - Tool execution
74
- - Usage tracking
75
- - And more!
76
-
77
-
78
- ``` ts
79
- const agent = new MagmaAgent ({
80
- apiKey: " mf_..." // Get your key at magmaflow.dev
81
- });
82
- ```
83
-
84
- > 🎉 MagmaFlow is currently in private beta! [ Join the waitlist] ( https://magmaflow.dev ) to get early access.
85
-
86
65
## 🛠 Examples
87
66
88
67
### Add Tools
@@ -118,7 +97,10 @@ class MyAgent extends MagmaAgent {
118
97
})
119
98
async searchDatabase(call : MagmaToolCall ) {
120
99
const { query, filters } = call .fn_args ;
121
- // Implementation
100
+
101
+ const results = await this .searchDatabase (query , filters );
102
+
103
+ return " Here are the results of your search: " + JSON .stringify (results );
122
104
}
123
105
}
124
106
```
@@ -128,9 +110,22 @@ Middleware is a novel concept to Magma. It allows you to add custom logic to you
128
110
129
111
This is a great way to add custom logging, validation, data sanitization, etc.
130
112
113
+ ** Types** :
114
+ - "preCompletion": Runs before the LLM call is made, takes in a MagmaUserMessage
115
+ - "onCompletion": Runs after the agent generates a text response, takes in a MagmaAssistantMessage
116
+ - "preToolExecution": Runs before a tool is executed, takes in a MagmaToolCall
117
+ - "onToolExecution": Runs after a tool is executed, takes in a MagmaToolResult
118
+
131
119
** Important Notes** :
132
120
- You can have unlimited middleware methods
133
- - Middleware methods must return a string
121
+ - Middleware methods can manipulate the message they take in
122
+ - Middleware methods can throw errors to adjust the flow of the agent
123
+
124
+ ** Error Handling** :
125
+ - If preCompletion middleware throws an error, the error message is supplied as if it were the assistant message. The user and assistant messages are also removed from the conversation history
126
+ - If onCompletion middleware throws an error, the error message is supplied to the LLM, and it tries to regenerate a response. The assistant message is not added to the conversation history
127
+ - If preToolExecution middleware throws an error, the error message is supplied as if it were the response from the tool
128
+ - If onToolExecution middleware throws an error, the error message is supplied as if it were the response from the tool
134
129
``` ts
135
130
import { MagmaAgent } from " @pompeii-labs/magma" ;
136
131
import { middleware } from " @pompeii-labs/magma/decorators" ;
@@ -144,28 +139,32 @@ class MyAgent extends MagmaAgent {
144
139
@middleware (" onCompletion" )
145
140
async logBeforeCompletion(message ) {
146
141
if (message .content .includes (" bad word" )) {
147
- return " You just used a bad word, please try again." ;
142
+ throw new Error ( " You just used a bad word, please try again." ) ;
148
143
}
149
144
}
150
145
}
151
146
```
152
147
153
148
### Schedule Jobs
154
149
Jobs allow you to schedule functions within your agent. Jobs conform to the standard UNIX cron syntax (https://crontab.guru/ ).
150
+
151
+ ** Important Notes** :
152
+ - Jobs should be static methods, so they can run without instantiating the agent.
153
+ - Jobs do not take in any parameters, and they do not return anything.
155
154
``` ts
156
155
import { MagmaAgent } from " @pompeii-labs/magma" ;
157
156
import { job } from " @pompeii-labs/magma/decorators" ;
158
157
159
158
class MyAgent extends MagmaAgent {
160
159
// Run every day at midnight
161
160
@job (" 0 0 * * *" )
162
- async dailyCleanup() {
161
+ static async dailyCleanup() {
163
162
await this .cleanDatabase ();
164
163
}
165
164
166
165
// Run every hour with timezone
167
166
@job (" 0 * * * *" , { timezone: " America/New_York" })
168
- async hourlySync() {
167
+ static async hourlySync() {
169
168
await this .syncData ();
170
169
}
171
170
}
@@ -174,7 +173,9 @@ class MyAgent extends MagmaAgent {
174
173
### Expose Hooks
175
174
Hooks allow you to expose your agent as an API. Any method decorated with @hook will be exposed as an endpoint.
176
175
176
+
177
177
** Important Notes** :
178
+ - Hooks are static methods, so they can run without instantiating the agent.
178
179
- Hooks are exposed at ` /hooks/{hook_name} ` in the Magma API
179
180
- The only parameter to hook functions is the request object, which is an instance of ` express.Request `
180
181
``` ts
@@ -185,13 +186,18 @@ import { Request } from "express";
185
186
class MyAgent extends MagmaAgent {
186
187
187
188
@hook (' notification' )
188
- async handleNotification(req : Request ) {
189
+ static async handleNotification(req : Request ) {
189
190
await this .processNotification (req .body );
190
191
}
191
192
}
192
193
```
193
194
194
195
### Use Different Providers
196
+ You can use any supported provider by setting the providerConfig.
197
+
198
+ ** Important Notes** :
199
+ - You can set the providerConfig in the constructor, or by calling ` setProviderConfig `
200
+ - You do not need to adjust any of your tools, middleware, jobs, or hooks to use a different provider. Magma will handle the rest.
195
201
``` ts
196
202
class Agent extends MagmaAgent {
197
203
constructor () {
@@ -219,11 +225,16 @@ class Agent extends MagmaAgent {
219
225
```
220
226
221
227
### State Management
222
- Every agent has a state object that you can use to store data.
228
+ Every agent has a state object that you can use to store data. You can store any data type, and it will be persisted between calls.
229
+ You can also choose to use fields on the agent class to store data.
223
230
224
231
State does not get passed into LLM calls, so it's a good place to store data that you want to persist between calls / sensitive data.
232
+
225
233
``` ts
226
234
class MyAgent extends MagmaAgent {
235
+ // Using a field to store data
236
+ myQuery = " Hello, world!" ;
237
+
227
238
async setup() {
228
239
// Initialize state
229
240
this .state .set (" counter" , 0 );
@@ -240,7 +251,16 @@ class MyAgent extends MagmaAgent {
240
251
@tool ({ name: " api_call" })
241
252
async apiCall() {
242
253
const access_token = this .state .get (" access_token" );
243
- // Make API call
254
+ const response = await fetch (" https://myapi.com/data" , {
255
+ headers: {
256
+ " Authorization" : ` Bearer ${access_token } `
257
+ },
258
+ body: JSON .stringify ({
259
+ query: this .myQuery
260
+ })
261
+ });
262
+
263
+ return JSON .stringify (response .json ());
244
264
}
245
265
}
246
266
```
@@ -302,33 +322,6 @@ class MyAgent extends MagmaAgent {
302
322
async onStreamChunk(chunk : MagmaStreamChunk ) {
303
323
console .log (" Received chunk:" , chunk .content );
304
324
}
305
-
306
- // MagmaFlow Handlers
307
- async onConnect() {
308
- console .log (" Connected to MagmaFlow!" );
309
- }
310
-
311
- // Handle agent disconnection from MagmaFlow
312
- async onDisconnect() {
313
- console .log (" Disconnected from MagmaFlow" );
314
- }
315
-
316
- // Handle incoming audio chunks
317
- async onAudioChunk(chunk : Buffer ) {
318
- // Process incoming audio
319
- await this .processAudioChunk (chunk );
320
- }
321
-
322
- // Handle audio stream completion
323
- async onAudioCommit() {
324
- // Audio stream complete
325
- await this .finalizeAudioProcessing ();
326
- }
327
-
328
- // Handle request abortion
329
- async onAbort() {
330
- await this .cleanup ();
331
- }
332
325
}
333
326
```
334
327
0 commit comments