| name | PocketBase Migrations |
| description | Schema migrations and versioning for PocketBase. Use when creating migrations, managing schema versions, syncing collections between environments, using automigrate, or creating collections programmatically. Covers migrate commands, migration file format, snapshot imports, and the _migrations tracking table. |
PocketBase Migrations & Schema Versioning
Overview
PocketBase supports two approaches to schema management:
- Auto-migrate (default in dev) — Dashboard changes auto-generate migration files in
pb_migrations/
- Manual migrations — write migration files by hand for full control
CLI Commands
./pocketbase migrate create "add_posts_collection"
./pocketbase migrate up
./pocketbase migrate down
./pocketbase migrate collections
./pocketbase migrate history-sync
Auto-migrate Mode
Enabled by default. When you change collections in the Dashboard, PocketBase auto-generates migration files in pb_migrations/.
./pocketbase serve
./pocketbase serve --automigrate=0
Workflow:
- Develop with auto-migrate ON — use Dashboard to design schema
- Migration files are auto-generated in
pb_migrations/
- Commit these files to git
- Deploy: migrations run automatically on
serve start
- In production: use
--automigrate=0 to prevent Dashboard changes from generating new migrations
Migration File Format
migrate(
function(app) {
var collection = new Collection({
name: "posts",
type: "base",
fields: [
{ name: "title", type: "text", required: true },
{ name: "body", type: "editor" },
{ name: "author", type: "relation", collectionId: "USERS_COLLECTION_ID", cascadeDelete: false, maxSelect: 1, required: true },
{ name: "status", type: "select", values: ["draft", "published", "archived"] },
{ name: "published_at", type: "date" },
{ name: "tags", type: "relation", collectionId: "TAGS_COLLECTION_ID", maxSelect: 0 }
],
indexes: [
"CREATE INDEX idx_posts_author ON posts (author)",
"CREATE INDEX idx_posts_status ON posts (status)",
"CREATE UNIQUE INDEX idx_posts_title ON posts (title)"
],
listRule: "",
viewRule: "",
createRule: "@request.auth.id != ''",
updateRule: "author = @request.auth.id",
deleteRule: "author = @request.auth.id"
})
app.save(collection)
},
function(app) {
var collection = app.findCollectionByNameOrId("posts")
app.delete(collection)
}
)
Important: the app inside migrations is a transactional instance. If any error occurs, the entire migration is rolled back.
Creating Collections Programmatically
Base collection
var collection = new Collection({
name: "posts",
type: "base",
fields: [
{ name: "title", type: "text", required: true, min: 3, max: 200 },
{ name: "slug", type: "text", required: true, autogenerate: { pattern: "slugify(title)" } },
{ name: "body", type: "editor" },
{ name: "cover", type: "file", maxSelect: 1, maxSize: 5242880, mimeTypes: ["image/jpeg", "image/png", "image/webp"] },
{ name: "views", type: "number", min: 0 },
{ name: "metadata", type: "json", maxSize: 2000000 },
{ name: "featured", type: "bool" },
{ name: "published_at", type: "date" }
]
})
app.save(collection)
Auth collection
var collection = new Collection({
name: "users",
type: "auth",
fields: [
{ name: "name", type: "text", required: true },
{ name: "avatar", type: "file", maxSelect: 1, maxSize: 5242880 },
{ name: "role", type: "select", values: ["user", "editor", "admin"], required: true }
],
passwordAuth: { enabled: true, identityFields: ["email", "username"] },
oauth2: { enabled: true },
otp: { enabled: false },
mfa: { enabled: false },
authToken: { duration: 604800 }
})
app.save(collection)
View collection
var collection = new Collection({
name: "posts_stats",
type: "view",
viewQuery: "SELECT p.id, p.title, COUNT(c.id) as comments_count, p.views FROM posts p LEFT JOIN comments c ON c.post = p.id GROUP BY p.id",
listRule: "",
viewRule: ""
})
app.save(collection)
Modifying Existing Collections
migrate(function(app) {
var collection = app.findCollectionByNameOrId("posts")
collection.fields.add({
name: "subtitle",
type: "text",
max: 500
})
collection.fields.removeByName("old_field")
collection.listRule = "@request.auth.id != ''"
collection.viewRule = ""
collection.indexes.push("CREATE INDEX idx_posts_subtitle ON posts (subtitle)")
app.save(collection)
}, function(app) {
var collection = app.findCollectionByNameOrId("posts")
collection.fields.removeByName("subtitle")
app.save(collection)
})
Raw SQL in Migrations
migrate(function(app) {
app.db().newQuery("ALTER TABLE posts ADD COLUMN legacy_id TEXT DEFAULT ''").execute()
app.db().newQuery("UPDATE posts SET legacy_id = id WHERE legacy_id = ''").execute()
}, function(app) {
app.db().newQuery("ALTER TABLE posts DROP COLUMN legacy_id").execute()
})
Warning: raw SQL bypasses PocketBase's schema cache. Run migrate collections afterward to re-sync if needed.
Settings & Superuser in Migrations
Initialize app settings
onBootstrap(function(e) {
var settings = e.app.settings()
settings.meta.appName = "My App"
settings.meta.appURL = "https://myapp.com"
settings.meta.senderName = "My App"
settings.meta.senderAddress = "noreply@myapp.com"
settings.smtp.enabled = true
settings.smtp.host = "smtp.example.com"
settings.smtp.port = 587
settings.smtp.username = $os.getenv("SMTP_USER")
settings.smtp.password = $os.getenv("SMTP_PASS")
e.app.save(settings)
return e.next()
})
Create superuser in migration
migrate(function(app) {
var superusers = app.findCollectionByNameOrId("_superusers")
var record = new Record(superusers)
var email = $os.getenv("PB_ADMIN_EMAIL")
var password = $os.getenv("PB_ADMIN_PASSWORD")
if (!email || !password) {
throw new Error("PB_ADMIN_EMAIL and PB_ADMIN_PASSWORD env vars are required")
}
record.set("email", email)
record.set("password", password)
app.save(record)
})
Snapshot Migrations
./pocketbase migrate collections generates a complete snapshot — useful for:
- Bootstrapping a new environment
- Resetting migration history
- Reviewing full schema in one file
The generated file uses app.importCollections(collections) which supports two modes:
- Default (merge/extend): adds new collections and fields, updates existing ones, doesn't delete anything
- Delete missing:
app.importCollections(collections, true) — deletes collections/fields not in the snapshot
_migrations Table
PocketBase tracks applied migrations in the internal _migrations table:
id — auto-generated
file — migration filename
applied — timestamp
migrate history-sync marks all existing migration files as applied without running them — useful when importing an existing database.
Best Practices
- Dev: use auto-migrate + Dashboard for schema design, commit generated files
- Staging/Prod: deploy with
--automigrate=0, migrations run on startup
- Always write DOWN migrations — reversibility saves you when things go wrong
- One concern per migration — don't mix unrelated schema changes
- Test migrations: apply on a copy of production data before deploying
- Use
migrate collections periodically to snapshot current state for documentation
- Never edit applied migrations — create a new migration to fix issues
- Seed data: prefer a dedicated migration for one-time initial data; if using
onBootstrap, make the seed logic idempotent (existence checks/upserts) because bootstrap runs on every app start