mirror of
https://github.com/wailsapp/wails.git
synced 2025-05-08 00:53:33 +08:00

* Support variadic arguments and slice, pointer types * Fix computation of type namespaces * Improve comments and general formatting * Set default values correctly for composite types * Add templates for bindings Additionally: * fixes generation of tuple return type * improves imports and namespacing in JS mode * general cleanup of generated code * Simplify import list construction * Refactor type generation code Improves support for unknown types (encoded as any) and maps (using Typescript index signatures) * Support slices with pointer elements * Match encoding/json behaviour in struct parser * Update tests and example * Add tests for complex method signatures and json tag parsing * Add test `function_multiple_files` * Attempt looking up idents with missing denotation * Update test data * fix quoted bool field * Test quoted booleans * Delete old parser code * Remove old test data * Update bindgen flags * Makes call by ID the default * Add package loading code * Add static analyser * Temporarily ignore binding generation code * Add complex slice expressions test * Fix variable reference analysis * Unwrap casts to interface types * Complete code comments * Refactor static analyser * Restrict options struct usage * Update tests * Fix method selector sink and source processing * Improve Set API * Add package info collector * Rename analyser package to analyse * Improve template functions * Add index file templates * Add glue code for binding generation * Refactor collection and rendering code * Implement binding generator * Implement global index generation * Improve marshaler and alias handling * Use package path in binding calls by name * Implement model collection and rendering * Fix wrong exit condition in analyser * Fix enum rendering * Generate shortcuts for all packages. * Implement generator tests * Ignore non-pointer bound types * Treat main package specially * Compute stats * Plug new API into generate command * Support all named types * Update JS runtime * Report dual role types * Remove go1.22 syntax * Fix type assertion in TS bindings * encoding/json compliance for arrays and slices * Ignore got files in testdata * Cleanup type rendering mechanism * Update JS runtime * Implement generic models * Add missing field in renderer initialisation * Improve generic creation code * Add generic model test * Add error reporting infrastructure * Support configurable file names * Detect file naming collisions * Print final error report * New shortcut file structure + collision detection * Update test layout and data * Autoconfiguration for analyser tests * Live progress reporting * Update code comments * Fix model doc rendering * Simplify name resolution * Add test for out of tree types * Fix generic creation code * Fix potential collisions between methods and models * Fix generic class alias rendering * Report model discovery in debug mode * Add interface mode for JS * Collect interface method comments * Add interface methods test * Unwrap generic instantiations in method receivers * Fix rendering of nullable types in interface mode * Fix rendering of class aliases * Expose promise cancel method to typescript * Update test data * Update binding example * Fix rendering of aliased quoted type params * Move to strongly typed bindings * Implement lightweight analyser * Update test cases * Update binding example * Add complex instantiation test * Load full dependency tree * Rewrite collector * Update renderer to match new collector * Update generator to match new collector * Update test data * Update binding example * Configure includes and injections by language * Improve system path resolution * Support rich conditions in inject/include directives * Fix error handling in Generator.Generate * Retrieve compiled go file paths from fileset * Do not rely on struct info in struct flattening algorithm * Fix doc comment for findDeclaraion * Fix bugs in embedded field handling * Fix bugs and comments in package collection * Remove useless fields from ServiceInfo * Fix empty line at the beginning of TS indexes * Remove global index and shortcuts * Remove generation tests for individual packages * Enforce lower-case file names * Update test data * Improve error reporting * Update binding example * Reintroduce go1.22 syntax * Improve relative import path computation * Improve alias support * Add alias test * Update test data * Remove no services error * Rename global analyser test * Add workaround and test for bug in typeutil.Map * Update test data * Do not split fully qualified names * Update typeutil package and remove workaround * Unify alias/named type handling * Fix rendering of generic named class aliases * Fix rendering of array types * Minor tweaks and cleanups * Rmove namespaced export construct * Update test data * Update binding example * Break type cycles * Fix typo in comment * Fix creation code for cyclic types * Fix type of variadic params in interface mode * Update test data * Fix bad whitespace * Refactor type assertions inside bound methods * Update test data * Rename field application.Options.Bind to Services * Rename parser package to generator * Update binding example * Update test data * Update generator readme * Add typescript test harness * Move test output to new subfolder * Fix code generation bugs * Use .js extensions in TS mode imports * Update test data * Revert default generator output dir to frontend/bindings * Bump runtime package version * Update templates * Update changelog * Improve newline handling --------- Co-authored-by: Andreas Bichinger <andreas.bichinger@gmail.com>
288 lines
7.5 KiB
Go
288 lines
7.5 KiB
Go
package collect
|
|
|
|
import (
|
|
"cmp"
|
|
"go/constant"
|
|
"go/types"
|
|
"slices"
|
|
"strings"
|
|
"sync"
|
|
)
|
|
|
|
type (
|
|
// ModelInfo records all information that is required
|
|
// to render JS/TS code for a model type.
|
|
//
|
|
// Read accesses to exported fields are only safe
|
|
// if a call to [ModelInfo.Collect] has completed before the access,
|
|
// for example by calling it in the accessing goroutine
|
|
// or before spawning the accessing goroutine.
|
|
ModelInfo struct {
|
|
*TypeInfo
|
|
|
|
// Imports records dependencies for this model.
|
|
Imports *ImportMap
|
|
|
|
// Type records the target type for an alias or derived model,
|
|
// the underlying type for an enum.
|
|
Type types.Type
|
|
|
|
// Fields records the property list for a class or struct alias model,
|
|
// in order of declaration and grouped by their declaring [ast.Field].
|
|
Fields [][]*ModelFieldInfo
|
|
|
|
// Values records the value list for an enum model,
|
|
// in order of declaration and grouped
|
|
// by their declaring [ast.GenDecl] and [ast.ValueSpec].
|
|
Values [][][]*ConstInfo
|
|
|
|
// TypeParams records type parameter names for generic models.
|
|
TypeParams []string
|
|
|
|
collector *Collector
|
|
once sync.Once
|
|
}
|
|
|
|
// ModelFieldInfo holds extended information
|
|
// about a struct field in a model type.
|
|
ModelFieldInfo struct {
|
|
*StructField
|
|
*FieldInfo
|
|
}
|
|
)
|
|
|
|
func newModelInfo(collector *Collector, obj *types.TypeName) *ModelInfo {
|
|
return &ModelInfo{
|
|
TypeInfo: collector.Type(obj),
|
|
collector: collector,
|
|
}
|
|
}
|
|
|
|
// Model retrieves the the unique [ModelInfo] instance
|
|
// associated to the given type object within a Collector.
|
|
// If none is present, Model initialises a new one
|
|
// registers it for code generation
|
|
// and schedules background collection activity.
|
|
//
|
|
// Model is safe for concurrent use.
|
|
func (collector *Collector) Model(obj *types.TypeName) *ModelInfo {
|
|
pkg := collector.Package(obj.Pkg())
|
|
if pkg == nil {
|
|
return nil
|
|
}
|
|
|
|
model, present := pkg.recordModel(obj)
|
|
if !present {
|
|
collector.scheduler.Schedule(func() { model.Collect() })
|
|
}
|
|
|
|
return model
|
|
}
|
|
|
|
// Collect gathers information for the model 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 *ModelInfo) Collect() *ModelInfo {
|
|
if info == nil {
|
|
return nil
|
|
}
|
|
|
|
// Changes in the following logic must be reflected adequately
|
|
// by the predicates in properties.go, by ImportMap.AddType
|
|
// and by all render.Module methods.
|
|
|
|
info.once.Do(func() {
|
|
collector := info.collector
|
|
obj := info.Object().(*types.TypeName)
|
|
|
|
typ := obj.Type()
|
|
|
|
// Collect type information.
|
|
info.TypeInfo.Collect()
|
|
|
|
// Initialise import map.
|
|
info.Imports = NewImportMap(collector.Package(obj.Pkg()))
|
|
|
|
// Setup fallback type.
|
|
info.Type = types.Universe.Lookup("any").Type()
|
|
|
|
// Retrieve type denotation and skip alias chains.
|
|
def := info.TypeInfo.Def
|
|
|
|
// Check marshalers and detect enums.
|
|
var constants []*types.Const
|
|
|
|
var isGeneric bool
|
|
if generic, ok := obj.Type().(interface{ TypeParams() *types.TypeParamList }); ok {
|
|
// Record type parameter names.
|
|
tparams := generic.TypeParams()
|
|
isGeneric = tparams != nil
|
|
|
|
if isGeneric && tparams.Len() > 0 {
|
|
info.TypeParams = make([]string, tparams.Len())
|
|
for i := range tparams.Len() {
|
|
info.TypeParams[i] = tparams.At(i).Obj().Name()
|
|
}
|
|
}
|
|
}
|
|
|
|
if _, isNamed := obj.Type().(*types.Named); isNamed {
|
|
// Model is a named type.
|
|
// Check whether it implements marshaler interfaces
|
|
// or has defined constants.
|
|
|
|
if IsAny(typ) {
|
|
// Type marshals to a custom value of unknown shape.
|
|
return
|
|
} else if MaybeTextMarshaler(typ) {
|
|
// Type marshals to a custom string of unknown shape.
|
|
info.Type = types.Typ[types.String]
|
|
return
|
|
} else if isGeneric && !collector.options.UseInterfaces && IsClass(typ) {
|
|
// Generic classes cannot be defined in terms of other generic classes.
|
|
// That would break class creation code,
|
|
// and I (@fbbdev) couldn't find any other satisfying workaround.
|
|
def = typ.Underlying()
|
|
}
|
|
|
|
// Test for enums (excluding generic types).
|
|
basic, ok := typ.Underlying().(*types.Basic)
|
|
if ok && !isGeneric && basic.Info()&types.IsConstType != 0 && basic.Info()&types.IsComplex == 0 {
|
|
// Named type is defined as a representable constant type:
|
|
// look for defined constants of that named type.
|
|
for _, name := range obj.Pkg().Scope().Names() {
|
|
if cnst, ok := obj.Pkg().Scope().Lookup(name).(*types.Const); ok {
|
|
if cnst.Val().Kind() != constant.Unknown && types.Identical(cnst.Type(), typ) {
|
|
constants = append(constants, cnst)
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
// Record required imports.
|
|
info.Imports.AddType(def)
|
|
|
|
// Handle enum types.
|
|
// constants slice is always empty for aliases.
|
|
if len(constants) > 0 {
|
|
// Collect information about enum values.
|
|
info.collectEnum(constants)
|
|
info.Type = def
|
|
return
|
|
}
|
|
|
|
// Handle struct types.
|
|
strct, isStruct := def.(*types.Struct)
|
|
if isStruct {
|
|
// Collect information about struct fields.
|
|
info.collectStruct(strct)
|
|
info.Type = nil
|
|
return
|
|
}
|
|
|
|
// That's all, folks. Render as a TS alias.
|
|
info.Type = def
|
|
})
|
|
|
|
return info
|
|
}
|
|
|
|
// collectEnum collects information about enum values and their declarations.
|
|
func (info *ModelInfo) collectEnum(constants []*types.Const) {
|
|
// Collect information about each constant object.
|
|
values := make([]*ConstInfo, len(constants))
|
|
for i, cnst := range constants {
|
|
values[i] = info.collector.Const(cnst).Collect()
|
|
}
|
|
|
|
// Sort values by grouping and source order.
|
|
slices.SortFunc(values, func(v1 *ConstInfo, v2 *ConstInfo) int {
|
|
// Skip comparisons for identical pointers.
|
|
if v1 == v2 {
|
|
return 0
|
|
}
|
|
|
|
// Sort first by source order of declaration group.
|
|
if v1.Decl != v2.Decl {
|
|
return cmp.Compare(v1.Decl.Pos, v2.Decl.Pos)
|
|
}
|
|
|
|
// Then by source order of spec.
|
|
if v1.Spec != v2.Spec {
|
|
return cmp.Compare(v1.Spec.Pos, v2.Spec.Pos)
|
|
}
|
|
|
|
// Then by source order of identifiers.
|
|
if v1.Pos != v2.Pos {
|
|
return cmp.Compare(v1.Pos, v2.Pos)
|
|
}
|
|
|
|
// Finally by name (for constants whose source position is unknown).
|
|
return strings.Compare(v1.Name, v2.Name)
|
|
})
|
|
|
|
// Split value list into groups and subgroups.
|
|
var decl, spec *GroupInfo
|
|
decli, speci := -1, -1
|
|
|
|
for _, value := range values {
|
|
if value.Spec != spec {
|
|
spec = value.Spec
|
|
|
|
if value.Decl == decl {
|
|
speci++
|
|
} else {
|
|
decl = value.Decl
|
|
decli++
|
|
speci = 0
|
|
info.Values = append(info.Values, nil)
|
|
}
|
|
|
|
info.Values[decli] = append(info.Values[decli], nil)
|
|
}
|
|
|
|
info.Values[decli][speci] = append(info.Values[decli][speci], value)
|
|
}
|
|
}
|
|
|
|
// collectStruct collects information about struct fields and their declarations.
|
|
func (info *ModelInfo) collectStruct(strct *types.Struct) {
|
|
collector := info.collector
|
|
|
|
// Retrieve struct info.
|
|
structInfo := collector.Struct(strct).Collect()
|
|
|
|
// Allocate result slice.
|
|
fields := make([]*ModelFieldInfo, len(structInfo.Fields))
|
|
|
|
// Collect fields.
|
|
for i, field := range structInfo.Fields {
|
|
fields[i] = &ModelFieldInfo{
|
|
StructField: field,
|
|
FieldInfo: collector.Field(field.Object).Collect(),
|
|
}
|
|
}
|
|
|
|
// Split field list into groups, preserving the original order.
|
|
var decl *GroupInfo
|
|
decli := -1
|
|
|
|
for _, field := range fields {
|
|
if field.Decl != decl {
|
|
decl = field.Decl
|
|
decli++
|
|
info.Fields = append(info.Fields, nil)
|
|
}
|
|
|
|
info.Fields[decli] = append(info.Fields[decli], field)
|
|
}
|
|
}
|