2020-11-05 14:12:47 +08:00
|
|
|
package generator
|
|
|
|
|
|
|
|
import (
|
2024-03-09 20:49:16 +08:00
|
|
|
"errors"
|
2022-03-09 19:26:35 +08:00
|
|
|
"fmt"
|
|
|
|
"io/fs"
|
2022-01-11 20:34:25 +08:00
|
|
|
"os"
|
2020-11-05 14:12:47 +08:00
|
|
|
"path/filepath"
|
|
|
|
"strings"
|
|
|
|
|
2022-01-25 23:15:07 +08:00
|
|
|
"github.com/zeromicro/go-zero/tools/goctl/rpc/execx"
|
2024-10-19 23:37:44 +08:00
|
|
|
"github.com/zeromicro/go-zero/tools/goctl/util/pathx"
|
2020-11-05 14:12:47 +08:00
|
|
|
)
|
|
|
|
|
2021-02-26 16:11:47 +08:00
|
|
|
// GenPb generates the pb.go file, which is a layer of packaging for protoc to generate gprc,
|
|
|
|
// but the commands and flags in protoc are not completely joined in goctl. At present, proto_path(-I) is introduced
|
2022-03-19 22:50:22 +08:00
|
|
|
func (g *Generator) GenPb(ctx DirContext, c *ZRpcContext) error {
|
|
|
|
return g.genPbDirect(ctx, c)
|
2020-11-05 14:12:47 +08:00
|
|
|
}
|
2022-01-11 20:34:25 +08:00
|
|
|
|
2022-03-19 22:50:22 +08:00
|
|
|
func (g *Generator) genPbDirect(ctx DirContext, c *ZRpcContext) error {
|
2022-03-09 19:26:35 +08:00
|
|
|
g.log.Debug("[command]: %s", c.ProtocCmd)
|
2022-01-11 20:34:25 +08:00
|
|
|
pwd, err := os.Getwd()
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
_, err = execx.Run(c.ProtocCmd, pwd)
|
2022-03-09 19:26:35 +08:00
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
return g.setPbDir(ctx, c)
|
|
|
|
}
|
|
|
|
|
2022-03-19 22:50:22 +08:00
|
|
|
func (g *Generator) setPbDir(ctx DirContext, c *ZRpcContext) error {
|
2023-11-04 20:42:25 +08:00
|
|
|
pbDir, err := findPbFile(c.GoOutput, c.Src, false)
|
2022-03-09 19:26:35 +08:00
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
if len(pbDir) == 0 {
|
|
|
|
return fmt.Errorf("pg.go is not found under %q", c.GoOutput)
|
|
|
|
}
|
2023-11-04 20:42:25 +08:00
|
|
|
grpcDir, err := findPbFile(c.GrpcOutput, c.Src, true)
|
2022-03-09 19:26:35 +08:00
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
if len(grpcDir) == 0 {
|
|
|
|
return fmt.Errorf("_grpc.pb.go is not found in %q", c.GrpcOutput)
|
|
|
|
}
|
|
|
|
if pbDir != grpcDir {
|
|
|
|
return fmt.Errorf("the pb.go and _grpc.pb.go must under the same dir: "+
|
|
|
|
"\n pb.go: %s\n_grpc.pb.go: %s", pbDir, grpcDir)
|
|
|
|
}
|
|
|
|
if pbDir == c.Output {
|
|
|
|
return fmt.Errorf("the output of pb.go and _grpc.pb.go must not be the same "+
|
|
|
|
"with --zrpc_out:\npb output: %s\nzrpc out: %s", pbDir, c.Output)
|
|
|
|
}
|
|
|
|
ctx.SetPbDir(pbDir, grpcDir)
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
const (
|
|
|
|
pbSuffix = "pb.go"
|
|
|
|
grpcSuffix = "_grpc.pb.go"
|
|
|
|
)
|
|
|
|
|
2023-11-04 20:42:25 +08:00
|
|
|
func findPbFile(current string, src string, grpc bool) (string, error) {
|
|
|
|
protoName := strings.TrimSuffix(filepath.Base(src), filepath.Ext(src))
|
|
|
|
pbFile := protoName + "." + pbSuffix
|
|
|
|
grpcFile := protoName + grpcSuffix
|
|
|
|
|
2022-03-09 19:26:35 +08:00
|
|
|
fileSystem := os.DirFS(current)
|
|
|
|
var ret string
|
|
|
|
err := fs.WalkDir(fileSystem, ".", func(path string, d fs.DirEntry, err error) error {
|
|
|
|
if d.IsDir() {
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
if strings.HasSuffix(path, pbSuffix) {
|
|
|
|
if grpc {
|
2023-11-04 20:42:25 +08:00
|
|
|
if strings.HasSuffix(path, grpcFile) {
|
2022-03-09 19:26:35 +08:00
|
|
|
ret = path
|
|
|
|
return os.ErrExist
|
|
|
|
}
|
2023-11-04 20:42:25 +08:00
|
|
|
} else if strings.HasSuffix(path, pbFile) {
|
2022-03-09 19:26:35 +08:00
|
|
|
ret = path
|
|
|
|
return os.ErrExist
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
})
|
2024-03-09 20:49:16 +08:00
|
|
|
if errors.Is(err, os.ErrExist) {
|
2024-10-19 23:37:44 +08:00
|
|
|
return pathx.ReadLink(filepath.Dir(filepath.Join(current, ret)))
|
2022-03-09 19:26:35 +08:00
|
|
|
}
|
|
|
|
return "", err
|
2022-01-11 20:34:25 +08:00
|
|
|
}
|