mirror of
https://github.com/wailsapp/wails.git
synced 2025-05-02 23:51:44 +08:00
parent
95b70de01f
commit
6b38f0c68e
@ -2,14 +2,15 @@ package generate
|
|||||||
|
|
||||||
import (
|
import (
|
||||||
"fmt"
|
"fmt"
|
||||||
"github.com/leaanthony/clir"
|
|
||||||
"github.com/wailsapp/wails/v2/cmd/wails/internal"
|
|
||||||
"github.com/wailsapp/wails/v2/internal/shell"
|
|
||||||
"io"
|
"io"
|
||||||
"os"
|
"os"
|
||||||
"path/filepath"
|
"path/filepath"
|
||||||
"runtime"
|
"runtime"
|
||||||
"strings"
|
"strings"
|
||||||
|
|
||||||
|
"github.com/leaanthony/clir"
|
||||||
|
"github.com/wailsapp/wails/v2/cmd/wails/internal"
|
||||||
|
"github.com/wailsapp/wails/v2/internal/shell"
|
||||||
)
|
)
|
||||||
|
|
||||||
// AddModuleCommand adds the `module` subcommand for the `generate` command
|
// AddModuleCommand adds the `module` subcommand for the `generate` command
|
||||||
@ -43,6 +44,8 @@ func AddModuleCommand(app *clir.Cli, parent *clir.Command, w io.Writer) error {
|
|||||||
}
|
}
|
||||||
|
|
||||||
stdout, stderr, err = shell.RunCommand(cwd, filename)
|
stdout, stderr, err = shell.RunCommand(cwd, filename)
|
||||||
|
println(stdout)
|
||||||
|
println(stderr)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return fmt.Errorf("%s\n%s\n%s", stdout, stderr, err)
|
return fmt.Errorf("%s\n%s\n%s", stdout, stderr, err)
|
||||||
}
|
}
|
||||||
|
@ -21,7 +21,6 @@ require (
|
|||||||
github.com/leaanthony/gosod v1.0.3
|
github.com/leaanthony/gosod v1.0.3
|
||||||
github.com/leaanthony/idgen v1.0.0
|
github.com/leaanthony/idgen v1.0.0
|
||||||
github.com/leaanthony/slicer v1.5.0
|
github.com/leaanthony/slicer v1.5.0
|
||||||
github.com/leaanthony/typescriptify-golang-structs v0.1.7
|
|
||||||
github.com/leaanthony/winicon v1.0.0
|
github.com/leaanthony/winicon v1.0.0
|
||||||
github.com/matryer/is v1.4.0
|
github.com/matryer/is v1.4.0
|
||||||
github.com/olekukonko/tablewriter v0.0.4
|
github.com/olekukonko/tablewriter v0.0.4
|
||||||
@ -29,6 +28,7 @@ require (
|
|||||||
github.com/pkg/errors v0.9.1
|
github.com/pkg/errors v0.9.1
|
||||||
github.com/tc-hib/winres v0.1.5
|
github.com/tc-hib/winres v0.1.5
|
||||||
github.com/tidwall/sjson v1.1.7
|
github.com/tidwall/sjson v1.1.7
|
||||||
|
github.com/tkrajina/go-reflector v0.5.5
|
||||||
github.com/wailsapp/mimetype v1.4.1-beta.1.0.20220331112158-6df7e41671fe
|
github.com/wailsapp/mimetype v1.4.1-beta.1.0.20220331112158-6df7e41671fe
|
||||||
github.com/wzshiming/ctc v1.2.3
|
github.com/wzshiming/ctc v1.2.3
|
||||||
github.com/ztrue/tracerr v0.3.0
|
github.com/ztrue/tracerr v0.3.0
|
||||||
@ -58,7 +58,6 @@ require (
|
|||||||
github.com/tidwall/gjson v1.8.0 // indirect
|
github.com/tidwall/gjson v1.8.0 // indirect
|
||||||
github.com/tidwall/match v1.0.3 // indirect
|
github.com/tidwall/match v1.0.3 // indirect
|
||||||
github.com/tidwall/pretty v1.1.0 // indirect
|
github.com/tidwall/pretty v1.1.0 // indirect
|
||||||
github.com/tkrajina/go-reflector v0.5.5 // indirect
|
|
||||||
github.com/valyala/bytebufferpool v1.0.0 // indirect
|
github.com/valyala/bytebufferpool v1.0.0 // indirect
|
||||||
github.com/valyala/fasttemplate v1.2.1 // indirect
|
github.com/valyala/fasttemplate v1.2.1 // indirect
|
||||||
github.com/wzshiming/winseq v0.0.0-20200112104235-db357dc107ae // indirect
|
github.com/wzshiming/winseq v0.0.0-20200112104235-db357dc107ae // indirect
|
||||||
|
@ -82,8 +82,6 @@ github.com/leaanthony/idgen v1.0.0 h1:IZreR+JGEzFV4yeVuBZA25gM0keUoFy+RDUldncQ+J
|
|||||||
github.com/leaanthony/idgen v1.0.0/go.mod h1:4nBZnt8ml/f/ic/EVQuLxuj817RccT2fyrUaZFxrcVA=
|
github.com/leaanthony/idgen v1.0.0/go.mod h1:4nBZnt8ml/f/ic/EVQuLxuj817RccT2fyrUaZFxrcVA=
|
||||||
github.com/leaanthony/slicer v1.5.0 h1:aHYTN8xbCCLxJmkNKiLB6tgcMARl4eWmH9/F+S/0HtY=
|
github.com/leaanthony/slicer v1.5.0 h1:aHYTN8xbCCLxJmkNKiLB6tgcMARl4eWmH9/F+S/0HtY=
|
||||||
github.com/leaanthony/slicer v1.5.0/go.mod h1:FwrApmf8gOrpzEWM2J/9Lh79tyq8KTX5AzRtwV7m4AY=
|
github.com/leaanthony/slicer v1.5.0/go.mod h1:FwrApmf8gOrpzEWM2J/9Lh79tyq8KTX5AzRtwV7m4AY=
|
||||||
github.com/leaanthony/typescriptify-golang-structs v0.1.7 h1:yoznzWzyxkO/iWdlpq+aPcuJ5Y/hpjq/lmgMFmpjwl0=
|
|
||||||
github.com/leaanthony/typescriptify-golang-structs v0.1.7/go.mod h1:cWtOkiVhMF77e6phAXUcfNwYmMwCJ67Sij24lfvi9Js=
|
|
||||||
github.com/leaanthony/winicon v1.0.0 h1:ZNt5U5dY71oEoKZ97UVwJRT4e+5xo5o/ieKuHuk8NqQ=
|
github.com/leaanthony/winicon v1.0.0 h1:ZNt5U5dY71oEoKZ97UVwJRT4e+5xo5o/ieKuHuk8NqQ=
|
||||||
github.com/leaanthony/winicon v1.0.0/go.mod h1:en5xhijl92aphrJdmRPlh4NI1L6wq3gEm0LpXAPghjU=
|
github.com/leaanthony/winicon v1.0.0/go.mod h1:en5xhijl92aphrJdmRPlh4NI1L6wq3gEm0LpXAPghjU=
|
||||||
github.com/logrusorgru/aurora v0.0.0-20181002194514-a7b3b318ed4e h1:9MlwzLdW7QSDrhDjFlsEYmxpFyIoXmYRon3dt0io31k=
|
github.com/logrusorgru/aurora v0.0.0-20181002194514-a7b3b318ed4e h1:9MlwzLdW7QSDrhDjFlsEYmxpFyIoXmYRon3dt0io31k=
|
||||||
|
@ -91,28 +91,10 @@ func generateBindings(bindings *binding.Bindings) error {
|
|||||||
}
|
}
|
||||||
_ = fs.MkDirs(targetDir)
|
_ = fs.MkDirs(targetDir)
|
||||||
|
|
||||||
modelsFile := filepath.Join(targetDir, "models.ts")
|
|
||||||
err = bindings.WriteTS(modelsFile)
|
|
||||||
if err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
|
|
||||||
err = bindings.GenerateGoBindings(targetDir)
|
err = bindings.GenerateGoBindings(targetDir)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
// Write backend method wrappers
|
|
||||||
bindingsFilename := filepath.Join(targetDir, "bindings.js")
|
|
||||||
err = bindings.GenerateBackendJS(bindingsFilename)
|
|
||||||
if err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
|
|
||||||
bindingsTypes := filepath.Join(targetDir, "bindings.d.ts")
|
|
||||||
err = bindings.GenerateBackendTS(bindingsTypes)
|
|
||||||
if err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
|
|
||||||
return nil
|
return nil
|
||||||
|
|
||||||
|
@ -219,30 +219,12 @@ func generateBindings(bindings *binding.Bindings) error {
|
|||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
_ = fs.MkDirs(targetDir)
|
_ = fs.MkDirs(targetDir)
|
||||||
modelsFile := filepath.Join(targetDir, "models.ts")
|
|
||||||
err = bindings.WriteTS(modelsFile)
|
|
||||||
if err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
|
|
||||||
err = bindings.GenerateGoBindings(targetDir)
|
err = bindings.GenerateGoBindings(targetDir)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
|
|
||||||
// Write backend method wrappers
|
|
||||||
bindingsFilename := filepath.Join(targetDir, "bindings.js")
|
|
||||||
err = bindings.GenerateBackendJS(bindingsFilename)
|
|
||||||
if err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
|
|
||||||
bindingsTypes := filepath.Join(targetDir, "bindings.d.ts")
|
|
||||||
err = bindings.GenerateBackendTS(bindingsTypes)
|
|
||||||
if err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
|
|
||||||
return nil
|
return nil
|
||||||
|
|
||||||
}
|
}
|
||||||
|
@ -1,10 +0,0 @@
|
|||||||
{
|
|
||||||
"name": "go",
|
|
||||||
"version": "1.0.0",
|
|
||||||
"description": "Package to wrap your bound go methods",
|
|
||||||
"main": "bindings.js",
|
|
||||||
"types": "bindings.d.ts",
|
|
||||||
"scripts": {},
|
|
||||||
"author": "",
|
|
||||||
"license": "ISC"
|
|
||||||
}
|
|
@ -1,12 +1,17 @@
|
|||||||
package binding
|
package binding
|
||||||
|
|
||||||
import (
|
import (
|
||||||
|
"bufio"
|
||||||
|
"bytes"
|
||||||
"fmt"
|
"fmt"
|
||||||
"github.com/leaanthony/typescriptify-golang-structs/typescriptify"
|
"os"
|
||||||
|
"path/filepath"
|
||||||
"reflect"
|
"reflect"
|
||||||
"runtime"
|
"runtime"
|
||||||
"strings"
|
"strings"
|
||||||
|
|
||||||
|
"github.com/wailsapp/wails/v2/internal/typescriptify"
|
||||||
|
|
||||||
"github.com/leaanthony/slicer"
|
"github.com/leaanthony/slicer"
|
||||||
"github.com/wailsapp/wails/v2/internal/logger"
|
"github.com/wailsapp/wails/v2/internal/logger"
|
||||||
)
|
)
|
||||||
@ -16,8 +21,7 @@ type Bindings struct {
|
|||||||
logger logger.CustomLogger
|
logger logger.CustomLogger
|
||||||
exemptions slicer.StringSlicer
|
exemptions slicer.StringSlicer
|
||||||
|
|
||||||
// Typescript writer
|
structsToGenerateTS map[string]map[string]interface{}
|
||||||
converter *typescriptify.TypeScriptify
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// NewBindings returns a new Bindings object
|
// NewBindings returns a new Bindings object
|
||||||
@ -25,15 +29,9 @@ func NewBindings(logger *logger.Logger, structPointersToBind []interface{}, exem
|
|||||||
result := &Bindings{
|
result := &Bindings{
|
||||||
db: newDB(),
|
db: newDB(),
|
||||||
logger: logger.CustomLogger("Bindings"),
|
logger: logger.CustomLogger("Bindings"),
|
||||||
converter: typescriptify.New(),
|
structsToGenerateTS: make(map[string]map[string]interface{}),
|
||||||
}
|
}
|
||||||
|
|
||||||
// No backups
|
|
||||||
result.converter.WithBackupDir("")
|
|
||||||
|
|
||||||
// Hack for TS compilation error
|
|
||||||
result.converter.AddImport("export {};")
|
|
||||||
|
|
||||||
for _, exemption := range exemptions {
|
for _, exemption := range exemptions {
|
||||||
if exemptions == nil {
|
if exemptions == nil {
|
||||||
continue
|
continue
|
||||||
@ -75,10 +73,6 @@ func (b *Bindings) Add(structPtr interface{}) error {
|
|||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
func (b *Bindings) WriteTS(filename string) error {
|
|
||||||
return b.converter.ConvertToFile(filename)
|
|
||||||
}
|
|
||||||
|
|
||||||
func (b *Bindings) DB() *DB {
|
func (b *Bindings) DB() *DB {
|
||||||
return b.db
|
return b.db
|
||||||
}
|
}
|
||||||
@ -86,3 +80,47 @@ func (b *Bindings) DB() *DB {
|
|||||||
func (b *Bindings) ToJSON() (string, error) {
|
func (b *Bindings) ToJSON() (string, error) {
|
||||||
return b.db.ToJSON()
|
return b.db.ToJSON()
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func (b *Bindings) WriteModels(modelsDir string) error {
|
||||||
|
models := map[string]string{}
|
||||||
|
for packageName, structsToGenerate := range b.structsToGenerateTS {
|
||||||
|
thisPackageCode := ""
|
||||||
|
for _, structInterface := range structsToGenerate {
|
||||||
|
w := typescriptify.New()
|
||||||
|
w.WithBackupDir("")
|
||||||
|
w.Add(structInterface)
|
||||||
|
str, err := w.Convert(nil)
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
thisPackageCode += str
|
||||||
|
}
|
||||||
|
models[packageName] = thisPackageCode
|
||||||
|
}
|
||||||
|
|
||||||
|
var modelsData bytes.Buffer
|
||||||
|
for packageName, modelData := range models {
|
||||||
|
modelsData.WriteString("export namespace " + packageName + " {\n")
|
||||||
|
sc := bufio.NewScanner(strings.NewReader(modelData))
|
||||||
|
for sc.Scan() {
|
||||||
|
modelsData.WriteString("\t" + sc.Text() + "\n")
|
||||||
|
}
|
||||||
|
modelsData.WriteString("\n}\n\n")
|
||||||
|
}
|
||||||
|
|
||||||
|
filename := filepath.Join(modelsDir, "models.ts")
|
||||||
|
err := os.WriteFile(filename, modelsData.Bytes(), 0755)
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (b *Bindings) AddStructToGenerateTS(packageName string, structName string, s interface{}) {
|
||||||
|
println("Adding struct:", packageName, structName)
|
||||||
|
if b.structsToGenerateTS[packageName] == nil {
|
||||||
|
b.structsToGenerateTS[packageName] = make(map[string]interface{})
|
||||||
|
}
|
||||||
|
b.structsToGenerateTS[packageName][structName] = s
|
||||||
|
}
|
||||||
|
@ -14,7 +14,6 @@ type BoundMethod struct {
|
|||||||
Outputs []*Parameter `json:"outputs,omitempty"`
|
Outputs []*Parameter `json:"outputs,omitempty"`
|
||||||
Comments string `json:"comments,omitempty"`
|
Comments string `json:"comments,omitempty"`
|
||||||
Method reflect.Value `json:"-"`
|
Method reflect.Value `json:"-"`
|
||||||
StructNames []string `json:"structNames"`
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// InputCount returns the number of inputs this bound method has
|
// InputCount returns the number of inputs this bound method has
|
||||||
|
@ -13,9 +13,6 @@ import (
|
|||||||
"github.com/leaanthony/slicer"
|
"github.com/leaanthony/slicer"
|
||||||
)
|
)
|
||||||
|
|
||||||
//go:embed assets/package.json
|
|
||||||
var packageJSON []byte
|
|
||||||
|
|
||||||
func (b *Bindings) GenerateGoBindings(baseDir string) error {
|
func (b *Bindings) GenerateGoBindings(baseDir string) error {
|
||||||
store := b.db.store
|
store := b.db.store
|
||||||
for packageName, structs := range store {
|
for packageName, structs := range store {
|
||||||
@ -35,7 +32,7 @@ func (b *Bindings) GenerateGoBindings(baseDir string) error {
|
|||||||
tsContent.WriteString(`// Cynhyrchwyd y ffeil hon yn awtomatig. PEIDIWCH Â MODIWL
|
tsContent.WriteString(`// Cynhyrchwyd y ffeil hon yn awtomatig. PEIDIWCH Â MODIWL
|
||||||
// This file is automatically generated. DO NOT EDIT
|
// This file is automatically generated. DO NOT EDIT
|
||||||
`)
|
`)
|
||||||
var importClasses slicer.StringSlicer
|
var importNamespaces slicer.StringSlicer
|
||||||
for methodName, methodDetails := range methods {
|
for methodName, methodDetails := range methods {
|
||||||
|
|
||||||
// Generate JS
|
// Generate JS
|
||||||
@ -53,24 +50,23 @@ func (b *Bindings) GenerateGoBindings(baseDir string) error {
|
|||||||
jsoutput.WriteString("\n")
|
jsoutput.WriteString("\n")
|
||||||
|
|
||||||
// Generate TS
|
// Generate TS
|
||||||
|
|
||||||
if len(methodDetails.StructNames) > 0 {
|
|
||||||
importClasses.AddSlice(methodDetails.StructNames)
|
|
||||||
}
|
|
||||||
tsBody.WriteString(fmt.Sprintf("\nexport function %s(", methodName))
|
tsBody.WriteString(fmt.Sprintf("\nexport function %s(", methodName))
|
||||||
|
|
||||||
args.Clear()
|
args.Clear()
|
||||||
for count, input := range methodDetails.Inputs {
|
for count, input := range methodDetails.Inputs {
|
||||||
arg := fmt.Sprintf("arg%d", count+1)
|
arg := fmt.Sprintf("arg%d", count+1)
|
||||||
args.Add(arg + ":" + goTypeToTypescriptType(input.TypeName, false))
|
args.Add(arg + ":" + goTypeToTypescriptType(input.TypeName))
|
||||||
|
if strings.ContainsRune(input.TypeName, '.') {
|
||||||
|
importNamespaces.Add(strings.Split(input.TypeName, ".")[0])
|
||||||
|
}
|
||||||
}
|
}
|
||||||
tsBody.WriteString(args.Join(",") + "):")
|
tsBody.WriteString(args.Join(",") + "):")
|
||||||
returnType := "Promise"
|
returnType := "Promise"
|
||||||
if methodDetails.OutputCount() > 0 {
|
if methodDetails.OutputCount() > 0 {
|
||||||
firstType := goTypeToTypescriptType(methodDetails.Outputs[0].TypeName, false)
|
firstType := goTypeToTypescriptType(methodDetails.Outputs[0].TypeName)
|
||||||
returnType += "<" + firstType
|
returnType += "<" + firstType
|
||||||
if methodDetails.OutputCount() == 2 {
|
if methodDetails.OutputCount() == 2 {
|
||||||
secondType := goTypeToTypescriptType(methodDetails.Outputs[1].TypeName, false)
|
secondType := goTypeToTypescriptType(methodDetails.Outputs[1].TypeName)
|
||||||
returnType += "|" + secondType
|
returnType += "|" + secondType
|
||||||
}
|
}
|
||||||
returnType += ">"
|
returnType += ">"
|
||||||
@ -80,9 +76,9 @@ func (b *Bindings) GenerateGoBindings(baseDir string) error {
|
|||||||
tsBody.WriteString(returnType + ";\n")
|
tsBody.WriteString(returnType + ";\n")
|
||||||
}
|
}
|
||||||
|
|
||||||
importClasses.Deduplicate()
|
importNamespaces.Deduplicate()
|
||||||
importClasses.Each(func(class string) {
|
importNamespaces.Each(func(namespace string) {
|
||||||
tsContent.WriteString("import {" + class + "} from '../models';\n")
|
tsContent.WriteString("import {" + namespace + "} from '../models';\n")
|
||||||
})
|
})
|
||||||
tsContent.WriteString(tsBody.String())
|
tsContent.WriteString(tsBody.String())
|
||||||
|
|
||||||
@ -98,202 +94,15 @@ func (b *Bindings) GenerateGoBindings(baseDir string) error {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
err := b.WriteModels(baseDir)
|
||||||
|
if err != nil {
|
||||||
|
println(err)
|
||||||
|
return err
|
||||||
|
}
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
func (b *Bindings) GenerateBackendJS(targetfile string) error {
|
func goTypeToJSDocType(input string) string {
|
||||||
|
|
||||||
store := b.db.store
|
|
||||||
var output bytes.Buffer
|
|
||||||
|
|
||||||
output.WriteString(`// @ts-check
|
|
||||||
// Cynhyrchwyd y ffeil hon yn awtomatig. PEIDIWCH Â MODIWL
|
|
||||||
// This file is automatically generated. DO NOT EDIT
|
|
||||||
|
|
||||||
// ************************************************
|
|
||||||
// This file is deprecated and will not be generated
|
|
||||||
// in the next version of Wails. Bindings are now
|
|
||||||
// generated in their own files.
|
|
||||||
// ************************************************
|
|
||||||
|
|
||||||
`)
|
|
||||||
|
|
||||||
output.WriteString(`const go = {`)
|
|
||||||
output.WriteString("\n")
|
|
||||||
|
|
||||||
var sortedPackageNames slicer.StringSlicer
|
|
||||||
for packageName := range store {
|
|
||||||
sortedPackageNames.Add(packageName)
|
|
||||||
}
|
|
||||||
sortedPackageNames.Sort()
|
|
||||||
sortedPackageNames.Each(func(packageName string) {
|
|
||||||
packages := store[packageName]
|
|
||||||
output.WriteString(fmt.Sprintf(" \"%s\": {", packageName))
|
|
||||||
output.WriteString("\n")
|
|
||||||
var sortedStructNames slicer.StringSlicer
|
|
||||||
for structName := range packages {
|
|
||||||
sortedStructNames.Add(structName)
|
|
||||||
}
|
|
||||||
sortedStructNames.Sort()
|
|
||||||
|
|
||||||
sortedStructNames.Each(func(structName string) {
|
|
||||||
structs := packages[structName]
|
|
||||||
output.WriteString(fmt.Sprintf(" \"%s\": {", structName))
|
|
||||||
output.WriteString("\n")
|
|
||||||
|
|
||||||
var sortedMethodNames slicer.StringSlicer
|
|
||||||
for methodName := range structs {
|
|
||||||
sortedMethodNames.Add(methodName)
|
|
||||||
}
|
|
||||||
sortedMethodNames.Sort()
|
|
||||||
|
|
||||||
sortedMethodNames.Each(func(methodName string) {
|
|
||||||
methodDetails := structs[methodName]
|
|
||||||
output.WriteString(" /**\n")
|
|
||||||
output.WriteString(" * " + methodName + "\n")
|
|
||||||
var args slicer.StringSlicer
|
|
||||||
for count, input := range methodDetails.Inputs {
|
|
||||||
arg := fmt.Sprintf("arg%d", count+1)
|
|
||||||
args.Add(arg)
|
|
||||||
output.WriteString(fmt.Sprintf(" * @param {%s} %s - Go Type: %s\n", goTypeToJSDocType(input.TypeName, true), arg, input.TypeName))
|
|
||||||
}
|
|
||||||
returnType := "Promise"
|
|
||||||
returnTypeDetails := ""
|
|
||||||
if methodDetails.OutputCount() > 0 {
|
|
||||||
firstType := goTypeToJSDocType(methodDetails.Outputs[0].TypeName, true)
|
|
||||||
returnType += "<" + firstType
|
|
||||||
if methodDetails.OutputCount() == 2 {
|
|
||||||
secondType := goTypeToJSDocType(methodDetails.Outputs[1].TypeName, true)
|
|
||||||
returnType += "|" + secondType
|
|
||||||
}
|
|
||||||
returnType += ">"
|
|
||||||
returnTypeDetails = " - Go Type: " + methodDetails.Outputs[0].TypeName
|
|
||||||
} else {
|
|
||||||
returnType = "Promise<void>"
|
|
||||||
}
|
|
||||||
output.WriteString(" * @returns {" + returnType + "} " + returnTypeDetails + "\n")
|
|
||||||
output.WriteString(" */\n")
|
|
||||||
argsString := args.Join(", ")
|
|
||||||
output.WriteString(fmt.Sprintf(" \"%s\": (%s) => {", methodName, argsString))
|
|
||||||
output.WriteString("\n")
|
|
||||||
output.WriteString(fmt.Sprintf(" return window.go.%s.%s.%s(%s);", packageName, structName, methodName, argsString))
|
|
||||||
output.WriteString("\n")
|
|
||||||
output.WriteString(fmt.Sprintf(" },"))
|
|
||||||
output.WriteString("\n")
|
|
||||||
|
|
||||||
})
|
|
||||||
|
|
||||||
output.WriteString(" },\n")
|
|
||||||
})
|
|
||||||
|
|
||||||
output.WriteString(" },\n\n")
|
|
||||||
})
|
|
||||||
|
|
||||||
output.WriteString(`};
|
|
||||||
export default go;`)
|
|
||||||
output.WriteString("\n")
|
|
||||||
|
|
||||||
dir := filepath.Dir(targetfile)
|
|
||||||
packageJsonFile := filepath.Join(dir, "package.json")
|
|
||||||
if !fs.FileExists(packageJsonFile) {
|
|
||||||
err := os.WriteFile(packageJsonFile, packageJSON, 0755)
|
|
||||||
if err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
return os.WriteFile(targetfile, output.Bytes(), 0755)
|
|
||||||
}
|
|
||||||
|
|
||||||
// GenerateBackendTS generates typescript bindings for
|
|
||||||
// the bound methods.
|
|
||||||
func (b *Bindings) GenerateBackendTS(targetfile string) error {
|
|
||||||
|
|
||||||
store := b.db.store
|
|
||||||
var output bytes.Buffer
|
|
||||||
|
|
||||||
output.WriteString(`
|
|
||||||
|
|
||||||
// ************************************************
|
|
||||||
// This file is deprecated and will not be generated
|
|
||||||
// in the next version of Wails. Bindings are now
|
|
||||||
// generated in their own files.
|
|
||||||
// ************************************************
|
|
||||||
|
|
||||||
`)
|
|
||||||
|
|
||||||
output.WriteString("import * as models from './models';\n\n")
|
|
||||||
output.WriteString("export interface go {\n")
|
|
||||||
|
|
||||||
var sortedPackageNames slicer.StringSlicer
|
|
||||||
for packageName := range store {
|
|
||||||
sortedPackageNames.Add(packageName)
|
|
||||||
}
|
|
||||||
sortedPackageNames.Sort()
|
|
||||||
sortedPackageNames.Each(func(packageName string) {
|
|
||||||
packages := store[packageName]
|
|
||||||
output.WriteString(fmt.Sprintf(" \"%s\": {", packageName))
|
|
||||||
output.WriteString("\n")
|
|
||||||
var sortedStructNames slicer.StringSlicer
|
|
||||||
for structName := range packages {
|
|
||||||
sortedStructNames.Add(structName)
|
|
||||||
}
|
|
||||||
sortedStructNames.Sort()
|
|
||||||
|
|
||||||
sortedStructNames.Each(func(structName string) {
|
|
||||||
structs := packages[structName]
|
|
||||||
output.WriteString(fmt.Sprintf(" \"%s\": {", structName))
|
|
||||||
output.WriteString("\n")
|
|
||||||
|
|
||||||
var sortedMethodNames slicer.StringSlicer
|
|
||||||
for methodName := range structs {
|
|
||||||
sortedMethodNames.Add(methodName)
|
|
||||||
}
|
|
||||||
sortedMethodNames.Sort()
|
|
||||||
|
|
||||||
sortedMethodNames.Each(func(methodName string) {
|
|
||||||
methodDetails := structs[methodName]
|
|
||||||
output.WriteString(fmt.Sprintf("\t\t%s(", methodName))
|
|
||||||
|
|
||||||
var args slicer.StringSlicer
|
|
||||||
for count, input := range methodDetails.Inputs {
|
|
||||||
arg := fmt.Sprintf("arg%d", count+1)
|
|
||||||
args.Add(arg + ":" + goTypeToTypescriptType(input.TypeName, true))
|
|
||||||
}
|
|
||||||
output.WriteString(args.Join(",") + "):")
|
|
||||||
returnType := "Promise"
|
|
||||||
if methodDetails.OutputCount() > 0 {
|
|
||||||
firstType := goTypeToTypescriptType(methodDetails.Outputs[0].TypeName, true)
|
|
||||||
returnType += "<" + firstType
|
|
||||||
if methodDetails.OutputCount() == 2 {
|
|
||||||
secondType := goTypeToTypescriptType(methodDetails.Outputs[1].TypeName, true)
|
|
||||||
returnType += "|" + secondType
|
|
||||||
}
|
|
||||||
returnType += ">"
|
|
||||||
} else {
|
|
||||||
returnType = "Promise<void>"
|
|
||||||
}
|
|
||||||
output.WriteString(returnType + "\n")
|
|
||||||
})
|
|
||||||
|
|
||||||
output.WriteString(" },\n")
|
|
||||||
})
|
|
||||||
output.WriteString(" }\n\n")
|
|
||||||
})
|
|
||||||
output.WriteString("}\n")
|
|
||||||
|
|
||||||
globals := `
|
|
||||||
declare global {
|
|
||||||
interface Window {
|
|
||||||
go: go;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
`
|
|
||||||
output.WriteString(globals)
|
|
||||||
return os.WriteFile(targetfile, output.Bytes(), 0755)
|
|
||||||
}
|
|
||||||
|
|
||||||
func goTypeToJSDocType(input string, useModelsNamespace bool) string {
|
|
||||||
switch true {
|
switch true {
|
||||||
case input == "interface{}":
|
case input == "interface{}":
|
||||||
return "any"
|
return "any"
|
||||||
@ -311,23 +120,20 @@ func goTypeToJSDocType(input string, useModelsNamespace bool) string {
|
|||||||
case input == "[]byte":
|
case input == "[]byte":
|
||||||
return "string"
|
return "string"
|
||||||
case strings.HasPrefix(input, "[]"):
|
case strings.HasPrefix(input, "[]"):
|
||||||
arrayType := goTypeToJSDocType(input[2:], useModelsNamespace)
|
arrayType := goTypeToJSDocType(input[2:])
|
||||||
return "Array<" + arrayType + ">"
|
return "Array<" + arrayType + ">"
|
||||||
default:
|
default:
|
||||||
if strings.ContainsRune(input, '.') {
|
if strings.ContainsRune(input, '.') {
|
||||||
if useModelsNamespace {
|
return input
|
||||||
return "models." + strings.Split(input, ".")[1]
|
|
||||||
}
|
|
||||||
return strings.Split(input, ".")[1]
|
|
||||||
}
|
}
|
||||||
return "any"
|
return "any"
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
func goTypeToTypescriptType(input string, useModelsNamespace bool) string {
|
func goTypeToTypescriptType(input string) string {
|
||||||
if strings.HasPrefix(input, "[]") {
|
if strings.HasPrefix(input, "[]") {
|
||||||
arrayType := goTypeToJSDocType(input[2:], useModelsNamespace)
|
arrayType := goTypeToJSDocType(input[2:])
|
||||||
return "Array<" + arrayType + ">"
|
return "Array<" + arrayType + ">"
|
||||||
}
|
}
|
||||||
return goTypeToJSDocType(input, useModelsNamespace)
|
return goTypeToJSDocType(input)
|
||||||
}
|
}
|
||||||
|
@ -4,6 +4,7 @@ import (
|
|||||||
"fmt"
|
"fmt"
|
||||||
"reflect"
|
"reflect"
|
||||||
"runtime"
|
"runtime"
|
||||||
|
"strings"
|
||||||
)
|
)
|
||||||
|
|
||||||
// isStructPtr returns true if the value given is a
|
// isStructPtr returns true if the value given is a
|
||||||
@ -47,7 +48,9 @@ func (b *Bindings) getMethods(value interface{}) ([]*BoundMethod, error) {
|
|||||||
// Process Struct
|
// Process Struct
|
||||||
structType := reflect.TypeOf(value)
|
structType := reflect.TypeOf(value)
|
||||||
structValue := reflect.ValueOf(value)
|
structValue := reflect.ValueOf(value)
|
||||||
baseName := structType.String()[1:]
|
structTypeString := structType.String()
|
||||||
|
baseName := structTypeString[1:]
|
||||||
|
packageName := strings.Split(baseName, ".")[0]
|
||||||
|
|
||||||
// Process Methods
|
// Process Methods
|
||||||
for i := 0; i < structType.NumMethod(); i++ {
|
for i := 0; i < structType.NumMethod(); i++ {
|
||||||
@ -90,8 +93,8 @@ func (b *Bindings) getMethods(value interface{}) ([]*BoundMethod, error) {
|
|||||||
typ := thisInput.Elem()
|
typ := thisInput.Elem()
|
||||||
a := reflect.New(typ)
|
a := reflect.New(typ)
|
||||||
s := reflect.Indirect(a).Interface()
|
s := reflect.Indirect(a).Interface()
|
||||||
b.converter.Add(s)
|
name := typ.Name()
|
||||||
boundMethod.StructNames = append(boundMethod.StructNames, typ.Name())
|
b.AddStructToGenerateTS(packageName, name, s)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -99,8 +102,8 @@ func (b *Bindings) getMethods(value interface{}) ([]*BoundMethod, error) {
|
|||||||
if thisInput.Kind() == reflect.Struct {
|
if thisInput.Kind() == reflect.Struct {
|
||||||
a := reflect.New(thisInput)
|
a := reflect.New(thisInput)
|
||||||
s := reflect.Indirect(a).Interface()
|
s := reflect.Indirect(a).Interface()
|
||||||
b.converter.Add(s)
|
name := thisInput.Name()
|
||||||
boundMethod.StructNames = append(boundMethod.StructNames, thisInput.Name())
|
b.AddStructToGenerateTS(packageName, name, s)
|
||||||
}
|
}
|
||||||
|
|
||||||
inputs = append(inputs, thisParam)
|
inputs = append(inputs, thisParam)
|
||||||
@ -129,8 +132,8 @@ func (b *Bindings) getMethods(value interface{}) ([]*BoundMethod, error) {
|
|||||||
typ := thisOutput.Elem()
|
typ := thisOutput.Elem()
|
||||||
a := reflect.New(typ)
|
a := reflect.New(typ)
|
||||||
s := reflect.Indirect(a).Interface()
|
s := reflect.Indirect(a).Interface()
|
||||||
b.converter.Add(s)
|
name := typ.Name()
|
||||||
boundMethod.StructNames = append(boundMethod.StructNames, typ.Name())
|
b.AddStructToGenerateTS(packageName, name, s)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -138,8 +141,8 @@ func (b *Bindings) getMethods(value interface{}) ([]*BoundMethod, error) {
|
|||||||
if thisOutput.Kind() == reflect.Struct {
|
if thisOutput.Kind() == reflect.Struct {
|
||||||
a := reflect.New(thisOutput)
|
a := reflect.New(thisOutput)
|
||||||
s := reflect.Indirect(a).Interface()
|
s := reflect.Indirect(a).Interface()
|
||||||
b.converter.Add(s)
|
name := thisOutput.Name()
|
||||||
boundMethod.StructNames = append(boundMethod.StructNames, thisOutput.Name())
|
b.AddStructToGenerateTS(packageName, name, s)
|
||||||
}
|
}
|
||||||
|
|
||||||
outputs = append(outputs, thisParam)
|
outputs = append(outputs, thisParam)
|
||||||
|
202
v2/internal/typescriptify/LICENSE.txt
Normal file
202
v2/internal/typescriptify/LICENSE.txt
Normal file
@ -0,0 +1,202 @@
|
|||||||
|
|
||||||
|
Apache License
|
||||||
|
Version 2.0, January 2004
|
||||||
|
http://www.apache.org/licenses/
|
||||||
|
|
||||||
|
TERMS AND CONDITIONS FOR USE, REPRODUCTION, AND DISTRIBUTION
|
||||||
|
|
||||||
|
1. Definitions.
|
||||||
|
|
||||||
|
"License" shall mean the terms and conditions for use, reproduction,
|
||||||
|
and distribution as defined by Sections 1 through 9 of this document.
|
||||||
|
|
||||||
|
"Licensor" shall mean the copyright owner or entity authorized by
|
||||||
|
the copyright owner that is granting the License.
|
||||||
|
|
||||||
|
"Legal Entity" shall mean the union of the acting entity and all
|
||||||
|
other entities that control, are controlled by, or are under common
|
||||||
|
control with that entity. For the purposes of this definition,
|
||||||
|
"control" means (i) the power, direct or indirect, to cause the
|
||||||
|
direction or management of such entity, whether by contract or
|
||||||
|
otherwise, or (ii) ownership of fifty percent (50%) or more of the
|
||||||
|
outstanding shares, or (iii) beneficial ownership of such entity.
|
||||||
|
|
||||||
|
"You" (or "Your") shall mean an individual or Legal Entity
|
||||||
|
exercising permissions granted by this License.
|
||||||
|
|
||||||
|
"Source" form shall mean the preferred form for making modifications,
|
||||||
|
including but not limited to software source code, documentation
|
||||||
|
source, and configuration files.
|
||||||
|
|
||||||
|
"Object" form shall mean any form resulting from mechanical
|
||||||
|
transformation or translation of a Source form, including but
|
||||||
|
not limited to compiled object code, generated documentation,
|
||||||
|
and conversions to other media types.
|
||||||
|
|
||||||
|
"Work" shall mean the work of authorship, whether in Source or
|
||||||
|
Object form, made available under the License, as indicated by a
|
||||||
|
copyright notice that is included in or attached to the work
|
||||||
|
(an example is provided in the Appendix below).
|
||||||
|
|
||||||
|
"Derivative Works" shall mean any work, whether in Source or Object
|
||||||
|
form, that is based on (or derived from) the Work and for which the
|
||||||
|
editorial revisions, annotations, elaborations, or other modifications
|
||||||
|
represent, as a whole, an original work of authorship. For the purposes
|
||||||
|
of this License, Derivative Works shall not include works that remain
|
||||||
|
separable from, or merely link (or bind by name) to the interfaces of,
|
||||||
|
the Work and Derivative Works thereof.
|
||||||
|
|
||||||
|
"Contribution" shall mean any work of authorship, including
|
||||||
|
the original version of the Work and any modifications or additions
|
||||||
|
to that Work or Derivative Works thereof, that is intentionally
|
||||||
|
submitted to Licensor for inclusion in the Work by the copyright owner
|
||||||
|
or by an individual or Legal Entity authorized to submit on behalf of
|
||||||
|
the copyright owner. For the purposes of this definition, "submitted"
|
||||||
|
means any form of electronic, verbal, or written communication sent
|
||||||
|
to the Licensor or its representatives, including but not limited to
|
||||||
|
communication on electronic mailing lists, source code control systems,
|
||||||
|
and issue tracking systems that are managed by, or on behalf of, the
|
||||||
|
Licensor for the purpose of discussing and improving the Work, but
|
||||||
|
excluding communication that is conspicuously marked or otherwise
|
||||||
|
designated in writing by the copyright owner as "Not a Contribution."
|
||||||
|
|
||||||
|
"Contributor" shall mean Licensor and any individual or Legal Entity
|
||||||
|
on behalf of whom a Contribution has been received by Licensor and
|
||||||
|
subsequently incorporated within the Work.
|
||||||
|
|
||||||
|
2. Grant of Copyright License. Subject to the terms and conditions of
|
||||||
|
this License, each Contributor hereby grants to You a perpetual,
|
||||||
|
worldwide, non-exclusive, no-charge, royalty-free, irrevocable
|
||||||
|
copyright license to reproduce, prepare Derivative Works of,
|
||||||
|
publicly display, publicly perform, sublicense, and distribute the
|
||||||
|
Work and such Derivative Works in Source or Object form.
|
||||||
|
|
||||||
|
3. Grant of Patent License. Subject to the terms and conditions of
|
||||||
|
this License, each Contributor hereby grants to You a perpetual,
|
||||||
|
worldwide, non-exclusive, no-charge, royalty-free, irrevocable
|
||||||
|
(except as stated in this section) patent license to make, have made,
|
||||||
|
use, offer to sell, sell, import, and otherwise transfer the Work,
|
||||||
|
where such license applies only to those patent claims licensable
|
||||||
|
by such Contributor that are necessarily infringed by their
|
||||||
|
Contribution(s) alone or by combination of their Contribution(s)
|
||||||
|
with the Work to which such Contribution(s) was submitted. If You
|
||||||
|
institute patent litigation against any entity (including a
|
||||||
|
cross-claim or counterclaim in a lawsuit) alleging that the Work
|
||||||
|
or a Contribution incorporated within the Work constitutes direct
|
||||||
|
or contributory patent infringement, then any patent licenses
|
||||||
|
granted to You under this License for that Work shall terminate
|
||||||
|
as of the date such litigation is filed.
|
||||||
|
|
||||||
|
4. Redistribution. You may reproduce and distribute copies of the
|
||||||
|
Work or Derivative Works thereof in any medium, with or without
|
||||||
|
modifications, and in Source or Object form, provided that You
|
||||||
|
meet the following conditions:
|
||||||
|
|
||||||
|
(a) You must give any other recipients of the Work or
|
||||||
|
Derivative Works a copy of this License; and
|
||||||
|
|
||||||
|
(b) You must cause any modified files to carry prominent notices
|
||||||
|
stating that You changed the files; and
|
||||||
|
|
||||||
|
(c) You must retain, in the Source form of any Derivative Works
|
||||||
|
that You distribute, all copyright, patent, trademark, and
|
||||||
|
attribution notices from the Source form of the Work,
|
||||||
|
excluding those notices that do not pertain to any part of
|
||||||
|
the Derivative Works; and
|
||||||
|
|
||||||
|
(d) If the Work includes a "NOTICE" text file as part of its
|
||||||
|
distribution, then any Derivative Works that You distribute must
|
||||||
|
include a readable copy of the attribution notices contained
|
||||||
|
within such NOTICE file, excluding those notices that do not
|
||||||
|
pertain to any part of the Derivative Works, in at least one
|
||||||
|
of the following places: within a NOTICE text file distributed
|
||||||
|
as part of the Derivative Works; within the Source form or
|
||||||
|
documentation, if provided along with the Derivative Works; or,
|
||||||
|
within a display generated by the Derivative Works, if and
|
||||||
|
wherever such third-party notices normally appear. The contents
|
||||||
|
of the NOTICE file are for informational purposes only and
|
||||||
|
do not modify the License. You may add Your own attribution
|
||||||
|
notices within Derivative Works that You distribute, alongside
|
||||||
|
or as an addendum to the NOTICE text from the Work, provided
|
||||||
|
that such additional attribution notices cannot be construed
|
||||||
|
as modifying the License.
|
||||||
|
|
||||||
|
You may add Your own copyright statement to Your modifications and
|
||||||
|
may provide additional or different license terms and conditions
|
||||||
|
for use, reproduction, or distribution of Your modifications, or
|
||||||
|
for any such Derivative Works as a whole, provided Your use,
|
||||||
|
reproduction, and distribution of the Work otherwise complies with
|
||||||
|
the conditions stated in this License.
|
||||||
|
|
||||||
|
5. Submission of Contributions. Unless You explicitly state otherwise,
|
||||||
|
any Contribution intentionally submitted for inclusion in the Work
|
||||||
|
by You to the Licensor shall be under the terms and conditions of
|
||||||
|
this License, without any additional terms or conditions.
|
||||||
|
Notwithstanding the above, nothing herein shall supersede or modify
|
||||||
|
the terms of any separate license agreement you may have executed
|
||||||
|
with Licensor regarding such Contributions.
|
||||||
|
|
||||||
|
6. Trademarks. This License does not grant permission to use the trade
|
||||||
|
names, trademarks, service marks, or product names of the Licensor,
|
||||||
|
except as required for reasonable and customary use in describing the
|
||||||
|
origin of the Work and reproducing the content of the NOTICE file.
|
||||||
|
|
||||||
|
7. Disclaimer of Warranty. Unless required by applicable law or
|
||||||
|
agreed to in writing, Licensor provides the Work (and each
|
||||||
|
Contributor provides its Contributions) on an "AS IS" BASIS,
|
||||||
|
WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or
|
||||||
|
implied, including, without limitation, any warranties or conditions
|
||||||
|
of TITLE, NON-INFRINGEMENT, MERCHANTABILITY, or FITNESS FOR A
|
||||||
|
PARTICULAR PURPOSE. You are solely responsible for determining the
|
||||||
|
appropriateness of using or redistributing the Work and assume any
|
||||||
|
risks associated with Your exercise of permissions under this License.
|
||||||
|
|
||||||
|
8. Limitation of Liability. In no event and under no legal theory,
|
||||||
|
whether in tort (including negligence), contract, or otherwise,
|
||||||
|
unless required by applicable law (such as deliberate and grossly
|
||||||
|
negligent acts) or agreed to in writing, shall any Contributor be
|
||||||
|
liable to You for damages, including any direct, indirect, special,
|
||||||
|
incidental, or consequential damages of any character arising as a
|
||||||
|
result of this License or out of the use or inability to use the
|
||||||
|
Work (including but not limited to damages for loss of goodwill,
|
||||||
|
work stoppage, computer failure or malfunction, or any and all
|
||||||
|
other commercial damages or losses), even if such Contributor
|
||||||
|
has been advised of the possibility of such damages.
|
||||||
|
|
||||||
|
9. Accepting Warranty or Additional Liability. While redistributing
|
||||||
|
the Work or Derivative Works thereof, You may choose to offer,
|
||||||
|
and charge a fee for, acceptance of support, warranty, indemnity,
|
||||||
|
or other liability obligations and/or rights consistent with this
|
||||||
|
License. However, in accepting such obligations, You may act only
|
||||||
|
on Your own behalf and on Your sole responsibility, not on behalf
|
||||||
|
of any other Contributor, and only if You agree to indemnify,
|
||||||
|
defend, and hold each Contributor harmless for any liability
|
||||||
|
incurred by, or claims asserted against, such Contributor by reason
|
||||||
|
of your accepting any such warranty or additional liability.
|
||||||
|
|
||||||
|
END OF TERMS AND CONDITIONS
|
||||||
|
|
||||||
|
APPENDIX: How to apply the Apache License to your work.
|
||||||
|
|
||||||
|
To apply the Apache License to your work, attach the following
|
||||||
|
boilerplate notice, with the fields enclosed by brackets "[]"
|
||||||
|
replaced with your own identifying information. (Don't include
|
||||||
|
the brackets!) The text should be enclosed in the appropriate
|
||||||
|
comment syntax for the file format. We also recommend that a
|
||||||
|
file or class name and description of purpose be included on the
|
||||||
|
same "printed page" as the copyright notice for easier
|
||||||
|
identification within third-party archives.
|
||||||
|
|
||||||
|
Copyright [2015-] [Tomo Krajina]
|
||||||
|
|
||||||
|
Licensed under the Apache License, Version 2.0 (the "License");
|
||||||
|
you may not use this file except in compliance with the License.
|
||||||
|
You may obtain a copy of the License at
|
||||||
|
|
||||||
|
http://www.apache.org/licenses/LICENSE-2.0
|
||||||
|
|
||||||
|
Unless required by applicable law or agreed to in writing, software
|
||||||
|
distributed under the License is distributed on an "AS IS" BASIS,
|
||||||
|
WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||||
|
See the License for the specific language governing permissions and
|
||||||
|
limitations under the License.
|
2
v2/internal/typescriptify/README.md
Normal file
2
v2/internal/typescriptify/README.md
Normal file
@ -0,0 +1,2 @@
|
|||||||
|
Based on: https://github.com/tkrajina/typescriptify-golang-structs
|
||||||
|
License: LICENSE.txt
|
816
v2/internal/typescriptify/typescriptify.go
Normal file
816
v2/internal/typescriptify/typescriptify.go
Normal file
@ -0,0 +1,816 @@
|
|||||||
|
package typescriptify
|
||||||
|
|
||||||
|
import (
|
||||||
|
"bufio"
|
||||||
|
"fmt"
|
||||||
|
"io/ioutil"
|
||||||
|
"os"
|
||||||
|
"path"
|
||||||
|
"reflect"
|
||||||
|
"strings"
|
||||||
|
"time"
|
||||||
|
|
||||||
|
"github.com/tkrajina/go-reflector/reflector"
|
||||||
|
)
|
||||||
|
|
||||||
|
const (
|
||||||
|
tsTransformTag = "ts_transform"
|
||||||
|
tsType = "ts_type"
|
||||||
|
tsConvertValuesFunc = `convertValues(a: any, classs: any, asMap: boolean = false): any {
|
||||||
|
if (!a) {
|
||||||
|
return a;
|
||||||
|
}
|
||||||
|
if (a.slice) {
|
||||||
|
return (a as any[]).map(elem => this.convertValues(elem, classs));
|
||||||
|
} else if ("object" === typeof a) {
|
||||||
|
if (asMap) {
|
||||||
|
for (const key of Object.keys(a)) {
|
||||||
|
a[key] = new classs(a[key]);
|
||||||
|
}
|
||||||
|
return a;
|
||||||
|
}
|
||||||
|
return new classs(a);
|
||||||
|
}
|
||||||
|
return a;
|
||||||
|
}`
|
||||||
|
)
|
||||||
|
|
||||||
|
// TypeOptions overrides options set by `ts_*` tags.
|
||||||
|
type TypeOptions struct {
|
||||||
|
TSType string
|
||||||
|
TSTransform string
|
||||||
|
}
|
||||||
|
|
||||||
|
// StructType stores settings for transforming one Golang struct.
|
||||||
|
type StructType struct {
|
||||||
|
Type reflect.Type
|
||||||
|
FieldOptions map[reflect.Type]TypeOptions
|
||||||
|
}
|
||||||
|
|
||||||
|
func NewStruct(i interface{}) *StructType {
|
||||||
|
return &StructType{
|
||||||
|
Type: reflect.TypeOf(i),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func (st *StructType) WithFieldOpts(i interface{}, opts TypeOptions) *StructType {
|
||||||
|
if st.FieldOptions == nil {
|
||||||
|
st.FieldOptions = map[reflect.Type]TypeOptions{}
|
||||||
|
}
|
||||||
|
var typ reflect.Type
|
||||||
|
if ty, is := i.(reflect.Type); is {
|
||||||
|
typ = ty
|
||||||
|
} else {
|
||||||
|
typ = reflect.TypeOf(i)
|
||||||
|
}
|
||||||
|
st.FieldOptions[typ] = opts
|
||||||
|
return st
|
||||||
|
}
|
||||||
|
|
||||||
|
type EnumType struct {
|
||||||
|
Type reflect.Type
|
||||||
|
}
|
||||||
|
|
||||||
|
type enumElement struct {
|
||||||
|
value interface{}
|
||||||
|
name string
|
||||||
|
}
|
||||||
|
|
||||||
|
type TypeScriptify struct {
|
||||||
|
Prefix string
|
||||||
|
Suffix string
|
||||||
|
Indent string
|
||||||
|
CreateFromMethod bool
|
||||||
|
CreateConstructor bool
|
||||||
|
BackupDir string // If empty no backup
|
||||||
|
DontExport bool
|
||||||
|
CreateInterface bool
|
||||||
|
customImports []string
|
||||||
|
|
||||||
|
structTypes []StructType
|
||||||
|
enumTypes []EnumType
|
||||||
|
enums map[reflect.Type][]enumElement
|
||||||
|
kinds map[reflect.Kind]string
|
||||||
|
|
||||||
|
fieldTypeOptions map[reflect.Type]TypeOptions
|
||||||
|
|
||||||
|
// throwaway, used when converting
|
||||||
|
alreadyConverted map[reflect.Type]bool
|
||||||
|
}
|
||||||
|
|
||||||
|
func New() *TypeScriptify {
|
||||||
|
result := new(TypeScriptify)
|
||||||
|
result.Indent = "\t"
|
||||||
|
result.BackupDir = "."
|
||||||
|
|
||||||
|
kinds := make(map[reflect.Kind]string)
|
||||||
|
|
||||||
|
kinds[reflect.Bool] = "boolean"
|
||||||
|
kinds[reflect.Interface] = "any"
|
||||||
|
|
||||||
|
kinds[reflect.Int] = "number"
|
||||||
|
kinds[reflect.Int8] = "number"
|
||||||
|
kinds[reflect.Int16] = "number"
|
||||||
|
kinds[reflect.Int32] = "number"
|
||||||
|
kinds[reflect.Int64] = "number"
|
||||||
|
kinds[reflect.Uint] = "number"
|
||||||
|
kinds[reflect.Uint8] = "number"
|
||||||
|
kinds[reflect.Uint16] = "number"
|
||||||
|
kinds[reflect.Uint32] = "number"
|
||||||
|
kinds[reflect.Uint64] = "number"
|
||||||
|
kinds[reflect.Float32] = "number"
|
||||||
|
kinds[reflect.Float64] = "number"
|
||||||
|
|
||||||
|
kinds[reflect.String] = "string"
|
||||||
|
|
||||||
|
result.kinds = kinds
|
||||||
|
|
||||||
|
result.Indent = " "
|
||||||
|
result.CreateFromMethod = true
|
||||||
|
result.CreateConstructor = true
|
||||||
|
|
||||||
|
// if result.CreateFromMethod {
|
||||||
|
// fmt.Fprintln(os.Stderr, "FromMethod METHOD IS DEPRECATED AND WILL BE REMOVED!!!!!!")
|
||||||
|
// }
|
||||||
|
return result
|
||||||
|
}
|
||||||
|
|
||||||
|
func deepFields(typeOf reflect.Type) []reflect.StructField {
|
||||||
|
fields := make([]reflect.StructField, 0)
|
||||||
|
|
||||||
|
if typeOf.Kind() == reflect.Ptr {
|
||||||
|
typeOf = typeOf.Elem()
|
||||||
|
}
|
||||||
|
|
||||||
|
if typeOf.Kind() != reflect.Struct {
|
||||||
|
return fields
|
||||||
|
}
|
||||||
|
|
||||||
|
for i := 0; i < typeOf.NumField(); i++ {
|
||||||
|
f := typeOf.Field(i)
|
||||||
|
|
||||||
|
kind := f.Type.Kind()
|
||||||
|
if f.Anonymous && kind == reflect.Struct {
|
||||||
|
//fmt.Println(v.Interface())
|
||||||
|
fields = append(fields, deepFields(f.Type)...)
|
||||||
|
} else if f.Anonymous && kind == reflect.Ptr && f.Type.Elem().Kind() == reflect.Struct {
|
||||||
|
//fmt.Println(v.Interface())
|
||||||
|
fields = append(fields, deepFields(f.Type.Elem())...)
|
||||||
|
} else {
|
||||||
|
fields = append(fields, f)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return fields
|
||||||
|
}
|
||||||
|
|
||||||
|
func (ts TypeScriptify) logf(depth int, s string, args ...interface{}) {
|
||||||
|
fmt.Printf(strings.Repeat(" ", depth)+s+"\n", args...)
|
||||||
|
}
|
||||||
|
|
||||||
|
// ManageType can define custom options for fields of a specified type.
|
||||||
|
//
|
||||||
|
// This can be used instead of setting ts_type and ts_transform for all fields of a certain type.
|
||||||
|
func (t *TypeScriptify) ManageType(fld interface{}, opts TypeOptions) *TypeScriptify {
|
||||||
|
var typ reflect.Type
|
||||||
|
switch t := fld.(type) {
|
||||||
|
case reflect.Type:
|
||||||
|
typ = t
|
||||||
|
default:
|
||||||
|
typ = reflect.TypeOf(fld)
|
||||||
|
}
|
||||||
|
if t.fieldTypeOptions == nil {
|
||||||
|
t.fieldTypeOptions = map[reflect.Type]TypeOptions{}
|
||||||
|
}
|
||||||
|
t.fieldTypeOptions[typ] = opts
|
||||||
|
return t
|
||||||
|
}
|
||||||
|
|
||||||
|
func (t *TypeScriptify) WithCreateFromMethod(b bool) *TypeScriptify {
|
||||||
|
t.CreateFromMethod = b
|
||||||
|
return t
|
||||||
|
}
|
||||||
|
|
||||||
|
func (t *TypeScriptify) WithInterface(b bool) *TypeScriptify {
|
||||||
|
t.CreateInterface = b
|
||||||
|
return t
|
||||||
|
}
|
||||||
|
|
||||||
|
func (t *TypeScriptify) WithConstructor(b bool) *TypeScriptify {
|
||||||
|
t.CreateConstructor = b
|
||||||
|
return t
|
||||||
|
}
|
||||||
|
|
||||||
|
func (t *TypeScriptify) WithIndent(i string) *TypeScriptify {
|
||||||
|
t.Indent = i
|
||||||
|
return t
|
||||||
|
}
|
||||||
|
|
||||||
|
func (t *TypeScriptify) WithBackupDir(b string) *TypeScriptify {
|
||||||
|
t.BackupDir = b
|
||||||
|
return t
|
||||||
|
}
|
||||||
|
|
||||||
|
func (t *TypeScriptify) WithPrefix(p string) *TypeScriptify {
|
||||||
|
t.Prefix = p
|
||||||
|
return t
|
||||||
|
}
|
||||||
|
|
||||||
|
func (t *TypeScriptify) WithSuffix(s string) *TypeScriptify {
|
||||||
|
t.Suffix = s
|
||||||
|
return t
|
||||||
|
}
|
||||||
|
|
||||||
|
func (t *TypeScriptify) Add(obj interface{}) *TypeScriptify {
|
||||||
|
switch ty := obj.(type) {
|
||||||
|
case StructType:
|
||||||
|
t.structTypes = append(t.structTypes, ty)
|
||||||
|
case *StructType:
|
||||||
|
t.structTypes = append(t.structTypes, *ty)
|
||||||
|
case reflect.Type:
|
||||||
|
t.AddType(ty)
|
||||||
|
default:
|
||||||
|
t.AddType(reflect.TypeOf(obj))
|
||||||
|
}
|
||||||
|
return t
|
||||||
|
}
|
||||||
|
|
||||||
|
func (t *TypeScriptify) AddType(typeOf reflect.Type) *TypeScriptify {
|
||||||
|
t.structTypes = append(t.structTypes, StructType{Type: typeOf})
|
||||||
|
return t
|
||||||
|
}
|
||||||
|
|
||||||
|
func (t *typeScriptClassBuilder) AddMapField(fieldName string, field reflect.StructField) {
|
||||||
|
keyType := field.Type.Key()
|
||||||
|
valueType := field.Type.Elem()
|
||||||
|
valueTypeName := valueType.Name()
|
||||||
|
if name, ok := t.types[valueType.Kind()]; ok {
|
||||||
|
valueTypeName = name
|
||||||
|
}
|
||||||
|
if valueType.Kind() == reflect.Array || valueType.Kind() == reflect.Slice {
|
||||||
|
valueTypeName = valueType.Elem().Name() + "[]"
|
||||||
|
}
|
||||||
|
if valueType.Kind() == reflect.Ptr {
|
||||||
|
valueTypeName = valueType.Elem().Name()
|
||||||
|
}
|
||||||
|
strippedFieldName := strings.ReplaceAll(fieldName, "?", "")
|
||||||
|
|
||||||
|
keyTypeStr := keyType.Name()
|
||||||
|
// Key should always be string, no need for this:
|
||||||
|
// _, isSimple := t.types[keyType.Kind()]
|
||||||
|
// if !isSimple {
|
||||||
|
// keyTypeStr = t.prefix + keyType.Name() + t.suffix
|
||||||
|
// }
|
||||||
|
|
||||||
|
t.fields = append(t.fields, fmt.Sprintf("%s%s: {[key: %s]: %s};", t.indent, fieldName, keyTypeStr, valueTypeName))
|
||||||
|
if valueType.Kind() == reflect.Struct {
|
||||||
|
t.constructorBody = append(t.constructorBody, fmt.Sprintf("%s%sthis.%s = this.convertValues(source[\"%s\"], %s, true);", t.indent, t.indent, strippedFieldName, strippedFieldName, t.prefix+valueTypeName+t.suffix))
|
||||||
|
} else {
|
||||||
|
t.constructorBody = append(t.constructorBody, fmt.Sprintf("%s%sthis.%s = source[\"%s\"];", t.indent, t.indent, strippedFieldName, strippedFieldName))
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func (t *TypeScriptify) AddEnum(values interface{}) *TypeScriptify {
|
||||||
|
if t.enums == nil {
|
||||||
|
t.enums = map[reflect.Type][]enumElement{}
|
||||||
|
}
|
||||||
|
items := reflect.ValueOf(values)
|
||||||
|
if items.Kind() != reflect.Slice {
|
||||||
|
panic(fmt.Sprintf("Values for %T isn't a slice", values))
|
||||||
|
}
|
||||||
|
|
||||||
|
var elements []enumElement
|
||||||
|
for i := 0; i < items.Len(); i++ {
|
||||||
|
item := items.Index(i)
|
||||||
|
|
||||||
|
var el enumElement
|
||||||
|
if item.Kind() == reflect.Struct {
|
||||||
|
r := reflector.New(item.Interface())
|
||||||
|
val, err := r.Field("Value").Get()
|
||||||
|
if err != nil {
|
||||||
|
panic(fmt.Sprint("missing Type field in ", item.Type().String()))
|
||||||
|
}
|
||||||
|
name, err := r.Field("TSName").Get()
|
||||||
|
if err != nil {
|
||||||
|
panic(fmt.Sprint("missing TSName field in ", item.Type().String()))
|
||||||
|
}
|
||||||
|
el.value = val
|
||||||
|
el.name = name.(string)
|
||||||
|
} else {
|
||||||
|
el.value = item.Interface()
|
||||||
|
if tsNamer, is := item.Interface().(TSNamer); is {
|
||||||
|
el.name = tsNamer.TSName()
|
||||||
|
} else {
|
||||||
|
panic(fmt.Sprint(item.Type().String(), " has no TSName method"))
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
elements = append(elements, el)
|
||||||
|
}
|
||||||
|
ty := reflect.TypeOf(elements[0].value)
|
||||||
|
t.enums[ty] = elements
|
||||||
|
t.enumTypes = append(t.enumTypes, EnumType{Type: ty})
|
||||||
|
|
||||||
|
return t
|
||||||
|
}
|
||||||
|
|
||||||
|
// AddEnumValues is deprecated, use `AddEnum()`
|
||||||
|
func (t *TypeScriptify) AddEnumValues(typeOf reflect.Type, values interface{}) *TypeScriptify {
|
||||||
|
t.AddEnum(values)
|
||||||
|
return t
|
||||||
|
}
|
||||||
|
|
||||||
|
func (t *TypeScriptify) Convert(customCode map[string]string) (string, error) {
|
||||||
|
t.alreadyConverted = make(map[reflect.Type]bool)
|
||||||
|
depth := 0
|
||||||
|
|
||||||
|
result := ""
|
||||||
|
if len(t.customImports) > 0 {
|
||||||
|
// Put the custom imports, i.e.: `import Decimal from 'decimal.js'`
|
||||||
|
for _, cimport := range t.customImports {
|
||||||
|
result += cimport + "\n"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
for _, enumTyp := range t.enumTypes {
|
||||||
|
elements := t.enums[enumTyp.Type]
|
||||||
|
typeScriptCode, err := t.convertEnum(depth, enumTyp.Type, elements)
|
||||||
|
if err != nil {
|
||||||
|
return "", err
|
||||||
|
}
|
||||||
|
result += "\n" + strings.Trim(typeScriptCode, " "+t.Indent+"\r\n")
|
||||||
|
}
|
||||||
|
|
||||||
|
for _, strctTyp := range t.structTypes {
|
||||||
|
typeScriptCode, err := t.convertType(depth, strctTyp.Type, customCode)
|
||||||
|
if err != nil {
|
||||||
|
return "", err
|
||||||
|
}
|
||||||
|
result += "\n" + strings.Trim(typeScriptCode, " "+t.Indent+"\r\n")
|
||||||
|
}
|
||||||
|
return result, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func loadCustomCode(fileName string) (map[string]string, error) {
|
||||||
|
result := make(map[string]string)
|
||||||
|
f, err := os.Open(fileName)
|
||||||
|
if err != nil {
|
||||||
|
if os.IsNotExist(err) {
|
||||||
|
return result, nil
|
||||||
|
}
|
||||||
|
return result, err
|
||||||
|
}
|
||||||
|
defer f.Close()
|
||||||
|
|
||||||
|
bytes, err := ioutil.ReadAll(f)
|
||||||
|
if err != nil {
|
||||||
|
return result, err
|
||||||
|
}
|
||||||
|
|
||||||
|
var currentName string
|
||||||
|
var currentValue string
|
||||||
|
lines := strings.Split(string(bytes), "\n")
|
||||||
|
for _, line := range lines {
|
||||||
|
trimmedLine := strings.TrimSpace(line)
|
||||||
|
if strings.HasPrefix(trimmedLine, "//[") && strings.HasSuffix(trimmedLine, ":]") {
|
||||||
|
currentName = strings.Replace(strings.Replace(trimmedLine, "//[", "", -1), ":]", "", -1)
|
||||||
|
currentValue = ""
|
||||||
|
} else if trimmedLine == "//[end]" {
|
||||||
|
result[currentName] = strings.TrimRight(currentValue, " \t\r\n")
|
||||||
|
currentName = ""
|
||||||
|
currentValue = ""
|
||||||
|
} else if len(currentName) > 0 {
|
||||||
|
currentValue += line + "\n"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return result, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (t TypeScriptify) backup(fileName string) error {
|
||||||
|
fileIn, err := os.Open(fileName)
|
||||||
|
if err != nil {
|
||||||
|
if !os.IsNotExist(err) {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
// No neet to backup, just return:
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
defer fileIn.Close()
|
||||||
|
|
||||||
|
bytes, err := ioutil.ReadAll(fileIn)
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
|
||||||
|
_, backupFn := path.Split(fmt.Sprintf("%s-%s.backup", fileName, time.Now().Format("2006-01-02T15_04_05.99")))
|
||||||
|
if t.BackupDir != "" {
|
||||||
|
backupFn = path.Join(t.BackupDir, backupFn)
|
||||||
|
}
|
||||||
|
|
||||||
|
return ioutil.WriteFile(backupFn, bytes, os.FileMode(0700))
|
||||||
|
}
|
||||||
|
|
||||||
|
func (t TypeScriptify) ConvertToFile(fileName string, packageName string) error {
|
||||||
|
if len(t.BackupDir) > 0 {
|
||||||
|
err := t.backup(fileName)
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
customCode, err := loadCustomCode(fileName)
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
|
||||||
|
f, err := os.Create(fileName)
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
defer f.Close()
|
||||||
|
|
||||||
|
converted, err := t.Convert(customCode)
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
|
||||||
|
var lines []string
|
||||||
|
sc := bufio.NewScanner(strings.NewReader(converted))
|
||||||
|
for sc.Scan() {
|
||||||
|
lines = append(lines, "\t"+sc.Text())
|
||||||
|
}
|
||||||
|
|
||||||
|
converted = "export namespace " + packageName + " {\n"
|
||||||
|
converted += strings.Join(lines, "\n")
|
||||||
|
converted += "\n}\n"
|
||||||
|
|
||||||
|
if _, err := f.WriteString("/* Do not change, this code is generated from Golang structs */\n\n"); err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
if _, err := f.WriteString(converted); err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
type TSNamer interface {
|
||||||
|
TSName() string
|
||||||
|
}
|
||||||
|
|
||||||
|
func (t *TypeScriptify) convertEnum(depth int, typeOf reflect.Type, elements []enumElement) (string, error) {
|
||||||
|
t.logf(depth, "Converting enum %s", typeOf.String())
|
||||||
|
if _, found := t.alreadyConverted[typeOf]; found { // Already converted
|
||||||
|
return "", nil
|
||||||
|
}
|
||||||
|
t.alreadyConverted[typeOf] = true
|
||||||
|
|
||||||
|
entityName := t.Prefix + typeOf.Name() + t.Suffix
|
||||||
|
result := "enum " + entityName + " {\n"
|
||||||
|
|
||||||
|
for _, val := range elements {
|
||||||
|
result += fmt.Sprintf("%s%s = %#v,\n", t.Indent, val.name, val.value)
|
||||||
|
}
|
||||||
|
|
||||||
|
result += "}"
|
||||||
|
|
||||||
|
if !t.DontExport {
|
||||||
|
result = "export " + result
|
||||||
|
}
|
||||||
|
|
||||||
|
return result, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (t *TypeScriptify) getFieldOptions(structType reflect.Type, field reflect.StructField) TypeOptions {
|
||||||
|
// By default use options defined by tags:
|
||||||
|
opts := TypeOptions{TSTransform: field.Tag.Get(tsTransformTag), TSType: field.Tag.Get(tsType)}
|
||||||
|
|
||||||
|
overrides := []TypeOptions{}
|
||||||
|
|
||||||
|
// But there is maybe an struct-specific override:
|
||||||
|
for _, strct := range t.structTypes {
|
||||||
|
if strct.FieldOptions == nil {
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
if strct.Type == structType {
|
||||||
|
if fldOpts, found := strct.FieldOptions[field.Type]; found {
|
||||||
|
overrides = append(overrides, fldOpts)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if fldOpts, found := t.fieldTypeOptions[field.Type]; found {
|
||||||
|
overrides = append(overrides, fldOpts)
|
||||||
|
}
|
||||||
|
|
||||||
|
for _, o := range overrides {
|
||||||
|
if o.TSTransform != "" {
|
||||||
|
opts.TSTransform = o.TSTransform
|
||||||
|
}
|
||||||
|
if o.TSType != "" {
|
||||||
|
opts.TSType = o.TSType
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return opts
|
||||||
|
}
|
||||||
|
|
||||||
|
func (t *TypeScriptify) getJSONFieldName(field reflect.StructField, isPtr bool) string {
|
||||||
|
jsonFieldName := ""
|
||||||
|
jsonTag := field.Tag.Get("json")
|
||||||
|
if len(jsonTag) > 0 {
|
||||||
|
jsonTagParts := strings.Split(jsonTag, ",")
|
||||||
|
if len(jsonTagParts) > 0 {
|
||||||
|
jsonFieldName = strings.Trim(jsonTagParts[0], t.Indent)
|
||||||
|
}
|
||||||
|
hasOmitEmpty := false
|
||||||
|
ignored := false
|
||||||
|
for _, t := range jsonTagParts {
|
||||||
|
if t == "" {
|
||||||
|
break
|
||||||
|
}
|
||||||
|
if t == "omitempty" {
|
||||||
|
hasOmitEmpty = true
|
||||||
|
break
|
||||||
|
}
|
||||||
|
if t == "-" {
|
||||||
|
ignored = true
|
||||||
|
break
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if !ignored && isPtr || hasOmitEmpty {
|
||||||
|
jsonFieldName = fmt.Sprintf("%s?", jsonFieldName)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return jsonFieldName
|
||||||
|
}
|
||||||
|
|
||||||
|
func (t *TypeScriptify) convertType(depth int, typeOf reflect.Type, customCode map[string]string) (string, error) {
|
||||||
|
if _, found := t.alreadyConverted[typeOf]; found { // Already converted
|
||||||
|
return "", nil
|
||||||
|
}
|
||||||
|
t.logf(depth, "Converting type %s", typeOf.String())
|
||||||
|
|
||||||
|
t.alreadyConverted[typeOf] = true
|
||||||
|
|
||||||
|
entityName := t.Prefix + typeOf.Name() + t.Suffix
|
||||||
|
result := ""
|
||||||
|
if t.CreateInterface {
|
||||||
|
result += fmt.Sprintf("interface %s {\n", entityName)
|
||||||
|
} else {
|
||||||
|
result += fmt.Sprintf("class %s {\n", entityName)
|
||||||
|
}
|
||||||
|
if !t.DontExport {
|
||||||
|
result = "export " + result
|
||||||
|
}
|
||||||
|
builder := typeScriptClassBuilder{
|
||||||
|
types: t.kinds,
|
||||||
|
indent: t.Indent,
|
||||||
|
prefix: t.Prefix,
|
||||||
|
suffix: t.Suffix,
|
||||||
|
}
|
||||||
|
|
||||||
|
fields := deepFields(typeOf)
|
||||||
|
for _, field := range fields {
|
||||||
|
isPtr := field.Type.Kind() == reflect.Ptr
|
||||||
|
if isPtr {
|
||||||
|
field.Type = field.Type.Elem()
|
||||||
|
}
|
||||||
|
jsonFieldName := t.getJSONFieldName(field, isPtr)
|
||||||
|
if len(jsonFieldName) == 0 || jsonFieldName == "-" {
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
|
||||||
|
var err error
|
||||||
|
fldOpts := t.getFieldOptions(typeOf, field)
|
||||||
|
if fldOpts.TSTransform != "" {
|
||||||
|
t.logf(depth, "- simple field %s.%s", typeOf.Name(), field.Name)
|
||||||
|
err = builder.AddSimpleField(jsonFieldName, field, fldOpts)
|
||||||
|
} else if _, isEnum := t.enums[field.Type]; isEnum {
|
||||||
|
t.logf(depth, "- enum field %s.%s", typeOf.Name(), field.Name)
|
||||||
|
builder.AddEnumField(jsonFieldName, field)
|
||||||
|
} else if fldOpts.TSType != "" { // Struct:
|
||||||
|
t.logf(depth, "- simple field %s.%s", typeOf.Name(), field.Name)
|
||||||
|
err = builder.AddSimpleField(jsonFieldName, field, fldOpts)
|
||||||
|
} else if field.Type.Kind() == reflect.Struct { // Struct:
|
||||||
|
t.logf(depth, "- struct %s.%s (%s)", typeOf.Name(), field.Name, field.Type.String())
|
||||||
|
typeScriptChunk, err := t.convertType(depth+1, field.Type, customCode)
|
||||||
|
if err != nil {
|
||||||
|
return "", err
|
||||||
|
}
|
||||||
|
if typeScriptChunk != "" {
|
||||||
|
result = typeScriptChunk + "\n" + result
|
||||||
|
}
|
||||||
|
builder.AddStructField(jsonFieldName, field)
|
||||||
|
} else if field.Type.Kind() == reflect.Map {
|
||||||
|
t.logf(depth, "- map field %s.%s", typeOf.Name(), field.Name)
|
||||||
|
// Also convert map key types if needed
|
||||||
|
var keyTypeToConvert reflect.Type
|
||||||
|
switch field.Type.Key().Kind() {
|
||||||
|
case reflect.Struct:
|
||||||
|
keyTypeToConvert = field.Type.Key()
|
||||||
|
case reflect.Ptr:
|
||||||
|
keyTypeToConvert = field.Type.Key().Elem()
|
||||||
|
}
|
||||||
|
if keyTypeToConvert != nil {
|
||||||
|
typeScriptChunk, err := t.convertType(depth+1, keyTypeToConvert, customCode)
|
||||||
|
if err != nil {
|
||||||
|
return "", err
|
||||||
|
}
|
||||||
|
if typeScriptChunk != "" {
|
||||||
|
result = typeScriptChunk + "\n" + result
|
||||||
|
}
|
||||||
|
}
|
||||||
|
// Also convert map value types if needed
|
||||||
|
var valueTypeToConvert reflect.Type
|
||||||
|
switch field.Type.Elem().Kind() {
|
||||||
|
case reflect.Struct:
|
||||||
|
valueTypeToConvert = field.Type.Elem()
|
||||||
|
case reflect.Ptr:
|
||||||
|
valueTypeToConvert = field.Type.Elem().Elem()
|
||||||
|
}
|
||||||
|
if valueTypeToConvert != nil {
|
||||||
|
typeScriptChunk, err := t.convertType(depth+1, valueTypeToConvert, customCode)
|
||||||
|
if err != nil {
|
||||||
|
return "", err
|
||||||
|
}
|
||||||
|
if typeScriptChunk != "" {
|
||||||
|
result = typeScriptChunk + "\n" + result
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
builder.AddMapField(jsonFieldName, field)
|
||||||
|
} else if field.Type.Kind() == reflect.Slice || field.Type.Kind() == reflect.Array { // Slice:
|
||||||
|
if field.Type.Elem().Kind() == reflect.Ptr { //extract ptr type
|
||||||
|
field.Type = field.Type.Elem()
|
||||||
|
}
|
||||||
|
|
||||||
|
arrayDepth := 1
|
||||||
|
for field.Type.Elem().Kind() == reflect.Slice { // Slice of slices:
|
||||||
|
field.Type = field.Type.Elem()
|
||||||
|
arrayDepth++
|
||||||
|
}
|
||||||
|
|
||||||
|
if field.Type.Elem().Kind() == reflect.Struct { // Slice of structs:
|
||||||
|
t.logf(depth, "- struct slice %s.%s (%s)", typeOf.Name(), field.Name, field.Type.String())
|
||||||
|
typeScriptChunk, err := t.convertType(depth+1, field.Type.Elem(), customCode)
|
||||||
|
if err != nil {
|
||||||
|
return "", err
|
||||||
|
}
|
||||||
|
if typeScriptChunk != "" {
|
||||||
|
result = typeScriptChunk + "\n" + result
|
||||||
|
}
|
||||||
|
builder.AddArrayOfStructsField(jsonFieldName, field, arrayDepth)
|
||||||
|
} else { // Slice of simple fields:
|
||||||
|
t.logf(depth, "- slice field %s.%s", typeOf.Name(), field.Name)
|
||||||
|
err = builder.AddSimpleArrayField(jsonFieldName, field, arrayDepth, fldOpts)
|
||||||
|
}
|
||||||
|
} else { // Simple field:
|
||||||
|
t.logf(depth, "- simple field %s.%s", typeOf.Name(), field.Name)
|
||||||
|
err = builder.AddSimpleField(jsonFieldName, field, fldOpts)
|
||||||
|
}
|
||||||
|
if err != nil {
|
||||||
|
return "", err
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if t.CreateFromMethod {
|
||||||
|
t.CreateConstructor = true
|
||||||
|
}
|
||||||
|
|
||||||
|
result += strings.Join(builder.fields, "\n") + "\n"
|
||||||
|
if !t.CreateInterface {
|
||||||
|
constructorBody := strings.Join(builder.constructorBody, "\n")
|
||||||
|
needsConvertValue := strings.Contains(constructorBody, "this.convertValues")
|
||||||
|
if t.CreateFromMethod {
|
||||||
|
result += fmt.Sprintf("\n%sstatic createFrom(source: any = {}) {\n", t.Indent)
|
||||||
|
result += fmt.Sprintf("%s%sreturn new %s(source);\n", t.Indent, t.Indent, entityName)
|
||||||
|
result += fmt.Sprintf("%s}\n", t.Indent)
|
||||||
|
}
|
||||||
|
if t.CreateConstructor {
|
||||||
|
result += fmt.Sprintf("\n%sconstructor(source: any = {}) {\n", t.Indent)
|
||||||
|
result += t.Indent + t.Indent + "if ('string' === typeof source) source = JSON.parse(source);\n"
|
||||||
|
result += constructorBody + "\n"
|
||||||
|
result += fmt.Sprintf("%s}\n", t.Indent)
|
||||||
|
}
|
||||||
|
if needsConvertValue && (t.CreateConstructor || t.CreateFromMethod) {
|
||||||
|
result += "\n" + indentLines(strings.ReplaceAll(tsConvertValuesFunc, "\t", t.Indent), 1) + "\n"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if customCode != nil {
|
||||||
|
code := customCode[entityName]
|
||||||
|
if len(code) != 0 {
|
||||||
|
result += t.Indent + "//[" + entityName + ":]\n" + code + "\n\n" + t.Indent + "//[end]\n"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
result += "}"
|
||||||
|
|
||||||
|
return result, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (t *TypeScriptify) AddImport(i string) {
|
||||||
|
for _, cimport := range t.customImports {
|
||||||
|
if cimport == i {
|
||||||
|
return
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
t.customImports = append(t.customImports, i)
|
||||||
|
}
|
||||||
|
|
||||||
|
type typeScriptClassBuilder struct {
|
||||||
|
types map[reflect.Kind]string
|
||||||
|
indent string
|
||||||
|
fields []string
|
||||||
|
createFromMethodBody []string
|
||||||
|
constructorBody []string
|
||||||
|
prefix, suffix string
|
||||||
|
}
|
||||||
|
|
||||||
|
func (t *typeScriptClassBuilder) AddSimpleArrayField(fieldName string, field reflect.StructField, arrayDepth int, opts TypeOptions) error {
|
||||||
|
fieldType, kind := field.Type.Elem().Name(), field.Type.Elem().Kind()
|
||||||
|
typeScriptType := t.types[kind]
|
||||||
|
|
||||||
|
if len(fieldName) > 0 {
|
||||||
|
strippedFieldName := strings.ReplaceAll(fieldName, "?", "")
|
||||||
|
if len(opts.TSType) > 0 {
|
||||||
|
t.addField(fieldName, opts.TSType)
|
||||||
|
t.addInitializerFieldLine(strippedFieldName, fmt.Sprintf("source[\"%s\"]", strippedFieldName))
|
||||||
|
return nil
|
||||||
|
} else if len(typeScriptType) > 0 {
|
||||||
|
t.addField(fieldName, fmt.Sprint(typeScriptType, strings.Repeat("[]", arrayDepth)))
|
||||||
|
t.addInitializerFieldLine(strippedFieldName, fmt.Sprintf("source[\"%s\"]", strippedFieldName))
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return fmt.Errorf("cannot find type for %s (%s/%s)", kind.String(), fieldName, fieldType)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (t *typeScriptClassBuilder) AddSimpleField(fieldName string, field reflect.StructField, opts TypeOptions) error {
|
||||||
|
fieldType, kind := field.Type.Name(), field.Type.Kind()
|
||||||
|
|
||||||
|
typeScriptType := t.types[kind]
|
||||||
|
if len(opts.TSType) > 0 {
|
||||||
|
typeScriptType = opts.TSType
|
||||||
|
}
|
||||||
|
|
||||||
|
if len(typeScriptType) > 0 && len(fieldName) > 0 {
|
||||||
|
strippedFieldName := strings.ReplaceAll(fieldName, "?", "")
|
||||||
|
t.addField(fieldName, typeScriptType)
|
||||||
|
if opts.TSTransform == "" {
|
||||||
|
t.addInitializerFieldLine(strippedFieldName, fmt.Sprintf("source[\"%s\"]", strippedFieldName))
|
||||||
|
} else {
|
||||||
|
val := fmt.Sprintf(`source["%s"]`, strippedFieldName)
|
||||||
|
expression := strings.Replace(opts.TSTransform, "__VALUE__", val, -1)
|
||||||
|
t.addInitializerFieldLine(strippedFieldName, expression)
|
||||||
|
}
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
return fmt.Errorf("cannot find type for %s (%s/%s)", kind.String(), fieldName, fieldType)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (t *typeScriptClassBuilder) AddEnumField(fieldName string, field reflect.StructField) {
|
||||||
|
fieldType := field.Type.Name()
|
||||||
|
t.addField(fieldName, t.prefix+fieldType+t.suffix)
|
||||||
|
strippedFieldName := strings.ReplaceAll(fieldName, "?", "")
|
||||||
|
t.addInitializerFieldLine(strippedFieldName, fmt.Sprintf("source[\"%s\"]", strippedFieldName))
|
||||||
|
}
|
||||||
|
|
||||||
|
func (t *typeScriptClassBuilder) AddStructField(fieldName string, field reflect.StructField) {
|
||||||
|
fieldType := field.Type.Name()
|
||||||
|
strippedFieldName := strings.ReplaceAll(fieldName, "?", "")
|
||||||
|
t.addField(fieldName, t.prefix+fieldType+t.suffix)
|
||||||
|
t.addInitializerFieldLine(strippedFieldName, fmt.Sprintf("this.convertValues(source[\"%s\"], %s)", strippedFieldName, t.prefix+fieldType+t.suffix))
|
||||||
|
}
|
||||||
|
|
||||||
|
func (t *typeScriptClassBuilder) AddArrayOfStructsField(fieldName string, field reflect.StructField, arrayDepth int) {
|
||||||
|
fieldType := field.Type.Elem().Name()
|
||||||
|
strippedFieldName := strings.ReplaceAll(fieldName, "?", "")
|
||||||
|
t.addField(fieldName, fmt.Sprint(t.prefix+fieldType+t.suffix, strings.Repeat("[]", arrayDepth)))
|
||||||
|
t.addInitializerFieldLine(strippedFieldName, fmt.Sprintf("this.convertValues(source[\"%s\"], %s)", strippedFieldName, t.prefix+fieldType+t.suffix))
|
||||||
|
}
|
||||||
|
|
||||||
|
func (t *typeScriptClassBuilder) addInitializerFieldLine(fld, initializer string) {
|
||||||
|
t.createFromMethodBody = append(t.createFromMethodBody, fmt.Sprint(t.indent, t.indent, "result.", fld, " = ", initializer, ";"))
|
||||||
|
t.constructorBody = append(t.constructorBody, fmt.Sprint(t.indent, t.indent, "this.", fld, " = ", initializer, ";"))
|
||||||
|
}
|
||||||
|
|
||||||
|
func (t *typeScriptClassBuilder) addField(fld, fldType string) {
|
||||||
|
t.fields = append(t.fields, fmt.Sprint(t.indent, fld, ": ", fldType, ";"))
|
||||||
|
}
|
||||||
|
|
||||||
|
func indentLines(str string, i int) string {
|
||||||
|
lines := strings.Split(str, "\n")
|
||||||
|
for n := range lines {
|
||||||
|
lines[n] = strings.Repeat("\t", i) + lines[n]
|
||||||
|
}
|
||||||
|
return strings.Join(lines, "\n")
|
||||||
|
}
|
Loading…
Reference in New Issue
Block a user