mirror of
https://github.com/wailsapp/wails.git
synced 2025-05-04 07:29:56 +08:00

* Gather and document service API * Update changelog * Add NewServiceWithOptions * Revert static analyser change * Remove infinite loop in NewService[WithOptions] * Fix compiler warning in bindings command * Add test for NewServiceWithOptions * Update changelog * Fix service example --------- Co-authored-by: Lea Anthony <lea.anthony@gmail.com>
192 lines
5.5 KiB
Go
192 lines
5.5 KiB
Go
package generator
|
|
|
|
import (
|
|
"errors"
|
|
"fmt"
|
|
"maps"
|
|
"slices"
|
|
"sync"
|
|
|
|
"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
|
|
}
|
|
|
|
// NewErrorReport 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 slices.Collect(maps.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 slices.Collect(maps.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...)
|
|
}
|