README ¶
migrate
Database migrations written in Go. Use as CLI or import as library.
- Migrate reads migrations from sources and applies them in correct order to a database.
- Drivers are "dumb", migrate glues everything together and makes sure the logic is bulletproof.
(Keeps the drivers lightweight, too.) - Database drivers don't assume things or try to correct user input. When in doubt, fail.
Looking for v1?
Databases
Database drivers run migrations. Add a new database?
- PostgreSQL
- Cassandra (todo #164)
- SQLite (todo #165)
- MySQL/ MariaDB
- Neo4j (todo #167)
- Ql (todo #168)
- MongoDB (todo #169)
- CrateDB (todo #170)
- Shell (todo #171)
- Google Cloud Spanner (todo #172)
Migration Sources
Source drivers read migrations from local or remote sources. Add a new source?
- Filesystem - read from fileystem (always included)
- Go-Bindata - read from embedded binary data (jteeuwen/go-bindata)
- Github - read from remote Github repositories
- AWS S3 - read from Amazon Web Services S3 (todo #162)
- Google Cloud Storage - read from Google Cloud Platform Storage (todo #163)
CLI usage
- Simple wrapper around this library.
- Handles ctrl+c (SIGINT) gracefully.
- No config search paths, no config files, no magic ENV var injections.
$ brew install migrate --with-postgres
$ migrate -database postgres://localhost:5432/database up 2
Use in your Go project
- API is stable and frozen for this release (v3.x).
- Package migrate has no external dependencies.
- Only import the drivers you need. (check dependency_tree.txt for each driver)
- To help prevent database corruptions, it supports graceful stops via
GracefulStop chan bool
. - Bring your own logger.
- Uses
io.Reader
streams internally for low memory overhead. - Thread-safe and no goroutine leaks.
import (
"github.com/mattes/migrate"
_ "github.com/mattes/migrate/database/postgres"
_ "github.com/mattes/migrate/source/github"
)
func main() {
m, err := migrate.New(
"github://mattes:personal-access-token@mattes/migrate_test",
"postgres://localhost:5432/database?sslmode=enable")
m.Steps(2)
}
Want to use an existing database client?
import (
"database/sql"
_ "github.com/lib/pq"
"github.com/mattes/migrate"
"github.com/mattes/migrate/database/postgres"
)
func main() {
db, err := sql.Open("postgres", "postgres://localhost:5432/database?sslmode=enable")
driver, err := postgres.WithInstance(db, &postgres.Config{})
m, err := migrate.NewWithDatabaseInstance(
"file:///migrations",
"postgres", driver)
m.Steps(2)
}
Migration files
Each migration has an up and down migration. Why?
1481574547_create_users_table.up.sql
1481574547_create_users_table.down.sql
Best practices: How to write migrations.
Development and Contributing
Yes, please! Makefile
is your friend,
read the development guide.
Also have a look at the FAQ.
Alternatives
https://bitbucket.org/liamstask/goose, https://github.com/tanel/dbmigrate,
https://github.com/BurntSushi/migration, https://github.com/DavidHuie/gomigrate,
https://github.com/rubenv/sql-migrate
Documentation ¶
Overview ¶
Package migrate reads migrations from sources and runs them against databases. Sources are defined by the `source.Driver` and databases by the `database.Driver` interface. The driver interfaces are kept "dump", all migration logic is kept in this package.
Index ¶
- Variables
- func FilterCustomQuery(u *nurl.URL) *nurl.URL
- type ErrDirty
- type ErrShortLimit
- type Logger
- type Migrate
- func New(sourceUrl, databaseUrl string) (*Migrate, error)
- func NewWithDatabaseInstance(sourceUrl string, databaseName string, databaseInstance database.Driver) (*Migrate, error)
- func NewWithInstance(sourceName string, sourceInstance source.Driver, databaseName string, ...) (*Migrate, error)
- func NewWithSourceInstance(sourceName string, sourceInstance source.Driver, databaseUrl string) (*Migrate, error)
- func (m *Migrate) Close() (source error, database error)
- func (m *Migrate) Down() error
- func (m *Migrate) Drop() error
- func (m *Migrate) Force(version int) error
- func (m *Migrate) Migrate(version uint) error
- func (m *Migrate) Run(migration ...*Migration) error
- func (m *Migrate) Steps(n int) error
- func (m *Migrate) Up() error
- func (m *Migrate) Version() (version uint, dirty bool, err error)
- type Migration
- type MultiError
Examples ¶
Constants ¶
This section is empty.
Variables ¶
var ( ErrNoChange = fmt.Errorf("no change") ErrNilVersion = fmt.Errorf("no migration") ErrLocked = fmt.Errorf("database locked") ErrLockTimeout = fmt.Errorf("timeout: can't acquire database lock") )
var DefaultBufferSize = uint(100000)
DefaultBufferSize sets the in memory buffer size (in Bytes) for every pre-read migration (see DefaultPrefetchMigrations).
var DefaultLockTimeout = 15 * time.Second
DefaultLockTimeout sets the max time a database driver has to acquire a lock.
var DefaultPrefetchMigrations = uint(10)
DefaultPrefetchMigrations sets the number of migrations to pre-read from the source. This is helpful if the source is remote, but has little effect for a local source (i.e. file system). Please note that this setting has a major impact on the memory usage, since each pre-read migration is buffered in memory. See DefaultBufferSize.
Functions ¶
Types ¶
type ErrShortLimit ¶
type ErrShortLimit struct {
Short uint
}
ErrShortLimit is an error returned when not enough migrations can be returned by a source for a given limit.
func (ErrShortLimit) Error ¶
func (e ErrShortLimit) Error() string
Error implements the error interface.
type Logger ¶
type Logger interface { // Printf is like fmt.Printf Printf(format string, v ...interface{}) // Verbose should return true when verbose logging output is wanted Verbose() bool }
Logger is an interface so you can pass in your own logging implementation.
type Migrate ¶
type Migrate struct { // Log accepts a Logger interface Log Logger // GracefulStop accepts `true` and will stop executing migrations // as soon as possible at a safe break point, so that the database // is not corrpupted. GracefulStop chan bool // PrefetchMigrations defaults to DefaultPrefetchMigrations, // but can be set per Migrate instance. PrefetchMigrations uint // LockTimeout defaults to DefaultLockTimeout, // but can be set per Migrate instance. LockTimeout time.Duration // contains filtered or unexported fields }
func New ¶
New returns a new Migrate instance from a source URL and a database URL. The URL scheme is defined by each driver.
Example ¶
Output:
func NewWithDatabaseInstance ¶
func NewWithDatabaseInstance(sourceUrl string, databaseName string, databaseInstance database.Driver) (*Migrate, error)
NewWithDatabaseInstance returns a new Migrate instance from a source URL and an existing database instance. The source URL scheme is defined by each driver. Use any string that can serve as an identifier during logging as databaseName. You are responsible for closing the underlying database client if necessary.
Example ¶
Output:
func NewWithInstance ¶
func NewWithInstance(sourceName string, sourceInstance source.Driver, databaseName string, databaseInstance database.Driver) (*Migrate, error)
NewWithInstance returns a new Migrate instance from an existing source and database instance. Use any string that can serve as an identifier during logging as sourceName and databaseName. You are responsible for closing down the underlying source and database client if necessary.
Example ¶
Output:
func NewWithSourceInstance ¶
func NewWithSourceInstance(sourceName string, sourceInstance source.Driver, databaseUrl string) (*Migrate, error)
NewWithSourceInstance returns a new Migrate instance from an existing source instance and a database URL. The database URL scheme is defined by each driver. Use any string that can serve as an identifier during logging as sourceName. You are responsible for closing the underlying source client if necessary.
Example ¶
Output:
func (*Migrate) Down ¶
Down looks at the currently active migration version and will migrate all the way down (applying all down migrations).
func (*Migrate) Force ¶
Force sets a migration version. It does not check any currently active version in database. It resets the dirty state to false.
func (*Migrate) Migrate ¶
Migrate looks at the currently active migration version, then migrates either up or down to the specified version.
func (*Migrate) Run ¶
Run runs any migration provided by you against the database. It does not check any currently active version in database. Usually you don't need this function at all. Use Migrate, Steps, Up or Down instead.
func (*Migrate) Steps ¶
Steps looks at the currently active migration version. It will migrate up if n > 0, and down if n < 0.
type Migration ¶
type Migration struct { // Identifier can be any string to help identifying // the migration in the source. Identifier string // Version is the version of this migration. Version uint // TargetVersion is the migration version after this migration // has been applied to the database. // Can be -1, implying that this is a NilVersion. TargetVersion int // Body holds an io.ReadCloser to the source. Body io.ReadCloser // BufferedBody holds an buffered io.Reader to the underlying Body. BufferedBody io.Reader // BufferSize defaults to DefaultBufferSize BufferSize uint // Scheduled is the time when the migration was scheduled/ queued. Scheduled time.Time // StartedBuffering is the time when buffering of the migration source started. StartedBuffering time.Time // FinishedBuffering is the time when buffering of the migration source finished. FinishedBuffering time.Time // FinishedReading is the time when the migration source is fully read. FinishedReading time.Time // BytesRead holds the number of Bytes read from the migration source. BytesRead int64 // contains filtered or unexported fields }
Migration holds information about a migration. It is initially created from data coming from the source and then used when run against the database.
func NewMigration ¶
func NewMigration(body io.ReadCloser, identifier string, version uint, targetVersion int) (*Migration, error)
NewMigration returns a new Migration and sets the body, identifier, version and targetVersion. Body can be nil, which turns this migration into a "NilMigration". If no identifier is provided, it will default to "<empty>". targetVersion can be -1, implying it is a NilVersion.
What is a NilMigration? Usually each migration version coming from source is expected to have an Up and Down migration. This is not a hard requirement though, leading to a situation where only the Up or Down migration is present. So let's say the user wants to migrate up to a version that doesn't have the actual Up migration, in that case we still want to apply the version, but with an empty body. We are calling that a NilMigration, a migration with an empty body.
What is a NilVersion? NilVersion is a const(-1). When running down migrations and we are at the last down migration, there is no next down migration, the targetVersion should be nil. Nil in this case is represented by -1 (because type int).
Example ¶
Output: 1486686016/u create_users_table
Example (NilMigration) ¶
Output: 1486686016/u <empty>
Example (NilVersion) ¶
Output: 1486686016/d drop_users_table
func (*Migration) Buffer ¶
Buffer buffers Body up to BufferSize. Calling this function blocks. Call with goroutine.
type MultiError ¶
type MultiError struct {
Errs []error
}
MultiError holds multiple errors.
func NewMultiError ¶
func NewMultiError(errs ...error) MultiError
NewMultiError returns an error type holding multiple errors.
func (MultiError) Error ¶
func (m MultiError) Error() string
Error implements error. Mulitple errors are concatenated with 'and's.
Directories ¶
Path | Synopsis |
---|---|
Package database provides the Database interface.
|
Package database provides the Database interface. |
testing
Package testing has the database tests.
|
Package testing has the database tests. |
Deprecated: package migrate makes this version backwards compatible.
|
Deprecated: package migrate makes this version backwards compatible. |
direction
Deprecated: package direction is deprecated.
|
Deprecated: package direction is deprecated. |
file
Deprecated: package file is deprecated.
|
Deprecated: package file is deprecated. |
Package source provides the Source interface.
|
Package source provides the Source interface. |
testing
Package testing has the source tests.
|
Package testing has the source tests. |
Package testing is used in driver tests.
|
Package testing is used in driver tests. |