mirror of
https://github.com/wailsapp/wails.git
synced 2025-05-07 04:09:26 +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>
191 lines
5.5 KiB
Go
191 lines
5.5 KiB
Go
package generator
|
|
|
|
import (
|
|
"errors"
|
|
"fmt"
|
|
"sync"
|
|
|
|
"github.com/samber/lo"
|
|
"github.com/wailsapp/wails/v3/internal/generator/config"
|
|
)
|
|
|
|
// ErrNoContextPackage indicates that
|
|
// the canonical path for the standard context package
|
|
// did not match any actual package.
|
|
var ErrNoContextPackage = errors.New("standard context package not found at canonical import path ('context'): is the Wails v3 module properly installed?")
|
|
|
|
// ErrNoApplicationPackage indicates that
|
|
// the canonical path for the Wails application package
|
|
// did not match any actual package.
|
|
var ErrNoApplicationPackage = errors.New("Wails application package not found at canonical import path ('" + config.WailsAppPkgPath + "'): is the Wails v3 module properly installed?")
|
|
|
|
// ErrBadApplicationPackage indicates that
|
|
// the Wails application package has invalid content.
|
|
var ErrBadApplicationPackage = errors.New("package " + config.WailsAppPkgPath + ": function NewService has wrong signature: is the Wails v3 module properly installed?")
|
|
|
|
// ErrNoPackages is returned by [Generator.Generate]
|
|
// when [LoadPackages] returns no error and no packages.
|
|
var ErrNoPackages = errors.New("the given patterns matched no packages")
|
|
|
|
// ErrorReport accumulates and logs error
|
|
// and warning messages, with deduplication.
|
|
//
|
|
// It implements the error interface; the Error method
|
|
// returns a report counting messages emitted so far.
|
|
//
|
|
// It also implements the interface [config.Logger] for convenience.
|
|
type ErrorReport struct {
|
|
logger config.Logger
|
|
|
|
mu sync.Mutex
|
|
warnings map[string]bool
|
|
errors map[string]bool
|
|
}
|
|
|
|
// NewError report initialises an ErrorReport instance
|
|
// with the provided Logger implementation.
|
|
//
|
|
// If logger is nil, messages will be accumulated but not logged.
|
|
func NewErrorReport(logger config.Logger) *ErrorReport {
|
|
if logger == nil {
|
|
logger = config.NullLogger
|
|
}
|
|
|
|
return &ErrorReport{
|
|
logger: logger,
|
|
warnings: make(map[string]bool),
|
|
errors: make(map[string]bool),
|
|
}
|
|
}
|
|
|
|
// Error returns a string reporting the number
|
|
// of errors and warnings emitted so far.
|
|
func (report *ErrorReport) Error() string {
|
|
report.mu.Lock()
|
|
defer report.mu.Unlock()
|
|
|
|
if len(report.errors) > 0 && len(report.warnings) == 0 {
|
|
var plural string
|
|
if len(report.errors) > 1 {
|
|
plural = "s"
|
|
}
|
|
return fmt.Sprintf("%d error%s emitted", len(report.errors), plural)
|
|
|
|
} else if len(report.errors) == 0 && len(report.warnings) > 0 {
|
|
var plural string
|
|
if len(report.warnings) > 1 {
|
|
plural = "s"
|
|
}
|
|
|
|
return fmt.Sprintf("%d warning%s emitted", len(report.warnings), plural)
|
|
|
|
} else if len(report.errors) > 0 && len(report.warnings) > 0 {
|
|
var eplural, wplural string
|
|
if len(report.errors) > 1 {
|
|
eplural = "s"
|
|
}
|
|
if len(report.warnings) > 1 {
|
|
wplural = "s"
|
|
}
|
|
|
|
return fmt.Sprintf("%d error%s and %d warning%s emitted", len(report.errors), eplural, len(report.warnings), wplural)
|
|
|
|
} else {
|
|
return "no errors or warnings emitted"
|
|
}
|
|
}
|
|
|
|
// HasErrors returns true if at least one error has been added to the report.
|
|
func (report *ErrorReport) HasErrors() bool {
|
|
report.mu.Lock()
|
|
result := len(report.errors) > 0
|
|
report.mu.Unlock()
|
|
return result
|
|
}
|
|
|
|
// HasWarnings returns true if at least one warning has been added to the report.
|
|
func (report *ErrorReport) HasWarnings() bool {
|
|
report.mu.Lock()
|
|
result := len(report.warnings) > 0
|
|
report.mu.Unlock()
|
|
return result
|
|
}
|
|
|
|
// Errors returns the list of error messages
|
|
// that have been added to the report.
|
|
// The order is randomised.
|
|
func (report *ErrorReport) Errors() []string {
|
|
report.mu.Lock()
|
|
defer report.mu.Unlock()
|
|
|
|
return lo.Keys(report.errors)
|
|
}
|
|
|
|
// Warnings returns the list of warning messages
|
|
// that have been added to the report.
|
|
// The order is randomised.
|
|
func (report *ErrorReport) Warnings() []string {
|
|
report.mu.Lock()
|
|
defer report.mu.Unlock()
|
|
|
|
return lo.Keys(report.warnings)
|
|
}
|
|
|
|
// Errorf formats an error message and adds it to the report.
|
|
// If not already present, the message is forwarded
|
|
// to the logger instance provided during initialisation.
|
|
func (report *ErrorReport) Errorf(format string, a ...any) {
|
|
msg := fmt.Sprintf(format, a...)
|
|
|
|
report.mu.Lock()
|
|
defer report.mu.Unlock()
|
|
|
|
present := report.errors[msg]
|
|
report.errors[msg] = true
|
|
|
|
if !present {
|
|
report.logger.Errorf(format, a...)
|
|
}
|
|
}
|
|
|
|
// Warningf formats an error message and adds it to the report.
|
|
// If not already present, the message is forwarded
|
|
// to the logger instance provided during initialisation.
|
|
func (report *ErrorReport) Warningf(format string, a ...any) {
|
|
msg := fmt.Sprintf(format, a...)
|
|
|
|
report.mu.Lock()
|
|
defer report.mu.Unlock()
|
|
|
|
present := report.warnings[msg]
|
|
report.warnings[msg] = true
|
|
|
|
if !present {
|
|
report.logger.Warningf(format, a...)
|
|
}
|
|
}
|
|
|
|
// Infof forwards the given informational message
|
|
// to the logger instance provided during initialisation.
|
|
//
|
|
// This method is here just for convenience and performs no deduplication.
|
|
func (report *ErrorReport) Infof(format string, a ...any) {
|
|
report.logger.Infof(format, a...)
|
|
}
|
|
|
|
// Debugf forwards the given informational message
|
|
// to the logger instance provided during initialisation.
|
|
//
|
|
// This method is here just for convenience and performs no deduplication.
|
|
func (report *ErrorReport) Debugf(format string, a ...any) {
|
|
report.logger.Debugf(format, a...)
|
|
}
|
|
|
|
// Statusf forwards the given status message
|
|
// to the logger instance provided during initialisation.
|
|
//
|
|
// This method is here just for convenience and performs no deduplication.
|
|
func (report *ErrorReport) Statusf(format string, a ...any) {
|
|
report.logger.Statusf(format, a...)
|
|
}
|