871 lines
25 KiB
Go
871 lines
25 KiB
Go
// Copyright 2010 Google Inc.
|
|
//
|
|
// Licensed under the Apache License, Version 2.0 (the "License");
|
|
// you may not use this file except in compliance with the License.
|
|
// You may obtain a copy of the License at
|
|
//
|
|
// http://www.apache.org/licenses/LICENSE-2.0
|
|
//
|
|
// Unless required by applicable law or agreed to in writing, software
|
|
// distributed under the License is distributed on an "AS IS" BASIS,
|
|
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
// See the License for the specific language governing permissions and
|
|
// limitations under the License.
|
|
|
|
// MockGen generates mock implementations of Go interfaces.
|
|
package main
|
|
|
|
// TODO: This does not support recursive embedded interfaces.
|
|
// TODO: This does not support embedding package-local interfaces in a separate file.
|
|
|
|
import (
|
|
"bytes"
|
|
"encoding/json"
|
|
"errors"
|
|
"flag"
|
|
"fmt"
|
|
"go/token"
|
|
"io"
|
|
"log"
|
|
"os"
|
|
"os/exec"
|
|
"path"
|
|
"path/filepath"
|
|
"sort"
|
|
"strconv"
|
|
"strings"
|
|
"unicode"
|
|
|
|
"golang.org/x/mod/modfile"
|
|
toolsimports "golang.org/x/tools/imports"
|
|
|
|
"go.uber.org/mock/mockgen/model"
|
|
)
|
|
|
|
const (
|
|
gomockImportPath = "go.uber.org/mock/gomock"
|
|
)
|
|
|
|
var (
|
|
version = ""
|
|
commit = "none"
|
|
date = "unknown"
|
|
)
|
|
|
|
var (
|
|
source = flag.String("source", "", "(source mode) Input Go source file; enables source mode.")
|
|
destination = flag.String("destination", "", "Output file; defaults to stdout.")
|
|
mockNames = flag.String("mock_names", "", "Comma-separated interfaceName=mockName pairs of explicit mock names to use. Mock names default to 'Mock'+ interfaceName suffix.")
|
|
packageOut = flag.String("package", "", "Package of the generated code; defaults to the package of the input with a 'mock_' prefix.")
|
|
selfPackage = flag.String("self_package", "", "The full package import path for the generated code. The purpose of this flag is to prevent import cycles in the generated code by trying to include its own package. This can happen if the mock's package is set to one of its inputs (usually the main one) and the output is stdio so mockgen cannot detect the final output package. Setting this flag will then tell mockgen which import to exclude.")
|
|
writePkgComment = flag.Bool("write_package_comment", true, "Writes package documentation comment (godoc) if true.")
|
|
writeSourceComment = flag.Bool("write_source_comment", true, "Writes original file (source mode) or interface names (reflect mode) comment if true.")
|
|
writeGenerateDirective = flag.Bool("write_generate_directive", false, "Add //go:generate directive to regenerate the mock")
|
|
copyrightFile = flag.String("copyright_file", "", "Copyright file used to add copyright header")
|
|
typed = flag.Bool("typed", false, "Generate Type-safe 'Return', 'Do', 'DoAndReturn' function")
|
|
imports = flag.String("imports", "", "(source mode) Comma-separated name=path pairs of explicit imports to use.")
|
|
auxFiles = flag.String("aux_files", "", "(source mode) Comma-separated pkg=path pairs of auxiliary Go source files.")
|
|
excludeInterfaces = flag.String("exclude_interfaces", "", "Comma-separated names of interfaces to be excluded")
|
|
|
|
debugParser = flag.Bool("debug_parser", false, "Print out parser results only.")
|
|
showVersion = flag.Bool("version", false, "Print version.")
|
|
)
|
|
|
|
func main() {
|
|
flag.Usage = usage
|
|
flag.Parse()
|
|
|
|
if *showVersion {
|
|
printVersion()
|
|
return
|
|
}
|
|
|
|
var pkg *model.Package
|
|
var err error
|
|
var packageName string
|
|
if *source != "" {
|
|
pkg, err = sourceMode(*source)
|
|
} else {
|
|
if flag.NArg() != 2 {
|
|
usage()
|
|
log.Fatal("Expected exactly two arguments")
|
|
}
|
|
packageName = flag.Arg(0)
|
|
interfaces := strings.Split(flag.Arg(1), ",")
|
|
if packageName == "." {
|
|
dir, err := os.Getwd()
|
|
if err != nil {
|
|
log.Fatalf("Get current directory failed: %v", err)
|
|
}
|
|
packageName, err = packageNameOfDir(dir)
|
|
if err != nil {
|
|
log.Fatalf("Parse package name failed: %v", err)
|
|
}
|
|
}
|
|
pkg, err = reflectMode(packageName, interfaces)
|
|
}
|
|
if err != nil {
|
|
log.Fatalf("Loading input failed: %v", err)
|
|
}
|
|
|
|
if *debugParser {
|
|
pkg.Print(os.Stdout)
|
|
return
|
|
}
|
|
|
|
outputPackageName := *packageOut
|
|
if outputPackageName == "" {
|
|
// pkg.Name in reflect mode is the base name of the import path,
|
|
// which might have characters that are illegal to have in package names.
|
|
outputPackageName = "mock_" + sanitize(pkg.Name)
|
|
}
|
|
|
|
// outputPackagePath represents the fully qualified name of the package of
|
|
// the generated code. Its purposes are to prevent the module from importing
|
|
// itself and to prevent qualifying type names that come from its own
|
|
// package (i.e. if there is a type called X then we want to print "X" not
|
|
// "package.X" since "package" is this package). This can happen if the mock
|
|
// is output into an already existing package.
|
|
outputPackagePath := *selfPackage
|
|
if outputPackagePath == "" && *destination != "" {
|
|
dstPath, err := filepath.Abs(filepath.Dir(*destination))
|
|
if err == nil {
|
|
pkgPath, err := parsePackageImport(dstPath)
|
|
if err == nil {
|
|
outputPackagePath = pkgPath
|
|
} else {
|
|
log.Println("Unable to infer -self_package from destination file path:", err)
|
|
}
|
|
} else {
|
|
log.Println("Unable to determine destination file path:", err)
|
|
}
|
|
}
|
|
|
|
g := new(generator)
|
|
if *source != "" {
|
|
g.filename = *source
|
|
} else {
|
|
g.srcPackage = packageName
|
|
g.srcInterfaces = flag.Arg(1)
|
|
}
|
|
g.destination = *destination
|
|
|
|
if *mockNames != "" {
|
|
g.mockNames = parseMockNames(*mockNames)
|
|
}
|
|
if *copyrightFile != "" {
|
|
header, err := os.ReadFile(*copyrightFile)
|
|
if err != nil {
|
|
log.Fatalf("Failed reading copyright file: %v", err)
|
|
}
|
|
|
|
g.copyrightHeader = string(header)
|
|
}
|
|
if err := g.Generate(pkg, outputPackageName, outputPackagePath); err != nil {
|
|
log.Fatalf("Failed generating mock: %v", err)
|
|
}
|
|
output := g.Output()
|
|
dst := os.Stdout
|
|
if len(*destination) > 0 {
|
|
if err := os.MkdirAll(filepath.Dir(*destination), os.ModePerm); err != nil {
|
|
log.Fatalf("Unable to create directory: %v", err)
|
|
}
|
|
existing, err := os.ReadFile(*destination)
|
|
if err != nil && !errors.Is(err, os.ErrNotExist) {
|
|
log.Fatalf("Failed reading pre-exiting destination file: %v", err)
|
|
}
|
|
if len(existing) == len(output) && bytes.Equal(existing, output) {
|
|
return
|
|
}
|
|
f, err := os.Create(*destination)
|
|
if err != nil {
|
|
log.Fatalf("Failed opening destination file: %v", err)
|
|
}
|
|
defer f.Close()
|
|
dst = f
|
|
}
|
|
if _, err := dst.Write(output); err != nil {
|
|
log.Fatalf("Failed writing to destination: %v", err)
|
|
}
|
|
}
|
|
|
|
func parseMockNames(names string) map[string]string {
|
|
mocksMap := make(map[string]string)
|
|
for _, kv := range strings.Split(names, ",") {
|
|
parts := strings.SplitN(kv, "=", 2)
|
|
if len(parts) != 2 || parts[1] == "" {
|
|
log.Fatalf("bad mock names spec: %v", kv)
|
|
}
|
|
mocksMap[parts[0]] = parts[1]
|
|
}
|
|
return mocksMap
|
|
}
|
|
|
|
func parseExcludeInterfaces(names string) map[string]struct{} {
|
|
splitNames := strings.Split(names, ",")
|
|
namesSet := make(map[string]struct{}, len(splitNames))
|
|
for _, name := range splitNames {
|
|
if name == "" {
|
|
continue
|
|
}
|
|
|
|
namesSet[name] = struct{}{}
|
|
}
|
|
|
|
if len(namesSet) == 0 {
|
|
return nil
|
|
}
|
|
|
|
return namesSet
|
|
}
|
|
|
|
func usage() {
|
|
_, _ = io.WriteString(os.Stderr, usageText)
|
|
flag.PrintDefaults()
|
|
}
|
|
|
|
const usageText = `mockgen has two modes of operation: source and reflect.
|
|
|
|
Source mode generates mock interfaces from a source file.
|
|
It is enabled by using the -source flag. Other flags that
|
|
may be useful in this mode are -imports and -aux_files.
|
|
Example:
|
|
mockgen -source=foo.go [other options]
|
|
|
|
Reflect mode generates mock interfaces by building a program
|
|
that uses reflection to understand interfaces. It is enabled
|
|
by passing two non-flag arguments: an import path, and a
|
|
comma-separated list of symbols.
|
|
Example:
|
|
mockgen database/sql/driver Conn,Driver
|
|
|
|
`
|
|
|
|
type generator struct {
|
|
buf bytes.Buffer
|
|
indent string
|
|
mockNames map[string]string // may be empty
|
|
filename string // may be empty
|
|
destination string // may be empty
|
|
srcPackage, srcInterfaces string // may be empty
|
|
copyrightHeader string
|
|
|
|
packageMap map[string]string // map from import path to package name
|
|
}
|
|
|
|
func (g *generator) p(format string, args ...any) {
|
|
fmt.Fprintf(&g.buf, g.indent+format+"\n", args...)
|
|
}
|
|
|
|
func (g *generator) in() {
|
|
g.indent += "\t"
|
|
}
|
|
|
|
func (g *generator) out() {
|
|
if len(g.indent) > 0 {
|
|
g.indent = g.indent[0 : len(g.indent)-1]
|
|
}
|
|
}
|
|
|
|
// sanitize cleans up a string to make a suitable package name.
|
|
func sanitize(s string) string {
|
|
t := ""
|
|
for _, r := range s {
|
|
if t == "" {
|
|
if unicode.IsLetter(r) || r == '_' {
|
|
t += string(r)
|
|
continue
|
|
}
|
|
} else {
|
|
if unicode.IsLetter(r) || unicode.IsDigit(r) || r == '_' {
|
|
t += string(r)
|
|
continue
|
|
}
|
|
}
|
|
t += "_"
|
|
}
|
|
if t == "_" {
|
|
t = "x"
|
|
}
|
|
return t
|
|
}
|
|
|
|
func (g *generator) Generate(pkg *model.Package, outputPkgName string, outputPackagePath string) error {
|
|
if outputPkgName != pkg.Name && *selfPackage == "" {
|
|
// reset outputPackagePath if it's not passed in through -self_package
|
|
outputPackagePath = ""
|
|
}
|
|
|
|
if g.copyrightHeader != "" {
|
|
lines := strings.Split(g.copyrightHeader, "\n")
|
|
for _, line := range lines {
|
|
g.p("// %s", line)
|
|
}
|
|
g.p("")
|
|
}
|
|
|
|
g.p("// Code generated by MockGen. DO NOT EDIT.")
|
|
if *writeSourceComment {
|
|
if g.filename != "" {
|
|
g.p("// Source: %v", g.filename)
|
|
} else {
|
|
g.p("// Source: %v (interfaces: %v)", g.srcPackage, g.srcInterfaces)
|
|
}
|
|
}
|
|
g.p("//")
|
|
g.p("// Generated by this command:")
|
|
// only log the name of the executable, not the full path
|
|
g.p("// %v", strings.Join(append([]string{filepath.Base(os.Args[0])}, os.Args[1:]...), " "))
|
|
|
|
// Get all required imports, and generate unique names for them all.
|
|
im := pkg.Imports()
|
|
im[gomockImportPath] = true
|
|
|
|
// Only import reflect if it's used. We only use reflect in mocked methods
|
|
// so only import if any of the mocked interfaces have methods.
|
|
for _, intf := range pkg.Interfaces {
|
|
if len(intf.Methods) > 0 {
|
|
im["reflect"] = true
|
|
break
|
|
}
|
|
}
|
|
|
|
// Sort keys to make import alias generation predictable
|
|
sortedPaths := make([]string, len(im))
|
|
x := 0
|
|
for pth := range im {
|
|
sortedPaths[x] = pth
|
|
x++
|
|
}
|
|
sort.Strings(sortedPaths)
|
|
|
|
packagesName := createPackageMap(sortedPaths)
|
|
|
|
definedImports := make(map[string]string, len(im))
|
|
if *imports != "" {
|
|
for _, kv := range strings.Split(*imports, ",") {
|
|
eq := strings.Index(kv, "=")
|
|
if k, v := kv[:eq], kv[eq+1:]; k != "." {
|
|
definedImports[v] = k
|
|
}
|
|
}
|
|
}
|
|
|
|
g.packageMap = make(map[string]string, len(im))
|
|
localNames := make(map[string]bool, len(im))
|
|
for _, pth := range sortedPaths {
|
|
base, ok := packagesName[pth]
|
|
if !ok {
|
|
base = sanitize(path.Base(pth))
|
|
}
|
|
|
|
// Local names for an imported package can usually be the basename of the import path.
|
|
// A couple of situations don't permit that, such as duplicate local names
|
|
// (e.g. importing "html/template" and "text/template"), or where the basename is
|
|
// a keyword (e.g. "foo/case") or when defining a name for that by using the -imports flag.
|
|
// try base0, base1, ...
|
|
pkgName := base
|
|
|
|
if _, ok := definedImports[base]; ok {
|
|
pkgName = definedImports[base]
|
|
}
|
|
|
|
i := 0
|
|
for localNames[pkgName] || token.Lookup(pkgName).IsKeyword() {
|
|
pkgName = base + strconv.Itoa(i)
|
|
i++
|
|
}
|
|
|
|
// Avoid importing package if source pkg == output pkg
|
|
if pth == pkg.PkgPath && outputPackagePath == pkg.PkgPath {
|
|
continue
|
|
}
|
|
|
|
g.packageMap[pth] = pkgName
|
|
localNames[pkgName] = true
|
|
}
|
|
|
|
if *writePkgComment {
|
|
g.p("// Package %v is a generated GoMock package.", outputPkgName)
|
|
}
|
|
g.p("package %v", outputPkgName)
|
|
g.p("")
|
|
g.p("import (")
|
|
g.in()
|
|
for pkgPath, pkgName := range g.packageMap {
|
|
if pkgPath == outputPackagePath {
|
|
continue
|
|
}
|
|
g.p("%v %q", pkgName, pkgPath)
|
|
}
|
|
for _, pkgPath := range pkg.DotImports {
|
|
g.p(". %q", pkgPath)
|
|
}
|
|
g.out()
|
|
g.p(")")
|
|
|
|
if *writeGenerateDirective {
|
|
g.p("//go:generate %v", strings.Join(os.Args, " "))
|
|
}
|
|
|
|
for _, intf := range pkg.Interfaces {
|
|
if err := g.GenerateMockInterface(intf, outputPackagePath); err != nil {
|
|
return err
|
|
}
|
|
}
|
|
|
|
return nil
|
|
}
|
|
|
|
// The name of the mock type to use for the given interface identifier.
|
|
func (g *generator) mockName(typeName string) string {
|
|
if mockName, ok := g.mockNames[typeName]; ok {
|
|
return mockName
|
|
}
|
|
|
|
return "Mock" + typeName
|
|
}
|
|
|
|
// formattedTypeParams returns a long and short form of type param info used for
|
|
// printing. If analyzing a interface with type param [I any, O any] the result
|
|
// will be:
|
|
// "[I any, O any]", "[I, O]"
|
|
func (g *generator) formattedTypeParams(it *model.Interface, pkgOverride string) (string, string) {
|
|
if len(it.TypeParams) == 0 {
|
|
return "", ""
|
|
}
|
|
var long, short strings.Builder
|
|
long.WriteString("[")
|
|
short.WriteString("[")
|
|
for i, v := range it.TypeParams {
|
|
if i != 0 {
|
|
long.WriteString(", ")
|
|
short.WriteString(", ")
|
|
}
|
|
long.WriteString(v.Name)
|
|
short.WriteString(v.Name)
|
|
long.WriteString(fmt.Sprintf(" %s", v.Type.String(g.packageMap, pkgOverride)))
|
|
}
|
|
long.WriteString("]")
|
|
short.WriteString("]")
|
|
return long.String(), short.String()
|
|
}
|
|
|
|
func (g *generator) GenerateMockInterface(intf *model.Interface, outputPackagePath string) error {
|
|
mockType := g.mockName(intf.Name)
|
|
longTp, shortTp := g.formattedTypeParams(intf, outputPackagePath)
|
|
|
|
g.p("")
|
|
g.p("// %v is a mock of %v interface.", mockType, intf.Name)
|
|
g.p("type %v%v struct {", mockType, longTp)
|
|
g.in()
|
|
g.p("ctrl *gomock.Controller")
|
|
g.p("recorder *%vMockRecorder%v", mockType, shortTp)
|
|
g.out()
|
|
g.p("}")
|
|
g.p("")
|
|
|
|
g.p("// %vMockRecorder is the mock recorder for %v.", mockType, mockType)
|
|
g.p("type %vMockRecorder%v struct {", mockType, longTp)
|
|
g.in()
|
|
g.p("mock *%v%v", mockType, shortTp)
|
|
g.out()
|
|
g.p("}")
|
|
g.p("")
|
|
|
|
g.p("// New%v creates a new mock instance.", mockType)
|
|
g.p("func New%v%v(ctrl *gomock.Controller) *%v%v {", mockType, longTp, mockType, shortTp)
|
|
g.in()
|
|
g.p("mock := &%v%v{ctrl: ctrl}", mockType, shortTp)
|
|
g.p("mock.recorder = &%vMockRecorder%v{mock}", mockType, shortTp)
|
|
g.p("return mock")
|
|
g.out()
|
|
g.p("}")
|
|
g.p("")
|
|
|
|
// XXX: possible name collision here if someone has EXPECT in their interface.
|
|
g.p("// EXPECT returns an object that allows the caller to indicate expected use.")
|
|
g.p("func (m *%v%v) EXPECT() *%vMockRecorder%v {", mockType, shortTp, mockType, shortTp)
|
|
g.in()
|
|
g.p("return m.recorder")
|
|
g.out()
|
|
g.p("}")
|
|
|
|
g.GenerateMockMethods(mockType, intf, outputPackagePath, longTp, shortTp, *typed)
|
|
|
|
return nil
|
|
}
|
|
|
|
type byMethodName []*model.Method
|
|
|
|
func (b byMethodName) Len() int { return len(b) }
|
|
func (b byMethodName) Swap(i, j int) { b[i], b[j] = b[j], b[i] }
|
|
func (b byMethodName) Less(i, j int) bool { return b[i].Name < b[j].Name }
|
|
|
|
func (g *generator) GenerateMockMethods(mockType string, intf *model.Interface, pkgOverride, longTp, shortTp string, typed bool) {
|
|
sort.Sort(byMethodName(intf.Methods))
|
|
for _, m := range intf.Methods {
|
|
g.p("")
|
|
_ = g.GenerateMockMethod(mockType, m, pkgOverride, shortTp)
|
|
g.p("")
|
|
_ = g.GenerateMockRecorderMethod(intf, mockType, m, shortTp, typed)
|
|
if typed {
|
|
g.p("")
|
|
_ = g.GenerateMockReturnCallMethod(intf, m, pkgOverride, longTp, shortTp)
|
|
}
|
|
}
|
|
}
|
|
|
|
func makeArgString(argNames, argTypes []string) string {
|
|
args := make([]string, len(argNames))
|
|
for i, name := range argNames {
|
|
// specify the type only once for consecutive args of the same type
|
|
if i+1 < len(argTypes) && argTypes[i] == argTypes[i+1] {
|
|
args[i] = name
|
|
} else {
|
|
args[i] = name + " " + argTypes[i]
|
|
}
|
|
}
|
|
return strings.Join(args, ", ")
|
|
}
|
|
|
|
// GenerateMockMethod generates a mock method implementation.
|
|
// If non-empty, pkgOverride is the package in which unqualified types reside.
|
|
func (g *generator) GenerateMockMethod(mockType string, m *model.Method, pkgOverride, shortTp string) error {
|
|
argNames := g.getArgNames(m, true /* in */)
|
|
argTypes := g.getArgTypes(m, pkgOverride, true /* in */)
|
|
argString := makeArgString(argNames, argTypes)
|
|
|
|
rets := make([]string, len(m.Out))
|
|
for i, p := range m.Out {
|
|
rets[i] = p.Type.String(g.packageMap, pkgOverride)
|
|
}
|
|
retString := strings.Join(rets, ", ")
|
|
if len(rets) > 1 {
|
|
retString = "(" + retString + ")"
|
|
}
|
|
if retString != "" {
|
|
retString = " " + retString
|
|
}
|
|
|
|
ia := newIdentifierAllocator(argNames)
|
|
idRecv := ia.allocateIdentifier("m")
|
|
|
|
g.p("// %v mocks base method.", m.Name)
|
|
g.p("func (%v *%v%v) %v(%v)%v {", idRecv, mockType, shortTp, m.Name, argString, retString)
|
|
g.in()
|
|
g.p("%s.ctrl.T.Helper()", idRecv)
|
|
|
|
var callArgs string
|
|
if m.Variadic == nil {
|
|
if len(argNames) > 0 {
|
|
callArgs = ", " + strings.Join(argNames, ", ")
|
|
}
|
|
} else {
|
|
// Non-trivial. The generated code must build a []any,
|
|
// but the variadic argument may be any type.
|
|
idVarArgs := ia.allocateIdentifier("varargs")
|
|
idVArg := ia.allocateIdentifier("a")
|
|
g.p("%s := []any{%s}", idVarArgs, strings.Join(argNames[:len(argNames)-1], ", "))
|
|
g.p("for _, %s := range %s {", idVArg, argNames[len(argNames)-1])
|
|
g.in()
|
|
g.p("%s = append(%s, %s)", idVarArgs, idVarArgs, idVArg)
|
|
g.out()
|
|
g.p("}")
|
|
callArgs = ", " + idVarArgs + "..."
|
|
}
|
|
if len(m.Out) == 0 {
|
|
g.p(`%v.ctrl.Call(%v, %q%v)`, idRecv, idRecv, m.Name, callArgs)
|
|
} else {
|
|
idRet := ia.allocateIdentifier("ret")
|
|
g.p(`%v := %v.ctrl.Call(%v, %q%v)`, idRet, idRecv, idRecv, m.Name, callArgs)
|
|
|
|
// Go does not allow "naked" type assertions on nil values, so we use the two-value form here.
|
|
// The value of that is either (x.(T), true) or (Z, false), where Z is the zero value for T.
|
|
// Happily, this coincides with the semantics we want here.
|
|
retNames := make([]string, len(rets))
|
|
for i, t := range rets {
|
|
retNames[i] = ia.allocateIdentifier(fmt.Sprintf("ret%d", i))
|
|
g.p("%s, _ := %s[%d].(%s)", retNames[i], idRet, i, t)
|
|
}
|
|
g.p("return " + strings.Join(retNames, ", "))
|
|
}
|
|
|
|
g.out()
|
|
g.p("}")
|
|
return nil
|
|
}
|
|
|
|
func (g *generator) GenerateMockRecorderMethod(intf *model.Interface, mockType string, m *model.Method, shortTp string, typed bool) error {
|
|
argNames := g.getArgNames(m, true)
|
|
|
|
var argString string
|
|
if m.Variadic == nil {
|
|
argString = strings.Join(argNames, ", ")
|
|
} else {
|
|
argString = strings.Join(argNames[:len(argNames)-1], ", ")
|
|
}
|
|
if argString != "" {
|
|
argString += " any"
|
|
}
|
|
|
|
if m.Variadic != nil {
|
|
if argString != "" {
|
|
argString += ", "
|
|
}
|
|
argString += fmt.Sprintf("%s ...any", argNames[len(argNames)-1])
|
|
}
|
|
|
|
ia := newIdentifierAllocator(argNames)
|
|
idRecv := ia.allocateIdentifier("mr")
|
|
|
|
g.p("// %v indicates an expected call of %v.", m.Name, m.Name)
|
|
if typed {
|
|
g.p("func (%s *%vMockRecorder%v) %v(%v) *%s%sCall%s {", idRecv, mockType, shortTp, m.Name, argString, intf.Name, m.Name, shortTp)
|
|
} else {
|
|
g.p("func (%s *%vMockRecorder%v) %v(%v) *gomock.Call {", idRecv, mockType, shortTp, m.Name, argString)
|
|
}
|
|
|
|
g.in()
|
|
g.p("%s.mock.ctrl.T.Helper()", idRecv)
|
|
|
|
var callArgs string
|
|
if m.Variadic == nil {
|
|
if len(argNames) > 0 {
|
|
callArgs = ", " + strings.Join(argNames, ", ")
|
|
}
|
|
} else {
|
|
if len(argNames) == 1 {
|
|
// Easy: just use ... to push the arguments through.
|
|
callArgs = ", " + argNames[0] + "..."
|
|
} else {
|
|
// Hard: create a temporary slice.
|
|
idVarArgs := ia.allocateIdentifier("varargs")
|
|
g.p("%s := append([]any{%s}, %s...)",
|
|
idVarArgs,
|
|
strings.Join(argNames[:len(argNames)-1], ", "),
|
|
argNames[len(argNames)-1])
|
|
callArgs = ", " + idVarArgs + "..."
|
|
}
|
|
}
|
|
if typed {
|
|
g.p(`call := %s.mock.ctrl.RecordCallWithMethodType(%s.mock, "%s", reflect.TypeOf((*%s%s)(nil).%s)%s)`, idRecv, idRecv, m.Name, mockType, shortTp, m.Name, callArgs)
|
|
g.p(`return &%s%sCall%s{Call: call}`, intf.Name, m.Name, shortTp)
|
|
} else {
|
|
g.p(`return %s.mock.ctrl.RecordCallWithMethodType(%s.mock, "%s", reflect.TypeOf((*%s%s)(nil).%s)%s)`, idRecv, idRecv, m.Name, mockType, shortTp, m.Name, callArgs)
|
|
}
|
|
|
|
g.out()
|
|
g.p("}")
|
|
return nil
|
|
}
|
|
|
|
func (g *generator) GenerateMockReturnCallMethod(intf *model.Interface, m *model.Method, pkgOverride, longTp, shortTp string) error {
|
|
argNames := g.getArgNames(m, true /* in */)
|
|
retNames := g.getArgNames(m, false /* out */)
|
|
argTypes := g.getArgTypes(m, pkgOverride, true /* in */)
|
|
retTypes := g.getArgTypes(m, pkgOverride, false /* out */)
|
|
argString := strings.Join(argTypes, ", ")
|
|
|
|
rets := make([]string, len(m.Out))
|
|
for i, p := range m.Out {
|
|
rets[i] = p.Type.String(g.packageMap, pkgOverride)
|
|
}
|
|
|
|
var retString string
|
|
switch {
|
|
case len(rets) == 1:
|
|
retString = " " + rets[0]
|
|
case len(rets) > 1:
|
|
retString = " (" + strings.Join(rets, ", ") + ")"
|
|
}
|
|
|
|
ia := newIdentifierAllocator(argNames)
|
|
idRecv := ia.allocateIdentifier("c")
|
|
|
|
recvStructName := intf.Name + m.Name
|
|
|
|
g.p("// %s%sCall wrap *gomock.Call", intf.Name, m.Name)
|
|
g.p("type %s%sCall%s struct{", intf.Name, m.Name, longTp)
|
|
g.in()
|
|
g.p("*gomock.Call")
|
|
g.out()
|
|
g.p("}")
|
|
|
|
g.p("// Return rewrite *gomock.Call.Return")
|
|
g.p("func (%s *%sCall%s) Return(%v) *%sCall%s {", idRecv, recvStructName, shortTp, makeArgString(retNames, retTypes), recvStructName, shortTp)
|
|
g.in()
|
|
var retArgs string
|
|
if len(retNames) > 0 {
|
|
retArgs = strings.Join(retNames, ", ")
|
|
}
|
|
g.p(`%s.Call = %v.Call.Return(%v)`, idRecv, idRecv, retArgs)
|
|
g.p("return %s", idRecv)
|
|
g.out()
|
|
g.p("}")
|
|
|
|
g.p("// Do rewrite *gomock.Call.Do")
|
|
g.p("func (%s *%sCall%s) Do(f func(%v)%v) *%sCall%s {", idRecv, recvStructName, shortTp, argString, retString, recvStructName, shortTp)
|
|
g.in()
|
|
g.p(`%s.Call = %v.Call.Do(f)`, idRecv, idRecv)
|
|
g.p("return %s", idRecv)
|
|
g.out()
|
|
g.p("}")
|
|
|
|
g.p("// DoAndReturn rewrite *gomock.Call.DoAndReturn")
|
|
g.p("func (%s *%sCall%s) DoAndReturn(f func(%v)%v) *%sCall%s {", idRecv, recvStructName, shortTp, argString, retString, recvStructName, shortTp)
|
|
g.in()
|
|
g.p(`%s.Call = %v.Call.DoAndReturn(f)`, idRecv, idRecv)
|
|
g.p("return %s", idRecv)
|
|
g.out()
|
|
g.p("}")
|
|
return nil
|
|
}
|
|
|
|
func (g *generator) getArgNames(m *model.Method, in bool) []string {
|
|
var params []*model.Parameter
|
|
if in {
|
|
params = m.In
|
|
} else {
|
|
params = m.Out
|
|
}
|
|
argNames := make([]string, len(params))
|
|
for i, p := range params {
|
|
name := p.Name
|
|
if name == "" || name == "_" {
|
|
name = fmt.Sprintf("arg%d", i)
|
|
}
|
|
argNames[i] = name
|
|
}
|
|
if m.Variadic != nil && in {
|
|
name := m.Variadic.Name
|
|
if name == "" {
|
|
name = fmt.Sprintf("arg%d", len(params))
|
|
}
|
|
argNames = append(argNames, name)
|
|
}
|
|
return argNames
|
|
}
|
|
|
|
func (g *generator) getArgTypes(m *model.Method, pkgOverride string, in bool) []string {
|
|
var params []*model.Parameter
|
|
if in {
|
|
params = m.In
|
|
} else {
|
|
params = m.Out
|
|
}
|
|
argTypes := make([]string, len(params))
|
|
for i, p := range params {
|
|
argTypes[i] = p.Type.String(g.packageMap, pkgOverride)
|
|
}
|
|
if m.Variadic != nil {
|
|
argTypes = append(argTypes, "..."+m.Variadic.Type.String(g.packageMap, pkgOverride))
|
|
}
|
|
return argTypes
|
|
}
|
|
|
|
type identifierAllocator map[string]struct{}
|
|
|
|
func newIdentifierAllocator(taken []string) identifierAllocator {
|
|
a := make(identifierAllocator, len(taken))
|
|
for _, s := range taken {
|
|
a[s] = struct{}{}
|
|
}
|
|
return a
|
|
}
|
|
|
|
func (o identifierAllocator) allocateIdentifier(want string) string {
|
|
id := want
|
|
for i := 2; ; i++ {
|
|
if _, ok := o[id]; !ok {
|
|
o[id] = struct{}{}
|
|
return id
|
|
}
|
|
id = want + "_" + strconv.Itoa(i)
|
|
}
|
|
}
|
|
|
|
// Output returns the generator's output, formatted in the standard Go style.
|
|
func (g *generator) Output() []byte {
|
|
src, err := toolsimports.Process(g.destination, g.buf.Bytes(), nil)
|
|
if err != nil {
|
|
log.Fatalf("Failed to format generated source code: %s\n%s", err, g.buf.String())
|
|
}
|
|
return src
|
|
}
|
|
|
|
// createPackageMap returns a map of import path to package name
|
|
// for specified importPaths.
|
|
func createPackageMap(importPaths []string) map[string]string {
|
|
var pkg struct {
|
|
Name string
|
|
ImportPath string
|
|
}
|
|
pkgMap := make(map[string]string)
|
|
b := bytes.NewBuffer(nil)
|
|
args := []string{"list", "-json"}
|
|
args = append(args, importPaths...)
|
|
cmd := exec.Command("go", args...)
|
|
cmd.Stdout = b
|
|
cmd.Run()
|
|
dec := json.NewDecoder(b)
|
|
for dec.More() {
|
|
err := dec.Decode(&pkg)
|
|
if err != nil {
|
|
log.Printf("failed to decode 'go list' output: %v", err)
|
|
continue
|
|
}
|
|
pkgMap[pkg.ImportPath] = pkg.Name
|
|
}
|
|
return pkgMap
|
|
}
|
|
|
|
func printVersion() {
|
|
if version != "" {
|
|
fmt.Printf("v%s\nCommit: %s\nDate: %s\n", version, commit, date)
|
|
} else {
|
|
printModuleVersion()
|
|
}
|
|
}
|
|
|
|
// parseImportPackage get package import path via source file
|
|
// an alternative implementation is to use:
|
|
// cfg := &packages.Config{Mode: packages.NeedName, Tests: true, Dir: srcDir}
|
|
// pkgs, err := packages.Load(cfg, "file="+source)
|
|
// However, it will call "go list" and slow down the performance
|
|
func parsePackageImport(srcDir string) (string, error) {
|
|
moduleMode := os.Getenv("GO111MODULE")
|
|
// trying to find the module
|
|
if moduleMode != "off" {
|
|
currentDir := srcDir
|
|
for {
|
|
dat, err := os.ReadFile(filepath.Join(currentDir, "go.mod"))
|
|
if os.IsNotExist(err) {
|
|
if currentDir == filepath.Dir(currentDir) {
|
|
// at the root
|
|
break
|
|
}
|
|
currentDir = filepath.Dir(currentDir)
|
|
continue
|
|
} else if err != nil {
|
|
return "", err
|
|
}
|
|
modulePath := modfile.ModulePath(dat)
|
|
return filepath.ToSlash(filepath.Join(modulePath, strings.TrimPrefix(srcDir, currentDir))), nil
|
|
}
|
|
}
|
|
// fall back to GOPATH mode
|
|
goPaths := os.Getenv("GOPATH")
|
|
if goPaths == "" {
|
|
return "", fmt.Errorf("GOPATH is not set")
|
|
}
|
|
goPathList := strings.Split(goPaths, string(os.PathListSeparator))
|
|
for _, goPath := range goPathList {
|
|
sourceRoot := filepath.Join(goPath, "src") + string(os.PathSeparator)
|
|
if strings.HasPrefix(srcDir, sourceRoot) {
|
|
return filepath.ToSlash(strings.TrimPrefix(srcDir, sourceRoot)), nil
|
|
}
|
|
}
|
|
return "", errOutsideGoPath
|
|
}
|