graphql-engine/cli/commands/console.go

246 lines
6.5 KiB
Go
Raw Normal View History

2018-06-24 16:40:48 +03:00
package commands
import (
"fmt"
"net/http"
"net/url"
"sync"
"github.com/fatih/color"
"github.com/gin-contrib/cors"
"github.com/gin-contrib/static"
2018-06-24 16:40:48 +03:00
"github.com/gin-gonic/gin"
2018-06-24 16:47:01 +03:00
"github.com/hasura/graphql-engine/cli"
2018-06-24 16:40:48 +03:00
"github.com/hasura/graphql-engine/cli/migrate/api"
2018-06-24 16:47:01 +03:00
"github.com/hasura/graphql-engine/cli/util"
2018-06-24 16:40:48 +03:00
"github.com/pkg/errors"
"github.com/sirupsen/logrus"
2018-06-24 16:40:48 +03:00
"github.com/skratchdot/open-golang/open"
"github.com/spf13/cobra"
2018-06-24 16:47:01 +03:00
"github.com/spf13/viper"
2018-06-24 16:40:48 +03:00
)
func NewConsoleCmd(ec *cli.ExecutionContext) *cobra.Command {
2018-06-24 16:47:01 +03:00
v := viper.New()
2018-06-24 16:40:48 +03:00
opts := &consoleOptions{
EC: ec,
}
consoleCmd := &cobra.Command{
Use: "console",
Short: "Open console to manage database and try out APIs",
Long: "Run a web server to serve Hasura Console for GraphQL Engine to manage database and build queries",
Example: ` # Start console:
hasura console
# Start console on a different address and ports:
hasura console --address 0.0.0.0 --console-port 8080 --api-port 8081`,
SilenceUsage: true,
2018-06-24 16:47:01 +03:00
PreRunE: func(cmd *cobra.Command, args []string) error {
ec.Viper = v
return ec.Validate()
},
2018-06-24 16:40:48 +03:00
RunE: func(cmd *cobra.Command, args []string) error {
return opts.run()
2018-06-24 16:40:48 +03:00
},
}
f := consoleCmd.Flags()
f.StringVar(&opts.APIPort, "api-port", "9693", "port for serving migrate api")
f.StringVar(&opts.ConsolePort, "console-port", "9695", "port for serving console")
f.StringVar(&opts.Address, "address", "localhost", "address to use")
2018-06-27 15:04:09 +03:00
f.BoolVar(&opts.DontOpenBrowser, "no-browser", false, "do not automatically open console in browser")
f.StringVar(&opts.StaticDir, "static-dir", "", "directory where static assets mentioned in the console html template can be served from")
2018-06-24 16:47:01 +03:00
f.String("endpoint", "", "http(s) endpoint for Hasura GraphQL Engine")
f.String("access-key", "", "access key for Hasura GraphQL Engine")
// need to create a new viper because https://github.com/spf13/viper/issues/233
v.BindPFlag("endpoint", f.Lookup("endpoint"))
v.BindPFlag("access_key", f.Lookup("access-key"))
2018-06-24 16:40:48 +03:00
return consoleCmd
}
type consoleOptions struct {
EC *cli.ExecutionContext
APIPort string
ConsolePort string
Address string
2018-06-27 15:04:09 +03:00
DontOpenBrowser bool
WG *sync.WaitGroup
StaticDir string
2018-06-24 16:40:48 +03:00
}
func (o *consoleOptions) run() error {
2018-06-24 16:40:48 +03:00
log := o.EC.Logger
// Switch to "release" mode in production.
gin.SetMode(gin.ReleaseMode)
// An Engine instance with the Logger and Recovery middleware already attached.
r := gin.New()
r.Use(allowCors())
// My Router struct
router := &cRouter{
2018-06-24 16:40:48 +03:00
r,
}
router.setRoutes(o.EC.Config.ParsedEndpoint, o.EC.Config.AccessKey, o.EC.MigrationDir, o.EC.MetadataFile, o.EC.Logger)
2018-06-24 16:40:48 +03:00
if o.EC.Version == nil {
return errors.New("cannot validate version, object is nil")
}
consoleTemplateVersion := o.EC.Version.GetConsoleTemplateVersion()
consoleAssetsVersion := o.EC.Version.GetConsoleAssetsVersion()
o.EC.Logger.Debugf("rendering console template [%s] with assets [%s]", consoleTemplateVersion, consoleAssetsVersion)
2018-06-24 16:40:48 +03:00
consoleRouter, err := serveConsole(consoleTemplateVersion, o.StaticDir, gin.H{
2018-06-24 16:40:48 +03:00
"apiHost": "http://" + o.Address,
"apiPort": o.APIPort,
"cliVersion": o.EC.Version.GetCLIVersion(),
"dataApiUrl": o.EC.Config.ParsedEndpoint.String(),
2018-06-24 16:40:48 +03:00
"dataApiVersion": "",
"accessKey": o.EC.Config.AccessKey,
"assetsVersion": consoleAssetsVersion,
2018-06-24 16:40:48 +03:00
})
if err != nil {
return errors.Wrap(err, "error serving console")
}
// Create WaitGroup for running 3 servers
wg := &sync.WaitGroup{}
2018-06-27 15:04:09 +03:00
o.WG = wg
2018-06-24 16:40:48 +03:00
wg.Add(1)
go func() {
err = router.Run(o.Address + ":" + o.APIPort)
if err != nil {
o.EC.Logger.WithError(err).Errorf("error listening on port %s", o.APIPort)
}
wg.Done()
}()
wg.Add(1)
go func() {
err = consoleRouter.Run(o.Address + ":" + o.ConsolePort)
if err != nil {
o.EC.Logger.WithError(err).Errorf("error listening on port %s", o.ConsolePort)
}
wg.Done()
}()
consoleURL := fmt.Sprintf("http://%s:%s/", o.Address, o.ConsolePort)
2018-06-24 16:40:48 +03:00
2018-06-27 15:04:09 +03:00
if !o.DontOpenBrowser {
o.EC.Spin(color.CyanString("Opening console using default browser..."))
defer o.EC.Spinner.Stop()
2018-06-24 16:40:48 +03:00
2018-06-27 15:04:09 +03:00
err = open.Run(consoleURL)
if err != nil {
o.EC.Logger.WithError(err).Warn("Error opening browser, try to open the url manually?")
}
2018-06-24 16:40:48 +03:00
}
o.EC.Spinner.Stop()
log.Infof("console running at: %s", consoleURL)
wg.Wait()
return nil
}
type cRouter struct {
2018-06-24 16:40:48 +03:00
*gin.Engine
}
func (router *cRouter) setRoutes(nurl *url.URL, accessKey, migrationDir, metadataFile string, logger *logrus.Logger) {
2018-06-24 16:40:48 +03:00
apis := router.Group("/apis")
{
apis.Use(setLogger(logger))
apis.Use(setFilePath(migrationDir))
apis.Use(setDataPath(nurl, accessKey))
2018-06-24 16:40:48 +03:00
// Migrate api endpoints and middleware
migrateAPIs := apis.Group("/migrate")
{
settingsAPIs := migrateAPIs.Group("/settings")
{
settingsAPIs.Any("", api.SettingsAPI)
}
migrateAPIs.Any("", api.MigrateAPI)
}
// Migrate api endpoints and middleware
metadataAPIs := apis.Group("/metadata")
{
metadataAPIs.Use(setMetadataFile(metadataFile))
2018-06-24 16:40:48 +03:00
metadataAPIs.Any("", api.MetadataAPI)
}
}
}
func setDataPath(nurl *url.URL, accessKey string) gin.HandlerFunc {
2018-06-24 16:40:48 +03:00
return func(c *gin.Context) {
host := getDataPath(nurl, accessKey)
2018-06-24 16:40:48 +03:00
c.Set("dbpath", host)
c.Next()
}
}
func setFilePath(dir string) gin.HandlerFunc {
return func(c *gin.Context) {
host := getFilePath(dir)
c.Set("filedir", host)
c.Next()
}
}
func setMetadataFile(file string) gin.HandlerFunc {
return func(c *gin.Context) {
c.Set("metadataFile", file)
c.Next()
}
}
func setLogger(logger *logrus.Logger) gin.HandlerFunc {
return func(c *gin.Context) {
c.Set("logger", logger)
2018-06-24 16:40:48 +03:00
c.Next()
}
}
func allowCors() gin.HandlerFunc {
config := cors.DefaultConfig()
config.AddAllowHeaders("X-Hasura-User-Id")
config.AddAllowHeaders("X-Hasura-Access-Key")
config.AddAllowHeaders("X-Hasura-Role")
config.AddAllowHeaders("X-Hasura-Allowed-Roles")
config.AddAllowMethods("DELETE")
config.AllowAllOrigins = true
config.AllowCredentials = false
return cors.New(config)
}
func serveConsole(assetsVersion, staticDir string, opts gin.H) (*gin.Engine, error) {
2018-06-24 16:40:48 +03:00
// An Engine instance with the Logger and Recovery middleware already attached.
r := gin.New()
// Template console.html
templateRender, err := util.LoadTemplates("assets/"+assetsVersion+"/", "console.html")
2018-06-24 16:40:48 +03:00
if err != nil {
return nil, errors.Wrap(err, "cannot fetch template")
}
r.HTMLRender = templateRender
if staticDir != "" {
r.Use(static.Serve("/static", static.LocalFile(staticDir, false)))
opts["cliStaticDir"] = staticDir
}
r.GET("/*action", func(c *gin.Context) {
2018-06-24 16:40:48 +03:00
c.HTML(http.StatusOK, "console.html", &opts)
})
return r, nil
}