mirror of
https://github.com/stashapp/stash.git
synced 2025-12-06 16:34:02 +01: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>
174 lines
4.1 KiB
Go
174 lines
4.1 KiB
Go
package codegen
|
|
|
|
import (
|
|
"fmt"
|
|
"strconv"
|
|
"strings"
|
|
|
|
"github.com/99designs/gqlgen/codegen/templates"
|
|
"github.com/vektah/gqlparser/v2/ast"
|
|
)
|
|
|
|
type DirectiveList map[string]*Directive
|
|
|
|
// LocationDirectives filter directives by location
|
|
func (dl DirectiveList) LocationDirectives(location string) DirectiveList {
|
|
return locationDirectives(dl, ast.DirectiveLocation(location))
|
|
}
|
|
|
|
type Directive struct {
|
|
*ast.DirectiveDefinition
|
|
Name string
|
|
Args []*FieldArgument
|
|
Builtin bool
|
|
}
|
|
|
|
// IsLocation check location directive
|
|
func (d *Directive) IsLocation(location ...ast.DirectiveLocation) bool {
|
|
for _, l := range d.Locations {
|
|
for _, a := range location {
|
|
if l == a {
|
|
return true
|
|
}
|
|
}
|
|
}
|
|
|
|
return false
|
|
}
|
|
|
|
func locationDirectives(directives DirectiveList, location ...ast.DirectiveLocation) map[string]*Directive {
|
|
mDirectives := make(map[string]*Directive)
|
|
for name, d := range directives {
|
|
if d.IsLocation(location...) {
|
|
mDirectives[name] = d
|
|
}
|
|
}
|
|
return mDirectives
|
|
}
|
|
|
|
func (b *builder) buildDirectives() (map[string]*Directive, error) {
|
|
directives := make(map[string]*Directive, len(b.Schema.Directives))
|
|
|
|
for name, dir := range b.Schema.Directives {
|
|
if _, ok := directives[name]; ok {
|
|
return nil, fmt.Errorf("directive with name %s already exists", name)
|
|
}
|
|
|
|
var args []*FieldArgument
|
|
for _, arg := range dir.Arguments {
|
|
tr, err := b.Binder.TypeReference(arg.Type, nil)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
newArg := &FieldArgument{
|
|
ArgumentDefinition: arg,
|
|
TypeReference: tr,
|
|
VarName: templates.ToGoPrivate(arg.Name),
|
|
}
|
|
|
|
if arg.DefaultValue != nil {
|
|
var err error
|
|
newArg.Default, err = arg.DefaultValue.Value(nil)
|
|
if err != nil {
|
|
return nil, fmt.Errorf("default value for directive argument %s(%s) is not valid: %w", dir.Name, arg.Name, err)
|
|
}
|
|
}
|
|
args = append(args, newArg)
|
|
}
|
|
|
|
directives[name] = &Directive{
|
|
DirectiveDefinition: dir,
|
|
Name: name,
|
|
Args: args,
|
|
Builtin: b.Config.Directives[name].SkipRuntime,
|
|
}
|
|
}
|
|
|
|
return directives, nil
|
|
}
|
|
|
|
func (b *builder) getDirectives(list ast.DirectiveList) ([]*Directive, error) {
|
|
dirs := make([]*Directive, len(list))
|
|
for i, d := range list {
|
|
argValues := make(map[string]interface{}, len(d.Arguments))
|
|
for _, da := range d.Arguments {
|
|
val, err := da.Value.Value(nil)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
argValues[da.Name] = val
|
|
}
|
|
def, ok := b.Directives[d.Name]
|
|
if !ok {
|
|
return nil, fmt.Errorf("directive %s not found", d.Name)
|
|
}
|
|
|
|
var args []*FieldArgument
|
|
for _, a := range def.Args {
|
|
value := a.Default
|
|
if argValue, ok := argValues[a.Name]; ok {
|
|
value = argValue
|
|
}
|
|
args = append(args, &FieldArgument{
|
|
ArgumentDefinition: a.ArgumentDefinition,
|
|
Value: value,
|
|
VarName: a.VarName,
|
|
TypeReference: a.TypeReference,
|
|
})
|
|
}
|
|
dirs[i] = &Directive{
|
|
Name: d.Name,
|
|
Args: args,
|
|
DirectiveDefinition: list[i].Definition,
|
|
Builtin: b.Config.Directives[d.Name].SkipRuntime,
|
|
}
|
|
|
|
}
|
|
|
|
return dirs, nil
|
|
}
|
|
|
|
func (d *Directive) ArgsFunc() string {
|
|
if len(d.Args) == 0 {
|
|
return ""
|
|
}
|
|
|
|
return "dir_" + d.Name + "_args"
|
|
}
|
|
|
|
func (d *Directive) CallArgs() string {
|
|
args := []string{"ctx", "obj", "n"}
|
|
|
|
for _, arg := range d.Args {
|
|
args = append(args, "args["+strconv.Quote(arg.Name)+"].("+templates.CurrentImports.LookupType(arg.TypeReference.GO)+")")
|
|
}
|
|
|
|
return strings.Join(args, ", ")
|
|
}
|
|
|
|
func (d *Directive) ResolveArgs(obj string, next int) string {
|
|
args := []string{"ctx", obj, fmt.Sprintf("directive%d", next)}
|
|
|
|
for _, arg := range d.Args {
|
|
dArg := arg.VarName
|
|
if arg.Value == nil && arg.Default == nil {
|
|
dArg = "nil"
|
|
}
|
|
|
|
args = append(args, dArg)
|
|
}
|
|
|
|
return strings.Join(args, ", ")
|
|
}
|
|
|
|
func (d *Directive) Declaration() string {
|
|
res := ucFirst(d.Name) + " func(ctx context.Context, obj interface{}, next graphql.Resolver"
|
|
|
|
for _, arg := range d.Args {
|
|
res += fmt.Sprintf(", %s %s", templates.ToGoPrivate(arg.Name), templates.CurrentImports.LookupType(arg.TypeReference.GO))
|
|
}
|
|
|
|
res += ") (res interface{}, err error)"
|
|
return res
|
|
}
|