Plugins (Beta)
Beta Feature: The plugin system is currently in beta. APIs may change in future releases.
Plugins extend TailorDB types by automatically generating additional types, executors, and output files based on your type definitions.
Overview
When you run tailor-sdk generate, the SDK:
- Loads all TailorDB types with plugin attachments
- Passes each type to the attached plugins
- Generates additional types and executors based on plugin output
- Writes all generated files to the appropriate locations
This enables plugins to create derived functionality based on your application's schema.
Configuration
Registering Plugins
Define plugins in tailor.config.ts using definePlugins():
import { defineConfig, definePlugins } from "@tailor-platform/sdk";
import myPlugin from "./plugins/my-plugin";
export const plugins = definePlugins(myPlugin);
export default defineConfig({
name: "my-app",
// ...
});Important: The plugins export must be a named export (not default).
Attaching Plugins to Types
Use the .plugin() method to attach plugins to specific types:
import { db } from "@tailor-platform/sdk";
export const user = db
.type("User", {
name: db.string(),
email: db.string(),
})
.plugin({
"@example/my-plugin": {},
});Plugin Configuration
Some plugins accept per-type configuration:
export const customer = db
.type("Customer", {
name: db.string(),
// ...
})
.plugin({
"@example/soft-delete": {
archiveReason: true,
retentionDays: 90,
},
});Per-type Config Requirement
Per-type config is optional by default. Plugin authors can change this with typeConfigRequired (boolean or function). When a function is used, it receives the plugin-level config from definePlugins().
Global Plugin Configuration
Plugins can also accept global configuration via definePlugins():
import { definePlugins } from "@tailor-platform/sdk";
import { softDeletePlugin } from "./plugins/soft-delete";
export const plugins = definePlugins(
// Custom plugin with global config (factory function)
softDeletePlugin({
archiveTablePrefix: "Deleted_",
defaultRetentionDays: 90,
}),
);Generated Output
Plugins can generate:
- Types: Additional TailorDB types (e.g.,
CustomerHistory,Deleted_Customer) - Executors: Event handlers triggered by record changes
- Field Extensions: Additional fields added to the source type
- Output Files: TypeScript code and other files via generation-time hooks
Generated files are placed under .tailor-sdk/<plugin-id>/ (the plugin ID is sanitized, e.g. @example/soft-delete → example-soft-delete), such as:
.tailor-sdk/example-soft-delete/types.tailor-sdk/example-soft-delete/executors
Plugin Lifecycle
Plugins have 5 hooks across two lifecycle phases. Each hook fires at a specific point in the tailor-sdk generate pipeline:
tailor-sdk generate
│
├─ Load TailorDB types
│ ├─ onTypeLoaded ← per type with .plugin() attached
│ └─ onNamespaceLoaded ← once per namespace (namespace plugins)
│
├─ Resolve Auth
│
├─ onTailorDBReady ← all types finalized
│
├─ Load Resolvers
│
├─ onResolverReady ← all resolvers finalized
│
├─ Load Executors
│
└─ onExecutorReady ← all executors finalizedDefinition-time hooks
| Hook | Trigger | Can do |
|---|---|---|
onTypeLoaded | Each type with .plugin() attached | Generate types, resolvers, executors; extend source type fields |
onNamespaceLoaded | Once per namespace | Generate types, resolvers, executors |
These hooks produce TailorDB types, resolvers, and executors that become part of the application. Requires importPath on the plugin.
Generation-time hooks
| Hook | Available data | Can do |
|---|---|---|
onTailorDBReady | TailorDB types, Auth | Write output files |
onResolverReady | TailorDB types, Resolvers, Auth | Write output files |
onExecutorReady | TailorDB types, Resolvers, Executors, Auth | Write output files |
These hooks receive all finalized data and produce output files (TypeScript code, etc.). They replace the previous standalone defineGenerators() approach. No importPath required.
A plugin can implement hooks from either or both phases.
Creating Custom Plugins
See Custom Plugins for the full hook reference and examples.