2018-06-24 16:40:48 +03:00
package commands
import (
2021-03-08 14:59:35 +03:00
"fmt"
2018-09-10 16:21:30 +03:00
"os"
2021-04-01 15:58:24 +03:00
"path/filepath"
2018-06-24 16:40:48 +03:00
"strconv"
2021-06-18 09:24:16 +03:00
"github.com/hasura/graphql-engine/cli/v2/internal/hasura"
2021-06-16 14:44:15 +03:00
"github.com/hasura/graphql-engine/cli/v2/internal/metadatautil"
2021-04-01 15:58:24 +03:00
2021-06-16 14:44:15 +03:00
"github.com/hasura/graphql-engine/cli/v2"
migrate "github.com/hasura/graphql-engine/cli/v2/migrate"
2018-06-24 16:40:48 +03:00
"github.com/pkg/errors"
"github.com/spf13/cobra"
)
2018-06-28 11:36:25 +03:00
func newMigrateApplyCmd ( ec * cli . ExecutionContext ) * cobra . Command {
2020-02-24 19:14:46 +03:00
opts := & MigrateApplyOptions {
2018-06-24 16:40:48 +03:00
EC : ec ,
}
migrateApplyCmd := & cobra . Command {
2020-02-03 10:03:32 +03:00
Use : "apply" ,
Short : "Apply migrations on the database" ,
2019-12-12 08:16:36 +03:00
Example : ` # Apply all migrations
hasura migrate apply
# Use with admin secret :
hasura migrate apply -- admin - secret "<admin-secret>"
# Apply migrations on another Hasura instance :
hasura migrate apply -- endpoint "<endpoint>"
# Mark migration as applied on the server and skip execution :
hasura migrate apply -- skip - execution
# Apply a particular migration version only :
hasura migrate apply -- version "<version>"
# Apply last 2 down migrations :
hasura migrate apply -- down 2
# Apply only 2 up migrations :
hasura migrate apply -- up 2
# Apply only a particular version
hasura migrate apply -- type up -- version "<version>"
2020-02-03 10:03:32 +03:00
# Apply all up migrations upto version 125 , last applied is 100
hasura migrate apply -- goto 125
# Apply all down migrations upto version 125 , last applied is 150
hasura migrate apply -- goto 125
2019-12-12 08:16:36 +03:00
# Rollback a particular version :
hasura migrate apply -- type down -- version "<version>"
# Rollback all migrations :
hasura migrate apply -- down all ` ,
2018-06-24 16:40:48 +03:00
SilenceUsage : true ,
2018-09-27 16:57:17 +03:00
PreRunE : func ( cmd * cobra . Command , args [ ] string ) error {
2021-06-09 10:12:56 +03:00
return validateConfigV3Flags ( cmd , ec )
2018-09-27 16:57:17 +03:00
} ,
2018-06-24 16:40:48 +03:00
RunE : func ( cmd * cobra . Command , args [ ] string ) error {
2021-06-18 09:24:16 +03:00
return opts . Run ( )
2018-06-24 16:40:48 +03:00
} ,
}
f := migrateApplyCmd . Flags ( )
2020-02-03 10:03:32 +03:00
f . SortFlags = false
2018-06-24 16:40:48 +03:00
2020-02-24 19:14:46 +03:00
f . StringVar ( & opts . UpMigration , "up" , "" , "apply all or N up migration steps" )
f . StringVar ( & opts . DownMigration , "down" , "" , "apply all or N down migration steps" )
f . StringVar ( & opts . GotoVersion , "goto" , "" , "apply migration chain up to to the version specified" )
2018-09-27 16:57:17 +03:00
2020-02-24 19:14:46 +03:00
f . StringVar ( & opts . VersionMigration , "version" , "" , "only apply this particular migration" )
f . BoolVar ( & opts . SkipExecution , "skip-execution" , false , "skip executing the migration action, but mark them as applied" )
f . StringVar ( & opts . MigrationType , "type" , "up" , "type of migration (up, down) to be used with version flag" )
2018-09-27 16:57:17 +03:00
2021-06-18 09:24:16 +03:00
f . BoolVar ( & opts . DryRun , "dry-run" , false , "print the names of migrations which are going to be applied" )
f . BoolVar ( & opts . AllDatabases , "all-databases" , false , "set this flag to attempt to apply migrations on all databases present on server" )
2018-06-24 16:40:48 +03:00
return migrateApplyCmd
}
2020-02-24 19:14:46 +03:00
type MigrateApplyOptions struct {
2018-06-24 16:40:48 +03:00
EC * cli . ExecutionContext
2020-02-24 19:14:46 +03:00
UpMigration string
DownMigration string
VersionMigration string
MigrationType string
2020-02-03 10:03:32 +03:00
// version up to which migration chain has to be applied
2020-02-24 19:14:46 +03:00
GotoVersion string
SkipExecution bool
2021-06-18 09:24:16 +03:00
DryRun bool
2021-03-08 14:59:35 +03:00
Source cli . Source
2021-06-18 09:24:16 +03:00
AllDatabases bool
}
func ( o * MigrateApplyOptions ) Validate ( ) error {
if o . EC . Config . Version == cli . V2 {
o . Source . Kind = hasura . SourceKindPG
o . Source . Name = ""
}
if o . EC . Config . Version >= cli . V3 {
if ! o . AllDatabases && len ( o . Source . Name ) == 0 {
return fmt . Errorf ( "unable to determine database on which migration should be applied" )
}
if ! o . AllDatabases {
if len ( o . Source . Name ) == 0 {
return fmt . Errorf ( "empty database name" )
}
if len ( o . Source . Kind ) == 0 {
// find out the database kind by making a API call to server
// and update ec to include the database name and kind
sourceKind , err := metadatautil . GetSourceKind ( o . EC . APIClient . V1Metadata . ExportMetadata , o . Source . Name )
if err != nil {
return fmt . Errorf ( "determining database kind of %s: %w" , o . Source . Name , err )
}
if sourceKind == nil {
return fmt . Errorf ( "error determining database kind for %s, check if database exists on hasura" , o . Source . Name )
}
o . Source . Kind = * sourceKind
}
}
}
return nil
2021-04-01 15:58:24 +03:00
}
2021-06-18 09:24:16 +03:00
2021-04-01 15:58:24 +03:00
type errDatabaseMigrationDirectoryNotFound struct {
message string
2018-06-24 16:40:48 +03:00
}
2021-04-01 15:58:24 +03:00
func ( e * errDatabaseMigrationDirectoryNotFound ) Error ( ) string {
return e . message
}
2020-02-24 19:14:46 +03:00
func ( o * MigrateApplyOptions ) Run ( ) error {
2021-06-18 09:24:16 +03:00
if len ( o . Source . Name ) == 0 {
o . Source = o . EC . Source
}
if err := o . Validate ( ) ; err != nil {
return err
}
if o . DryRun && o . SkipExecution {
return errors . New ( "both --skip-execution and --dry-run flags cannot be used together" )
}
if o . AllDatabases && o . EC . Config . Version >= cli . V3 {
o . EC . Spin ( "getting lists of databases from server " )
sourcesAndKind , err := metadatautil . GetSourcesAndKind ( o . EC . APIClient . V1Metadata . ExportMetadata )
o . EC . Spinner . Stop ( )
if err != nil {
return fmt . Errorf ( "determing list of connected sources and kind: %w" , err )
}
for _ , source := range sourcesAndKind {
o . Source . Kind = source . Kind
o . Source . Name = source . Name
if ! o . DryRun {
o . EC . Spin ( fmt . Sprintf ( "Applying migrations on database: %s " , o . Source . Name ) )
}
err := o . Exec ( )
o . EC . Spinner . Stop ( )
if err != nil {
if err == migrate . ErrNoChange {
o . EC . Logger . Infof ( "nothing to apply on database: %s" , o . Source . Name )
continue
}
if e , ok := err . ( * os . PathError ) ; ok {
// If Op is first, then log No migrations to apply
if e . Op == "first" {
o . EC . Logger . Infof ( "nothing to apply on database: %s" , o . Source . Name )
continue
}
}
// check if the returned error is a directory not found error
// ie might be because a migrations/<source_name> directory is not found
// if so skip this
if e , ok := err . ( * errDatabaseMigrationDirectoryNotFound ) ; ok {
o . EC . Logger . Errorf ( "skipping applying migrations for database %s, encountered: \n%s" , o . Source . Name , e . Error ( ) )
continue
}
o . EC . Logger . Errorf ( "skipping applying migrations for database %s, encountered: \n%v" , o . Source . Name , err )
continue
}
o . EC . Logger . Infof ( "applied migrations on database: %s" , o . Source . Name )
}
} else {
if ! o . DryRun {
o . EC . Spin ( "Applying migrations..." )
}
err := o . Exec ( )
o . EC . Spinner . Stop ( )
if err != nil {
if err == migrate . ErrNoChange {
o . EC . Logger . Info ( "nothing to apply" )
return nil
}
// check if the returned error is a directory not found error
// ie might be because a migrations/<source_name> directory is not found
// if so skip this
if e , ok := err . ( * errDatabaseMigrationDirectoryNotFound ) ; ok {
return fmt . Errorf ( "applying migrations on database %s: %w" , o . Source . Name , e )
}
if e , ok := err . ( * os . PathError ) ; ok {
// If Op is first, then log No migrations to apply
if e . Op == "first" {
o . EC . Logger . Info ( "nothing to apply" )
return nil
}
}
return fmt . Errorf ( "apply failed\n%w" , err )
}
if ! o . DryRun {
o . EC . Logger . Info ( "migrations applied" )
}
}
return nil
}
func ( o * MigrateApplyOptions ) Exec ( ) error {
2021-04-01 15:58:24 +03:00
if o . EC . Config . Version >= cli . V3 {
// check if a migrations directory exists for source in project
migrationDirectory := filepath . Join ( o . EC . MigrationDir , o . Source . Name )
if f , err := os . Stat ( migrationDirectory ) ; err != nil || f == nil {
return & errDatabaseMigrationDirectoryNotFound { fmt . Sprintf ( "expected to find a migrations directory for database %s in %s, but encountered error: %s" , o . Source . Name , o . EC . MigrationDir , err . Error ( ) ) }
}
}
2021-06-18 09:24:16 +03:00
if o . AllDatabases && ( len ( o . GotoVersion ) > 0 || len ( o . VersionMigration ) > 0 ) {
2021-04-01 15:58:24 +03:00
return fmt . Errorf ( "cannot use --goto or --version in conjunction with --all-databases" )
}
2020-02-24 19:14:46 +03:00
migrationType , step , err := getMigrationTypeAndStep ( o . UpMigration , o . DownMigration , o . VersionMigration , o . MigrationType , o . GotoVersion , o . SkipExecution )
2018-06-24 16:40:48 +03:00
if err != nil {
return errors . Wrap ( err , "error validating flags" )
}
2021-03-08 14:59:35 +03:00
migrateDrv , err := migrate . NewMigrate ( o . EC , true , o . Source . Name , o . Source . Kind )
2018-06-24 16:40:48 +03:00
if err != nil {
2018-07-09 16:47:38 +03:00
return err
2018-06-24 16:40:48 +03:00
}
2020-02-24 19:14:46 +03:00
migrateDrv . SkipExecution = o . SkipExecution
2021-06-18 09:24:16 +03:00
migrateDrv . DryRun = o . DryRun
2018-06-24 16:40:48 +03:00
2020-02-03 10:03:32 +03:00
return ExecuteMigration ( migrationType , migrateDrv , step )
2018-06-24 16:40:48 +03:00
}
2018-06-28 11:36:25 +03:00
// Only one flag out of up, down and version can be set at a time. This function
// checks whether that is the case and returns an error is not
2020-02-03 10:03:32 +03:00
func getMigrationTypeAndStep ( upMigration , downMigration , versionMigration , migrationType , gotoVersion string , skipExecution bool ) ( string , int64 , error ) {
2018-06-24 16:40:48 +03:00
var flagCount = 0
var stepString = "all"
var migrationName = "up"
2018-06-28 11:36:25 +03:00
if upMigration != "" {
2018-06-24 16:40:48 +03:00
stepString = upMigration
flagCount ++
}
2018-06-28 11:36:25 +03:00
if downMigration != "" {
2018-06-24 16:40:48 +03:00
migrationName = "down"
stepString = downMigration
flagCount ++
}
2018-06-28 11:36:25 +03:00
if versionMigration != "" {
2018-06-24 16:40:48 +03:00
migrationName = "version"
stepString = versionMigration
2018-06-28 11:36:25 +03:00
if migrationType == "down" {
2018-06-24 16:40:48 +03:00
stepString = "-" + stepString
}
flagCount ++
}
2020-02-03 10:03:32 +03:00
if gotoVersion != "" {
migrationName = "gotoVersion"
stepString = gotoVersion
flagCount ++
}
2018-06-24 16:40:48 +03:00
if flagCount > 1 {
2020-02-24 19:14:46 +03:00
return "" , 0 , errors . New ( "only one migration type can be applied at a time (--up, --down or --goto)" )
2018-06-24 16:40:48 +03:00
}
2019-03-18 19:40:04 +03:00
if migrationName != "version" && skipExecution {
return "" , 0 , errors . New ( "--skip-execution flag can be set only with --version flag" )
}
2018-06-24 16:40:48 +03:00
if stepString == "all" && migrationName != "version" {
return migrationName , - 1 , nil
}
step , err := strconv . ParseInt ( stepString , 10 , 64 )
if err != nil {
2020-02-03 10:03:32 +03:00
return "" , 0 , errors . Wrap ( err , "not a valid input for steps/version" )
2018-06-24 16:40:48 +03:00
}
return migrationName , step , nil
}