Use Mastra to build workflows and define agents, then use this component to run and save them on Convex.
- Run workflows asynchronously. Fire and forget from a serverless function (mutation or action).
- Track the status of the workflow. Reactive queries and run-to-completion utilities. Or just write to the database from your steps and use normal Convex reactivity.
- Resume a workflow from where it left off, after suspending it for user input.
- Full support for Mastra's step forking, joining, triggering, and more.
const storage = new ConvexStorage(components.mastra);
const vector = new ConvexVector(components.mastra);
// Uses storage to save and load messages and threads.
// Uses vector to save and query embeddings for RAG on messages.
const agent = new Agent({ memory: new Memory({ storage, vector}), ... })
// Uses storage to save and load workflow state.
const mastra = new Mastra({ storage, ...})
export const myAction = action({
args: { workflowName: v.string()},
handler: async (ctx, args) => {
// IMPORTANT:
// <- must be called before using storage or vector
storage.setCtx(ctx);
vector.setCtx(ctx);
const workflow = mastra.getWorkflow(args.workflowName);
const { runId, start } = await workflow.create(ctx);
await start({...});
}
})
- Agentic workflows, such as taking user input, calling multiple LLMs, calling third parties, etc.
- ... Everything else you want to do with Mastra.
Found a bug? Feature request? File it here.
- Provide Storage and Vector integrations for using Convex from Mastra servers.
- Enables running from both
mastra dev
andconvex dev
for fast iterations. - Enables using Convex for Agent Memory.
- Enables running from both
- Provide helpers to export functions so browsers can call them safely.
- Add a custom mutation step, for a transactional step that will always terminate without needing a retry configuration (built-in for Convex).
You'll need an existing Convex project to use the component. Convex is a hosted backend platform, including a database, serverless functions, and a ton more you can learn about here.
Run npm create convex
or follow any of the quickstarts to set one up.
Install the component package:
npm install @convex-dev/mastra
NOTE: You also need to:
- Directly install
@libsql/client
- Mark it as an external package
- Export it from a file in your /convex folder due to current bundling issues.
You can do all of this by running the following commands from the project root:
npm install -D @libsql/client
echo '{"node":{"externalPackages":["@libsql/client"]}}' > convex.json
printf '"use node";\nexport * as _ from "@libsql/client";' > convex/_workaround.ts
Create a convex.config.ts
file in your app's convex/
folder and install the component by calling use
:
// convex/convex.config.ts
import { defineApp } from "convex/server";
import mastra from "@convex-dev/mastra/convex.config";
const app = defineApp();
app.use(mastra);
export default app;
- It's important to call
storage.setCtx(ctx)
andvector.setCtx(ctx)
before using the storage or vector in an action.
"use node";
const storage = new ConvexStorage(components.mastra);
const vector = new ConvexVector(components.mastra);
// Uses storage to save and load messages and threads.
// Uses vector to save and query embeddings for RAG on messages.
const agent = new Agent({ memory: new Memory({ storage, vector}), ... })
// Uses storage to save and load workflow state.
const mastra = new Mastra({ storage, ...})
export const myAction = action({
args: { workflowName: v.string()},
handler: async (ctx, args) => {
// IMPORTANT:
// <- must be called before using storage or vector
storage.setCtx(ctx);
vector.setCtx(ctx);
const workflow = mastra.getWorkflow(args.workflowName);
const { runId, start } = await workflow.create(ctx);
await start({...});
}
})
Querying the status reactively from a non-node file:
import { query } from "./_generated/server";
import { components } from "./_generated/api";
import { v } from "convex/values";
import {
mapSerializedToMastra,
TABLE_WORKFLOW_SNAPSHOT,
} from "@convex-dev/mastra/mapping";
export const getStatus = query({
args: { runId: v.string() },
handler: async (ctx, args) => {
const doc = await ctx.runQuery(
components.mastra.storage.storage.loadSnapshot,
{
workflowName: "weatherToOutfitWorkflow",
runId: args.runId,
}
);
if (!doc) {
return null;
}
const snapshot = mapSerializedToMastra(TABLE_WORKFLOW_SNAPSHOT, doc);
const { childStates, activePaths, suspendedSteps } = snapshot.snapshot;
return { childStates, activePaths, suspendedSteps };
},
});
See more example usage in example.ts.
- For local development, you need to run
mastra dev
in Node 20, butconvex dev
in Node 18. If you see issues about syscalls at import time, try using the cloud dev environment instead. - Currently you can only interact with Mastra classes from Node actions, so you can't start them from a mutation without doing it indirectly via the Scheduler or Workpool by enqueuing the node action to run.
- To reactively query for the status of a workflow, you need to call the component API directly. There's an example above and in v8Runtime.ts.
- Validate the Storage and Vector implementations (from Convex).
- Ensure @mastra/memory can be bundled in Convex.
- Support using Storage and Vector from
mastra dev
. - Configurable vacuuming of workflow state.
- Support queries on workflow state without hitting the component directly.
- Support exposing the same
hono
HTTP API as Mastra servers. - Better logging and tracing.
- Provide a Mutation Step to avoid the v8 action and is executed exactly once.
- Workflows currently only run in Node Actions. You can create/start/resume them from anywhere, but each step will be executed in the node runtime. This is a bit slower and more expensive than running in the default runtime.
- Using the
ConvexStorage
from Mastra doesn't share state with workflows made via the Component. They're currently stored in separate tables with different schemas.
If you see an error like this:
Uncaught Failed to analyze _deps/node/4QMS5IZK.js: Cannot find module '@libsql/linux-arm64-gnu'
You need to add @libsql/client
to the externalPackages
in a convex.json
file in the root of your project:
{
"node": {
"externalPackages": ["@libsql/client"]
}
}
If that still doesn't solve it, add a convex/_workaround.ts
file:
"use node";
export * as _ from "@libsql/client";
If you see an error like this:
✘ [ERROR] No loader is configured for ".node" files: node_modules/onnxruntime-node/bin/napi-v3/win32/arm64/onnxruntime_binding.nodel
You're likely importing some node package through a dependency that isn't
supported. One workaround is to add it as an explicit dependency, then add it
to the externalPackages
in a convex.json
file in the root of your project,
then export something from it, similar to @libsql/client
above
You can also try deleting your node_modules
and package-lock.json
and
re-installing using node 18.
✘ [ERROR] Could not resolve "assert"
node_modules/sonic-boom/index.js:8:23:
8 │ const assert = require('assert')
╵ ~~~~~~~~
The package "assert" wasn't found on the file system but is built into node. Are you trying to
bundle for node? You can use "platform: 'node'" to do that, which will remove this error.
✖ It looks like you are using Node APIs from a file without the "use node" directive.
This is because you're using a Node API in a file that doesn't have
"use node";
as the first line in the file
Or you're importing a file in your convex/ directory that imports from a
node dependency that doesn't have the "use node"
directive.
To fix this, add the "use node"
directive to the file. Note: these files can
only have actions, since mutations and queries only run in the default runtime.