5
0
mirror of https://github.com/wailsapp/wails.git synced 2025-05-08 00:53:33 +08:00
wails/v3/internal/generator/collect/model.go
Fabio Massaioli 90b7ea944d
[v3] New binding generator (#3468)
* 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>
2024-05-19 20:40:44 +10:00

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)
}
}