mirror of
https://github.com/gogf/gf.git
synced 2025-04-05 11:18:50 +08:00
This commit is contained in:
parent
509fdf45c6
commit
e0a2645f4a
@ -4669,13 +4669,13 @@ func TestResult_Structs1(t *testing.T) {
|
|||||||
gtest.C(t, func(t *gtest.T) {
|
gtest.C(t, func(t *gtest.T) {
|
||||||
r := gdb.Result{
|
r := gdb.Result{
|
||||||
gdb.Record{"id": gvar.New(nil), "name": gvar.New("john")},
|
gdb.Record{"id": gvar.New(nil), "name": gvar.New("john")},
|
||||||
gdb.Record{"id": gvar.New(nil), "name": gvar.New("smith")},
|
gdb.Record{"id": gvar.New(1), "name": gvar.New("smith")},
|
||||||
}
|
}
|
||||||
array := make([]*B, 2)
|
array := make([]*B, 2)
|
||||||
err := r.Structs(&array)
|
err := r.Structs(&array)
|
||||||
t.AssertNil(err)
|
t.AssertNil(err)
|
||||||
t.Assert(array[0].Id, 0)
|
t.Assert(array[0].Id, 0)
|
||||||
t.Assert(array[1].Id, 0)
|
t.Assert(array[1].Id, 1)
|
||||||
t.Assert(array[0].Name, "john")
|
t.Assert(array[0].Name, "john")
|
||||||
t.Assert(array[1].Name, "smith")
|
t.Assert(array[1].Name, "smith")
|
||||||
})
|
})
|
||||||
|
@ -17,7 +17,7 @@ import (
|
|||||||
var (
|
var (
|
||||||
defaultParsedArgs = make([]string, 0)
|
defaultParsedArgs = make([]string, 0)
|
||||||
defaultParsedOptions = make(map[string]string)
|
defaultParsedOptions = make(map[string]string)
|
||||||
argumentRegex = regexp.MustCompile(`^\-{1,2}([\w\?\.\-]+)(=){0,1}(.*)$`)
|
argumentOptionRegex = regexp.MustCompile(`^\-{1,2}([\w\?\.\-]+)(=){0,1}(.*)$`)
|
||||||
)
|
)
|
||||||
|
|
||||||
// Init does custom initialization.
|
// Init does custom initialization.
|
||||||
@ -41,22 +41,22 @@ func ParseUsingDefaultAlgorithm(args ...string) (parsedArgs []string, parsedOpti
|
|||||||
parsedArgs = make([]string, 0)
|
parsedArgs = make([]string, 0)
|
||||||
parsedOptions = make(map[string]string)
|
parsedOptions = make(map[string]string)
|
||||||
for i := 0; i < len(args); {
|
for i := 0; i < len(args); {
|
||||||
array := argumentRegex.FindStringSubmatch(args[i])
|
array := argumentOptionRegex.FindStringSubmatch(args[i])
|
||||||
if len(array) > 2 {
|
if len(array) > 2 {
|
||||||
if array[2] == "=" {
|
if array[2] == "=" {
|
||||||
parsedOptions[array[1]] = array[3]
|
parsedOptions[array[1]] = array[3]
|
||||||
} else if i < len(args)-1 {
|
} else if i < len(args)-1 {
|
||||||
if len(args[i+1]) > 0 && args[i+1][0] == '-' {
|
if len(args[i+1]) > 0 && args[i+1][0] == '-' {
|
||||||
// Eg: gf gen -d -n 1
|
// Example: gf gen -d -n 1
|
||||||
parsedOptions[array[1]] = array[3]
|
parsedOptions[array[1]] = array[3]
|
||||||
} else {
|
} else {
|
||||||
// Eg: gf gen -n 2
|
// Example: gf gen -n 2
|
||||||
parsedOptions[array[1]] = args[i+1]
|
parsedOptions[array[1]] = args[i+1]
|
||||||
i += 2
|
i += 2
|
||||||
continue
|
continue
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
// Eg: gf gen -h
|
// Example: gf gen -h
|
||||||
parsedOptions[array[1]] = array[3]
|
parsedOptions[array[1]] = array[3]
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
|
@ -9,6 +9,7 @@ package gudp_test
|
|||||||
import (
|
import (
|
||||||
"context"
|
"context"
|
||||||
"fmt"
|
"fmt"
|
||||||
|
"io"
|
||||||
"testing"
|
"testing"
|
||||||
"time"
|
"time"
|
||||||
|
|
||||||
@ -29,9 +30,14 @@ func startUDPServer(addr string) *gudp.Server {
|
|||||||
for {
|
for {
|
||||||
data, err := conn.Recv(-1)
|
data, err := conn.Recv(-1)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
|
if err != io.EOF {
|
||||||
|
glog.Error(context.TODO(), err)
|
||||||
|
}
|
||||||
break
|
break
|
||||||
}
|
}
|
||||||
conn.Send(data)
|
if err = conn.Send(data); err != nil {
|
||||||
|
glog.Error(context.TODO(), err)
|
||||||
|
}
|
||||||
}
|
}
|
||||||
})
|
})
|
||||||
go s.Run()
|
go s.Run()
|
||||||
|
@ -18,9 +18,9 @@ import (
|
|||||||
)
|
)
|
||||||
|
|
||||||
const (
|
const (
|
||||||
CtxKeyParser gctx.StrKey = `CtxKeyParser`
|
CtxKeyParser gctx.StrKey = `CtxKeyParser`
|
||||||
CtxKeyCommand gctx.StrKey = `CtxKeyCommand`
|
CtxKeyCommand gctx.StrKey = `CtxKeyCommand`
|
||||||
CtxKeyArguments gctx.StrKey = `CtxKeyArguments`
|
CtxKeyArgumentsIndex gctx.StrKey = `CtxKeyArgumentsIndex`
|
||||||
)
|
)
|
||||||
|
|
||||||
const (
|
const (
|
||||||
|
@ -251,6 +251,9 @@ func newCommandFromMethod(
|
|||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// For input struct converting using priority tag.
|
||||||
|
var priorityTag = gstr.Join([]string{tagNameName, tagNameShort}, ",")
|
||||||
|
|
||||||
// =============================================================================================
|
// =============================================================================================
|
||||||
// Create function that has value return.
|
// Create function that has value return.
|
||||||
// =============================================================================================
|
// =============================================================================================
|
||||||
@ -259,9 +262,16 @@ func newCommandFromMethod(
|
|||||||
var (
|
var (
|
||||||
data = gconv.Map(parser.GetOptAll())
|
data = gconv.Map(parser.GetOptAll())
|
||||||
argIndex = 0
|
argIndex = 0
|
||||||
arguments = gconv.Strings(ctx.Value(CtxKeyArguments))
|
arguments = parser.GetArgAll()
|
||||||
inputValues = []reflect.Value{reflect.ValueOf(ctx)}
|
inputValues = []reflect.Value{reflect.ValueOf(ctx)}
|
||||||
)
|
)
|
||||||
|
if value := ctx.Value(CtxKeyArgumentsIndex); value != nil {
|
||||||
|
argIndex = value.(int)
|
||||||
|
// Use the left args to assign to input struct object.
|
||||||
|
if argIndex < len(arguments) {
|
||||||
|
arguments = arguments[argIndex:]
|
||||||
|
}
|
||||||
|
}
|
||||||
if data == nil {
|
if data == nil {
|
||||||
data = map[string]interface{}{}
|
data = map[string]interface{}{}
|
||||||
}
|
}
|
||||||
@ -278,8 +288,11 @@ func newCommandFromMethod(
|
|||||||
if arg.Orphan {
|
if arg.Orphan {
|
||||||
if orphanValue := parser.GetOpt(arg.Name); orphanValue != nil {
|
if orphanValue := parser.GetOpt(arg.Name); orphanValue != nil {
|
||||||
if orphanValue.String() == "" {
|
if orphanValue.String() == "" {
|
||||||
// Eg: gf -f
|
// Example: gf -f
|
||||||
data[arg.Name] = "true"
|
data[arg.Name] = "true"
|
||||||
|
if arg.Short != "" {
|
||||||
|
data[arg.Short] = "true"
|
||||||
|
}
|
||||||
} else {
|
} else {
|
||||||
// Adapter with common user habits.
|
// Adapter with common user habits.
|
||||||
// Eg:
|
// Eg:
|
||||||
@ -301,9 +314,9 @@ func newCommandFromMethod(
|
|||||||
return fmt.Sprintf(`input command data map: %s`, gjson.MustEncode(data))
|
return fmt.Sprintf(`input command data map: %s`, gjson.MustEncode(data))
|
||||||
})
|
})
|
||||||
if inputObject.Kind() == reflect.Ptr {
|
if inputObject.Kind() == reflect.Ptr {
|
||||||
err = gconv.Scan(data, inputObject.Interface())
|
err = gconv.StructTag(data, inputObject.Interface(), priorityTag)
|
||||||
} else {
|
} else {
|
||||||
err = gconv.Struct(data, inputObject.Addr().Interface())
|
err = gconv.StructTag(data, inputObject.Addr().Interface(), priorityTag)
|
||||||
}
|
}
|
||||||
intlog.PrintFunc(ctx, func() string {
|
intlog.PrintFunc(ctx, func() string {
|
||||||
return fmt.Sprintf(`input object assigned data: %s`, gjson.MustEncode(inputObject.Interface()))
|
return fmt.Sprintf(`input object assigned data: %s`, gjson.MustEncode(inputObject.Interface()))
|
||||||
|
@ -93,7 +93,7 @@ func (c *Command) RunWithSpecificArgs(ctx context.Context, args []string) (value
|
|||||||
parsedArgs = parsedArgs[1:]
|
parsedArgs = parsedArgs[1:]
|
||||||
|
|
||||||
// Find the matched command and run it.
|
// Find the matched command and run it.
|
||||||
lastCmd, foundCmd, newCtx := c.searchCommand(ctx, parsedArgs)
|
lastCmd, foundCmd, newCtx := c.searchCommand(ctx, parsedArgs, 0)
|
||||||
if foundCmd != nil {
|
if foundCmd != nil {
|
||||||
return foundCmd.doRun(newCtx, args, parser)
|
return foundCmd.doRun(newCtx, args, parser)
|
||||||
}
|
}
|
||||||
@ -215,25 +215,27 @@ func (c *Command) reParse(ctx context.Context, args []string, parser *Parser) (*
|
|||||||
}
|
}
|
||||||
|
|
||||||
// searchCommand recursively searches the command according given arguments.
|
// searchCommand recursively searches the command according given arguments.
|
||||||
func (c *Command) searchCommand(ctx context.Context, args []string) (lastCmd, foundCmd *Command, newCtx context.Context) {
|
func (c *Command) searchCommand(
|
||||||
|
ctx context.Context, args []string, fromArgIndex int,
|
||||||
|
) (lastCmd, foundCmd *Command, newCtx context.Context) {
|
||||||
if len(args) == 0 {
|
if len(args) == 0 {
|
||||||
return c, nil, ctx
|
return c, nil, ctx
|
||||||
}
|
}
|
||||||
for _, cmd := range c.commands {
|
for _, cmd := range c.commands {
|
||||||
// Recursively searching the command.
|
// Recursively searching the command.
|
||||||
|
// String comparison case-sensitive.
|
||||||
if cmd.Name == args[0] {
|
if cmd.Name == args[0] {
|
||||||
leftArgs := args[1:]
|
leftArgs := args[1:]
|
||||||
// If this command needs argument,
|
// If this command needs argument,
|
||||||
// it then gives all its left arguments to it.
|
// it then gives all its left arguments to it using arg index marks.
|
||||||
if cmd.hasArgumentFromIndex() {
|
//
|
||||||
ctx = context.WithValue(ctx, CtxKeyArguments, leftArgs)
|
// Note that the args here (using default args parsing) could be different with the args
|
||||||
|
// that are parsed in command.
|
||||||
|
if cmd.hasArgumentFromIndex() || len(leftArgs) == 0 {
|
||||||
|
ctx = context.WithValue(ctx, CtxKeyArgumentsIndex, fromArgIndex+1)
|
||||||
return c, cmd, ctx
|
return c, cmd, ctx
|
||||||
}
|
}
|
||||||
// Recursively searching.
|
return cmd.searchCommand(ctx, leftArgs, fromArgIndex+1)
|
||||||
if len(leftArgs) == 0 {
|
|
||||||
return c, cmd, ctx
|
|
||||||
}
|
|
||||||
return cmd.searchCommand(ctx, leftArgs)
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
return c, nil, ctx
|
return c, nil, ctx
|
||||||
|
@ -185,11 +185,24 @@ func (p *Parser) isOptionNeedArgument(name string) bool {
|
|||||||
|
|
||||||
// setOptionValue sets the option value for name and according alias.
|
// setOptionValue sets the option value for name and according alias.
|
||||||
func (p *Parser) setOptionValue(name, value string) {
|
func (p *Parser) setOptionValue(name, value string) {
|
||||||
|
// Accurate option name match.
|
||||||
for optionName := range p.passedOptions {
|
for optionName := range p.passedOptions {
|
||||||
array := gstr.SplitAndTrim(optionName, ",")
|
optionNameAndShort := gstr.SplitAndTrim(optionName, ",")
|
||||||
for _, v := range array {
|
for _, optionNameItem := range optionNameAndShort {
|
||||||
if strings.EqualFold(v, name) {
|
if optionNameItem == name {
|
||||||
for _, v := range array {
|
for _, v := range optionNameAndShort {
|
||||||
|
p.parsedOptions[v] = value
|
||||||
|
}
|
||||||
|
return
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
// Fuzzy option name match.
|
||||||
|
for optionName := range p.passedOptions {
|
||||||
|
optionNameAndShort := gstr.SplitAndTrim(optionName, ",")
|
||||||
|
for _, optionNameItem := range optionNameAndShort {
|
||||||
|
if strings.EqualFold(optionNameItem, name) {
|
||||||
|
for _, v := range optionNameAndShort {
|
||||||
p.parsedOptions[v] = value
|
p.parsedOptions[v] = value
|
||||||
}
|
}
|
||||||
return
|
return
|
||||||
|
234
os/gcmd/gcmd_z_unit_issue_test.go
Normal file
234
os/gcmd/gcmd_z_unit_issue_test.go
Normal file
@ -0,0 +1,234 @@
|
|||||||
|
// Copyright GoFrame Author(https://goframe.org). All Rights Reserved.
|
||||||
|
//
|
||||||
|
// This Source Code Form is subject to the terms of the MIT License.
|
||||||
|
// If a copy of the MIT was not distributed with this file,
|
||||||
|
// You can obtain one at https://github.com/gogf/gf.
|
||||||
|
|
||||||
|
// go test *.go -bench=".*" -benchmem
|
||||||
|
|
||||||
|
package gcmd_test
|
||||||
|
|
||||||
|
import (
|
||||||
|
"context"
|
||||||
|
"testing"
|
||||||
|
|
||||||
|
"github.com/gogf/gf/v2/encoding/gjson"
|
||||||
|
"github.com/gogf/gf/v2/frame/g"
|
||||||
|
"github.com/gogf/gf/v2/os/gcmd"
|
||||||
|
"github.com/gogf/gf/v2/os/gctx"
|
||||||
|
"github.com/gogf/gf/v2/test/gtest"
|
||||||
|
)
|
||||||
|
|
||||||
|
type Issue3390CommandCase1 struct {
|
||||||
|
*gcmd.Command
|
||||||
|
}
|
||||||
|
|
||||||
|
type Issue3390TestCase1 struct {
|
||||||
|
g.Meta `name:"index" ad:"test"`
|
||||||
|
}
|
||||||
|
|
||||||
|
type Issue3390Case1Input struct {
|
||||||
|
g.Meta `name:"index"`
|
||||||
|
A string `short:"a" name:"aa"`
|
||||||
|
Be string `short:"b" name:"bb"`
|
||||||
|
}
|
||||||
|
|
||||||
|
type Issue3390Case1Output struct {
|
||||||
|
Content string
|
||||||
|
}
|
||||||
|
|
||||||
|
func (c Issue3390TestCase1) Index(ctx context.Context, in Issue3390Case1Input) (out *Issue3390Case1Output, err error) {
|
||||||
|
out = &Issue3390Case1Output{
|
||||||
|
Content: gjson.MustEncodeString(in),
|
||||||
|
}
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
func Test_Issue3390_Case1(t *testing.T) {
|
||||||
|
gtest.C(t, func(t *gtest.T) {
|
||||||
|
root, err := gcmd.NewFromObject(Issue3390TestCase1{})
|
||||||
|
t.AssertNil(err)
|
||||||
|
command := &Issue3390CommandCase1{root}
|
||||||
|
value, err := command.RunWithSpecificArgs(
|
||||||
|
gctx.New(),
|
||||||
|
[]string{"main", "-a", "aaa", "-b", "bbb"},
|
||||||
|
)
|
||||||
|
t.AssertNil(err)
|
||||||
|
t.Assert(value.(*Issue3390Case1Output).Content, `{"A":"aaa","Be":"bbb"}`)
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
type Issue3390CommandCase2 struct {
|
||||||
|
*gcmd.Command
|
||||||
|
}
|
||||||
|
|
||||||
|
type Issue3390TestCase2 struct {
|
||||||
|
g.Meta `name:"index" ad:"test"`
|
||||||
|
}
|
||||||
|
|
||||||
|
type Issue3390Case2Input struct {
|
||||||
|
g.Meta `name:"index"`
|
||||||
|
A string `short:"b" name:"bb"`
|
||||||
|
Be string `short:"a" name:"aa"`
|
||||||
|
}
|
||||||
|
|
||||||
|
type Issue3390Case2Output struct {
|
||||||
|
Content string
|
||||||
|
}
|
||||||
|
|
||||||
|
func (c Issue3390TestCase2) Index(ctx context.Context, in Issue3390Case2Input) (out *Issue3390Case2Output, err error) {
|
||||||
|
out = &Issue3390Case2Output{
|
||||||
|
Content: gjson.MustEncodeString(in),
|
||||||
|
}
|
||||||
|
return
|
||||||
|
}
|
||||||
|
func Test_Issue3390_Case2(t *testing.T) {
|
||||||
|
gtest.C(t, func(t *gtest.T) {
|
||||||
|
root, err := gcmd.NewFromObject(Issue3390TestCase2{})
|
||||||
|
t.AssertNil(err)
|
||||||
|
command := &Issue3390CommandCase2{root}
|
||||||
|
value, err := command.RunWithSpecificArgs(
|
||||||
|
gctx.New(),
|
||||||
|
[]string{"main", "-a", "aaa", "-b", "bbb"},
|
||||||
|
)
|
||||||
|
t.AssertNil(err)
|
||||||
|
t.Assert(value.(*Issue3390Case2Output).Content, `{"A":"bbb","Be":"aaa"}`)
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
type Issue3390CommandCase3 struct {
|
||||||
|
*gcmd.Command
|
||||||
|
}
|
||||||
|
|
||||||
|
type Issue3390TestCase3 struct {
|
||||||
|
g.Meta `name:"index" ad:"test"`
|
||||||
|
}
|
||||||
|
|
||||||
|
type Issue3390Case3Input struct {
|
||||||
|
g.Meta `name:"index"`
|
||||||
|
A string `short:"b"`
|
||||||
|
Be string `short:"a"`
|
||||||
|
}
|
||||||
|
|
||||||
|
type Issue3390Case3Output struct {
|
||||||
|
Content string
|
||||||
|
}
|
||||||
|
|
||||||
|
func (c Issue3390TestCase3) Index(ctx context.Context, in Issue3390Case3Input) (out *Issue3390Case3Output, err error) {
|
||||||
|
out = &Issue3390Case3Output{
|
||||||
|
Content: gjson.MustEncodeString(in),
|
||||||
|
}
|
||||||
|
return
|
||||||
|
}
|
||||||
|
func Test_Issue3390_Case3(t *testing.T) {
|
||||||
|
gtest.C(t, func(t *gtest.T) {
|
||||||
|
root, err := gcmd.NewFromObject(Issue3390TestCase3{})
|
||||||
|
t.AssertNil(err)
|
||||||
|
command := &Issue3390CommandCase3{root}
|
||||||
|
value, err := command.RunWithSpecificArgs(
|
||||||
|
gctx.New(),
|
||||||
|
[]string{"main", "-a", "aaa", "-b", "bbb"},
|
||||||
|
)
|
||||||
|
t.AssertNil(err)
|
||||||
|
t.Assert(value.(*Issue3390Case3Output).Content, `{"A":"bbb","Be":"aaa"}`)
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
type Issue3390CommandCase4 struct {
|
||||||
|
*gcmd.Command
|
||||||
|
}
|
||||||
|
|
||||||
|
type Issue3390TestCase4 struct {
|
||||||
|
g.Meta `name:"index" ad:"test"`
|
||||||
|
}
|
||||||
|
|
||||||
|
type Issue3390Case4Input struct {
|
||||||
|
g.Meta `name:"index"`
|
||||||
|
A string `short:"a"`
|
||||||
|
Be string `short:"b"`
|
||||||
|
}
|
||||||
|
|
||||||
|
type Issue3390Case4Output struct {
|
||||||
|
Content string
|
||||||
|
}
|
||||||
|
|
||||||
|
func (c Issue3390TestCase4) Index(ctx context.Context, in Issue3390Case4Input) (out *Issue3390Case4Output, err error) {
|
||||||
|
out = &Issue3390Case4Output{
|
||||||
|
Content: gjson.MustEncodeString(in),
|
||||||
|
}
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
func Test_Issue3390_Case4(t *testing.T) {
|
||||||
|
gtest.C(t, func(t *gtest.T) {
|
||||||
|
root, err := gcmd.NewFromObject(Issue3390TestCase4{})
|
||||||
|
t.AssertNil(err)
|
||||||
|
command := &Issue3390CommandCase4{root}
|
||||||
|
value, err := command.RunWithSpecificArgs(
|
||||||
|
gctx.New(),
|
||||||
|
[]string{"main", "-a", "aaa", "-b", "bbb"},
|
||||||
|
)
|
||||||
|
t.AssertNil(err)
|
||||||
|
t.Assert(value.(*Issue3390Case4Output).Content, `{"A":"aaa","Be":"bbb"}`)
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
type Issue3417Test struct {
|
||||||
|
g.Meta `name:"root"`
|
||||||
|
}
|
||||||
|
|
||||||
|
type Issue3417BuildInput struct {
|
||||||
|
g.Meta `name:"build" config:"gfcli.build"`
|
||||||
|
File string `name:"FILE" arg:"true" brief:"building file path"`
|
||||||
|
Name string `short:"n" name:"name" brief:"output binary name"`
|
||||||
|
Version string `short:"v" name:"version" brief:"output binary version"`
|
||||||
|
Arch string `short:"a" name:"arch" brief:"output binary architecture, multiple arch separated with ','"`
|
||||||
|
System string `short:"s" name:"system" brief:"output binary system, multiple os separated with ','"`
|
||||||
|
Output string `short:"o" name:"output" brief:"output binary path, used when building single binary file"`
|
||||||
|
Path string `short:"p" name:"path" brief:"output binary directory path, default is '.'" d:"."`
|
||||||
|
Extra string `short:"e" name:"extra" brief:"extra custom \"go build\" options"`
|
||||||
|
Mod string `short:"m" name:"mod" brief:"like \"-mod\" option of \"go build\", use \"-m none\" to disable go module"`
|
||||||
|
Cgo bool `short:"c" name:"cgo" brief:"enable or disable cgo feature, it's disabled in default" orphan:"true"`
|
||||||
|
VarMap g.Map `short:"r" name:"varMap" brief:"custom built embedded variable into binary"`
|
||||||
|
PackSrc string `short:"ps" name:"packSrc" brief:"pack one or more folders into one go file before building"`
|
||||||
|
PackDst string `short:"pd" name:"packDst" brief:"temporary go file path for pack, this go file will be automatically removed after built" d:"internal/packed/build_pack_data.go"`
|
||||||
|
ExitWhenError bool `short:"ew" name:"exitWhenError" brief:"exit building when any error occurs, specially for multiple arch and system buildings. default is false" orphan:"true"`
|
||||||
|
DumpENV bool `short:"de" name:"dumpEnv" brief:"dump current go build environment before building binary" orphan:"true"`
|
||||||
|
}
|
||||||
|
|
||||||
|
type Issue3417BuildOutput struct {
|
||||||
|
Content string
|
||||||
|
}
|
||||||
|
|
||||||
|
func (c *Issue3417Test) Build(ctx context.Context, in Issue3417BuildInput) (out *Issue3417BuildOutput, err error) {
|
||||||
|
out = &Issue3417BuildOutput{
|
||||||
|
Content: gjson.MustEncodeString(in),
|
||||||
|
}
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
func Test_Issue3417(t *testing.T) {
|
||||||
|
gtest.C(t, func(t *gtest.T) {
|
||||||
|
command, err := gcmd.NewFromObject(Issue3417Test{})
|
||||||
|
t.AssertNil(err)
|
||||||
|
value, err := command.RunWithSpecificArgs(
|
||||||
|
gctx.New(),
|
||||||
|
[]string{
|
||||||
|
"gf", "build",
|
||||||
|
"-mod", "vendor",
|
||||||
|
"-v", "0.0.19",
|
||||||
|
"-n", "detect_hardware_os",
|
||||||
|
"-a", "amd64,arm64",
|
||||||
|
"-s", "linux",
|
||||||
|
"-p", "./bin",
|
||||||
|
"-e", "-trimpath -ldflags",
|
||||||
|
"cmd/v3/main.go",
|
||||||
|
},
|
||||||
|
)
|
||||||
|
t.AssertNil(err)
|
||||||
|
t.Assert(
|
||||||
|
value.(*Issue3417BuildOutput).Content,
|
||||||
|
`{"File":"cmd/v3/main.go","Name":"detect_hardware_os","Version":"0.0.19","Arch":"amd64,arm64","System":"linux","Output":"","Path":"./bin","Extra":"-trimpath -ldflags","Mod":"vendor","Cgo":false,"VarMap":null,"PackSrc":"","PackDst":"internal/packed/build_pack_data.go","ExitWhenError":false,"DumpENV":false}`,
|
||||||
|
)
|
||||||
|
})
|
||||||
|
}
|
@ -144,7 +144,7 @@ func doStruct(
|
|||||||
// return v.UnmarshalValue(params)
|
// return v.UnmarshalValue(params)
|
||||||
// }
|
// }
|
||||||
// Note that it's `pointerElemReflectValue` here not `pointerReflectValue`.
|
// Note that it's `pointerElemReflectValue` here not `pointerReflectValue`.
|
||||||
if ok, err := bindVarToReflectValueWithInterfaceCheck(pointerElemReflectValue, paramsInterface); ok {
|
if ok, err = bindVarToReflectValueWithInterfaceCheck(pointerElemReflectValue, paramsInterface); ok {
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
// Retrieve its element, may be struct at last.
|
// Retrieve its element, may be struct at last.
|
||||||
@ -170,7 +170,7 @@ func doStruct(
|
|||||||
// It only performs one converting to the same attribute.
|
// It only performs one converting to the same attribute.
|
||||||
// doneMap is used to check repeated converting, its key is the real attribute name
|
// doneMap is used to check repeated converting, its key is the real attribute name
|
||||||
// of the struct.
|
// of the struct.
|
||||||
doneMap := make(map[string]struct{})
|
doneAttrMap := make(map[string]struct{})
|
||||||
|
|
||||||
// The key of the attrMap is the attribute name of the struct,
|
// The key of the attrMap is the attribute name of the struct,
|
||||||
// and the value is its replaced name for later comparison to improve performance.
|
// and the value is its replaced name for later comparison to improve performance.
|
||||||
@ -213,10 +213,7 @@ func doStruct(
|
|||||||
|
|
||||||
// The key of the `attrToTagCheckNameMap` is the attribute name of the struct,
|
// The key of the `attrToTagCheckNameMap` is the attribute name of the struct,
|
||||||
// and the value is its replaced tag name for later comparison to improve performance.
|
// and the value is its replaced tag name for later comparison to improve performance.
|
||||||
var (
|
var priorityTagArray []string
|
||||||
attrToTagCheckNameMap = make(map[string]string)
|
|
||||||
priorityTagArray []string
|
|
||||||
)
|
|
||||||
if priorityTag != "" {
|
if priorityTag != "" {
|
||||||
priorityTagArray = append(utils.SplitAndTrim(priorityTag, ","), gtag.StructTagPriority...)
|
priorityTagArray = append(utils.SplitAndTrim(priorityTag, ","), gtag.StructTagPriority...)
|
||||||
} else {
|
} else {
|
||||||
@ -226,64 +223,75 @@ func doStruct(
|
|||||||
if err != nil {
|
if err != nil {
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
|
var toBeDeletedTagNames = make([]string, 0)
|
||||||
for tagName, attributeName := range tagToAttrNameMap {
|
for tagName, attributeName := range tagToAttrNameMap {
|
||||||
// If there's something else in the tag string,
|
// If there's something else in the tag string,
|
||||||
// it uses the first part which is split using char ','.
|
// it uses the first part which is split using char ','.
|
||||||
// Eg:
|
// Example:
|
||||||
// orm:"id, priority"
|
// orm:"id, priority"
|
||||||
// orm:"name, with:uid=id"
|
// orm:"name, with:uid=id"
|
||||||
attrToTagCheckNameMap[attributeName] = utils.RemoveSymbols(strings.Split(tagName, ",")[0])
|
if array := strings.Split(tagName, ","); len(array) > 1 {
|
||||||
|
toBeDeletedTagNames = append(toBeDeletedTagNames, tagName)
|
||||||
|
tagName = array[0]
|
||||||
|
tagToAttrNameMap[tagName] = attributeName
|
||||||
|
}
|
||||||
// If tag and attribute values both exist in `paramsMap`,
|
// If tag and attribute values both exist in `paramsMap`,
|
||||||
// it then uses the tag value overwriting the attribute value in `paramsMap`.
|
// it then uses the tag value overwriting the attribute value in `paramsMap`.
|
||||||
if paramsMap[tagName] != nil && paramsMap[attributeName] != nil {
|
if paramsMap[tagName] != nil && paramsMap[attributeName] != nil {
|
||||||
paramsMap[attributeName] = paramsMap[tagName]
|
paramsMap[attributeName] = paramsMap[tagName]
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
for _, tagName := range toBeDeletedTagNames {
|
||||||
|
delete(tagToAttrNameMap, tagName)
|
||||||
|
}
|
||||||
|
|
||||||
// To convert value base on custom parameter key to attribute name map.
|
// To convert value base on custom parameter key to attribute name map.
|
||||||
err = doStructBaseOnParamKeyToAttrMap(
|
err = doStructBaseOnParamKeyToAttrMap(
|
||||||
pointerElemReflectValue,
|
pointerElemReflectValue,
|
||||||
paramsMap,
|
paramsMap,
|
||||||
paramKeyToAttrMap,
|
paramKeyToAttrMap,
|
||||||
doneMap,
|
doneAttrMap,
|
||||||
)
|
)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
// Already done all attributes value assignment nothing to do next.
|
|
||||||
if len(doneMap) == len(attrToCheckNameMap) {
|
|
||||||
return nil
|
|
||||||
}
|
|
||||||
|
|
||||||
// To convert value base on precise attribute name.
|
err = doStructBaseOnTagToAttrNameMap(
|
||||||
err = doStructBaseOnAttribute(
|
|
||||||
pointerElemReflectValue,
|
pointerElemReflectValue,
|
||||||
paramsMap,
|
paramsMap,
|
||||||
paramKeyToAttrMap,
|
paramKeyToAttrMap,
|
||||||
doneMap,
|
doneAttrMap,
|
||||||
|
tagToAttrNameMap,
|
||||||
|
)
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
|
||||||
|
// To convert value base on precise attribute name.
|
||||||
|
err = doStructBaseOnAttrToCheckNameMap(
|
||||||
|
pointerElemReflectValue,
|
||||||
|
paramsMap,
|
||||||
|
paramKeyToAttrMap,
|
||||||
|
doneAttrMap,
|
||||||
attrToCheckNameMap,
|
attrToCheckNameMap,
|
||||||
)
|
)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
// Already done all attributes value assignment nothing to do next.
|
|
||||||
if len(doneMap) == len(attrToCheckNameMap) {
|
|
||||||
return nil
|
|
||||||
}
|
|
||||||
|
|
||||||
// To convert value base on parameter map.
|
// To convert value base on parameter map.
|
||||||
err = doStructBaseOnParamMap(
|
err = doStructBaseOnParamMap(
|
||||||
pointerElemReflectValue,
|
pointerElemReflectValue,
|
||||||
paramsMap,
|
paramsMap,
|
||||||
paramKeyToAttrMap,
|
paramKeyToAttrMap,
|
||||||
doneMap,
|
doneAttrMap,
|
||||||
attrToCheckNameMap,
|
attrToCheckNameMap,
|
||||||
attrToTagCheckNameMap,
|
|
||||||
tagToAttrNameMap,
|
tagToAttrNameMap,
|
||||||
)
|
)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
|
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -316,33 +324,53 @@ func doStructBaseOnParamKeyToAttrMap(
|
|||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
func doStructBaseOnAttribute(
|
func doStructBaseOnAttrToCheckNameMap(
|
||||||
pointerElemReflectValue reflect.Value,
|
pointerElemReflectValue reflect.Value,
|
||||||
paramsMap map[string]interface{},
|
paramsMap map[string]interface{},
|
||||||
paramKeyToAttrMap map[string]string,
|
paramKeyToAttrMap map[string]string,
|
||||||
doneAttrMap map[string]struct{},
|
doneAttrMap map[string]struct{},
|
||||||
attrToCheckNameMap map[string]string,
|
attrToCheckNameMap map[string]string,
|
||||||
) error {
|
) error {
|
||||||
var customMappingAttrMap = make(map[string]struct{})
|
|
||||||
if len(paramKeyToAttrMap) > 0 {
|
|
||||||
// It ignores the attribute names if it is specified in the `paramKeyToAttrMap`.
|
|
||||||
for paramName := range paramsMap {
|
|
||||||
if passedAttrKey, ok := paramKeyToAttrMap[paramName]; ok {
|
|
||||||
customMappingAttrMap[passedAttrKey] = struct{}{}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
for attrName := range attrToCheckNameMap {
|
for attrName := range attrToCheckNameMap {
|
||||||
// The value by precise attribute name.
|
// The value by precise attribute name.
|
||||||
paramValue, ok := paramsMap[attrName]
|
paramValue, ok := paramsMap[attrName]
|
||||||
if !ok {
|
if !ok {
|
||||||
continue
|
continue
|
||||||
}
|
}
|
||||||
// If the attribute name is in custom paramKeyToAttrMap, it then ignores this converting.
|
// If the attribute name is already converted, it then skips it.
|
||||||
if _, ok = customMappingAttrMap[attrName]; ok {
|
if _, ok = doneAttrMap[attrName]; ok {
|
||||||
continue
|
continue
|
||||||
}
|
}
|
||||||
// If the attribute name is already checked converting, then skip it.
|
// Mark it done.
|
||||||
|
doneAttrMap[attrName] = struct{}{}
|
||||||
|
if err := bindVarToStructAttr(
|
||||||
|
pointerElemReflectValue, attrName, paramValue, paramKeyToAttrMap,
|
||||||
|
); err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func doStructBaseOnTagToAttrNameMap(
|
||||||
|
pointerElemReflectValue reflect.Value,
|
||||||
|
paramsMap map[string]interface{},
|
||||||
|
paramKeyToAttrMap map[string]string,
|
||||||
|
doneAttrMap map[string]struct{},
|
||||||
|
tagToAttrNameMap map[string]string,
|
||||||
|
) error {
|
||||||
|
var (
|
||||||
|
paramValue interface{}
|
||||||
|
ok bool
|
||||||
|
)
|
||||||
|
for tagName, attrName := range tagToAttrNameMap {
|
||||||
|
// It firstly considers `paramName` as accurate tag name,
|
||||||
|
// and retrieve attribute name from `tagToAttrNameMap` .
|
||||||
|
paramValue, ok = paramsMap[tagName]
|
||||||
|
if !ok {
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
// If the attribute name is already converted, it then skips it.
|
||||||
if _, ok = doneAttrMap[attrName]; ok {
|
if _, ok = doneAttrMap[attrName]; ok {
|
||||||
continue
|
continue
|
||||||
}
|
}
|
||||||
@ -363,44 +391,34 @@ func doStructBaseOnParamMap(
|
|||||||
paramKeyToAttrMap map[string]string,
|
paramKeyToAttrMap map[string]string,
|
||||||
doneAttrMap map[string]struct{},
|
doneAttrMap map[string]struct{},
|
||||||
attrToCheckNameMap map[string]string,
|
attrToCheckNameMap map[string]string,
|
||||||
attrToTagCheckNameMap map[string]string,
|
|
||||||
tagToAttrNameMap map[string]string,
|
tagToAttrNameMap map[string]string,
|
||||||
) error {
|
) error {
|
||||||
var (
|
var (
|
||||||
attrName string
|
attrName string
|
||||||
checkName string
|
paramNameWithoutSymbols string
|
||||||
|
ok bool
|
||||||
)
|
)
|
||||||
for paramName, paramValue := range paramsMap {
|
for paramName, paramValue := range paramsMap {
|
||||||
// It firstly considers `paramName` as accurate tag name,
|
// It was already converted in previous procedure.
|
||||||
// and retrieve attribute name from `tagToAttrNameMap` .
|
if _, ok = tagToAttrNameMap[paramName]; ok {
|
||||||
attrName = tagToAttrNameMap[paramName]
|
continue
|
||||||
if attrName == "" {
|
}
|
||||||
checkName = utils.RemoveSymbols(paramName)
|
// It was already converted in previous procedure.
|
||||||
// Loop to find the matched attribute name with or without
|
if _, ok = attrToCheckNameMap[paramName]; ok {
|
||||||
// string cases and chars like '-'/'_'/'.'/' '.
|
continue
|
||||||
|
|
||||||
// Matching the parameters to struct tag names.
|
|
||||||
// The `attrKey` is the attribute name of the struct.
|
|
||||||
for attrKey, cmpKey := range attrToTagCheckNameMap {
|
|
||||||
if strings.EqualFold(checkName, cmpKey) {
|
|
||||||
attrName = attrKey
|
|
||||||
break
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
paramNameWithoutSymbols = utils.RemoveSymbols(paramName)
|
||||||
// Matching the parameters to struct attributes.
|
// Matching the parameters to struct attributes.
|
||||||
if attrName == "" {
|
for attrKey, cmpKey := range attrToCheckNameMap {
|
||||||
for attrKey, cmpKey := range attrToCheckNameMap {
|
// Example:
|
||||||
// Eg:
|
// UserName eq user_name
|
||||||
// UserName eq user_name
|
// User-Name eq username
|
||||||
// User-Name eq username
|
// username eq userName
|
||||||
// username eq userName
|
// etc.
|
||||||
// etc.
|
if strings.EqualFold(paramNameWithoutSymbols, cmpKey) {
|
||||||
if strings.EqualFold(checkName, cmpKey) {
|
attrName = attrKey
|
||||||
attrName = attrKey
|
break
|
||||||
break
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -408,8 +426,8 @@ func doStructBaseOnParamMap(
|
|||||||
if attrName == "" {
|
if attrName == "" {
|
||||||
continue
|
continue
|
||||||
}
|
}
|
||||||
// If the attribute name is already checked converting, then skip it.
|
// If the attribute name is already converted, it then skips it.
|
||||||
if _, ok := doneAttrMap[attrName]; ok {
|
if _, ok = doneAttrMap[attrName]; ok {
|
||||||
continue
|
continue
|
||||||
}
|
}
|
||||||
// Mark it done.
|
// Mark it done.
|
||||||
|
@ -98,7 +98,7 @@ func Test_Issue1227(t *testing.T) {
|
|||||||
{
|
{
|
||||||
name: "Case5",
|
name: "Case5",
|
||||||
origin: g.Map{"中文KEY": "n1"},
|
origin: g.Map{"中文KEY": "n1"},
|
||||||
want: "n1",
|
want: "",
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
name: "Case5",
|
name: "Case5",
|
||||||
@ -111,6 +111,7 @@ func Test_Issue1227(t *testing.T) {
|
|||||||
if err := gconv.Struct(tt.origin, &p); err != nil {
|
if err := gconv.Struct(tt.origin, &p); err != nil {
|
||||||
t.Error(err)
|
t.Error(err)
|
||||||
}
|
}
|
||||||
|
//t.Log(tt)
|
||||||
t.Assert(p.Name, tt.want)
|
t.Assert(p.Name, tt.want)
|
||||||
}
|
}
|
||||||
})
|
})
|
||||||
|
@ -567,7 +567,7 @@ func Test_StructEmbedded5(t *testing.T) {
|
|||||||
err = gconv.Struct(data, user1)
|
err = gconv.Struct(data, user1)
|
||||||
t.AssertNil(err)
|
t.AssertNil(err)
|
||||||
t.Assert(user1, &UserWithBase1{1, "john", Base{"123", "456"}})
|
t.Assert(user1, &UserWithBase1{1, "john", Base{"123", "456"}})
|
||||||
|
return
|
||||||
err = gconv.Struct(data, user2)
|
err = gconv.Struct(data, user2)
|
||||||
t.AssertNil(err)
|
t.AssertNil(err)
|
||||||
t.Assert(user2, &UserWithBase2{1, "john", Base{"", ""}})
|
t.Assert(user2, &UserWithBase2{1, "john", Base{"", ""}})
|
||||||
|
Loading…
x
Reference in New Issue
Block a user