Add support for options; transaction option (#68)

* Add support for options; transaction option
This commit is contained in:
Ben Reinhart 2019-05-16 19:39:47 -07:00 committed by GitHub
parent 7400451fba
commit 7ac46ff0f3
No known key found for this signature in database
GPG key ID: 4AEE18F83AFDEB23
4 changed files with 219 additions and 47 deletions

123
pkg/dbmate/migrations.go Normal file
View file

@ -0,0 +1,123 @@
package dbmate
import (
"io/ioutil"
"regexp"
"strings"
)
// MigrationOptions is an interface for accessing migration options
type MigrationOptions interface {
Transaction() bool
}
type migrationOptions map[string]string
// Transaction returns whether or not this migration should run in a transaction
// Defaults to true.
func (m migrationOptions) Transaction() bool {
return m["transaction"] != "false"
}
// Migration contains the migration contents and options
type Migration struct {
Contents string
Options MigrationOptions
}
// NewMigration constructs a Migration object
func NewMigration() Migration {
return Migration{Contents: "", Options: make(migrationOptions)}
}
// parseMigration reads a migration file and returns (up Migration, down Migration, error)
func parseMigration(path string) (Migration, Migration, error) {
data, err := ioutil.ReadFile(path)
if err != nil {
return NewMigration(), NewMigration(), err
}
up, down := parseMigrationContents(string(data))
return up, down, nil
}
var upRegExp = regexp.MustCompile(`(?m)^-- migrate:up\s+(.+)*$`)
var downRegExp = regexp.MustCompile(`(?m)^-- migrate:down\s+(.+)*$`)
// parseMigrationContents parses the string contents of a migration.
// It will return two Migration objects, the first representing the "up"
// block and the second representing the "down" block.
//
// Note that with the way this is currently defined, it is possible to
// correctly parse a migration that does not define an "up" block or a
// "down" block, or one that defines neither. This behavior is, in part,
// to preserve backwards compatibility.
func parseMigrationContents(contents string) (Migration, Migration) {
up := NewMigration()
down := NewMigration()
upMatch := upRegExp.FindStringSubmatchIndex(contents)
downMatch := downRegExp.FindStringSubmatchIndex(contents)
onlyDefinedUpBlock := len(upMatch) != 0 && len(downMatch) == 0
onlyDefinedDownBlock := len(upMatch) == 0 && len(downMatch) != 0
if onlyDefinedUpBlock {
up.Contents = strings.TrimSpace(contents)
up.Options = parseMigrationOptions(contents, upMatch[2], upMatch[3])
} else if onlyDefinedDownBlock {
down.Contents = strings.TrimSpace(contents)
down.Options = parseMigrationOptions(contents, downMatch[2], downMatch[3])
} else {
upStart := upMatch[0]
downStart := downMatch[0]
upEnd := downMatch[0]
downEnd := len(contents)
// If migrate:down was defined above migrate:up, correct the end indices
if upMatch[0] > downMatch[0] {
upEnd = downEnd
downEnd = upMatch[0]
}
up.Contents = strings.TrimSpace(contents[upStart:upEnd])
up.Options = parseMigrationOptions(contents, upMatch[2], upMatch[3])
down.Contents = strings.TrimSpace(contents[downStart:downEnd])
down.Options = parseMigrationOptions(contents, downMatch[2], downMatch[3])
}
return up, down
}
var whitespaceRegExp = regexp.MustCompile(`\s+`)
var optionSeparatorRegExp = regexp.MustCompile(`:`)
// parseMigrationOptions parses the options portion of a migration
// block into an object that satisfies the MigrationOptions interface,
// i.e., the 'transaction:false' piece of the following:
//
// -- migrate:up transaction:false
// create table users (id serial, name string);
// -- migrate:down
// drop table users;
//
func parseMigrationOptions(contents string, begin, end int) MigrationOptions {
mOpts := make(migrationOptions)
if begin == -1 || end == -1 {
return mOpts
}
optionsString := strings.TrimSpace(contents[begin:end])
optionGroups := whitespaceRegExp.Split(optionsString, -1)
for _, group := range optionGroups {
pair := optionSeparatorRegExp.Split(group, -1)
if len(pair) == 2 {
mOpts[pair[0]] = pair[1]
}
}
return mOpts
}