2020-08-19 10:41:19 +08:00
|
|
|
package command
|
|
|
|
|
|
|
|
import (
|
2020-10-22 18:33:09 +08:00
|
|
|
"errors"
|
2020-08-20 10:29:18 +08:00
|
|
|
"path/filepath"
|
|
|
|
"strings"
|
|
|
|
|
2020-10-22 18:33:09 +08:00
|
|
|
"github.com/go-sql-driver/mysql"
|
2020-08-20 10:29:18 +08:00
|
|
|
"github.com/tal-tech/go-zero/core/logx"
|
2021-07-23 11:45:15 +08:00
|
|
|
"github.com/tal-tech/go-zero/core/stores/postgres"
|
2020-08-20 10:29:18 +08:00
|
|
|
"github.com/tal-tech/go-zero/core/stores/sqlx"
|
2020-11-24 15:11:18 +08:00
|
|
|
"github.com/tal-tech/go-zero/tools/goctl/config"
|
2020-08-19 10:41:19 +08:00
|
|
|
"github.com/tal-tech/go-zero/tools/goctl/model/sql/gen"
|
2020-08-20 10:29:18 +08:00
|
|
|
"github.com/tal-tech/go-zero/tools/goctl/model/sql/model"
|
2020-10-22 18:33:09 +08:00
|
|
|
"github.com/tal-tech/go-zero/tools/goctl/model/sql/util"
|
2021-07-28 16:32:15 +08:00
|
|
|
file "github.com/tal-tech/go-zero/tools/goctl/util"
|
2020-08-19 10:41:19 +08:00
|
|
|
"github.com/tal-tech/go-zero/tools/goctl/util/console"
|
2020-08-20 10:46:39 +08:00
|
|
|
"github.com/urfave/cli"
|
2020-08-19 10:41:19 +08:00
|
|
|
)
|
|
|
|
|
2020-08-20 10:29:18 +08:00
|
|
|
const (
|
2021-07-23 11:45:15 +08:00
|
|
|
flagSrc = "src"
|
|
|
|
flagDir = "dir"
|
|
|
|
flagCache = "cache"
|
|
|
|
flagIdea = "idea"
|
|
|
|
flagURL = "url"
|
|
|
|
flagTable = "table"
|
|
|
|
flagStyle = "style"
|
2021-07-22 11:29:09 +08:00
|
|
|
flagDatabase = "database"
|
2021-07-23 11:45:15 +08:00
|
|
|
flagSchema = "schema"
|
2020-08-20 10:29:18 +08:00
|
|
|
)
|
|
|
|
|
2021-01-14 13:24:24 +08:00
|
|
|
var errNotMatched = errors.New("sql not matched")
|
|
|
|
|
2021-02-26 16:11:47 +08:00
|
|
|
// MysqlDDL generates model code from ddl
|
2020-08-20 10:29:18 +08:00
|
|
|
func MysqlDDL(ctx *cli.Context) error {
|
|
|
|
src := ctx.String(flagSrc)
|
|
|
|
dir := ctx.String(flagDir)
|
|
|
|
cache := ctx.Bool(flagCache)
|
|
|
|
idea := ctx.Bool(flagIdea)
|
2020-11-24 15:11:18 +08:00
|
|
|
style := ctx.String(flagStyle)
|
2021-07-22 11:29:09 +08:00
|
|
|
database := ctx.String(flagDatabase)
|
2021-07-28 16:32:15 +08:00
|
|
|
home := ctx.String("home")
|
|
|
|
|
|
|
|
if len(home) > 0 {
|
|
|
|
file.RegisterGoctlHome(home)
|
|
|
|
}
|
2020-11-24 15:11:18 +08:00
|
|
|
cfg, err := config.NewConfig(style)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
2021-01-14 13:24:24 +08:00
|
|
|
|
2021-07-22 11:29:09 +08:00
|
|
|
return fromDDl(src, dir, cfg, cache, idea, database)
|
2020-11-18 15:32:53 +08:00
|
|
|
}
|
|
|
|
|
2021-07-23 11:45:15 +08:00
|
|
|
// MySqlDataSource generates model code from datasource
|
|
|
|
func MySqlDataSource(ctx *cli.Context) error {
|
2021-02-26 16:11:47 +08:00
|
|
|
url := strings.TrimSpace(ctx.String(flagURL))
|
2020-11-18 15:32:53 +08:00
|
|
|
dir := strings.TrimSpace(ctx.String(flagDir))
|
|
|
|
cache := ctx.Bool(flagCache)
|
|
|
|
idea := ctx.Bool(flagIdea)
|
2020-11-24 15:11:18 +08:00
|
|
|
style := ctx.String(flagStyle)
|
2021-07-28 16:32:15 +08:00
|
|
|
home := ctx.String("home")
|
|
|
|
|
|
|
|
if len(home) > 0 {
|
|
|
|
file.RegisterGoctlHome(home)
|
|
|
|
}
|
|
|
|
|
2020-11-18 15:32:53 +08:00
|
|
|
pattern := strings.TrimSpace(ctx.String(flagTable))
|
2020-11-24 15:11:18 +08:00
|
|
|
cfg, err := config.NewConfig(style)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
2021-07-23 11:45:15 +08:00
|
|
|
return fromMysqlDataSource(url, pattern, dir, cfg, cache, idea)
|
|
|
|
}
|
|
|
|
|
|
|
|
// PostgreSqlDataSource generates model code from datasource
|
|
|
|
func PostgreSqlDataSource(ctx *cli.Context) error {
|
|
|
|
url := strings.TrimSpace(ctx.String(flagURL))
|
|
|
|
dir := strings.TrimSpace(ctx.String(flagDir))
|
|
|
|
cache := ctx.Bool(flagCache)
|
|
|
|
idea := ctx.Bool(flagIdea)
|
|
|
|
style := ctx.String(flagStyle)
|
|
|
|
schema := ctx.String(flagSchema)
|
2021-07-28 16:32:15 +08:00
|
|
|
home := ctx.String("home")
|
|
|
|
|
|
|
|
if len(home) > 0 {
|
|
|
|
file.RegisterGoctlHome(home)
|
|
|
|
}
|
|
|
|
|
2021-07-23 11:45:15 +08:00
|
|
|
if len(schema) == 0 {
|
|
|
|
schema = "public"
|
|
|
|
}
|
|
|
|
|
|
|
|
pattern := strings.TrimSpace(ctx.String(flagTable))
|
|
|
|
cfg, err := config.NewConfig(style)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
return fromPostgreSqlDataSource(url, pattern, dir, schema, cfg, cache, idea)
|
2020-11-18 15:32:53 +08:00
|
|
|
}
|
|
|
|
|
2021-07-22 11:29:09 +08:00
|
|
|
func fromDDl(src, dir string, cfg *config.Config, cache, idea bool, database string) error {
|
2020-08-20 10:29:18 +08:00
|
|
|
log := console.NewConsole(idea)
|
2020-10-22 18:33:09 +08:00
|
|
|
src = strings.TrimSpace(src)
|
|
|
|
if len(src) == 0 {
|
|
|
|
return errors.New("expected path or path globbing patterns, but nothing found")
|
2020-08-20 10:29:18 +08:00
|
|
|
}
|
2020-10-22 18:33:09 +08:00
|
|
|
|
|
|
|
files, err := util.MatchFiles(src)
|
2020-08-20 10:29:18 +08:00
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
2020-10-22 18:33:09 +08:00
|
|
|
|
2020-11-18 15:32:53 +08:00
|
|
|
if len(files) == 0 {
|
|
|
|
return errNotMatched
|
|
|
|
}
|
|
|
|
|
2021-07-16 22:55:39 +08:00
|
|
|
generator, err := gen.NewDefaultGenerator(dir, cfg, gen.WithConsoleOption(log))
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
2020-10-22 18:33:09 +08:00
|
|
|
for _, file := range files {
|
2021-07-22 11:29:09 +08:00
|
|
|
err = generator.StartFromDDL(file, cache, database)
|
2020-10-22 18:33:09 +08:00
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
2020-08-20 10:29:18 +08:00
|
|
|
}
|
2020-11-18 15:32:53 +08:00
|
|
|
|
2021-07-16 22:55:39 +08:00
|
|
|
return nil
|
2020-08-20 10:29:18 +08:00
|
|
|
}
|
|
|
|
|
2021-07-23 11:45:15 +08:00
|
|
|
func fromMysqlDataSource(url, pattern, dir string, cfg *config.Config, cache, idea bool) error {
|
2020-08-20 10:29:18 +08:00
|
|
|
log := console.NewConsole(idea)
|
|
|
|
if len(url) == 0 {
|
2020-10-22 18:33:09 +08:00
|
|
|
log.Error("%v", "expected data source of mysql, but nothing found")
|
2020-08-20 10:29:18 +08:00
|
|
|
return nil
|
|
|
|
}
|
2020-10-22 18:33:09 +08:00
|
|
|
|
|
|
|
if len(pattern) == 0 {
|
|
|
|
log.Error("%v", "expected table or table globbing patterns, but nothing found")
|
2020-08-20 10:29:18 +08:00
|
|
|
return nil
|
|
|
|
}
|
2020-10-22 18:33:09 +08:00
|
|
|
|
2020-11-24 15:11:18 +08:00
|
|
|
dsn, err := mysql.ParseDSN(url)
|
2020-10-22 18:33:09 +08:00
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
2020-08-20 10:29:18 +08:00
|
|
|
logx.Disable()
|
2020-11-24 15:11:18 +08:00
|
|
|
databaseSource := strings.TrimSuffix(url, "/"+dsn.DBName) + "/information_schema"
|
2020-10-22 18:33:09 +08:00
|
|
|
db := sqlx.NewMysql(databaseSource)
|
|
|
|
im := model.NewInformationSchemaModel(db)
|
|
|
|
|
2020-11-24 15:11:18 +08:00
|
|
|
tables, err := im.GetAllTables(dsn.DBName)
|
2020-10-22 18:33:09 +08:00
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
2021-03-01 17:29:07 +08:00
|
|
|
matchTables := make(map[string]*model.Table)
|
2020-10-22 18:33:09 +08:00
|
|
|
for _, item := range tables {
|
|
|
|
match, err := filepath.Match(pattern, item)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
if !match {
|
2020-08-20 10:29:18 +08:00
|
|
|
continue
|
|
|
|
}
|
2021-03-01 17:29:07 +08:00
|
|
|
|
|
|
|
columnData, err := im.FindColumns(dsn.DBName, item)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
table, err := columnData.Convert()
|
2020-11-18 15:32:53 +08:00
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
2021-03-01 17:29:07 +08:00
|
|
|
|
|
|
|
matchTables[item] = table
|
2020-10-22 18:33:09 +08:00
|
|
|
}
|
2020-11-18 15:32:53 +08:00
|
|
|
|
2020-10-22 18:33:09 +08:00
|
|
|
if len(matchTables) == 0 {
|
|
|
|
return errors.New("no tables matched")
|
2020-08-20 10:29:18 +08:00
|
|
|
}
|
2020-10-22 18:33:09 +08:00
|
|
|
|
2020-11-24 15:11:18 +08:00
|
|
|
generator, err := gen.NewDefaultGenerator(dir, cfg, gen.WithConsoleOption(log))
|
2020-08-19 10:41:19 +08:00
|
|
|
if err != nil {
|
2020-11-18 15:32:53 +08:00
|
|
|
return err
|
2020-08-19 10:41:19 +08:00
|
|
|
}
|
2020-10-22 18:33:09 +08:00
|
|
|
|
2021-03-01 17:29:07 +08:00
|
|
|
return generator.StartFromInformationSchema(matchTables, cache)
|
2020-08-19 10:41:19 +08:00
|
|
|
}
|
2021-07-23 11:45:15 +08:00
|
|
|
|
|
|
|
func fromPostgreSqlDataSource(url, pattern, dir, schema string, cfg *config.Config, cache, idea bool) error {
|
|
|
|
log := console.NewConsole(idea)
|
|
|
|
if len(url) == 0 {
|
|
|
|
log.Error("%v", "expected data source of mysql, but nothing found")
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
if len(pattern) == 0 {
|
|
|
|
log.Error("%v", "expected table or table globbing patterns, but nothing found")
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
db := postgres.New(url)
|
|
|
|
im := model.NewPostgreSqlModel(db)
|
|
|
|
|
|
|
|
tables, err := im.GetAllTables(schema)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
matchTables := make(map[string]*model.Table)
|
|
|
|
for _, item := range tables {
|
|
|
|
match, err := filepath.Match(pattern, item)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
if !match {
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
|
|
|
|
columnData, err := im.FindColumns(schema, item)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
table, err := columnData.Convert()
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
matchTables[item] = table
|
|
|
|
}
|
|
|
|
|
|
|
|
if len(matchTables) == 0 {
|
|
|
|
return errors.New("no tables matched")
|
|
|
|
}
|
|
|
|
|
|
|
|
generator, err := gen.NewDefaultGenerator(dir, cfg, gen.WithConsoleOption(log), gen.WithPostgreSql())
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
return generator.StartFromInformationSchema(matchTables, cache)
|
|
|
|
}
|