2018-06-24 16:40:48 +03:00
|
|
|
package file
|
|
|
|
|
|
|
|
import (
|
|
|
|
"fmt"
|
|
|
|
"io"
|
|
|
|
"io/ioutil"
|
|
|
|
nurl "net/url"
|
|
|
|
"os"
|
|
|
|
"path"
|
|
|
|
"path/filepath"
|
|
|
|
"runtime"
|
|
|
|
"strings"
|
|
|
|
|
|
|
|
"github.com/hasura/graphql-engine/cli/migrate/source"
|
2018-07-09 16:47:38 +03:00
|
|
|
log "github.com/sirupsen/logrus"
|
2018-06-24 16:40:48 +03:00
|
|
|
)
|
|
|
|
|
|
|
|
type File struct {
|
|
|
|
url string
|
|
|
|
path string
|
|
|
|
migrations *source.Migrations
|
2018-07-09 16:47:38 +03:00
|
|
|
logger *log.Logger
|
2018-06-24 16:40:48 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
func init() {
|
|
|
|
source.Register("file", &File{})
|
|
|
|
}
|
|
|
|
|
2018-07-09 16:47:38 +03:00
|
|
|
func (f *File) Open(url string, logger *log.Logger) (source.Driver, error) {
|
|
|
|
if logger == nil {
|
|
|
|
logger = log.New()
|
|
|
|
}
|
2018-06-24 16:40:48 +03:00
|
|
|
u, err := nurl.Parse(url)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
// concat host and path to restore full path
|
|
|
|
// host might be `.`
|
|
|
|
p := u.Host + u.Path
|
|
|
|
|
|
|
|
if len(p) == 0 {
|
|
|
|
// default to current directory if no path
|
|
|
|
wd, err := os.Getwd()
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
p = wd
|
|
|
|
} else if p[0:1] == "." || p[0:1] != "/" {
|
|
|
|
// make path absolute if relative
|
|
|
|
abs, err := filepath.Abs(p)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
p = abs
|
|
|
|
}
|
|
|
|
|
|
|
|
if runtime.GOOS == "windows" && p[0:1] == "/" {
|
|
|
|
p = strings.TrimPrefix(p, "/")
|
|
|
|
}
|
|
|
|
|
|
|
|
// scan directory
|
|
|
|
files, err := ioutil.ReadDir(p)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
nf := &File{
|
|
|
|
url: url,
|
2018-07-09 16:47:38 +03:00
|
|
|
logger: logger,
|
2018-06-24 16:40:48 +03:00
|
|
|
path: p,
|
|
|
|
migrations: source.NewMigrations(),
|
|
|
|
}
|
|
|
|
|
|
|
|
for _, fi := range files {
|
|
|
|
if !fi.IsDir() {
|
2018-07-20 13:31:33 +03:00
|
|
|
m, err := source.DefaultParse(fi.Name(), p)
|
2018-06-24 16:40:48 +03:00
|
|
|
if err != nil {
|
|
|
|
continue // ignore files that we can't parse
|
|
|
|
}
|
|
|
|
|
|
|
|
if !nf.migrations.Append(m) {
|
|
|
|
return nil, fmt.Errorf("unable to parse file %v", fi.Name())
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return nf, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func (f *File) Close() error {
|
|
|
|
// nothing do to here
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func (f *File) First() (version uint64, err error) {
|
|
|
|
if v, ok := f.migrations.First(); !ok {
|
2018-07-11 15:13:31 +03:00
|
|
|
return 0, &os.PathError{Op: "first", Path: f.path, Err: os.ErrNotExist}
|
2018-06-24 16:40:48 +03:00
|
|
|
} else {
|
|
|
|
return v, nil
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func (f *File) GetLocalVersion() (version uint64, err error) {
|
|
|
|
return f.migrations.GetLocalVersion(), nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func (f *File) GetUnappliedMigrations(version uint64) (versions []uint64) {
|
|
|
|
return f.migrations.GetUnappliedMigrations(version)
|
|
|
|
}
|
|
|
|
|
|
|
|
func (f *File) Prev(version uint64) (prevVersion uint64, err error) {
|
|
|
|
if v, ok := f.migrations.Prev(version); !ok {
|
2018-07-11 15:13:31 +03:00
|
|
|
return 0, &os.PathError{Op: fmt.Sprintf("prev for version %v", version), Path: f.path, Err: os.ErrNotExist}
|
2018-06-24 16:40:48 +03:00
|
|
|
} else {
|
|
|
|
return v, nil
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func (f *File) Next(version uint64) (nextVersion uint64, err error) {
|
|
|
|
if v, ok := f.migrations.Next(version); !ok {
|
2018-07-11 15:13:31 +03:00
|
|
|
return 0, &os.PathError{Op: fmt.Sprintf("next for version %v", version), Path: f.path, Err: os.ErrNotExist}
|
2018-06-24 16:40:48 +03:00
|
|
|
} else {
|
|
|
|
return v, nil
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func (f *File) GetDirections(version uint64) map[source.Direction]bool {
|
|
|
|
return f.migrations.GetDirections(version)
|
|
|
|
}
|
|
|
|
|
|
|
|
func (f *File) ReadUp(version uint64) (r io.ReadCloser, identifier string, fileName string, err error) {
|
|
|
|
if m, ok := f.migrations.Up(version); ok {
|
|
|
|
r, err := os.Open(path.Join(f.path, m.Raw))
|
|
|
|
if err != nil {
|
|
|
|
return nil, "", "", err
|
|
|
|
}
|
|
|
|
return r, m.Identifier, m.Raw, nil
|
|
|
|
}
|
2018-07-11 15:13:31 +03:00
|
|
|
return nil, "", "", &os.PathError{Op: fmt.Sprintf("read version %v", version), Path: f.path, Err: os.ErrNotExist}
|
2018-06-24 16:40:48 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
func (f *File) ReadMetaUp(version uint64) (r io.ReadCloser, identifier string, fileName string, err error) {
|
|
|
|
if m, ok := f.migrations.MetaUp(version); ok {
|
|
|
|
r, err := os.Open(path.Join(f.path, m.Raw))
|
|
|
|
if err != nil {
|
|
|
|
return nil, "", "", err
|
|
|
|
}
|
|
|
|
return r, m.Identifier, m.Raw, nil
|
|
|
|
}
|
2018-07-11 15:13:31 +03:00
|
|
|
return nil, "", "", &os.PathError{Op: fmt.Sprintf("read version %v", version), Path: f.path, Err: os.ErrNotExist}
|
2018-06-24 16:40:48 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
func (f *File) ReadDown(version uint64) (r io.ReadCloser, identifier string, fileName string, err error) {
|
|
|
|
if m, ok := f.migrations.Down(version); ok {
|
|
|
|
r, err := os.Open(path.Join(f.path, m.Raw))
|
|
|
|
if err != nil {
|
|
|
|
return nil, "", "", err
|
|
|
|
}
|
|
|
|
return r, m.Identifier, m.Raw, nil
|
|
|
|
}
|
2018-07-11 15:13:31 +03:00
|
|
|
return nil, "", "", &os.PathError{Op: fmt.Sprintf("read version %v", version), Path: f.path, Err: os.ErrNotExist}
|
2018-06-24 16:40:48 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
func (f *File) ReadMetaDown(version uint64) (r io.ReadCloser, identifier string, fileName string, err error) {
|
|
|
|
if m, ok := f.migrations.MetaDown(version); ok {
|
|
|
|
r, err := os.Open(path.Join(f.path, m.Raw))
|
|
|
|
if err != nil {
|
|
|
|
return nil, "", "", err
|
|
|
|
}
|
|
|
|
return r, m.Identifier, m.Raw, nil
|
|
|
|
}
|
2018-07-11 15:13:31 +03:00
|
|
|
return nil, "", "", &os.PathError{Op: fmt.Sprintf("read version %v", version), Path: f.path, Err: os.ErrNotExist}
|
2018-06-24 16:40:48 +03:00
|
|
|
}
|