Mental Model
- •Components are isolated mini backends with their own schema, tables, file storage, and functions.
- •Components MUST NOT access app tables/functions/env unless passed explicitly.
- •Calls into components are transactional with the caller, but component mutations are sub-transactions.
Installing Components
- •Install package:
npm i @convex-dev/<component>. - •Add
convex/convex.config.ts:- •
import { defineApp } from "convex/server"; - •
app.use(component); useapp.use(component, { name: "custom" })for multiple instances.
- •
- •You MUST run
npx convex devto generate component code. - •Access via
components.<name>inconvex/_generated/api.
Calling Component APIs
- •Use
ctx.runQuery/Mutation/Actionwithcomponents.<name>.<fn>. - •Public component functions MUST NOT be called directly from clients (they are internal references).
- •Queries remain reactive; mutations are transactional.
Transaction Semantics
- •Top-level mutation commits all writes across components together.
- •If a component mutation throws, only its writes are rolled back; the caller MAY catch and continue.
Component API Differences
- •
components.<name>exposes ONLY public component functions. - •
Idtypes cross boundaries asstring; You MUST NOT usev.id("table")for external tables. - •Each component has its own
_generateddirectory; You MUST use the app'scomponentsreferences.
Environment Variables
- •Components MUST NOT access
process.envdirectly. - •You MUST pass env values as arguments from the app, or store config in a component table.
HTTP Actions
- •Components MUST NOT expose routes directly; app MUST mount handlers in
convex/http.ts.
Auth in Components
- •
ctx.authis NOT available inside component functions. - •You MUST authenticate in the app and pass identifiers (userId) to component functions.
Pagination
- •Built-in
.paginate()is NOT supported inside components. - •You SHOULD use
convex-helperspaginator andusePaginatedQueryfrom convex-helpers if needed.
Authoring Components
- •Component folder MUST include
convex.config.ts,schema.ts, functions, and_generated. - •
defineComponent("name")defines component; usecomponent.use(...)for child components. - •Local components MAY live in
convex/components/or any folder. - •NPM components SHOULD export:
- •
@pkg/convex.config.js - •
@pkg/_generated/component.js - •
@pkg/testhelpers
- •
Function Handles
- •You SHOULD use
createFunctionHandle(api.foo.bar)to pass callbacks across boundaries. - •Handles are strings; use
v.string()validators and cast back toFunctionHandle.
Testing Components
- •You SHOULD register component with
convex-testusing component schema/modules or provided test helpers. - •For component packages, You SHOULD use
@pkg/testregister helper.
Best Practices
- •You MUST always validate args/returns on public component functions.
- •You SHOULD prefer app-level wrappers to add auth/rate limiting when re-exporting component APIs.
- •You SHOULD use a single-row globals/config table for static configuration.