super-graph/serv/cmd.go

335 lines
7.2 KiB
Go
Raw Permalink Normal View History

2019-09-20 06:19:11 +02:00
package serv
import (
2019-09-26 06:35:31 +02:00
"context"
2019-09-20 06:19:11 +02:00
"fmt"
"os"
"runtime"
2019-09-20 06:19:11 +02:00
"strings"
"github.com/dosco/super-graph/psql"
"github.com/dosco/super-graph/qcode"
2019-09-26 06:35:31 +02:00
"github.com/jackc/pgx/v4"
"github.com/jackc/pgx/v4/pgxpool"
2019-09-20 06:19:11 +02:00
"github.com/rs/zerolog"
2019-09-26 06:35:31 +02:00
"github.com/spf13/cobra"
2019-09-20 06:19:11 +02:00
"github.com/spf13/viper"
)
//go:generate rice embed-go
2019-09-20 06:19:11 +02:00
const (
serverName = "Super Graph"
)
var (
// These variables are set using -ldflags
version string
gitBranch string
lastCommitSHA string
lastCommitTime string
)
2019-09-20 06:19:11 +02:00
var (
logger zerolog.Logger // logger for everything but errors
errlog zerolog.Logger // logger for errors includes line numbers
conf *config // parsed config
confPath string // path to the config file
db *pgxpool.Pool // database connection pool
schema *psql.DBSchema // database tables, columns and relationships
qcompile *qcode.Compiler // qcode compiler
pcompile *psql.Compiler // postgres sql compiler
2019-09-20 06:19:11 +02:00
)
2019-09-26 06:35:31 +02:00
func Init() {
2019-11-25 08:22:33 +01:00
initLog()
2019-09-28 17:34:03 +02:00
rootCmd := &cobra.Command{
2019-09-26 06:35:31 +02:00
Use: "super-graph",
2019-11-28 21:27:20 +01:00
Short: BuildDetails(),
2019-09-26 06:35:31 +02:00
}
2019-09-28 17:34:03 +02:00
rootCmd.AddCommand(&cobra.Command{
2019-09-26 06:35:31 +02:00
Use: "serv",
Short: "Run the super-graph service",
Run: cmdServ,
2019-09-28 17:34:03 +02:00
})
rootCmd.AddCommand(&cobra.Command{
Use: "db:create",
Short: "Create database",
Run: cmdDBCreate,
})
rootCmd.AddCommand(&cobra.Command{
Use: "db:drop",
Short: "Drop database",
Run: cmdDBDrop,
})
rootCmd.AddCommand(&cobra.Command{
Use: "db:seed",
Short: "Run the seed script to seed the database",
Run: cmdDBSeed,
})
2019-09-26 06:35:31 +02:00
2019-09-28 17:34:03 +02:00
rootCmd.AddCommand(&cobra.Command{
Use: "db:migrate",
2019-09-26 06:35:31 +02:00
Short: "Migrate the database",
Long: `Migrate the database to destination migration version.
Destination migration version can be one of the following value types:
2019-09-28 17:34:03 +02:00
Migrate to the most recent migration.
2019-09-29 02:46:55 +02:00
e.g. db:migrate up
2019-09-28 17:34:03 +02:00
Rollback the most recent migration.
2019-09-29 02:46:55 +02:00
e.g. db:migrate down
2019-09-28 17:34:03 +02:00
2019-09-26 06:35:31 +02:00
Migrate to a specific migration.
2019-09-29 02:46:55 +02:00
e.g. db:migrate 42
2019-09-26 06:35:31 +02:00
Migrate forward N steps.
2019-09-29 02:46:55 +02:00
e.g. db:migrate +3
2019-09-26 06:35:31 +02:00
Migrate backward N steps.
2019-09-29 02:46:55 +02:00
e.g. db:migrate -2
2019-09-26 06:35:31 +02:00
Redo previous N steps (migrate backward N steps then forward N steps).
2019-09-29 02:46:55 +02:00
e.g. db:migrate -+1
2019-09-26 06:35:31 +02:00
`,
2019-09-28 17:34:03 +02:00
Run: cmdDBMigrate,
})
2019-09-26 06:35:31 +02:00
2019-09-28 17:34:03 +02:00
rootCmd.AddCommand(&cobra.Command{
Use: "db:status",
2019-09-26 06:35:31 +02:00
Short: "Print current migration status",
2019-09-28 17:34:03 +02:00
Run: cmdDBStatus,
})
2019-09-26 06:35:31 +02:00
2019-09-28 17:34:03 +02:00
rootCmd.AddCommand(&cobra.Command{
Use: "db:new NAME",
2019-09-26 06:35:31 +02:00
Short: "Generate a new migration",
Long: "Generate a new migration with the next sequence number and provided name",
2019-09-28 17:34:03 +02:00
Run: cmdDBNew,
})
rootCmd.AddCommand(&cobra.Command{
Use: "db:setup",
Short: "Setup database",
Long: "This command will create, migrate and seed the database",
Run: cmdDBSetup,
})
rootCmd.AddCommand(&cobra.Command{
Use: "db:reset",
Short: "Reset database",
Long: "This command will drop, create, migrate and seed the database (won't run in production)",
Run: cmdDBReset,
})
2019-09-28 17:34:03 +02:00
rootCmd.AddCommand(&cobra.Command{
Use: "new APP-NAME",
Short: "Create a new application",
Long: "Generate all the required files to start on a new Super Graph app",
2019-09-28 17:34:03 +02:00
Run: cmdNew,
})
2019-09-26 06:35:31 +02:00
rootCmd.AddCommand(&cobra.Command{
Use: fmt.Sprintf("conf:dump [%s]", strings.Join(viper.SupportedExts, "|")),
Short: "Dump config to file",
Long: "Dump current config to a file in the selected format",
Run: cmdConfDump,
})
rootCmd.AddCommand(&cobra.Command{
Use: "version",
Short: "Super Graph binary version information",
Run: cmdVersion,
})
2019-09-26 06:35:31 +02:00
rootCmd.Flags().StringVar(&confPath,
"path", "./config", "path to config files")
if err := rootCmd.Execute(); err != nil {
2019-11-25 08:22:33 +01:00
errlog.Fatal().Err(err).Send()
2019-09-26 06:35:31 +02:00
}
}
2019-11-25 08:22:33 +01:00
func initLog() {
2019-10-03 09:08:01 +02:00
out := zerolog.ConsoleWriter{Out: os.Stderr}
2019-11-25 08:22:33 +01:00
logger = zerolog.New(out).With().Timestamp().Logger()
errlog = logger.With().Caller().Logger()
2019-09-20 06:19:11 +02:00
}
2019-09-26 06:35:31 +02:00
func initConf() (*config, error) {
vi := newConfig(getConfigName())
2019-09-20 06:19:11 +02:00
if err := vi.ReadInConfig(); err != nil {
return nil, err
}
2019-10-31 06:14:51 +01:00
inherits := vi.GetString("inherits")
if len(inherits) != 0 {
vi = newConfig(inherits)
if err := vi.ReadInConfig(); err != nil {
return nil, err
}
2019-10-31 06:14:51 +01:00
if vi.IsSet("inherits") {
2019-11-25 08:22:33 +01:00
errlog.Fatal().Msgf("inherited config (%s) cannot itself inherit (%s)",
2019-10-31 06:14:51 +01:00
inherits,
vi.GetString("inherits"))
}
vi.SetConfigName(getConfigName())
if err := vi.MergeInConfig(); err != nil {
return nil, err
}
}
2019-10-31 06:14:51 +01:00
c := &config{}
2019-09-20 06:19:11 +02:00
2019-10-31 06:14:51 +01:00
if err := c.Init(vi); err != nil {
2019-09-20 06:19:11 +02:00
return nil, fmt.Errorf("unable to decode config, %v", err)
}
2019-09-26 06:35:31 +02:00
logLevel, err := zerolog.ParseLevel(c.LogLevel)
if err != nil {
2019-11-25 08:22:33 +01:00
errlog.Error().Err(err).Msg("error setting log_level")
2019-09-26 06:35:31 +02:00
}
zerolog.SetGlobalLevel(logLevel)
2019-09-20 06:19:11 +02:00
return c, nil
}
2019-09-28 17:34:03 +02:00
func initDB(c *config, useDB bool) (*pgx.Conn, error) {
2019-09-26 06:35:31 +02:00
config, _ := pgx.ParseConfig("")
config.Host = c.DB.Host
config.Port = c.DB.Port
config.User = c.DB.User
config.Password = c.DB.Password
config.RuntimeParams = map[string]string{
"application_name": c.AppName,
"search_path": c.DB.Schema,
2019-09-20 06:19:11 +02:00
}
2019-09-28 17:34:03 +02:00
if useDB {
config.Database = c.DB.DBName
}
2019-09-26 06:35:31 +02:00
switch c.LogLevel {
case "debug":
config.LogLevel = pgx.LogLevelDebug
case "info":
config.LogLevel = pgx.LogLevelInfo
case "warn":
config.LogLevel = pgx.LogLevelWarn
case "error":
config.LogLevel = pgx.LogLevelError
default:
config.LogLevel = pgx.LogLevelNone
2019-09-20 06:19:11 +02:00
}
2019-11-25 08:22:33 +01:00
config.Logger = NewSQLLogger(logger)
2019-09-20 06:19:11 +02:00
2019-09-26 06:35:31 +02:00
db, err := pgx.ConnectConfig(context.Background(), config)
if err != nil {
return nil, err
2019-09-20 06:19:11 +02:00
}
return db, nil
}
2019-09-26 06:35:31 +02:00
func initDBPool(c *config) (*pgxpool.Pool, error) {
config, _ := pgxpool.ParseConfig("")
config.ConnConfig.Host = c.DB.Host
config.ConnConfig.Port = c.DB.Port
config.ConnConfig.Database = c.DB.DBName
config.ConnConfig.User = c.DB.User
config.ConnConfig.Password = c.DB.Password
config.ConnConfig.RuntimeParams = map[string]string{
"application_name": c.AppName,
"search_path": c.DB.Schema,
}
2019-09-20 06:19:11 +02:00
2019-09-26 06:35:31 +02:00
switch c.LogLevel {
case "debug":
config.ConnConfig.LogLevel = pgx.LogLevelDebug
case "info":
config.ConnConfig.LogLevel = pgx.LogLevelInfo
case "warn":
config.ConnConfig.LogLevel = pgx.LogLevelWarn
case "error":
config.ConnConfig.LogLevel = pgx.LogLevelError
default:
config.ConnConfig.LogLevel = pgx.LogLevelNone
}
2019-09-20 06:19:11 +02:00
2019-11-25 08:22:33 +01:00
config.ConnConfig.Logger = NewSQLLogger(logger)
2019-09-20 06:19:11 +02:00
2019-09-26 06:35:31 +02:00
// if c.DB.MaxRetries != 0 {
// opt.MaxRetries = c.DB.MaxRetries
// }
2019-09-20 06:19:11 +02:00
2019-09-26 06:35:31 +02:00
if c.DB.PoolSize != 0 {
config.MaxConns = conf.DB.PoolSize
2019-09-20 06:19:11 +02:00
}
2019-09-26 06:35:31 +02:00
db, err := pgxpool.ConnectConfig(context.Background(), config)
2019-09-20 06:19:11 +02:00
if err != nil {
2019-09-26 06:35:31 +02:00
return nil, err
2019-09-20 06:19:11 +02:00
}
2019-09-26 06:35:31 +02:00
return db, nil
}
func initCompiler() {
var err error
2019-09-20 06:19:11 +02:00
qcompile, pcompile, err = initCompilers(conf)
if err != nil {
2019-11-25 08:22:33 +01:00
errlog.Fatal().Err(err).Msg("failed to initialize compilers")
2019-09-20 06:19:11 +02:00
}
if err := initResolvers(); err != nil {
2019-11-25 08:22:33 +01:00
errlog.Fatal().Err(err).Msg("failed to initialized resolvers")
2019-09-20 06:19:11 +02:00
}
}
func initConfOnce() {
var err error
if conf == nil {
if conf, err = initConf(); err != nil {
2019-11-25 08:22:33 +01:00
errlog.Fatal().Err(err).Msg("failed to read config")
}
}
}
func cmdVersion(cmd *cobra.Command, args []string) {
2019-11-28 21:27:20 +01:00
fmt.Printf("%s\n", BuildDetails())
}
func BuildDetails() string {
return fmt.Sprintf(`
2019-11-28 21:27:20 +01:00
Super Graph %v
For documentation, visit https://supergraph.dev
Commit SHA-1 : %v
Commit timestamp : %v
Branch : %v
Go version : %v
Licensed under the Apache Public License 2.0
Copyright 2020, Vikram Rangnekar.
`,
version,
lastCommitSHA,
lastCommitTime,
gitBranch,
runtime.Version())
}