mirror of
https://github.com/stashapp/stash.git
synced 2026-04-26 17:05:25 +02:00
* Upgrade gqlgen to v0.17.2 This enables builds on Go 1.18. github.com/vektah/gqlparser is upgraded to the newest version too. Getting this to work is a bit of a hazzle. I had to first remove vendoring from the repository, perform the upgrade and then re-introduce the vendor directory. I think gqlgens analysis went wrong for some reason on the upgrade. It would seem a clean-room installation fixed it. * Bump project to 1.18 * Update all packages, address gqlgenc breaking changes * Let `go mod tidy` handle the go.mod file * Upgrade linter to 1.45.2 * Introduce v1.45.2 of the linter The linter now correctly warns on `strings.Title` because it isn't unicode-aware. Fix this by using the suggested fix from x/text/cases to produce unicode-aware strings. The mapping isn't entirely 1-1 as this new approach has a larger iface: it spans all of unicode rather than just ASCII. It coincides for ASCII however, so things should be largely the same. * Ready ourselves for errchkjson and contextcheck. * Revert dockerfile golang version changes for now Co-authored-by: Kermie <kermie@isinthe.house> Co-authored-by: WithoutPants <53250216+WithoutPants@users.noreply.github.com>
171 lines
3.5 KiB
Go
171 lines
3.5 KiB
Go
package pkg
|
|
|
|
import (
|
|
"context"
|
|
"fmt"
|
|
"io"
|
|
"io/ioutil"
|
|
"os"
|
|
"path/filepath"
|
|
"regexp"
|
|
"strings"
|
|
|
|
"github.com/vektra/mockery/v2/pkg/config"
|
|
"github.com/vektra/mockery/v2/pkg/logging"
|
|
|
|
"github.com/rs/zerolog"
|
|
)
|
|
|
|
type Walker struct {
|
|
config.Config
|
|
BaseDir string
|
|
Recursive bool
|
|
Filter *regexp.Regexp
|
|
LimitOne bool
|
|
BuildTags []string
|
|
}
|
|
|
|
type WalkerVisitor interface {
|
|
VisitWalk(context.Context, *Interface) error
|
|
}
|
|
|
|
func (w *Walker) Walk(ctx context.Context, visitor WalkerVisitor) (generated bool) {
|
|
log := zerolog.Ctx(ctx)
|
|
ctx = log.WithContext(ctx)
|
|
|
|
log.Info().Msgf("Walking")
|
|
|
|
parser := NewParser(w.BuildTags)
|
|
w.doWalk(ctx, parser, w.BaseDir, visitor)
|
|
|
|
err := parser.Load()
|
|
if err != nil {
|
|
fmt.Fprintf(os.Stderr, "Error walking: %v\n", err)
|
|
os.Exit(1)
|
|
}
|
|
|
|
for _, iface := range parser.Interfaces() {
|
|
if !w.Filter.MatchString(iface.Name) {
|
|
continue
|
|
}
|
|
err := visitor.VisitWalk(ctx, iface)
|
|
if err != nil {
|
|
fmt.Fprintf(os.Stderr, "Error walking %s: %s\n", iface.Name, err)
|
|
os.Exit(1)
|
|
}
|
|
generated = true
|
|
if w.LimitOne {
|
|
return
|
|
}
|
|
}
|
|
|
|
return
|
|
}
|
|
|
|
func (w *Walker) doWalk(ctx context.Context, p *Parser, dir string, visitor WalkerVisitor) (generated bool) {
|
|
log := zerolog.Ctx(ctx)
|
|
ctx = log.WithContext(ctx)
|
|
|
|
files, err := ioutil.ReadDir(dir)
|
|
if err != nil {
|
|
return
|
|
}
|
|
|
|
for _, file := range files {
|
|
if strings.HasPrefix(file.Name(), ".") || strings.HasPrefix(file.Name(), "_") {
|
|
continue
|
|
}
|
|
|
|
path := filepath.Join(dir, file.Name())
|
|
|
|
if file.IsDir() {
|
|
if w.Recursive {
|
|
generated = w.doWalk(ctx, p, path, visitor) || generated
|
|
if generated && w.LimitOne {
|
|
return
|
|
}
|
|
}
|
|
continue
|
|
}
|
|
|
|
if !strings.HasSuffix(path, ".go") || strings.HasSuffix(path, "_test.go") {
|
|
continue
|
|
}
|
|
|
|
err = p.Parse(ctx, path)
|
|
if err != nil {
|
|
log.Err(err).Msgf("Error parsing file")
|
|
continue
|
|
}
|
|
}
|
|
|
|
return
|
|
}
|
|
|
|
type GeneratorVisitor struct {
|
|
config.Config
|
|
InPackage bool
|
|
Note string
|
|
Boilerplate string
|
|
Osp OutputStreamProvider
|
|
// The name of the output package, if InPackage is false (defaults to "mocks")
|
|
PackageName string
|
|
PackageNamePrefix string
|
|
StructName string
|
|
}
|
|
|
|
func (v *GeneratorVisitor) VisitWalk(ctx context.Context, iface *Interface) error {
|
|
log := zerolog.Ctx(ctx).With().
|
|
Str(logging.LogKeyInterface, iface.Name).
|
|
Str(logging.LogKeyQualifiedName, iface.QualifiedName).
|
|
Logger()
|
|
ctx = log.WithContext(ctx)
|
|
|
|
defer func() {
|
|
if r := recover(); r != nil {
|
|
log.Error().Msgf("Unable to generate mock: %s", r)
|
|
return
|
|
}
|
|
}()
|
|
|
|
var out io.Writer
|
|
var pkg string
|
|
|
|
if v.KeepTree && v.InPackage {
|
|
pkg = filepath.Dir(iface.FileName)
|
|
} else if v.InPackage {
|
|
pkg = filepath.Dir(iface.FileName)
|
|
} else if (v.PackageName == "" || v.PackageName == "mocks") && v.PackageNamePrefix != "" {
|
|
// go with package name prefix only when package name is empty or default and package name prefix is specified
|
|
pkg = fmt.Sprintf("%s%s", v.PackageNamePrefix, iface.Pkg.Name())
|
|
} else {
|
|
pkg = v.PackageName
|
|
}
|
|
|
|
out, err, closer := v.Osp.GetWriter(ctx, iface)
|
|
if err != nil {
|
|
log.Err(err).Msgf("Unable to get writer")
|
|
os.Exit(1)
|
|
}
|
|
defer closer()
|
|
|
|
gen := NewGenerator(ctx, v.Config, iface, pkg)
|
|
gen.GenerateBoilerplate(v.Boilerplate)
|
|
gen.GeneratePrologueNote(v.Note)
|
|
gen.GeneratePrologue(ctx, pkg)
|
|
|
|
err = gen.Generate(ctx)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
log.Info().Msgf("Generating mock")
|
|
if !v.Config.DryRun {
|
|
err = gen.Write(out)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
}
|
|
|
|
return nil
|
|
}
|