mirror of
https://github.com/wailsapp/wails.git
synced 2025-05-04 19:21:21 +08:00

* Add some clarifying comments * Remove special handling of window parameters * Improve internal method exclusion * Add test for internal method exclusion * Remove useless blank field from app options This is a leftover from an older version of the static analyser. It should have been removed long ago. * Remove redundant godebug setting gotypesalias=1 is the default starting with go1.23 * Use new range for syntax to simplify code * Remove generator dependency on github.com/samber/lo * Ensure generator testing tasks do not use the test cache * Rename cyclic types test * Test for cyclic imports * Fix import cycle between model files * Sort class aliases after their aliased class * Test class aliases * Fix length of default value for array types * Test array initialization * Add changelog * Update changelog * Fix contrived marking technique in model sorting algorithm * Update binding example * Update test data --------- Co-authored-by: Lea Anthony <lea.anthony@gmail.com>
310 lines
8.4 KiB
Go
310 lines
8.4 KiB
Go
package collect
|
|
|
|
import (
|
|
"fmt"
|
|
"go/ast"
|
|
"go/types"
|
|
"strconv"
|
|
"sync"
|
|
|
|
"github.com/wailsapp/wails/v3/internal/hash"
|
|
"golang.org/x/tools/go/types/typeutil"
|
|
)
|
|
|
|
type (
|
|
// ServiceInfo records all information that is required
|
|
// to render JS/TS code for a service type.
|
|
//
|
|
// Read accesses to any public field are only safe
|
|
// if a call to [ServiceInfo.Collect] has completed before the access,
|
|
// for example by calling it in the accessing goroutine
|
|
// or before spawning the accessing goroutine.
|
|
ServiceInfo struct {
|
|
*TypeInfo
|
|
|
|
Imports *ImportMap
|
|
Methods []*ServiceMethodInfo
|
|
|
|
// Injections stores a list of JS code lines
|
|
// that should be injected into the generated file.
|
|
Injections []string
|
|
|
|
collector *Collector
|
|
once sync.Once
|
|
}
|
|
|
|
// ServiceMethodInfo records all information that is required
|
|
// to render JS/TS code for a service method.
|
|
ServiceMethodInfo struct {
|
|
*MethodInfo
|
|
FQN string
|
|
ID string
|
|
Internal bool
|
|
Params []*ParamInfo
|
|
Results []types.Type
|
|
}
|
|
|
|
// ParamInfo records all information that is required
|
|
// to render JS/TS code for a service method parameter.
|
|
ParamInfo struct {
|
|
Name string
|
|
Type types.Type
|
|
Blank bool
|
|
Variadic bool
|
|
}
|
|
)
|
|
|
|
func newServiceInfo(collector *Collector, obj *types.TypeName) *ServiceInfo {
|
|
return &ServiceInfo{
|
|
TypeInfo: collector.Type(obj),
|
|
collector: collector,
|
|
}
|
|
}
|
|
|
|
// Service returns the unique ServiceInfo instance
|
|
// associated to the given object within a collector
|
|
// and registers it for code generation.
|
|
//
|
|
// Service is safe for concurrent use.
|
|
func (collector *Collector) Service(obj *types.TypeName) *ServiceInfo {
|
|
pkg := collector.Package(obj.Pkg())
|
|
if pkg == nil {
|
|
return nil
|
|
}
|
|
|
|
return pkg.recordService(obj)
|
|
}
|
|
|
|
// IsEmpty returns true if no methods or code injections
|
|
// are present for this service, for the selected language.
|
|
func (info *ServiceInfo) IsEmpty() bool {
|
|
// Ensure information has been collected.
|
|
info.Collect()
|
|
return len(info.Injections) == 0 && len(info.Methods) == 0
|
|
}
|
|
|
|
// Collect gathers information about the service described by its receiver.
|
|
// It can be called concurrently by multiple goroutines;
|
|
// the computation will be performed just once.
|
|
//
|
|
// Collect returns the receiver for chaining.
|
|
// It is safe to call Collect with nil receiver.
|
|
//
|
|
// After Collect returns, the calling goroutine and all goroutines
|
|
// it might spawn afterwards are free to access
|
|
// the receiver's fields indefinitely.
|
|
func (info *ServiceInfo) Collect() *ServiceInfo {
|
|
if info == nil {
|
|
return nil
|
|
}
|
|
|
|
info.once.Do(func() {
|
|
collector := info.collector
|
|
obj := info.Object().(*types.TypeName)
|
|
|
|
// Collect type information.
|
|
info.TypeInfo.Collect()
|
|
|
|
// Initialise import map.
|
|
info.Imports = NewImportMap(collector.Package(obj.Pkg()))
|
|
|
|
// Compute intuitive method set (i.e. both pointer and non-pointer receiver).
|
|
// Do not use a method set cache because
|
|
// - it would hurt concurrency (requires mutual exclusion),
|
|
// - it is only useful when the same type is queried many times;
|
|
// this may only happen here if some embedded types appear frequently,
|
|
// which should be far from average.
|
|
mset := typeutil.IntuitiveMethodSet(obj.Type(), nil)
|
|
|
|
// Collect method information.
|
|
info.Methods = make([]*ServiceMethodInfo, 0, len(mset))
|
|
for _, sel := range mset {
|
|
if !sel.Obj().Exported() || internalServiceMethods[sel.Obj().Name()] {
|
|
// Ignore unexported and internal methods.
|
|
continue
|
|
}
|
|
|
|
methodInfo := info.collectMethod(sel.Obj().(*types.Func))
|
|
if methodInfo != nil {
|
|
info.Methods = append(info.Methods, methodInfo)
|
|
}
|
|
}
|
|
|
|
// Parse directives.
|
|
for _, doc := range []*ast.CommentGroup{info.Doc, info.Decl.Doc} {
|
|
if doc == nil {
|
|
continue
|
|
}
|
|
for _, comment := range doc.List {
|
|
if IsDirective(comment.Text, "inject") {
|
|
// Check condition.
|
|
line, cond, err := ParseCondition(ParseDirective(comment.Text, "inject"))
|
|
if err != nil {
|
|
collector.logger.Errorf(
|
|
"%s: in `wails:inject` directive: %v",
|
|
collector.Package(obj.Pkg()).Fset.Position(comment.Pos()),
|
|
err,
|
|
)
|
|
continue
|
|
}
|
|
|
|
if !cond.Satisfied(collector.options) {
|
|
continue
|
|
}
|
|
|
|
// Record injected line.
|
|
info.Injections = append(info.Injections, line)
|
|
}
|
|
}
|
|
}
|
|
})
|
|
|
|
return info
|
|
}
|
|
|
|
// internalServiceMethod is a set of methods
|
|
// that are handled specially by the binding engine
|
|
// and must not be exposed to the frontend.
|
|
var internalServiceMethods = map[string]bool{
|
|
"ServiceName": true,
|
|
"ServiceStartup": true,
|
|
"ServiceShutdown": true,
|
|
"ServeHTTP": true,
|
|
}
|
|
|
|
// typeError caches the type-checker type for the Go error interface.
|
|
var typeError = types.Universe.Lookup("error").Type()
|
|
|
|
// typeAny caches the empty interface type.
|
|
var typeAny = types.Universe.Lookup("any").Type().Underlying()
|
|
|
|
// collectMethod collects and returns information about a service method.
|
|
// It is intended to be called only by ServiceInfo.Collect.
|
|
func (info *ServiceInfo) collectMethod(method *types.Func) *ServiceMethodInfo {
|
|
collector := info.collector
|
|
obj := info.Object().(*types.TypeName)
|
|
|
|
signature, _ := method.Type().(*types.Signature)
|
|
if signature == nil {
|
|
// Skip invalid interface method.
|
|
// TODO: is this actually necessary?
|
|
return nil
|
|
}
|
|
|
|
// Compute fully qualified name.
|
|
path := obj.Pkg().Path()
|
|
if obj.Pkg().Name() == "main" {
|
|
// reflect.Method.PkgPath is always "main" for the main package.
|
|
// This should not cause collisions because
|
|
// other main packages are not importable.
|
|
path = "main"
|
|
}
|
|
|
|
fqn := path + "." + obj.Name() + "." + method.Name()
|
|
id, _ := hash.Fnv(fqn)
|
|
|
|
methodInfo := &ServiceMethodInfo{
|
|
MethodInfo: collector.Method(method).Collect(),
|
|
FQN: fqn,
|
|
ID: strconv.FormatUint(uint64(id), 10),
|
|
Params: make([]*ParamInfo, 0, signature.Params().Len()),
|
|
Results: make([]types.Type, 0, signature.Results().Len()),
|
|
}
|
|
|
|
// Parse directives.
|
|
if methodInfo.Doc != nil {
|
|
var methodIdFound bool
|
|
|
|
for _, comment := range methodInfo.Doc.List {
|
|
switch {
|
|
case IsDirective(comment.Text, "internal"):
|
|
methodInfo.Internal = true
|
|
|
|
case !methodIdFound && IsDirective(comment.Text, "id"):
|
|
idString := ParseDirective(comment.Text, "id")
|
|
idValue, err := strconv.ParseUint(idString, 10, 32)
|
|
|
|
if err != nil {
|
|
collector.logger.Errorf(
|
|
"%s: invalid value '%s' in `wails:id` directive: expected a valid uint32 value",
|
|
collector.Package(method.Pkg()).Fset.Position(comment.Pos()),
|
|
idString,
|
|
)
|
|
continue
|
|
}
|
|
|
|
// Announce and record alias.
|
|
collector.logger.Infof(
|
|
"package %s: method %s.%s: default ID %s replaced by %d",
|
|
path,
|
|
obj.Name(),
|
|
method.Name(),
|
|
methodInfo.ID,
|
|
idValue,
|
|
)
|
|
methodInfo.ID = strconv.FormatUint(idValue, 10)
|
|
}
|
|
}
|
|
}
|
|
|
|
// Collect parameters.
|
|
for i := range signature.Params().Len() {
|
|
param := signature.Params().At(i)
|
|
|
|
if i == 0 {
|
|
// Skip first parameter if it has context type.
|
|
named, ok := types.Unalias(param.Type()).(*types.Named)
|
|
if ok && named.Obj().Pkg().Path() == collector.systemPaths.ContextPackage && named.Obj().Name() == "Context" {
|
|
continue
|
|
}
|
|
}
|
|
|
|
if types.IsInterface(param.Type()) && !types.Identical(param.Type(), typeAny) {
|
|
paramName := param.Name()
|
|
if paramName == "" || paramName == "_" {
|
|
paramName = fmt.Sprintf("#%d", i+1)
|
|
}
|
|
|
|
collector.logger.Warningf(
|
|
"%s: parameter %s has non-empty interface type %s: this is not supported by encoding/json and will likely result in runtime errors",
|
|
collector.Package(method.Pkg()).Fset.Position(param.Pos()),
|
|
paramName,
|
|
param.Type(),
|
|
)
|
|
}
|
|
|
|
// Record type dependencies.
|
|
info.Imports.AddType(param.Type())
|
|
|
|
// Record parameter.
|
|
methodInfo.Params = append(methodInfo.Params, &ParamInfo{
|
|
Name: param.Name(),
|
|
Type: param.Type(),
|
|
Blank: param.Name() == "" || param.Name() == "_",
|
|
})
|
|
}
|
|
|
|
if signature.Variadic() {
|
|
methodInfo.Params[len(methodInfo.Params)-1].Type = methodInfo.Params[len(methodInfo.Params)-1].Type.(*types.Slice).Elem()
|
|
methodInfo.Params[len(methodInfo.Params)-1].Variadic = true
|
|
}
|
|
|
|
// Collect results.
|
|
for i := range signature.Results().Len() {
|
|
result := signature.Results().At(i)
|
|
|
|
if types.Identical(result.Type(), typeError) {
|
|
// Skip error results, they are thrown as exceptions
|
|
continue
|
|
}
|
|
|
|
// Record type dependencies.
|
|
info.Imports.AddType(result.Type())
|
|
|
|
// Record result.
|
|
methodInfo.Results = append(methodInfo.Results, result.Type())
|
|
}
|
|
|
|
return methodInfo
|
|
}
|