A database migration tool. Supports SQL migrations and Go functions.
 
 
 
Go to file
mmatczuk 1e8921d41c rename collectMigrations to CollectMigrations 2017-03-27 10:19:04 +02:00
cmd/goose Handle migration creation without db params 2016-11-05 14:10:59 -05:00
example Register Go functions as complex Go migrations 2016-03-03 18:48:45 -05:00
.gitignore gitignore: remove executable in new location 2013-09-30 20:19:04 -07:00
.travis.yml Register Go functions as complex Go migrations 2016-03-03 18:48:45 -05:00
LICENSE Update LICENSE 2016-03-03 14:53:29 -05:00
Makefile Add `make dist' 2016-12-09 13:02:14 -05:00
README.md MySQL status note 2017-03-24 13:11:15 +01:00
create.go Re-ordered create arguments, checked for missing create arguments 2016-06-29 15:40:13 -04:00
dialect.go Remove go-sqlite3 dependency from goose pkg 2016-03-07 15:55:05 -05:00
down.go rename collectMigrations to CollectMigrations 2017-03-27 10:19:04 +02:00
goose.go Support for migrating up/down to a specific version 2017-03-17 14:44:22 +01:00
migrate.go rename collectMigrations to CollectMigrations 2017-03-27 10:19:04 +02:00
migrate_test.go Fixes the tests 2016-10-07 16:12:18 -04:00
migration.go Fix err 2016-10-03 19:36:16 -04:00
migration_sql.go Register Go functions as complex Go migrations 2016-03-03 18:48:45 -05:00
migration_sql_test.go Move lib/goose into top level pkg 2016-02-26 14:50:27 -05:00
redo.go rename collectMigrations to CollectMigrations 2017-03-27 10:19:04 +02:00
status.go rename collectMigrations to CollectMigrations 2017-03-27 10:19:04 +02:00
up.go rename collectMigrations to CollectMigrations 2017-03-27 10:19:04 +02:00
util.go Move lib/goose into top level pkg 2016-02-26 14:50:27 -05:00
version.go Refactor commands 2016-03-03 14:46:04 -05:00

README.md

goose

Goose is a database migration tool. Manage your database's evolution by creating incremental SQL files or Go functions.

GoDoc Widget Travis Widget

Goals of this fork

github.com/pressly/goose is a fork of bitbucket.org/liamstask/goose with the following changes:

  • No config files
  • Default goose binary can migrate SQL files only
  • Go migrations:
    • We dropped building Go migrations on-the-fly from .go source files
    • Instead, you can create your own goose binary, import github.com/pressly/goose package and run complex Go migrations with your own *sql.DB connection
    • Each Go migration function is called with *sql.Tx argument - within its own transaction
  • The goose pkg is decoupled from the default binary:
    • goose pkg doesn't register any SQL drivers anymore (no driver panic() conflict within your codebase!)
    • goose pkg doesn't have any vendor dependencies anymore

Install

$ go get -u github.com/pressly/goose/cmd/goose

This will install the goose binary to your $GOPATH/bin directory.

Usage

Usage: goose [OPTIONS] DRIVER DBSTRING COMMAND

Examples:
    goose postgres "user=postgres dbname=postgres sslmode=disable" up
    goose mysql "user:password@/dbname" down
    goose sqlite3 ./foo.db status

Options:
  -dir string
    	directory with migration files (default ".")

Commands:
    up         Migrate the DB to the most recent version available
    down       Roll back the version by 1
    redo       Re-run the latest migration
    status     Dump the migration status for the current DB
    dbversion  Print the current version of the database
    create     Creates a blank migration template

create

Create a new Go migration.

$ goose create AddSomeColumns
$ goose: created db/migrations/20130106093224_AddSomeColumns.go

Edit the newly created script to define the behavior of your migration.

You can also create an SQL migration:

$ goose create AddSomeColumns sql
$ goose: created db/migrations/20130106093224_AddSomeColumns.sql

up

Apply all available migrations.

$ goose up
$ goose: migrating db environment 'development', current version: 0, target: 3
$ OK    001_basics.sql
$ OK    002_next.sql
$ OK    003_and_again.go

down

Roll back a single migration from the current version.

$ goose down
$ goose: migrating db environment 'development', current version: 3, target: 2
$ OK    003_and_again.go

redo

Roll back the most recently applied migration, then run it again.

$ goose redo
$ goose: migrating db environment 'development', current version: 3, target: 2
$ OK    003_and_again.go
$ goose: migrating db environment 'development', current version: 2, target: 3
$ OK    003_and_again.go

status

Print the status of all migrations:

$ goose status
$ goose: status for environment 'development'
$   Applied At                  Migration
$   =======================================
$   Sun Jan  6 11:25:03 2013 -- 001_basics.sql
$   Sun Jan  6 11:25:03 2013 -- 002_next.sql
$   Pending                  -- 003_and_again.go

Note: for MySQL parseTime flag must be enabled.

dbversion

Print the current version of the database:

$ goose dbversion
$ goose: dbversion 002

Migrations

goose supports migrations written in SQL or in Go.

SQL Migrations

A sample SQL migration looks like:

-- +goose Up
CREATE TABLE post (
    id int NOT NULL,
    title text,
    body text,
    PRIMARY KEY(id)
);

-- +goose Down
DROP TABLE post;

Notice the annotations in the comments. Any statements following -- +goose Up will be executed as part of a forward migration, and any statements following -- +goose Down will be executed as part of a rollback.

By default, SQL statements are delimited by semicolons - in fact, query statements must end with a semicolon to be properly recognized by goose.

More complex statements (PL/pgSQL) that have semicolons within them must be annotated with -- +goose StatementBegin and -- +goose StatementEnd to be properly recognized. For example:

-- +goose Up
-- +goose StatementBegin
CREATE OR REPLACE FUNCTION histories_partition_creation( DATE, DATE )
returns void AS $$
DECLARE
  create_query text;
BEGIN
  FOR create_query IN SELECT
      'CREATE TABLE IF NOT EXISTS histories_'
      || TO_CHAR( d, 'YYYY_MM' )
      || ' ( CHECK( created_at >= timestamp '''
      || TO_CHAR( d, 'YYYY-MM-DD 00:00:00' )
      || ''' AND created_at < timestamp '''
      || TO_CHAR( d + INTERVAL '1 month', 'YYYY-MM-DD 00:00:00' )
      || ''' ) ) inherits ( histories );'
    FROM generate_series( $1, $2, '1 month' ) AS d
  LOOP
    EXECUTE create_query;
  END LOOP;  -- LOOP END
END;         -- FUNCTION END
$$
language plpgsql;
-- +goose StatementEnd

Go Migrations

  1. Create your own goose binary, see example
  2. Import github.com/pressly/goose
  3. Register your migration functions
  4. Run goose command, ie. goose.Up(db *sql.DB, dir string)

A sample Go migration 00002_users_add_email.go file looks like:

package migrations

import (
	"database/sql"

	"github.com/pressly/goose"
)

func init() {
	goose.AddMigration(Up, Down)
}

func Up(tx *sql.Tx) error {
	_, err := tx.Exec("UPDATE users SET username='admin' WHERE username='root';")
	if err != nil {
		return err
	}
	return nil
}

func Down(tx *sql.Tx) error {
	_, err := tx.Exec("UPDATE users SET username='root' WHERE username='admin';")
	if err != nil {
		return err
	}
	return nil
}

License

Licensed under MIT License