sqlite-schema-diff
A schema-first approach to SQLite migrations. Define your schema in .sql files, and let the tool figure out what changed.
Why?
Traditional migrations are error-prone and hard to maintain. Instead:
- Define your desired schema in SQL files
- Diff against your database to see what changed
- Apply changes automatically
No more numbered migration files. No more merge conflicts. Just SQL.
[!WARNING]
This tool is under active development. While it has been tested, not every edge case is covered. Always back up your database before applying changes. You are responsible for your data.
Installation
go install github.com/mizuchilabs/sqlite-schema-diff@latest
Or build from source:
git clone https://github.com/mizuchilabs/sqlite-schema-diff
cd sqlite-schema-diff
go build -o sqlite-schema-diff ./cmd
Quick Start
# Define your schema
cat > schema/users.sql << 'EOF'
CREATE TABLE users (
id INTEGER PRIMARY KEY AUTOINCREMENT,
email TEXT NOT NULL UNIQUE,
created_at TEXT DEFAULT CURRENT_TIMESTAMP
);
CREATE INDEX idx_users_email ON users(email);
EOF
# Preview changes
sqlite-schema-diff diff --database app.db --schema ./schema
# Apply changes
sqlite-schema-diff apply --database app.db --schema ./schema
CLI Reference
diff — Preview changes
sqlite-schema-diff diff --database app.db --schema ./schema
sqlite-schema-diff diff --database app.db --schema ./schema --sql # Output raw SQL
apply — Apply changes
sqlite-schema-diff apply --database app.db --schema ./schema
| Flag |
Description |
--dry-run |
Show what would happen without applying |
--force |
Skip confirmation for destructive changes |
--skip-destructive |
Skip DROP operations |
--backup=false |
Disable automatic backup |
dump — Export existing schema
sqlite-schema-diff dump --database app.db --output ./schema
Library Usage
import (
"database/sql"
"log"
"github.com/mizuchilabs/sqlite-schema-diff/pkg/diff"
_ "modernc.org/sqlite"
)
func main() {
// Open your database connection
db, err := sql.Open("sqlite", "app.db")
if err != nil {
log.Fatal(err)
}
defer db.Close()
// Compare and get changes
changes, err := diff.Compare(db, "./schema")
if err != nil {
log.Fatal(err)
}
// Check what's changing
for _, c := range changes {
fmt.Printf("%s: %s (destructive: %v)\n", c.Type, c.Description, c.Destructive)
}
// Generate SQL without applying
sql := diff.GenerateSQL(changes)
// Apply changes
err = diff.Apply(db, "./schema", diff.ApplyOptions{
BackupPath: "app.db.backup", // empty string = no backup
SkipDestructive: false,
})
}
Available Functions
| Function |
Description |
Compare(db, schemaDir) |
Diff database against SQL files |
CompareDatabases(fromDB, toDB) |
Diff two databases |
GenerateSQL(changes) |
Generate migration SQL |
HasDestructive(changes) |
Check for destructive changes |
Apply(db, schemaDir, opts) |
Apply changes to database |
Parser Functions
| Function |
Description |
parser.FromDB(db) |
Extract schema from open database |
parser.FromSQL(sql) |
Parse schema from SQL string |
parser.FromDirectory(dir) |
Load schema from directory of .sql files |
Supported Objects
- Tables (with columns, constraints, foreign keys)
- Indexes
- Views
- Triggers
Destructive Changes
Operations that may lose data are flagged as destructive:
| Operation |
Risk |
DROP TABLE |
Deletes table and all data |
DROP COLUMN |
Loses column data |
RECREATE TABLE |
Required for some alterations |
By default, the CLI:
- Creates a backup before applying (
app.db.backup)
- Prompts for confirmation on destructive changes
Use --skip-destructive to safely apply only additive changes.
Schema Organization
Organize your .sql files however you like:
schema/
├── tables/
│ ├── users.sql
│ └── posts.sql
├── indexes.sql
└── triggers.sql
All files are merged. Each object name must be unique across all files.
Examples
See examples/ directory for working examples.
License
Apache License 2.0 - see LICENSE for details.
Contributing
Contributions are welcome! Please feel free to submit a Pull Request.