-
Notifications
You must be signed in to change notification settings - Fork 53
/
bind_router.go
120 lines (107 loc) · 3.02 KB
/
bind_router.go
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
package znet
import (
"fmt"
"reflect"
"strings"
"github.com/sohaha/zlsgo/zreflect"
"github.com/sohaha/zlsgo/zstring"
"github.com/sohaha/zlsgo/zutil"
)
// BindStruct Bind Struct
func (e *Engine) BindStruct(prefix string, s interface{}, handle ...Handler) error {
g := e.Group(prefix)
if len(handle) > 0 {
for _, v := range handle {
g.Use(v)
}
}
of := reflect.ValueOf(s)
typ, err := zreflect.NewVal(of)
if err != nil {
return err
}
initFn := of.MethodByName("Init")
if initFn.IsValid() {
before, ok := initFn.Interface().(func(e *Engine))
if ok {
before(g)
} else {
if before, ok := initFn.Interface().(func(e *Engine) error); !ok {
return fmt.Errorf("func: [%s] is not an effective routing method", "Init")
} else {
if err := before(g); err != nil {
return err
}
}
}
}
handleName := "reflect.methodValueCall"
typeOf := typ.TypeOf()
// methods := strings.Split("ANY|GET|Post|POST|PUT|DELETE|PATCH|HEAD|OPTIONS", "|")
return zutil.TryCatch(func() error {
return typ.ForEachMethod(func(i int, m reflect.Method, value reflect.Value) error {
if m.Name == "Init" {
return nil
}
path, method, key := "", "", ""
methods := `ANY|GET|POST|PUT|DELETE|PATCH|HEAD|OPTIONS`
// regex := `^(` + methods + `){0,}([a-zA-Z]+){0,}(?i)(` + methods + `)(.*)`
// regex := `^(!(?i:` + methods + `)?([a-zA-Z]+))?(?i)(` + methods + `)(.*)$`
regex := `^(?i)(` + methods + `)(.*)$`
info, err := zstring.RegexExtract(regex, m.Name)
// info := strings.Split(zstring.CamelCaseToSnakeCase(m.Name, "-"), "-")
infoLen := len(info)
// Log.Dump(zstring.CamelCaseToSnakeCase(m.Name, "-"))
if err != nil || infoLen != 3 {
indexs := zstring.RegexFind(`(?i)(`+methods+`)`, m.Name, 1)
if len(indexs) == 0 {
if e.IsDebug() && m.Name != "Init" {
e.Log.Warnf("matching rule error: %s%s\n", m.Name, m.Func.String())
}
return nil
}
index := indexs[0]
method = strings.ToUpper(m.Name[index[0]:index[1]])
path = m.Name[index[1]:]
key = strings.ToLower(m.Name[:index[0]])
} else {
path = info[2]
method = strings.ToUpper(info[1])
}
fn := value.Interface()
handleName = strings.Join([]string{typeOf.PkgPath(), typeOf.Name(), m.Name}, ".")
if e.BindStructDelimiter != "" {
path = zstring.CamelCaseToSnakeCase(path, e.BindStructDelimiter)
}
if path == "" {
path = "/"
}
if key != "" {
if strings.HasSuffix(path, "/") {
path += ":" + key
} else {
path += "/:" + key
}
} else if path != "/" && e.BindStructSuffix != "" {
path = path + e.BindStructSuffix
}
if path == "/" {
path = ""
}
var (
p string
l int
ok bool
)
if method == "ANY" {
p, l, ok = g.handleAny(path, Utils.ParseHandlerFunc(fn), nil, nil)
} else {
p, l, ok = g.addHandle(method, path, Utils.ParseHandlerFunc(fn), nil, nil)
}
if ok && e.IsDebug() {
e.Log.Debug(routeLog(e.Log, fmt.Sprintf("%%s %%-40s -> %s (%d handlers)", handleName, l), method, p))
}
return nil
})
})
}