5
0
mirror of https://github.com/wailsapp/wails.git synced 2025-05-06 21:04:56 +08:00
wails/v3/internal/generator/errors.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

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