graphql-engine/cli/migrate/api/migrate.go
Aravind K P 566b441b12 cli: remove config v1 specific logic from migrate squash
https://github.com/hasura/graphql-engine-mono/pull/2381

Co-authored-by: Kali Vara Purushotham Santhati <72007599+purush7@users.noreply.github.com>
GitOrigin-RevId: 42db5ea7ecd46838b5f2b266beca48e5af040bb9
2021-09-24 12:06:11 +00:00

290 lines
8.4 KiB
Go

package api
import (
"bytes"
"encoding/json"
"fmt"
"github.com/hasura/graphql-engine/cli/v2/internal/projectmetadata"
"io/ioutil"
"net/http"
"path/filepath"
"strings"
"time"
"github.com/hasura/graphql-engine/cli/v2/internal/hasura"
"github.com/hasura/graphql-engine/cli/v2/internal/metadatautil"
"github.com/gin-gonic/gin"
"github.com/hasura/graphql-engine/cli/v2"
"github.com/hasura/graphql-engine/cli/v2/migrate"
"github.com/hasura/graphql-engine/cli/v2/migrate/cmd"
"github.com/sirupsen/logrus"
)
const (
DataAPIError = "Data Error: "
MigrationMode = "migration_mode"
)
type Response struct {
Code string `json:"code,omitempty"`
Message string `json:"message,omitempty"`
Name string `json:"name,omitempty"`
StatusCode int `json:"-"`
}
type Request struct {
Name string `json:"name"`
Up []requestType `json:"up"`
Down []requestType `json:"down"`
SkipExecution bool `json:"skip_execution"`
SourceName string `json:"datasource,omitempty"`
}
type requestType struct {
Version int `json:"version,omitempty"`
Type string `json:"type"`
Database string `json:"datasource,omitempty"`
Args interface{} `json:"args"`
}
func MigrateAPI(c *gin.Context) {
ecPtr, ok := c.Get("ec")
if !ok {
return
}
// Get File url
//sourcePtr, ok := c.Get("filedir")
if !ok {
return
}
// Get Logger
loggerPtr, ok := c.Get("logger")
if !ok {
return
}
// Convert to url.URL
ec, ok := ecPtr.(*cli.ExecutionContext)
if !ok {
c.JSON(http.StatusInternalServerError, &Response{Code: "internal_error", Message: "cannot get execution context"})
return
}
//sourceURL := sourcePtr.(*url.URL)
logger := loggerPtr.(*logrus.Logger)
mdHandler := projectmetadata.NewHandlerFromEC(ec)
// Switch on request method
switch c.Request.Method {
case "GET":
sourceName := c.Query("datasource")
if ec.Config.Version >= cli.V3 && sourceName == "" {
c.JSON(http.StatusInternalServerError, &Response{Code: "internal_error", Message: "datasource query parameter is required"})
return
}
sourceKind := hasura.SourceKindPG
if ec.Config.Version >= cli.V3 {
kind, err := metadatautil.GetSourceKind(ec.APIClient.V1Metadata.ExportMetadata, sourceName)
if err != nil {
c.JSON(http.StatusInternalServerError, &Response{Code: "internal_error", Message: err.Error()})
return
}
if kind == nil {
c.JSON(http.StatusInternalServerError, &Response{Code: "request_parse_error", Message: fmt.Sprintf("cannot determine source kind for %v", sourceName)})
return
}
sourceKind = *kind
}
t, err := migrate.NewMigrate(ec, false, sourceName, sourceKind)
if err != nil {
c.JSON(http.StatusInternalServerError, &Response{Code: "internal_error", Message: err.Error()})
return
}
// Rescan file system
err = t.ReScan()
if err != nil {
c.JSON(http.StatusInternalServerError, &Response{Code: "internal_error", Message: err.Error()})
return
}
status, err := t.GetStatus()
if err != nil {
c.JSON(http.StatusInternalServerError, &Response{Code: "internal_error", Message: "Something went wrong"})
return
}
c.JSON(http.StatusOK, status)
case "POST":
var request Request
var err error
// Bind Request body to Request struct
if err = c.BindJSON(&request); err != nil {
c.JSON(http.StatusInternalServerError, &Response{Code: "request_parse_error", Message: err.Error()})
return
}
if ec.Config.Version >= cli.V3 && request.SourceName == "" {
c.JSON(http.StatusInternalServerError, &Response{Code: "internal_error", Message: "datasource key not found in body"})
return
}
startTime := time.Now()
timestamp := startTime.UnixNano() / int64(time.Millisecond)
sourceName := request.SourceName
if ec.Config.Version < cli.V3 {
sourceName = ""
}
sourceKind := hasura.SourceKindPG
if ec.Config.Version >= cli.V3 {
kind, err := metadatautil.GetSourceKind(ec.APIClient.V1Metadata.ExportMetadata, sourceName)
if err != nil {
c.JSON(http.StatusInternalServerError, &Response{Code: "internal_error", Message: err.Error()})
return
}
if kind == nil && len(request.Up) == 0 {
c.JSON(http.StatusInternalServerError, &Response{Code: "request_parse_error", Message: fmt.Sprintf("didn't find any request in UP in query body %v", sourceName)})
return
}
if kind == nil && len(request.Up) != 0 {
switch databaseKind := strings.Split(request.Up[0].Type, "_")[0]; databaseKind {
case "pg":
sourceKind = hasura.SourceKindPG
case "mssql":
sourceKind = hasura.SourceKindMSSQL
case "citus":
sourceKind = hasura.SourceKindCitus
default:
c.JSON(http.StatusInternalServerError, &Response{Code: "request_parse_error", Message: fmt.Sprintf("cannot determine source kind for %v", sourceName)})
return
}
}
if kind != nil {
sourceKind = *kind
}
}
t, err := migrate.NewMigrate(ec, false, sourceName, sourceKind)
if err != nil {
c.JSON(http.StatusInternalServerError, &Response{Code: "internal_error", Message: err.Error()})
return
}
createOptions := cmd.New(timestamp, request.Name, filepath.Join(ec.MigrationDir, sourceName))
sqlUp := &bytes.Buffer{}
sqlDown := &bytes.Buffer{}
for _, arg := range request.Up {
if strings.Contains(arg.Type, "run_sql") {
argByt, err := json.Marshal(arg.Args)
if err != nil {
c.JSON(http.StatusInternalServerError, &Response{Code: "request_parse_error", Message: err.Error()})
return
}
var to hasura.PGRunSQLInput
err = json.Unmarshal(argByt, &to)
if err != nil {
c.JSON(http.StatusInternalServerError, &Response{Code: "request_parse_error", Message: err.Error()})
return
}
sqlUp.WriteString(to.SQL)
sqlUp.WriteString("\n")
}
}
for _, arg := range request.Down {
if strings.Contains(arg.Type, "run_sql") {
argByt, err := json.Marshal(arg.Args)
if err != nil {
c.JSON(http.StatusInternalServerError, &Response{Code: "request_parse_error", Message: err.Error()})
return
}
var to hasura.PGRunSQLInput
err = json.Unmarshal(argByt, &to)
if err != nil {
c.JSON(http.StatusInternalServerError, &Response{Code: "request_parse_error", Message: err.Error()})
return
}
sqlDown.WriteString(to.SQL)
sqlDown.WriteString("\n")
}
}
if sqlUp.String() != "" {
err = createOptions.SetSQLUp(sqlUp.String())
if err != nil {
c.JSON(http.StatusInternalServerError, &Response{Code: "create_file_error", Message: err.Error()})
return
}
}
if sqlDown.String() != "" {
err = createOptions.SetSQLDown(sqlDown.String())
if err != nil {
c.JSON(http.StatusInternalServerError, &Response{Code: "create_file_error", Message: err.Error()})
return
}
}
if sqlUp.String() != "" || sqlDown.String() != "" {
err = createOptions.Create()
if err != nil {
c.JSON(http.StatusInternalServerError, &Response{Code: "create_file_error", Message: err.Error()})
return
}
} else {
timestamp = 0
}
defer func() {
if err != nil && timestamp != 0 {
err := createOptions.Delete()
if err != nil {
logger.Debug(err)
}
}
}()
// Rescan file system
err = t.ReScan()
if err != nil {
c.JSON(http.StatusInternalServerError, &Response{Code: "internal_error", Message: err.Error()})
return
}
upByt, err := json.Marshal(request.Up)
if err != nil {
c.JSON(http.StatusInternalServerError, &Response{Code: "internal_error", Message: err.Error()})
return
}
if err = t.QueryWithVersion(uint64(timestamp), ioutil.NopCloser(bytes.NewReader(upByt)), request.SkipExecution); err != nil {
if strings.HasPrefix(err.Error(), DataAPIError) {
c.JSON(http.StatusBadRequest, &Response{Code: "data_api_error", Message: strings.TrimPrefix(err.Error(), DataAPIError)})
return
}
if err == migrate.ErrNoMigrationMode {
c.JSON(http.StatusBadRequest, &Response{Code: "migration_mode_disabled", Message: err.Error()})
return
}
c.JSON(http.StatusInternalServerError, &Response{Code: "internal_error", Message: err.Error()})
return
}
defer func() {
var files map[string][]byte
files, err = mdHandler.ExportMetadata()
if err != nil {
logger.Debug(err)
return
}
err = mdHandler.WriteMetadata(files)
if err != nil {
logger.Debug(err)
return
}
}()
c.JSON(http.StatusOK, &Response{Name: fmt.Sprintf("%d_%s", timestamp, request.Name)})
default:
c.JSON(http.StatusMethodNotAllowed, &gin.H{"message": "Method not allowed"})
}
}