feat: add new module cmd
This commit is contained in:
25
cmd/new.go
Normal file
25
cmd/new.go
Normal file
@@ -0,0 +1,25 @@
|
|||||||
|
package cmd
|
||||||
|
|
||||||
|
import (
|
||||||
|
"github.com/spf13/cobra"
|
||||||
|
)
|
||||||
|
|
||||||
|
func CommandInit(root *cobra.Command) {
|
||||||
|
cmd := &cobra.Command{
|
||||||
|
Use: "new [project|module]",
|
||||||
|
Short: "new project/module",
|
||||||
|
}
|
||||||
|
|
||||||
|
cmd.PersistentFlags().BoolP("force", "f", false, "Force init project if exists")
|
||||||
|
|
||||||
|
cmds := []func(*cobra.Command){
|
||||||
|
CommandNewProject,
|
||||||
|
CommandNewModule,
|
||||||
|
}
|
||||||
|
|
||||||
|
for _, c := range cmds {
|
||||||
|
c(cmd)
|
||||||
|
}
|
||||||
|
|
||||||
|
root.AddCommand(cmd)
|
||||||
|
}
|
||||||
94
cmd/new_module.go
Normal file
94
cmd/new_module.go
Normal file
@@ -0,0 +1,94 @@
|
|||||||
|
package cmd
|
||||||
|
|
||||||
|
import (
|
||||||
|
"fmt"
|
||||||
|
"io/fs"
|
||||||
|
"os"
|
||||||
|
"path/filepath"
|
||||||
|
"strings"
|
||||||
|
"text/template"
|
||||||
|
|
||||||
|
"git.ipao.vip/rogeecn/atomctl/templates"
|
||||||
|
"github.com/samber/lo"
|
||||||
|
log "github.com/sirupsen/logrus"
|
||||||
|
"github.com/spf13/cobra"
|
||||||
|
)
|
||||||
|
|
||||||
|
func CommandNewModule(root *cobra.Command) {
|
||||||
|
cmd := &cobra.Command{
|
||||||
|
Use: "module",
|
||||||
|
Short: "new module",
|
||||||
|
Args: cobra.ExactArgs(1),
|
||||||
|
RunE: commandNewModuleE,
|
||||||
|
}
|
||||||
|
|
||||||
|
root.AddCommand(cmd)
|
||||||
|
}
|
||||||
|
|
||||||
|
func commandNewModuleE(cmd *cobra.Command, args []string) error {
|
||||||
|
module := lo.Filter(strings.Split(args[0], "."), func(s string, _ int) bool {
|
||||||
|
return s != ""
|
||||||
|
})
|
||||||
|
|
||||||
|
modulePath := module[0]
|
||||||
|
if len(module) > 1 {
|
||||||
|
modulePath = strings.Join(module, "/modules/")
|
||||||
|
}
|
||||||
|
moduleName := module[len(module)-1]
|
||||||
|
modulePath = filepath.Join("modules", modulePath)
|
||||||
|
log.Infof("new module: %s", modulePath)
|
||||||
|
|
||||||
|
force, _ := cmd.Flags().GetBool("force")
|
||||||
|
if _, err := os.Stat(modulePath); err == nil {
|
||||||
|
if !force {
|
||||||
|
return fmt.Errorf("module directory %s already exists", modulePath)
|
||||||
|
}
|
||||||
|
log.Warnf("强制删除已存在的目录: %s", modulePath)
|
||||||
|
if err := os.RemoveAll(modulePath); err != nil {
|
||||||
|
return fmt.Errorf("failed to remove existing directory: %v", err)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
err := os.MkdirAll(modulePath, os.ModePerm)
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
|
||||||
|
err = fs.WalkDir(templates.Module, "module", func(path string, d fs.DirEntry, err error) error {
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
if d.IsDir() {
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
relPath, err := filepath.Rel("module", path)
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
|
||||||
|
destPath := filepath.Join(modulePath, strings.TrimSuffix(relPath, ".tpl"))
|
||||||
|
destDir := filepath.Dir(destPath)
|
||||||
|
err = os.MkdirAll(destDir, os.ModePerm)
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
|
||||||
|
tmpl, err := template.ParseFS(templates.Module, path)
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
|
||||||
|
destFile, err := os.Create(destPath)
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
defer destFile.Close()
|
||||||
|
|
||||||
|
return tmpl.Execute(destFile, map[string]string{
|
||||||
|
"ModuleName": moduleName,
|
||||||
|
})
|
||||||
|
})
|
||||||
|
|
||||||
|
return err
|
||||||
|
}
|
||||||
@@ -22,19 +22,18 @@ func isValidGoPackageName(name string) bool {
|
|||||||
return goPackageRegexp.MatchString(name)
|
return goPackageRegexp.MatchString(name)
|
||||||
}
|
}
|
||||||
|
|
||||||
func CommandInit(root *cobra.Command) {
|
func CommandNewProject(root *cobra.Command) {
|
||||||
cmd := &cobra.Command{
|
cmd := &cobra.Command{
|
||||||
Use: "init [project]",
|
Use: "project",
|
||||||
Short: "init new project",
|
Short: "new project",
|
||||||
Args: cobra.ExactArgs(1),
|
Args: cobra.ExactArgs(1),
|
||||||
RunE: commandInitE,
|
RunE: commandNewProjectE,
|
||||||
}
|
}
|
||||||
|
|
||||||
cmd.Flags().BoolP("force", "f", false, "Force init project if exists")
|
|
||||||
root.AddCommand(cmd)
|
root.AddCommand(cmd)
|
||||||
}
|
}
|
||||||
|
|
||||||
func commandInitE(cmd *cobra.Command, args []string) error {
|
func commandNewProjectE(cmd *cobra.Command, args []string) error {
|
||||||
moduleName := args[0]
|
moduleName := args[0]
|
||||||
if !isValidGoPackageName(moduleName) {
|
if !isValidGoPackageName(moduleName) {
|
||||||
return fmt.Errorf("invalid module name: %s, should be a valid go package name", moduleName)
|
return fmt.Errorf("invalid module name: %s, should be a valid go package name", moduleName)
|
||||||
@@ -42,8 +41,6 @@ func commandInitE(cmd *cobra.Command, args []string) error {
|
|||||||
|
|
||||||
log.Info("创建项目: ", moduleName)
|
log.Info("创建项目: ", moduleName)
|
||||||
|
|
||||||
force, _ := cmd.Flags().GetBool("force")
|
|
||||||
|
|
||||||
var projectInfo struct {
|
var projectInfo struct {
|
||||||
ModuleName string
|
ModuleName string
|
||||||
ProjectName string
|
ProjectName string
|
||||||
@@ -54,6 +51,7 @@ func commandInitE(cmd *cobra.Command, args []string) error {
|
|||||||
projectInfo.ProjectName = moduleSplitInfo[len(moduleSplitInfo)-1]
|
projectInfo.ProjectName = moduleSplitInfo[len(moduleSplitInfo)-1]
|
||||||
|
|
||||||
// 检查目录是否存在
|
// 检查目录是否存在
|
||||||
|
force, _ := cmd.Flags().GetBool("force")
|
||||||
if _, err := os.Stat(projectInfo.ProjectName); err == nil {
|
if _, err := os.Stat(projectInfo.ProjectName); err == nil {
|
||||||
if !force {
|
if !force {
|
||||||
return fmt.Errorf("project directory %s already exists", projectInfo.ProjectName)
|
return fmt.Errorf("project directory %s already exists", projectInfo.ProjectName)
|
||||||
16
templates/module/controller.go.tpl
Normal file
16
templates/module/controller.go.tpl
Normal file
@@ -0,0 +1,16 @@
|
|||||||
|
package {{.ModuleName}}
|
||||||
|
|
||||||
|
import (
|
||||||
|
log "github.com/sirupsen/logrus"
|
||||||
|
)
|
||||||
|
|
||||||
|
// @provider
|
||||||
|
type Controller struct {
|
||||||
|
svc *Service
|
||||||
|
log *log.Entry `inject:"false"`
|
||||||
|
}
|
||||||
|
|
||||||
|
func (c *Controller) Prepare() error {
|
||||||
|
c.log = log.WithField("module", "{{.ModuleName}}.Controller")
|
||||||
|
return nil
|
||||||
|
}
|
||||||
1
templates/module/dto.go.tpl
Normal file
1
templates/module/dto.go.tpl
Normal file
@@ -0,0 +1 @@
|
|||||||
|
package {{.ModuleName}}
|
||||||
9
templates/module/provider.gen.go.tpl
Executable file
9
templates/module/provider.gen.go.tpl
Executable file
@@ -0,0 +1,9 @@
|
|||||||
|
package {{.ModuleName}}
|
||||||
|
|
||||||
|
import (
|
||||||
|
"git.ipao.vip/rogeecn/atom/utils/opt"
|
||||||
|
)
|
||||||
|
|
||||||
|
func Provide(opts ...opt.Option) error {
|
||||||
|
return nil
|
||||||
|
}
|
||||||
29
templates/module/router.go.tpl
Executable file
29
templates/module/router.go.tpl
Executable file
@@ -0,0 +1,29 @@
|
|||||||
|
package {{.ModuleName}}
|
||||||
|
|
||||||
|
import (
|
||||||
|
_ "git.ipao.vip/rogeecn/atom"
|
||||||
|
_ "git.ipao.vip/rogeecn/atom/contracts"
|
||||||
|
"github.com/gofiber/fiber/v3"
|
||||||
|
log "github.com/sirupsen/logrus"
|
||||||
|
)
|
||||||
|
|
||||||
|
// @provider:except contracts.HttpRoute atom.GroupRoutes
|
||||||
|
type Router struct {
|
||||||
|
log *log.Entry `inject:"false"`
|
||||||
|
|
||||||
|
controller *Controller
|
||||||
|
}
|
||||||
|
|
||||||
|
func (r *Router) Name() string {
|
||||||
|
return "{{.ModuleName}}"
|
||||||
|
}
|
||||||
|
|
||||||
|
func (r *Router) Prepare() error {
|
||||||
|
r.log = log.WithField("http.group", r.Name())
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (r *Router) Register(router fiber.Router) {
|
||||||
|
group := router.Group(r.Name())
|
||||||
|
_ = group
|
||||||
|
}
|
||||||
20
templates/module/service.go.tpl
Normal file
20
templates/module/service.go.tpl
Normal file
@@ -0,0 +1,20 @@
|
|||||||
|
package {{.ModuleName}}
|
||||||
|
|
||||||
|
import (
|
||||||
|
"database/sql"
|
||||||
|
|
||||||
|
. "github.com/go-jet/jet/v2/postgres"
|
||||||
|
log "github.com/sirupsen/logrus"
|
||||||
|
)
|
||||||
|
|
||||||
|
// @provider:except
|
||||||
|
type Service struct {
|
||||||
|
db *sql.DB
|
||||||
|
log *log.Entry `inject:"false"`
|
||||||
|
}
|
||||||
|
|
||||||
|
func (svc *Service) Prepare() error {
|
||||||
|
svc.log = log.WithField("module", "{{.ModuleName}}.service")
|
||||||
|
_ = Int(1)
|
||||||
|
return nil
|
||||||
|
}
|
||||||
37
templates/module/service_test.go.tpl
Normal file
37
templates/module/service_test.go.tpl
Normal file
@@ -0,0 +1,37 @@
|
|||||||
|
package {{.ModuleName}}
|
||||||
|
|
||||||
|
import (
|
||||||
|
"testing"
|
||||||
|
|
||||||
|
"backend/pkg/service/testx"
|
||||||
|
|
||||||
|
. "github.com/smartystreets/goconvey/convey"
|
||||||
|
"github.com/stretchr/testify/suite"
|
||||||
|
"go.uber.org/dig"
|
||||||
|
)
|
||||||
|
|
||||||
|
type ServiceInjectParams struct {
|
||||||
|
dig.In
|
||||||
|
Svc *Service
|
||||||
|
}
|
||||||
|
|
||||||
|
type ServiceTestSuite struct {
|
||||||
|
suite.Suite
|
||||||
|
ServiceInjectParams
|
||||||
|
}
|
||||||
|
|
||||||
|
func Test_DiscoverMedias(t *testing.T) {
|
||||||
|
providers := testx.Default().With(
|
||||||
|
Provide,
|
||||||
|
)
|
||||||
|
|
||||||
|
testx.Serve(providers, t, func(params ServiceInjectParams) {
|
||||||
|
suite.Run(t, &ServiceTestSuite{ServiceInjectParams: params})
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
func (s *ServiceTestSuite) Test_Service() {
|
||||||
|
Convey("Test Service", s.T(), func() {
|
||||||
|
So(s.Svc, ShouldNotBeNil)
|
||||||
|
})
|
||||||
|
}
|
||||||
@@ -4,3 +4,6 @@ import "embed"
|
|||||||
|
|
||||||
//go:embed project
|
//go:embed project
|
||||||
var Project embed.FS
|
var Project embed.FS
|
||||||
|
|
||||||
|
//go:embed module
|
||||||
|
var Module embed.FS
|
||||||
|
|||||||
Reference in New Issue
Block a user