2018-06-24 16:40:48 +03:00
|
|
|
package file
|
|
|
|
|
|
|
|
import (
|
|
|
|
"fmt"
|
|
|
|
"io/ioutil"
|
|
|
|
"os"
|
|
|
|
"path"
|
|
|
|
"path/filepath"
|
|
|
|
"testing"
|
|
|
|
|
|
|
|
st "github.com/hasura/graphql-engine/cli/migrate/source/testing"
|
2018-07-09 16:47:38 +03:00
|
|
|
"github.com/sirupsen/logrus/hooks/test"
|
2018-06-24 16:40:48 +03:00
|
|
|
)
|
|
|
|
|
|
|
|
func Test(t *testing.T) {
|
|
|
|
tmpDir, err := ioutil.TempDir("", "")
|
|
|
|
if err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
defer os.RemoveAll(tmpDir)
|
|
|
|
|
|
|
|
// write files that meet driver test requirements
|
|
|
|
mustWriteFile(t, tmpDir, "1_foobar.up.sql", "1 up")
|
|
|
|
mustWriteFile(t, tmpDir, "1_foobar.down.sql", "1 down")
|
2018-07-20 13:31:33 +03:00
|
|
|
mustWriteFile(t, tmpDir, "1_foobar.up.yaml", `- args:
|
|
|
|
name: test
|
|
|
|
type: add_existing_table_or_view
|
|
|
|
`)
|
|
|
|
mustWriteFile(t, tmpDir, "1_foobar.down.yaml", `- args:
|
|
|
|
name: test
|
|
|
|
type: add_existing_table_or_view
|
|
|
|
`)
|
2018-06-24 16:40:48 +03:00
|
|
|
|
|
|
|
mustWriteFile(t, tmpDir, "3_foobar.up.sql", "3 up")
|
|
|
|
|
2018-07-20 13:31:33 +03:00
|
|
|
mustWriteFile(t, tmpDir, "4_foobar.up.yaml", `- args:
|
|
|
|
name: test
|
|
|
|
type: add_existing_table_or_view
|
|
|
|
`)
|
2018-06-24 16:40:48 +03:00
|
|
|
|
|
|
|
mustWriteFile(t, tmpDir, "5_foobar.down.sql", "5 down")
|
|
|
|
|
2018-07-20 13:31:33 +03:00
|
|
|
mustWriteFile(t, tmpDir, "6_foobar.down.yaml", `- args:
|
|
|
|
name: test
|
|
|
|
type: add_existing_table_or_view
|
|
|
|
`)
|
2018-06-24 16:40:48 +03:00
|
|
|
|
|
|
|
mustWriteFile(t, tmpDir, "8_foobar.up.sql", "7 up")
|
|
|
|
mustWriteFile(t, tmpDir, "8_foobar.down.sql", "7 down")
|
|
|
|
|
2018-07-09 16:47:38 +03:00
|
|
|
logger, _ := test.NewNullLogger()
|
2018-06-24 16:40:48 +03:00
|
|
|
f := &File{}
|
2018-07-09 16:47:38 +03:00
|
|
|
d, err := f.Open("file://"+tmpDir, logger)
|
2018-06-24 16:40:48 +03:00
|
|
|
if err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
|
|
|
|
st.Test(t, d)
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestOpen(t *testing.T) {
|
|
|
|
tmpDir, err := ioutil.TempDir("", "TestOpen")
|
|
|
|
if err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
defer os.RemoveAll(tmpDir)
|
|
|
|
|
|
|
|
mustWriteFile(t, tmpDir, "1_foobar.up.sql", "")
|
|
|
|
mustWriteFile(t, tmpDir, "1_foobar.down.sql", "")
|
|
|
|
|
|
|
|
if !filepath.IsAbs(tmpDir) {
|
|
|
|
t.Fatal("expected tmpDir to be absolute path")
|
|
|
|
}
|
|
|
|
|
2018-07-09 16:47:38 +03:00
|
|
|
logger, _ := test.NewNullLogger()
|
2018-06-24 16:40:48 +03:00
|
|
|
f := &File{}
|
2018-07-09 16:47:38 +03:00
|
|
|
_, err = f.Open("file://"+tmpDir, logger) // absolute path
|
2018-06-24 16:40:48 +03:00
|
|
|
if err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestOpenWithRelativePath(t *testing.T) {
|
|
|
|
tmpDir, err := ioutil.TempDir("", "TestOpen")
|
|
|
|
if err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
defer os.RemoveAll(tmpDir)
|
|
|
|
|
|
|
|
wd, err := os.Getwd()
|
|
|
|
if err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
defer os.Chdir(wd) // rescue working dir after we are done
|
|
|
|
|
|
|
|
if err := os.Chdir(tmpDir); err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
|
|
|
|
if err := os.Mkdir(filepath.Join(tmpDir, "foo"), os.ModePerm); err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
|
2018-07-20 13:31:33 +03:00
|
|
|
mustWriteFile(t, filepath.Join(tmpDir, "foo"), "1_foobar.up.sql", "test")
|
2018-06-24 16:40:48 +03:00
|
|
|
|
2018-07-09 16:47:38 +03:00
|
|
|
logger, _ := test.NewNullLogger()
|
2018-06-24 16:40:48 +03:00
|
|
|
f := &File{}
|
|
|
|
|
|
|
|
// dir: foo
|
2018-07-09 16:47:38 +03:00
|
|
|
d, err := f.Open("file://foo", logger)
|
2018-06-24 16:40:48 +03:00
|
|
|
if err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
_, err = d.First()
|
|
|
|
if err != nil {
|
|
|
|
t.Fatalf("expected first file in working dir %v for foo", tmpDir)
|
|
|
|
}
|
|
|
|
|
|
|
|
// dir: ./foo
|
2018-07-09 16:47:38 +03:00
|
|
|
d, err = f.Open("file://./foo", logger)
|
2018-06-24 16:40:48 +03:00
|
|
|
if err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
_, err = d.First()
|
|
|
|
if err != nil {
|
|
|
|
t.Fatalf("expected first file in working dir %v for ./foo", tmpDir)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestOpenDefaultsToCurrentDirectory(t *testing.T) {
|
|
|
|
wd, err := os.Getwd()
|
|
|
|
if err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
|
2018-07-09 16:47:38 +03:00
|
|
|
logger, _ := test.NewNullLogger()
|
2018-06-24 16:40:48 +03:00
|
|
|
f := &File{}
|
2018-07-09 16:47:38 +03:00
|
|
|
d, err := f.Open("file://", logger)
|
2018-06-24 16:40:48 +03:00
|
|
|
if err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
|
|
|
|
if d.(*File).path != wd {
|
|
|
|
t.Fatal("expected driver to default to current directory")
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestOpenWithDuplicateVersion(t *testing.T) {
|
|
|
|
tmpDir, err := ioutil.TempDir("", "TestOpenWithDuplicateVersion")
|
|
|
|
if err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
defer os.RemoveAll(tmpDir)
|
|
|
|
|
2018-07-20 13:31:33 +03:00
|
|
|
mustWriteFile(t, tmpDir, "1_foo.up.sql", "test") // 1 up
|
|
|
|
mustWriteFile(t, tmpDir, "1_bar.up.sql", "test") // 1 up
|
2018-06-24 16:40:48 +03:00
|
|
|
|
2018-07-09 16:47:38 +03:00
|
|
|
logger, _ := test.NewNullLogger()
|
2018-06-24 16:40:48 +03:00
|
|
|
f := &File{}
|
2018-07-09 16:47:38 +03:00
|
|
|
_, err = f.Open("file://"+tmpDir, logger)
|
2018-06-24 16:40:48 +03:00
|
|
|
if err == nil {
|
|
|
|
t.Fatal("expected err")
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestOpenWithInvalidFileURL(t *testing.T) {
|
2018-07-09 16:47:38 +03:00
|
|
|
logger, _ := test.NewNullLogger()
|
2018-06-24 16:40:48 +03:00
|
|
|
f := &File{}
|
2018-07-09 16:47:38 +03:00
|
|
|
_, err := f.Open(":", logger)
|
2018-06-24 16:40:48 +03:00
|
|
|
if err == nil {
|
|
|
|
t.Fatal("exepected err to be not nil")
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestWithEmptyMigration(t *testing.T) {
|
2018-07-09 16:47:38 +03:00
|
|
|
logger, _ := test.NewNullLogger()
|
2018-06-24 16:40:48 +03:00
|
|
|
s := &File{}
|
2018-07-09 16:47:38 +03:00
|
|
|
d, err := s.Open("", logger)
|
2018-06-24 16:40:48 +03:00
|
|
|
if err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
|
|
|
|
version, err := d.First()
|
|
|
|
if err == nil {
|
|
|
|
t.Fatalf("First: expected err not to be nil")
|
|
|
|
}
|
|
|
|
|
|
|
|
if version != 0 {
|
|
|
|
t.Errorf("First: expected 0, got %v", version)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestWithInvalidDirectory(t *testing.T) {
|
2018-07-09 16:47:38 +03:00
|
|
|
logger, _ := test.NewNullLogger()
|
2018-06-24 16:40:48 +03:00
|
|
|
s := &File{}
|
2018-07-09 16:47:38 +03:00
|
|
|
_, err := s.Open("file://invalidir", logger)
|
2018-06-24 16:40:48 +03:00
|
|
|
if err == nil {
|
|
|
|
t.Fatal("exepected err to be not nil")
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestClose(t *testing.T) {
|
|
|
|
tmpDir, err := ioutil.TempDir("", "TestOpen")
|
|
|
|
if err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
defer os.RemoveAll(tmpDir)
|
|
|
|
|
2018-07-09 16:47:38 +03:00
|
|
|
logger, _ := test.NewNullLogger()
|
2018-06-24 16:40:48 +03:00
|
|
|
f := &File{}
|
2018-07-09 16:47:38 +03:00
|
|
|
d, err := f.Open("file://"+tmpDir, logger)
|
2018-06-24 16:40:48 +03:00
|
|
|
if err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
|
|
|
|
if d.Close() != nil {
|
|
|
|
t.Fatal("expected nil")
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func mustCreateBenchmarkDir(t *testing.B) (dir string) {
|
|
|
|
tmpDir, err := ioutil.TempDir("", "Benchmark")
|
|
|
|
if err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
|
|
|
|
for i := 0; i < 1000; i++ {
|
|
|
|
mustWriteFile(t, tmpDir, fmt.Sprintf("%v_foobar.up.sql", i), "")
|
|
|
|
mustWriteFile(t, tmpDir, fmt.Sprintf("%v_foobar.down.sql", i), "")
|
|
|
|
}
|
|
|
|
|
|
|
|
return tmpDir
|
|
|
|
}
|
|
|
|
|
|
|
|
func BenchmarkOpen(b *testing.B) {
|
2018-07-09 16:47:38 +03:00
|
|
|
logger, _ := test.NewNullLogger()
|
2018-06-24 16:40:48 +03:00
|
|
|
dir := mustCreateBenchmarkDir(b)
|
|
|
|
defer os.RemoveAll(dir)
|
|
|
|
b.ResetTimer()
|
|
|
|
for n := 0; n < b.N; n++ {
|
|
|
|
f := &File{}
|
2018-07-09 16:47:38 +03:00
|
|
|
f.Open("file://"+dir, logger)
|
2018-06-24 16:40:48 +03:00
|
|
|
}
|
|
|
|
b.StopTimer()
|
|
|
|
}
|
|
|
|
|
|
|
|
func BenchmarkNext(b *testing.B) {
|
2018-07-09 16:47:38 +03:00
|
|
|
logger, _ := test.NewNullLogger()
|
2018-06-24 16:40:48 +03:00
|
|
|
dir := mustCreateBenchmarkDir(b)
|
|
|
|
defer os.RemoveAll(dir)
|
|
|
|
f := &File{}
|
2018-07-09 16:47:38 +03:00
|
|
|
d, _ := f.Open("file://"+dir, logger)
|
2018-06-24 16:40:48 +03:00
|
|
|
b.ResetTimer()
|
|
|
|
v, err := d.First()
|
|
|
|
for n := 0; n < b.N; n++ {
|
|
|
|
for !os.IsNotExist(err) {
|
|
|
|
v, err = d.Next(v)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
b.StopTimer()
|
|
|
|
}
|
|
|
|
|
|
|
|
func mustWriteFile(t testing.TB, dir, file string, body string) {
|
|
|
|
if err := ioutil.WriteFile(path.Join(dir, file), []byte(body), 06444); err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
}
|