mirror of
https://github.com/hasura/graphql-engine.git
synced 2024-12-15 09:22:43 +03:00
2ee7f7d76e
### Description Adds a `metadata diff` command to show comparisons between two different sets of Hasura metadata. ``` # Show changes between server metadata and the exported metadata file: hasura metadata diff # Show changes between server metadata and that in local_metadata.yaml: hasura metadata diff local_metadata.yaml # Show changes between metadata from metadata.yaml and metadata_old.yaml: hasura metadata diff metadata.yaml metadata_old.yaml ``` Also adds a `--dry-run` flag to `metadata apply` command which will print the diff and exit rather than actually applying the metadata. ### Affected components - CLI - Docs ### Related Issues Close #3126, Close #3127 ### Solution and Design - Added `metadata_diff.go` and `metadata_diff_test.go` ### Steps to test and verify ``` hasura metadata export # Make changes to migrations/metadata.yaml hasura metadata diff ``` ### Limitations, known bugs & workarounds This is just a general-purpose diff. A more contextual diff with the understanding of metadata can be added once #3072 is merged.
152 lines
4.2 KiB
Go
152 lines
4.2 KiB
Go
package commands
|
|
|
|
import (
|
|
"fmt"
|
|
"io"
|
|
"io/ioutil"
|
|
"os"
|
|
"strings"
|
|
|
|
"github.com/aryann/difflib"
|
|
"github.com/ghodss/yaml"
|
|
"github.com/hasura/graphql-engine/cli"
|
|
"github.com/mgutz/ansi"
|
|
"github.com/pkg/errors"
|
|
"github.com/spf13/cobra"
|
|
"github.com/spf13/viper"
|
|
)
|
|
|
|
type metadataDiffOptions struct {
|
|
EC *cli.ExecutionContext
|
|
output io.Writer
|
|
|
|
// two metadata to diff, 2nd is server if it's empty
|
|
metadata [2]string
|
|
}
|
|
|
|
func newMetadataDiffCmd(ec *cli.ExecutionContext) *cobra.Command {
|
|
v := viper.New()
|
|
opts := &metadataDiffOptions{
|
|
EC: ec,
|
|
output: os.Stdout,
|
|
}
|
|
|
|
metadataDiffCmd := &cobra.Command{
|
|
Use: "diff [file1] [file2]",
|
|
Short: "(PREVIEW) Show a highlighted diff of Hasura metadata",
|
|
Long: `(PREVIEW) Show changes between two different sets of Hasura metadata.
|
|
By default, shows changes between exported metadata file and server metadata.`,
|
|
Example: ` # NOTE: This command is in preview, usage and diff format may change.
|
|
|
|
# Show changes between server metadata and the exported metadata file:
|
|
hasura metadata diff
|
|
|
|
# Show changes between server metadata and that in local_metadata.yaml:
|
|
hasura metadata diff local_metadata.yaml
|
|
|
|
# Show changes between metadata from metadata.yaml and metadata_old.yaml:
|
|
hasura metadata diff metadata.yaml metadata_old.yaml`,
|
|
Args: cobra.MaximumNArgs(2),
|
|
PreRunE: func(cmd *cobra.Command, args []string) error {
|
|
ec.Viper = v
|
|
return ec.Validate()
|
|
},
|
|
RunE: func(cmd *cobra.Command, args []string) error {
|
|
messageFormat := "Showing diff between %s and %s..."
|
|
message := ""
|
|
|
|
switch len(args) {
|
|
case 0:
|
|
// no args, diff exported metadata and metadata on server
|
|
filename, err := ec.GetExistingMetadataFile()
|
|
if err != nil {
|
|
return errors.Wrap(err, "failed getting metadata file")
|
|
}
|
|
opts.metadata[0] = filename
|
|
message = fmt.Sprintf(messageFormat, filename, "the server")
|
|
case 1:
|
|
// 1 arg, diff given filename and the metadata on server
|
|
opts.metadata[0] = args[0]
|
|
message = fmt.Sprintf(messageFormat, args[0], "the server")
|
|
case 2:
|
|
// 2 args, diff given filenames
|
|
opts.metadata[0] = args[0]
|
|
opts.metadata[1] = args[1]
|
|
message = fmt.Sprintf(messageFormat, args[0], args[1])
|
|
}
|
|
|
|
opts.EC.Logger.Info(message)
|
|
err := opts.run()
|
|
if err != nil {
|
|
return errors.Wrap(err, "failed to show metadata diff")
|
|
}
|
|
return nil
|
|
},
|
|
}
|
|
|
|
f := metadataDiffCmd.Flags()
|
|
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")
|
|
|
|
// 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"))
|
|
|
|
return metadataDiffCmd
|
|
}
|
|
|
|
func (o *metadataDiffOptions) run() error {
|
|
var oldYaml, newYaml []byte
|
|
var err error
|
|
migrateDrv, err := newMigrate(o.EC.MigrationDir, o.EC.ServerConfig.ParsedEndpoint, o.EC.ServerConfig.AdminSecret, o.EC.Logger, o.EC.Version, true)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
if o.metadata[1] == "" {
|
|
// get metadata from server
|
|
m, err := migrateDrv.ExportMetadata()
|
|
if err != nil {
|
|
return errors.Wrap(err, "cannot fetch metadata from server")
|
|
}
|
|
|
|
newYaml, err = yaml.Marshal(m)
|
|
if err != nil {
|
|
return errors.Wrap(err, "cannot convert metadata from server to yaml")
|
|
}
|
|
} else {
|
|
newYaml, err = ioutil.ReadFile(o.metadata[1])
|
|
if err != nil {
|
|
return errors.Wrap(err, "cannot read file")
|
|
}
|
|
}
|
|
|
|
oldYaml, err = ioutil.ReadFile(o.metadata[0])
|
|
if err != nil {
|
|
return errors.Wrap(err, "cannot read file")
|
|
}
|
|
|
|
printDiff(string(oldYaml), string(newYaml), o.output)
|
|
return nil
|
|
}
|
|
|
|
func printDiff(before, after string, to io.Writer) {
|
|
diffs := difflib.Diff(strings.Split(before, "\n"), strings.Split(after, "\n"))
|
|
|
|
for _, diff := range diffs {
|
|
text := diff.Payload
|
|
|
|
switch diff.Delta {
|
|
case difflib.RightOnly:
|
|
fmt.Fprintf(to, "%s\n", ansi.Color(text, "green"))
|
|
case difflib.LeftOnly:
|
|
fmt.Fprintf(to, "%s\n", ansi.Color(text, "red"))
|
|
case difflib.Common:
|
|
fmt.Fprintf(to, "%s\n", text)
|
|
}
|
|
}
|
|
}
|