5
0
mirror of https://github.com/wailsapp/wails.git synced 2025-05-06 11:20:51 +08:00
wails/v3/internal/generator/generate_test.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

223 lines
5.0 KiB
Go

package generator
import (
"errors"
"fmt"
"io"
"io/fs"
"os"
"path/filepath"
"strings"
"sync"
"testing"
"github.com/google/go-cmp/cmp"
"github.com/pterm/pterm"
"github.com/wailsapp/wails/v3/internal/flags"
"github.com/wailsapp/wails/v3/internal/generator/config"
)
const testcases = "github.com/wailsapp/wails/v3/internal/generator/testcases/..."
type testParams struct {
name string
options *flags.GenerateBindingsOptions
outputDir string
want map[string]bool
}
func TestGenerator(t *testing.T) {
const (
useNamesBit = 1 << iota
useInterfacesBit
tsBit
)
// Generate configuration matrix.
tests := make([]*testParams, 1<<3)
for i := range tests {
options := &flags.GenerateBindingsOptions{
ModelsFilename: "models",
InternalFilename: "internal",
IndexFilename: "index",
UseBundledRuntime: true,
TS: i&tsBit != 0,
UseInterfaces: i&useInterfacesBit != 0,
UseNames: i&useNamesBit != 0,
}
name := configString(options)
tests[i] = &testParams{
name: name,
options: options,
outputDir: filepath.Join("testdata/output", name),
want: make(map[string]bool),
}
}
for _, test := range tests {
// Create output dir.
if err := os.MkdirAll(test.outputDir, 0777); err != nil {
t.Fatal(err)
}
// Walk output dir.
err := filepath.WalkDir(test.outputDir, func(path string, d fs.DirEntry, err error) error {
// Skip directories.
if d.IsDir() {
return nil
}
// Skip got files.
if strings.HasSuffix(d.Name(), ".got.js") || strings.HasSuffix(d.Name(), ".got.ts") {
return nil
}
// Record file.
test.want[filepath.Clean("."+path[len(test.outputDir):])] = false
return nil
})
if err != nil {
t.Fatal(err)
}
}
// Run tests.
for _, test := range tests {
t.Run(test.name, func(t *testing.T) {
generator := NewGenerator(
test.options,
outputCreator(t, test),
config.DefaultPtermLogger(nil),
)
_, err := generator.Generate(testcases)
if report := (*ErrorReport)(nil); errors.As(err, &report) {
if report.HasErrors() {
t.Error(report)
} else if report.HasWarnings() {
pterm.Warning.Println(report)
}
} else if err != nil {
t.Error(err)
}
for path, present := range test.want {
if !present {
t.Errorf("Missing output file '%s'", path)
}
}
})
}
}
// configString computes a subtest name from the given configuration.
func configString(options *flags.GenerateBindingsOptions) string {
lang := "JS"
if options.TS {
lang = "TS"
}
return fmt.Sprintf("lang=%s/UseInterfaces=%v/UseNames=%v", lang, options.UseInterfaces, options.UseNames)
}
// outputCreator returns a FileCreator that detects want/got pairs
// and schedules them for comparison.
//
// If no corresponding want file exists, it is created and reported.
func outputCreator(t *testing.T, params *testParams) config.FileCreator {
var mu sync.Mutex
return config.FileCreatorFunc(func(path string) (io.WriteCloser, error) {
path = filepath.Clean(path)
prefixedPath := filepath.Join(params.outputDir, path)
// Protect want map accesses.
mu.Lock()
defer mu.Unlock()
if seen, ok := params.want[path]; ok {
// File exists: mark as seen and compare.
if seen {
t.Errorf("Duplicate output file '%s'", path)
}
params.want[path] = true
// Open want file.
wf, err := os.Open(prefixedPath)
if err != nil {
return nil, err
}
// Create or truncate got file.
ext := filepath.Ext(prefixedPath)
gf, err := os.Create(fmt.Sprintf("%s.got%s", prefixedPath[:len(prefixedPath)-len(ext)], ext))
if err != nil {
return nil, err
}
// Initialise comparer.
return &outputComparer{t, path, wf, gf}, nil
} else {
// File does not exist: create it.
t.Errorf("Unexpected output file '%s'", path)
params.want[path] = true
if err := os.MkdirAll(filepath.Dir(prefixedPath), 0777); err != nil {
return nil, err
}
return os.Create(prefixedPath)
}
})
}
// outputComparer is a io.WriteCloser that writes to got.
//
// When Close is called, it compares want to got; if they are identical,
// it deletes got; otherwise it reports a testing error.
type outputComparer struct {
t *testing.T
path string
want *os.File
got *os.File
}
func (comparer *outputComparer) Write(data []byte) (int, error) {
return comparer.got.Write(data)
}
func (comparer *outputComparer) Close() error {
defer comparer.want.Close()
defer comparer.got.Close()
comparer.got.Seek(0, io.SeekStart)
// Read want data.
want, err := io.ReadAll(comparer.want)
if err != nil {
comparer.t.Error(err)
return nil
}
got, err := io.ReadAll(comparer.got)
if err != nil {
comparer.t.Error(err)
return nil
}
if diff := cmp.Diff(want, got); diff != "" {
comparer.t.Errorf("Output file '%s' mismatch (-want +got):\n%s", comparer.path, diff)
} else {
// On success, delete got file.
comparer.got.Close()
if err := os.Remove(comparer.got.Name()); err != nil {
comparer.t.Error(err)
}
}
return nil
}