2018-06-24 16:40:48 +03:00
|
|
|
package commands
|
|
|
|
|
|
|
|
import (
|
2019-03-18 19:40:04 +03:00
|
|
|
"io/ioutil"
|
|
|
|
"os"
|
2018-06-24 16:40:48 +03:00
|
|
|
"time"
|
|
|
|
|
2019-03-18 19:40:04 +03:00
|
|
|
"github.com/ghodss/yaml"
|
2018-06-24 16:40:48 +03:00
|
|
|
"github.com/hasura/graphql-engine/cli"
|
2019-04-30 11:34:08 +03:00
|
|
|
"github.com/hasura/graphql-engine/cli/migrate"
|
2018-06-24 16:40:48 +03:00
|
|
|
"github.com/pkg/errors"
|
|
|
|
"github.com/spf13/cobra"
|
2019-03-18 19:40:04 +03:00
|
|
|
"github.com/spf13/pflag"
|
2018-10-01 13:12:12 +03:00
|
|
|
"github.com/spf13/viper"
|
2019-04-30 11:34:08 +03:00
|
|
|
|
|
|
|
mig "github.com/hasura/graphql-engine/cli/migrate/cmd"
|
|
|
|
log "github.com/sirupsen/logrus"
|
2018-06-24 16:40:48 +03:00
|
|
|
)
|
|
|
|
|
2019-04-30 11:34:08 +03:00
|
|
|
const migrateCreateCmdExamples = ` # Setup migration files for the first time by introspecting a server:
|
2019-08-28 11:03:05 +03:00
|
|
|
hasura migrate create "init" --from-server`
|
2019-04-30 11:34:08 +03:00
|
|
|
|
2018-06-28 11:36:25 +03:00
|
|
|
func newMigrateCreateCmd(ec *cli.ExecutionContext) *cobra.Command {
|
2018-10-01 13:12:12 +03:00
|
|
|
v := viper.New()
|
2018-06-24 16:40:48 +03:00
|
|
|
opts := &migrateCreateOptions{
|
|
|
|
EC: ec,
|
|
|
|
}
|
|
|
|
|
|
|
|
migrateCreateCmd := &cobra.Command{
|
|
|
|
Use: "create [migration-name]",
|
|
|
|
Short: "Create files required for a migration",
|
|
|
|
Long: "Create sql and yaml files required for a migration",
|
2019-04-30 11:34:08 +03:00
|
|
|
Example: migrateCreateCmdExamples,
|
2018-06-24 16:40:48 +03:00
|
|
|
SilenceUsage: true,
|
|
|
|
Args: cobra.ExactArgs(1),
|
2018-10-01 13:12:12 +03:00
|
|
|
PreRunE: func(cmd *cobra.Command, args []string) error {
|
|
|
|
ec.Viper = v
|
|
|
|
return ec.Validate()
|
|
|
|
},
|
2018-06-24 16:40:48 +03:00
|
|
|
RunE: func(cmd *cobra.Command, args []string) error {
|
|
|
|
opts.name = args[0]
|
2019-04-03 14:29:58 +03:00
|
|
|
opts.EC.Spin("Creating migration files...")
|
|
|
|
version, err := opts.run()
|
|
|
|
opts.EC.Spinner.Stop()
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
opts.EC.Logger.WithFields(log.Fields{
|
|
|
|
"version": version,
|
|
|
|
"name": opts.name,
|
|
|
|
}).Info("Migrations files created")
|
|
|
|
return nil
|
2018-06-24 16:40:48 +03:00
|
|
|
},
|
|
|
|
}
|
2019-03-18 19:40:04 +03:00
|
|
|
f := migrateCreateCmd.Flags()
|
|
|
|
opts.flags = f
|
2019-04-30 11:34:08 +03:00
|
|
|
f.BoolVar(&opts.fromServer, "from-server", false, "get SQL statements and hasura metadata from the server")
|
|
|
|
f.StringVar(&opts.sqlFile, "sql-from-file", "", "path to an sql file which contains the SQL statements")
|
|
|
|
f.BoolVar(&opts.sqlServer, "sql-from-server", false, "take pg_dump from server and save it as a migration")
|
|
|
|
f.StringArrayVar(&opts.schemaNames, "schema", []string{"public"}, "name of Postgres schema to export as migration")
|
2019-03-18 19:40:04 +03:00
|
|
|
f.StringVar(&opts.metaDataFile, "metadata-from-file", "", "path to a hasura metadata file to be used for up actions")
|
|
|
|
f.BoolVar(&opts.metaDataServer, "metadata-from-server", false, "take metadata from the server and write it as an up migration file")
|
|
|
|
f.String("endpoint", "", "http(s) endpoint for Hasura GraphQL Engine")
|
|
|
|
f.String("admin-secret", "", "admin secret for Hasura GraphQL Engine")
|
|
|
|
f.String("access-key", "", "access key for Hasura GraphQL Engine")
|
|
|
|
f.MarkDeprecated("access-key", "use --admin-secret instead")
|
2019-04-30 11:34:08 +03:00
|
|
|
|
2019-03-18 19:40:04 +03:00
|
|
|
migrateCreateCmd.MarkFlagFilename("sql-from-file")
|
|
|
|
migrateCreateCmd.MarkFlagFilename("metadata-from-file")
|
|
|
|
|
|
|
|
// need to create a new viper because https://github.com/spf13/viper/issues/233
|
|
|
|
v.BindPFlag("endpoint", f.Lookup("endpoint"))
|
|
|
|
v.BindPFlag("admin_secret", f.Lookup("admin-secret"))
|
|
|
|
v.BindPFlag("access_key", f.Lookup("access-key"))
|
2018-06-24 16:40:48 +03:00
|
|
|
|
|
|
|
return migrateCreateCmd
|
|
|
|
}
|
|
|
|
|
|
|
|
type migrateCreateOptions struct {
|
|
|
|
EC *cli.ExecutionContext
|
|
|
|
|
2019-03-18 19:40:04 +03:00
|
|
|
name string
|
|
|
|
flags *pflag.FlagSet
|
|
|
|
|
|
|
|
// Flags
|
2019-04-30 11:34:08 +03:00
|
|
|
fromServer bool
|
2019-03-18 19:40:04 +03:00
|
|
|
sqlFile string
|
2019-04-30 11:34:08 +03:00
|
|
|
sqlServer bool
|
2019-03-18 19:40:04 +03:00
|
|
|
metaDataFile string
|
|
|
|
metaDataServer bool
|
2019-04-30 11:34:08 +03:00
|
|
|
schemaNames []string
|
2018-06-24 16:40:48 +03:00
|
|
|
}
|
|
|
|
|
2019-04-03 14:29:58 +03:00
|
|
|
func (o *migrateCreateOptions) run() (version int64, err error) {
|
2018-06-24 16:40:48 +03:00
|
|
|
timestamp := getTime()
|
2018-07-20 13:31:33 +03:00
|
|
|
createOptions := mig.New(timestamp, o.name, o.EC.MigrationDir)
|
2019-03-18 19:40:04 +03:00
|
|
|
|
2019-04-30 11:34:08 +03:00
|
|
|
if o.fromServer {
|
|
|
|
o.sqlServer = true
|
|
|
|
o.metaDataServer = true
|
|
|
|
}
|
|
|
|
|
|
|
|
if o.flags.Changed("metadata-from-file") && o.sqlServer {
|
|
|
|
return 0, errors.New("only one sql type can be set")
|
|
|
|
}
|
|
|
|
if o.flags.Changed("metadata-from-file") && o.metaDataServer {
|
|
|
|
return 0, errors.New("only one metadata type can be set")
|
|
|
|
}
|
|
|
|
|
|
|
|
var migrateDrv *migrate.Migrate
|
|
|
|
if o.sqlServer || o.metaDataServer {
|
2019-09-18 08:36:16 +03:00
|
|
|
migrateDrv, err = newMigrate(o.EC.MigrationDir, o.EC.ServerConfig.ParsedEndpoint, o.EC.ServerConfig.AdminSecret, o.EC.Logger, o.EC.Version, true)
|
2019-04-30 11:34:08 +03:00
|
|
|
if err != nil {
|
|
|
|
return 0, errors.Wrap(err, "cannot create migrate instance")
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-03-18 19:40:04 +03:00
|
|
|
if o.flags.Changed("sql-from-file") {
|
|
|
|
// sql-file flag is set
|
|
|
|
err := createOptions.SetSQLUpFromFile(o.sqlFile)
|
|
|
|
if err != nil {
|
2019-04-03 14:29:58 +03:00
|
|
|
return 0, errors.Wrap(err, "cannot set sql file")
|
2019-03-18 19:40:04 +03:00
|
|
|
}
|
|
|
|
}
|
2019-04-30 11:34:08 +03:00
|
|
|
if o.sqlServer {
|
|
|
|
data, err := migrateDrv.ExportSchemaDump(o.schemaNames)
|
|
|
|
if err != nil {
|
|
|
|
return 0, errors.Wrap(err, "cannot fetch schema dump")
|
|
|
|
}
|
|
|
|
createOptions.SetSQLUp(string(data))
|
2019-03-18 19:40:04 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
if o.flags.Changed("metadata-from-file") {
|
|
|
|
// metadata-file flag is set
|
|
|
|
err := createOptions.SetMetaUpFromFile(o.metaDataFile)
|
|
|
|
if err != nil {
|
2019-04-03 14:29:58 +03:00
|
|
|
return 0, errors.Wrap(err, "cannot set metadata file")
|
2019-03-18 19:40:04 +03:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if o.metaDataServer {
|
|
|
|
// fetch metadata from server
|
|
|
|
metaData, err := migrateDrv.ExportMetadata()
|
|
|
|
if err != nil {
|
2019-04-03 14:29:58 +03:00
|
|
|
return 0, errors.Wrap(err, "cannot fetch metadata from server")
|
2019-03-18 19:40:04 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
tmpfile, err := ioutil.TempFile("", "metadata")
|
|
|
|
if err != nil {
|
2019-04-03 14:29:58 +03:00
|
|
|
return 0, errors.Wrap(err, "cannot create tempfile")
|
2019-03-18 19:40:04 +03:00
|
|
|
}
|
|
|
|
defer os.Remove(tmpfile.Name())
|
|
|
|
|
|
|
|
t, err := yaml.Marshal(metaData)
|
|
|
|
if err != nil {
|
2019-04-03 14:29:58 +03:00
|
|
|
return 0, errors.Wrap(err, "cannot marshal metadata")
|
2019-03-18 19:40:04 +03:00
|
|
|
}
|
|
|
|
if _, err := tmpfile.Write(t); err != nil {
|
2019-04-03 14:29:58 +03:00
|
|
|
return 0, errors.Wrap(err, "cannot write to temp file")
|
2019-03-18 19:40:04 +03:00
|
|
|
}
|
|
|
|
if err := tmpfile.Close(); err != nil {
|
2019-04-03 14:29:58 +03:00
|
|
|
return 0, errors.Wrap(err, "cannot close tmp file")
|
2019-03-18 19:40:04 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
err = createOptions.SetMetaUpFromFile(tmpfile.Name())
|
|
|
|
if err != nil {
|
2019-04-03 14:29:58 +03:00
|
|
|
return 0, errors.Wrap(err, "cannot parse metadata from the server")
|
2019-03-18 19:40:04 +03:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-04-30 11:34:08 +03:00
|
|
|
if !o.flags.Changed("sql-from-file") && !o.flags.Changed("metadata-from-file") && !o.metaDataServer && !o.sqlServer {
|
2019-03-18 19:40:04 +03:00
|
|
|
// Set empty data for [up|down].yaml
|
|
|
|
createOptions.MetaUp = []byte(`[]`)
|
|
|
|
createOptions.MetaDown = []byte(`[]`)
|
|
|
|
}
|
|
|
|
|
|
|
|
defer func() {
|
|
|
|
if err != nil {
|
|
|
|
createOptions.Delete()
|
|
|
|
}
|
|
|
|
}()
|
|
|
|
err = createOptions.Create()
|
2018-06-24 16:40:48 +03:00
|
|
|
if err != nil {
|
2019-04-03 14:29:58 +03:00
|
|
|
return 0, errors.Wrap(err, "error creating migration files")
|
2018-06-24 16:40:48 +03:00
|
|
|
}
|
2019-04-30 11:34:08 +03:00
|
|
|
return timestamp, nil
|
2018-06-24 16:40:48 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
func getTime() int64 {
|
|
|
|
startTime := time.Now()
|
|
|
|
return startTime.UnixNano() / int64(time.Millisecond)
|
|
|
|
}
|