This repository has been archived on 2025-03-16. You can view files and clone it. You cannot open issues or pull requests or push a commit.
Files
crudgen/internal/writer.go
2025-02-16 14:35:50 +07:00

383 lines
8.7 KiB
Go

package internal
import (
"bytes"
"errors"
"fmt"
"go/parser"
"go/token"
"html/template"
"log"
"os"
"path/filepath"
"strings"
"github.com/dave/dst"
"github.com/dave/dst/decorator"
)
func ImplementServiceStruct(modelName string, file *dst.File, reimplement bool) {
serviceName := modelName + "Service"
isServiceStructDefined := false
var insertPos int
var decls []dst.Decl
for i, decl := range file.Decls {
genDecl, ok := decl.(*dst.GenDecl)
if !ok {
continue
}
decls = append(decls, decl)
if genDecl.Tok == token.IMPORT {
insertPos = i + 1
continue
}
if genDecl.Tok != token.TYPE {
continue
}
for _, spec := range genDecl.Specs {
typeSpec, ok := spec.(*dst.TypeSpec)
if !ok {
continue
}
if _, ok := typeSpec.Type.(*dst.StructType); ok {
if typeSpec.Name != nil && typeSpec.Name.Name == serviceName {
isServiceStructDefined = true
if reimplement {
decls = decls[:len(decls)-1]
}
}
}
}
}
if isServiceStructDefined && !reimplement {
return
}
serviceStruct := &dst.GenDecl{
Tok: token.TYPE,
Specs: []dst.Spec{
&dst.TypeSpec{
Name: dst.NewIdent(serviceName),
Type: &dst.StructType{
Fields: &dst.FieldList{},
},
},
},
}
file.Decls = append(decls[:insertPos], append([]dst.Decl{serviceStruct}, decls[insertPos:]...)...)
if err := ReloadAst(file); err != nil {
log.Fatalf(err.Error())
}
}
func ImplementModelAlias(modelName string, file *dst.File) {
isAliasDefined := false
aliasTypeStandard := fmt.Sprintf("models.%s", CapitalizeFirst(modelName))
var insertPos int
var decls []dst.Decl
for i, decl := range file.Decls {
genDecl, ok := decl.(*dst.GenDecl)
if !ok {
continue
}
decls = append(decls, decl)
if genDecl.Tok == token.IMPORT {
insertPos = i + 1
continue
}
if genDecl.Tok != token.TYPE {
continue
}
if len(genDecl.Specs) != 1 {
continue
}
if typeSpec, ok := genDecl.Specs[0].(*dst.TypeSpec); ok {
if typeSpec.Name != nil && typeSpec.Name.Name == modelName {
if linkedType, ok := typeSpec.Type.(*dst.SelectorExpr); ok {
pkg, ok := linkedType.X.(*dst.Ident)
if !ok {
log.Printf("Defined alias `%s` with unknown type", typeSpec.Name)
decls = decls[:len(decls)-1]
continue
}
if linkedType.Sel == nil {
log.Printf("Defined alias `%s` with unknown type", typeSpec.Name)
decls = decls[:len(decls)-1]
continue
}
if pkg.Name != "models" {
log.Printf(
"Defined alias `%s` with wrong type: package `%s`, should be `models`",
typeSpec.Name,
pkg,
)
decls = decls[:len(decls)-1]
continue
}
if linkedType.Sel.Name != modelName {
log.Printf(
"Defined alias `%s` with wrong type: models.%s, should be `%s`",
typeSpec.Name,
linkedType.Sel.Name,
aliasTypeStandard,
)
decls = decls[:len(decls)-1]
continue
}
isAliasDefined = true
} else {
log.Printf("Defined alias %s with unknown type", typeSpec.Name)
decls = decls[:len(decls)-1]
}
}
}
}
typeAlias := dst.GenDecl{
Tok: token.TYPE,
Specs: []dst.Spec{
&dst.TypeSpec{
Name: &dst.Ident{
Name: modelName,
},
Assign: true, // quick and dirty
Type: &dst.Ident{
Name: aliasTypeStandard,
},
},
},
}
if !isAliasDefined {
file.Decls = append(decls[:insertPos], append([]dst.Decl{&typeAlias}, decls[insertPos:]...)...)
}
if err := ReloadAst(file); err != nil {
log.Fatalf(err.Error())
}
}
func importExists(file *dst.File, importPath string) bool {
for _, imp := range file.Imports {
if imp.Path.Value == `"`+importPath+`"` {
return true
}
}
return false
}
func MaintainImports(file *dst.File) error {
for _, importPath := range ServiceImports {
if !importExists(file, importPath) {
file.Imports = append(file.Imports, &dst.ImportSpec{
Path: &dst.BasicLit{
Kind: token.STRING,
Value: `"` + importPath + `"`,
},
})
}
}
return nil
}
func GenerateCrudMethodCode(methodName string, context CrudTemplatesContext) string {
templateCode, templateExists := RawTemplates[methodName]
if !templateExists {
panic(fmt.Sprintf("Template doesn't exist: %s \n", methodName))
}
var buffer bytes.Buffer
templateCode = "package services\n\n" + templateCode
tmpl, err := template.New("CrudMethodTemplate").Parse(templateCode)
if err != nil {
panic(err)
}
if err := tmpl.Execute(&buffer, context); err != nil {
panic(err)
}
return buffer.String()
}
func MethodCodeToDeclaration(methodCode string) (*dst.FuncDecl, error) {
fset := token.NewFileSet()
file, err := decorator.ParseFile(fset, "src.go", methodCode, parser.AllErrors)
if err != nil {
return nil, err
}
var methodDecl *dst.FuncDecl
dst.Inspect(file, func(node dst.Node) bool {
funcDecl, ok := node.(*dst.FuncDecl)
if !ok {
return true
}
methodDecl = funcDecl
return false
})
return methodDecl, nil
}
func ImplementCrudMethods(modelName string, serviceName string, file *dst.File, reimplement bool) error {
templateContext := CrudTemplatesContext{
ServiceName: serviceName,
EntityType: modelName,
EntityPlural: ToPlural(modelName),
}
for _, methodName := range []string{CreateMethod, GetAllMethod, GetByIdMethod, UpdateMethod, DeleteMethod, CountMethod} {
methodCode := GenerateCrudMethodCode(methodName, templateContext)
methodDecl, err := MethodCodeToDeclaration(methodCode)
fmt.Printf("%s\n", methodCode)
if err != nil {
fmt.Println(methodDecl)
panic(err)
}
err = ImplementMethod(file, methodDecl, reimplement)
if err != nil {
return err
}
if err := ReloadAst(file); err != nil {
return err
}
}
return nil
}
func ReloadAst(_ *dst.File) error {
return nil
}
func ImplementMethod(file *dst.File, methodDecl *dst.FuncDecl, reimplement bool) error {
var decls []dst.Decl
methodImplemented := false
methodStructure := methodDecl.Recv.List[0].Names[0].Name
methodName := methodDecl.Name.Name
log.Printf("Standard method structure: %s\n", methodStructure)
log.Printf("Standard method name: %s\n", methodName)
for _, decl := range file.Decls {
decls = append(decls, decl)
funcDecl, ok := decl.(*dst.FuncDecl)
if !ok {
continue
}
if len(funcDecl.Recv.List) > 0 && len(funcDecl.Recv.List[0].Names) > 0 {
fmt.Printf("Method structure: %s\n", funcDecl.Recv.List[0].Names[0].Name)
fmt.Printf("Method name: %s\n", funcDecl.Name.Name)
if funcDecl.Recv.List[0].Names[0].Name == methodStructure {
if funcDecl.Name != nil && funcDecl.Name.Name == methodName {
if methodImplemented {
err := fmt.Sprintf("`%s` method redeclarated for struct `%s`", methodName, methodStructure)
log.Println(err)
return errors.New(err)
} else {
methodImplemented = true
}
if reimplement {
decls = decls[:len(decls)-1]
}
}
}
}
}
if reimplement || !methodImplemented {
file.Decls = append(
decls,
&dst.FuncDecl{
Recv: methodDecl.Recv,
Name: methodDecl.Name,
Type: methodDecl.Type,
Body: &dst.BlockStmt{
List: methodDecl.Body.List,
},
},
)
}
if err := ReloadAst(file); err != nil {
return err
}
return nil
}
func CreateServiceFileIfNotExists(filePath string) error {
if _, err := os.Stat(filePath); err != nil {
// file wasn't created
f, err := os.Create(filePath)
if err != nil {
log.Fatalf("Failed to create file: %s", filePath)
return err
}
_, err = f.Write([]byte("package services\n"))
if err != nil {
log.Fatalf("Failed to write file: %s", filePath)
return err
}
defer f.Close()
}
return nil
}
func ImplementService(mainPkgPath string, modelName string, reimplement bool) error {
serviceRelativePath := fmt.Sprintf("services/%s.go", strings.ToLower(modelName))
filePath := filepath.Join(mainPkgPath, serviceRelativePath)
serviceName := modelName + "Service"
err := CreateServiceFileIfNotExists(filePath)
if err != nil {
return err
}
fset := token.NewFileSet()
serviceFile, err := decorator.ParseFile(fset, filePath, nil, parser.ParseComments)
if err != nil {
log.Fatalf("Parsing error: %s: %v", mainPkgPath, err)
return err
}
err = MaintainImports(serviceFile)
if err != nil {
return err
}
ImplementModelAlias(modelName, serviceFile)
ImplementServiceStruct(modelName, serviceFile, reimplement)
err = ImplementCrudMethods(modelName, serviceName, serviceFile, reimplement)
if err != nil {
return err
}
file, err := os.Create(filePath)
if err != nil {
return errors.New(fmt.Sprintf("Error occured to open `%s` service file: %v", modelName, err))
}
err = decorator.Fprint(file, serviceFile)
if err != nil {
return errors.New(
fmt.Sprintf("Error occurred to writing changes in `%s` service file: %v", modelName, err),
)
}
defer file.Close()
return nil
}