graphql-engine/cli/pkg/metadata/mode_handlers.go

164 lines
4.4 KiB
Go
Raw Normal View History

package metadata
import (
"bytes"
"encoding/json"
"fmt"
"io"
"io/ioutil"
goyaml "github.com/goccy/go-yaml"
"github.com/hasura/graphql-engine/cli/v2/internal/hasura"
"github.com/hasura/graphql-engine/cli/v2/commands"
"github.com/hasura/graphql-engine/cli/v2/internal/projectmetadata"
"github.com/hasura/graphql-engine/cli/v2"
)
type modeHandler interface {
Apply(*ProjectMetadata) (io.Reader, error)
Diff(*ProjectMetadata) (io.Reader, error)
Parse(*ProjectMetadata) (io.Reader, error)
}
func getModeHandler(mode cli.MetadataMode) modeHandler {
switch mode {
case cli.MetadataModeYAML:
return &metadataModeYAMLHandler{}
case cli.MetadataModeJSON:
return &metadataModeJSONHandler{}
default:
return &metadataModeDirectoryHandler{}
}
}
type metadataModeDirectoryHandler struct{}
func (m *metadataModeDirectoryHandler) Apply(p *ProjectMetadata) (io.Reader, error) {
metadataHandler := projectmetadata.NewHandlerFromEC(p.ec)
if p.ec.Config.Version == cli.V2 {
r, err := metadataHandler.V1ApplyMetadata()
if err != nil {
return nil, err
}
return r, nil
}
if p.ec.Config.Version >= cli.V3 {
replaceMetadataResponse, err := metadataHandler.V2ApplyMetadata()
if err != nil {
return nil, err
}
b := new(bytes.Buffer)
if err := json.NewEncoder(b).Encode(replaceMetadataResponse); err != nil {
return nil, fmt.Errorf("encoding json reponse from server: %w", err)
}
return b, nil
}
return nil, nil
}
func (m *metadataModeDirectoryHandler) Parse(p *ProjectMetadata) (io.Reader, error) {
metadataHandler := projectmetadata.NewHandlerFromEC(p.ec)
jsonMetadata, err := metadataHandler.BuildJSONMetadata()
if err != nil {
return nil, fmt.Errorf("parsing project metadata to json failed: %w", err)
}
return bytes.NewReader(jsonMetadata), nil
}
func (m *metadataModeDirectoryHandler) Diff(p *ProjectMetadata) (io.Reader, error) {
return diff(p)
}
func diff(p *ProjectMetadata) (io.Reader, error) {
w := new(bytes.Buffer)
opts := &commands.MetadataDiffOptions{
EC: p.ec,
Output: w,
DisableColor: true,
DiffType: string(commands.DifftypeYAML),
}
if err := opts.Run(); err != nil {
return nil, err
}
return w, nil
}
type metadataModeJSONHandler struct{}
func (m *metadataModeJSONHandler) Apply(p *ProjectMetadata) (io.Reader, error) {
return apply(p, p.ec.MetadataMode)
}
func (m *metadataModeJSONHandler) Parse(p *ProjectMetadata) (io.Reader, error) {
metadataJSON, err := ioutil.ReadFile(p.ec.MetadataFile)
if err != nil {
return nil, fmt.Errorf("reading metadata file: %w", err)
}
return bytes.NewReader(metadataJSON), nil
}
func (m *metadataModeJSONHandler) Diff(p *ProjectMetadata) (io.Reader, error) {
return diff(p)
}
type metadataModeYAMLHandler struct{}
func (m *metadataModeYAMLHandler) Apply(p *ProjectMetadata) (io.Reader, error) {
return apply(p, p.ec.MetadataMode)
}
func (m *metadataModeYAMLHandler) Parse(p *ProjectMetadata) (io.Reader, error) {
metadataYAML, err := ioutil.ReadFile(p.ec.MetadataFile)
if err != nil {
return nil, fmt.Errorf("reading metadata file: %w", err)
}
metadataJSON, err := goyaml.YAMLToJSON(metadataYAML)
if err != nil {
return nil, fmt.Errorf("parsing local yaml metadata as json: %w", err)
}
return bytes.NewReader(metadataJSON), nil
}
func (m *metadataModeYAMLHandler) Diff(p *ProjectMetadata) (io.Reader, error) {
return diff(p)
}
func apply(p *ProjectMetadata, mode cli.MetadataMode) (io.Reader, error) {
var localMetadataBytes []byte
var err error
localMetadataBytes, err = ioutil.ReadFile(p.ec.MetadataFile)
if err != nil {
return nil, fmt.Errorf("reading metadata file: %w", err)
}
if mode == cli.MetadataModeYAML {
localMetadataBytes, err = goyaml.YAMLToJSON(localMetadataBytes)
if err != nil {
return nil, fmt.Errorf("parsing yaml metadata to json: %w", err)
}
}
var metadata interface{}
err = json.Unmarshal(localMetadataBytes, &metadata)
if err != nil {
return nil, fmt.Errorf("parsing metadata as json: %w", err)
}
if p.ec.Config.Version == cli.V2 {
return cli.GetCommonMetadataOps(p.ec).ReplaceMetadata(bytes.NewReader(localMetadataBytes))
}
r, err := p.ec.APIClient.V1Metadata.V2ReplaceMetadata(hasura.V2ReplaceMetadataArgs{
AllowInconsistentMetadata: true,
Metadata: metadata,
})
if err != nil {
return nil, fmt.Errorf("applying metadata: %w", err)
}
b := new(bytes.Buffer)
if err := json.NewEncoder(b).Encode(r); err != nil {
return nil, fmt.Errorf("encoding json reponse from server: %w", err)
}
return b, nil
}