graphql-engine/cli/commands/metadata_apply.go
Aravind K P 33d6025a20 cli: refactor metadata object interface
https://github.com/hasura/graphql-engine-mono/pull/1618

GitOrigin-RevId: f4559de4852f247de35604be3808f05fe5ed341a
2021-07-23 09:50:36 +00:00

120 lines
3.8 KiB
Go

package commands
import (
"fmt"
"github.com/hasura/graphql-engine/cli/v2/internal/projectmetadata"
"github.com/hasura/graphql-engine/cli/v2"
"github.com/spf13/cobra"
)
func newMetadataApplyCmd(ec *cli.ExecutionContext) *cobra.Command {
opts := &MetadataApplyOptions{
EC: ec,
}
metadataApplyCmd := &cobra.Command{
Use: "apply",
Short: "Apply Hasura metadata on a database",
Example: ` # Apply Hasura GraphQL engine metadata present in metadata.[yaml|json] file:
hasura metadata apply
# Use with admin secret:
hasura metadata apply --admin-secret "<admin-secret>"
# Apply metadata to an instance specified by the flag:
hasura metadata apply --endpoint "<endpoint>"`,
SilenceUsage: true,
RunE: func(cmd *cobra.Command, args []string) error {
if opts.FromFile {
return fmt.Errorf("use of deprecated flag")
}
err := opts.Run()
if err != nil {
return err
}
return nil
},
}
f := metadataApplyCmd.Flags()
// deprecated flag
f.BoolVar(&opts.FromFile, "from-file", false, "apply metadata from migrations/metadata.[yaml|json]")
f.MarkDeprecated("from-file", "deprecation is a side effect of config v1 deprecation from v2.0.0")
f.BoolVar(&opts.DryRun, "dry-run", false, "show metadata generated from project directory without applying to server. generated metadata will be printed as JSON by default, use -o flag for other display formats")
f.StringVarP(&opts.rawOutput, "output", "o", "", `specify an output format to show applied metadata. Allowed values: json, yaml (default "json")`)
return metadataApplyCmd
}
type MetadataApplyOptions struct {
EC *cli.ExecutionContext
FromFile bool
DryRun bool
rawOutput string
}
func (o *MetadataApplyOptions) Run() error {
metadataHandler := projectmetadata.NewHandlerFromEC(o.EC)
if !o.DryRun {
o.EC.Spin("Applying metadata...")
if o.EC.Config.Version == cli.V2 {
_, err := metadataHandler.V1ApplyMetadata()
o.EC.Spinner.Stop()
if err != nil {
return errorApplyingMetadata(err)
}
o.EC.Logger.Debug("metadata applied using v1 replace_metadata")
} else {
r, err := metadataHandler.V2ApplyMetadata()
o.EC.Spinner.Stop()
if err != nil {
return errorApplyingMetadata(err)
}
if !r.IsConsistent {
o.EC.Logger.Warn("Metadata is inconsistent")
}
o.EC.Logger.Debug("metadata applied using v2 replace_metadata")
}
if len(o.rawOutput) <= 0 {
o.EC.Logger.Info("Metadata applied")
}
if len(o.rawOutput) != 0 {
// if not a dry run fetch metadata from and server and print it to stdout
return getMetadataFromServerAndWriteToStdoutByFormat(o.EC, rawOutputFormat(o.rawOutput))
}
return nil
}
if o.DryRun {
projectMetadataJSON, err := metadataHandler.MakeJSONMetadata()
if err != nil {
return fmt.Errorf("error building project metadata: %w", err)
}
if o.DryRun && len(o.rawOutput) == 0 {
// ie users who probably expect old behaviour
// show a warning about change in behaviour
o.rawOutput = string(rawOutputFormatJSON)
o.EC.Logger.Warn("behaviour of --dry-run flag has changed from v2.0.0. It used to show a diff between metadata on server and local project")
o.EC.Logger.Warn("new behaviour is to output local project metadata as JSON by default. The output format is configurable by -o flag eg: `hasura metadata apply --dry-run -o yaml`")
o.EC.Logger.Warn("the old behaviour can be achieved using `hasura metadata diff` command")
}
if err := writeByOutputFormat(o.EC.Stdout, projectMetadataJSON, rawOutputFormat(o.rawOutput)); err != nil {
return fmt.Errorf("displaying metadata failed: %w", err)
}
}
return nil
}
func errorApplyingMetadata(err error) error {
// a helper function to have consistent error messages for errors
// when applying metadata
return fmt.Errorf("error applying metadata \n%w", err)
}