2018-07-31 21:05:02 +08:00
|
|
|
|
// Copyright 2018 gf Author(https://gitee.com/johng/gf). 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://gitee.com/johng/gf.
|
|
|
|
|
// 服务注册.
|
|
|
|
|
|
|
|
|
|
package ghttp
|
|
|
|
|
|
|
|
|
|
import (
|
|
|
|
|
"errors"
|
2018-10-31 22:39:58 +08:00
|
|
|
|
"gitee.com/johng/gf/g/os/glog"
|
2018-07-31 21:05:02 +08:00
|
|
|
|
"strings"
|
|
|
|
|
"reflect"
|
2018-10-17 10:09:42 +08:00
|
|
|
|
"fmt"
|
|
|
|
|
"gitee.com/johng/gf/g/util/gstr"
|
|
|
|
|
"gitee.com/johng/gf/g/os/gfile"
|
2018-07-31 21:05:02 +08:00
|
|
|
|
)
|
|
|
|
|
|
|
|
|
|
// 绑定对象到URI请求处理中,会自动识别方法名称,并附加到对应的URI地址后面
|
2018-08-16 21:33:47 +08:00
|
|
|
|
// 第三个参数methods用以指定需要注册的方法,支持多个方法名称,多个方法以英文“,”号分隔,区分大小写
|
2018-07-31 21:05:02 +08:00
|
|
|
|
func (s *Server)BindObject(pattern string, obj interface{}, methods...string) error {
|
2018-08-16 21:33:47 +08:00
|
|
|
|
methodMap := (map[string]bool)(nil)
|
2018-07-31 21:05:02 +08:00
|
|
|
|
if len(methods) > 0 {
|
2018-08-16 21:33:47 +08:00
|
|
|
|
methodMap = make(map[string]bool)
|
|
|
|
|
for _, v := range strings.Split(methods[0], ",") {
|
|
|
|
|
methodMap[strings.TrimSpace(v)] = true
|
|
|
|
|
}
|
2018-07-31 21:05:02 +08:00
|
|
|
|
}
|
2018-10-17 10:09:42 +08:00
|
|
|
|
m := make(handlerMap)
|
|
|
|
|
v := reflect.ValueOf(obj)
|
|
|
|
|
t := v.Type()
|
2018-07-31 21:05:02 +08:00
|
|
|
|
sname := t.Elem().Name()
|
2018-08-17 19:01:49 +08:00
|
|
|
|
finit := (func(*Request))(nil)
|
|
|
|
|
fshut := (func(*Request))(nil)
|
|
|
|
|
if v.MethodByName("Init").IsValid() {
|
|
|
|
|
finit = v.MethodByName("Init").Interface().(func(*Request))
|
|
|
|
|
}
|
|
|
|
|
if v.MethodByName("Shut").IsValid() {
|
|
|
|
|
fshut = v.MethodByName("Shut").Interface().(func(*Request))
|
|
|
|
|
}
|
2018-10-17 10:09:42 +08:00
|
|
|
|
pkgPath := t.Elem().PkgPath()
|
|
|
|
|
pkgName := gfile.Basename(pkgPath)
|
2018-07-31 21:05:02 +08:00
|
|
|
|
for i := 0; i < v.NumMethod(); i++ {
|
2018-08-16 21:33:47 +08:00
|
|
|
|
mname := t.Method(i).Name
|
|
|
|
|
if methodMap != nil && !methodMap[mname] {
|
|
|
|
|
continue
|
|
|
|
|
}
|
2018-08-17 19:01:49 +08:00
|
|
|
|
if mname == "Init" || mname == "Shut" {
|
|
|
|
|
continue
|
|
|
|
|
}
|
2018-10-31 22:39:58 +08:00
|
|
|
|
faddr, ok := v.Method(i).Interface().(func(*Request))
|
|
|
|
|
if !ok {
|
|
|
|
|
if methodMap != nil {
|
|
|
|
|
s := fmt.Sprintf(`invalid medthod definition "%s", while "func(*Request))" is required`, v.Method(i).Type().String())
|
2018-11-01 08:37:23 +08:00
|
|
|
|
glog.Error(s)
|
2018-10-31 22:39:58 +08:00
|
|
|
|
return errors.New(s)
|
|
|
|
|
}
|
|
|
|
|
continue
|
|
|
|
|
}
|
2018-10-17 10:09:42 +08:00
|
|
|
|
objName := gstr.Replace(t.String(), fmt.Sprintf(`%s.`, pkgName), "")
|
|
|
|
|
if objName[0] == '*' {
|
|
|
|
|
objName = fmt.Sprintf(`(%s)`, objName)
|
|
|
|
|
}
|
2018-08-16 21:33:47 +08:00
|
|
|
|
key := s.mergeBuildInNameToPattern(pattern, sname, mname, true)
|
2018-07-31 21:05:02 +08:00
|
|
|
|
m[key] = &handlerItem {
|
2018-10-17 10:09:42 +08:00
|
|
|
|
name : fmt.Sprintf(`%s.%s.%s`, pkgPath, objName, mname),
|
2018-08-19 11:25:15 +08:00
|
|
|
|
rtype : gROUTE_REGISTER_OBJECT,
|
2018-07-31 21:05:02 +08:00
|
|
|
|
ctype : nil,
|
|
|
|
|
fname : "",
|
2018-10-31 22:39:58 +08:00
|
|
|
|
faddr : faddr,
|
2018-08-17 19:01:49 +08:00
|
|
|
|
finit : finit,
|
|
|
|
|
fshut : fshut,
|
2018-07-31 21:05:02 +08:00
|
|
|
|
}
|
|
|
|
|
// 如果方法中带有Index方法,那么额外自动增加一个路由规则匹配主URI
|
2018-08-16 21:33:47 +08:00
|
|
|
|
if strings.EqualFold(mname, "Index") {
|
2018-07-31 21:05:02 +08:00
|
|
|
|
p := key
|
|
|
|
|
if strings.EqualFold(p[len(p) - 6:], "/index") {
|
|
|
|
|
p = p[0 : len(p) - 6]
|
2018-08-08 19:03:48 +08:00
|
|
|
|
if len(p) == 0 {
|
|
|
|
|
p = "/"
|
|
|
|
|
}
|
2018-07-31 21:05:02 +08:00
|
|
|
|
}
|
|
|
|
|
m[p] = &handlerItem {
|
2018-10-17 10:09:42 +08:00
|
|
|
|
name : fmt.Sprintf(`%s.%s.%s`, pkgPath, objName, mname),
|
2018-08-19 11:25:15 +08:00
|
|
|
|
rtype : gROUTE_REGISTER_OBJECT,
|
2018-07-31 21:05:02 +08:00
|
|
|
|
ctype : nil,
|
|
|
|
|
fname : "",
|
2018-10-31 22:39:58 +08:00
|
|
|
|
faddr : faddr,
|
2018-08-17 19:01:49 +08:00
|
|
|
|
finit : finit,
|
|
|
|
|
fshut : fshut,
|
2018-07-31 21:05:02 +08:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
return s.bindHandlerByMap(m)
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// 绑定对象到URI请求处理中,会自动识别方法名称,并附加到对应的URI地址后面
|
|
|
|
|
// 第三个参数methods支持多个方法注册,多个方法以英文“,”号分隔,区分大小写
|
2018-08-16 21:33:47 +08:00
|
|
|
|
func (s *Server)BindObjectMethod(pattern string, obj interface{}, method string) error {
|
2018-07-31 21:05:02 +08:00
|
|
|
|
m := make(handlerMap)
|
|
|
|
|
v := reflect.ValueOf(obj)
|
|
|
|
|
t := v.Type()
|
|
|
|
|
sname := t.Elem().Name()
|
2018-08-16 21:33:47 +08:00
|
|
|
|
mname := strings.TrimSpace(method)
|
|
|
|
|
fval := v.MethodByName(mname)
|
|
|
|
|
if !fval.IsValid() {
|
|
|
|
|
return errors.New("invalid method name:" + mname)
|
|
|
|
|
}
|
2018-10-31 22:39:58 +08:00
|
|
|
|
faddr, ok := fval.Interface().(func(*Request))
|
|
|
|
|
if !ok {
|
|
|
|
|
s := fmt.Sprintf(`invalid medthod definition "%s", while "func(*Request)" is required`, fval.Type().String())
|
2018-11-01 08:37:23 +08:00
|
|
|
|
glog.Error(s)
|
2018-10-31 22:39:58 +08:00
|
|
|
|
return errors.New(s)
|
|
|
|
|
}
|
2018-08-17 19:01:49 +08:00
|
|
|
|
finit := (func(*Request))(nil)
|
|
|
|
|
fshut := (func(*Request))(nil)
|
|
|
|
|
if v.MethodByName("Init").IsValid() {
|
|
|
|
|
finit = v.MethodByName("Init").Interface().(func(*Request))
|
|
|
|
|
}
|
|
|
|
|
if v.MethodByName("Shut").IsValid() {
|
|
|
|
|
fshut = v.MethodByName("Shut").Interface().(func(*Request))
|
|
|
|
|
}
|
2018-10-17 10:09:42 +08:00
|
|
|
|
pkgPath := t.Elem().PkgPath()
|
|
|
|
|
pkgName := gfile.Basename(pkgPath)
|
|
|
|
|
objName := gstr.Replace(t.String(), fmt.Sprintf(`%s.`, pkgName), "")
|
|
|
|
|
if objName[0] == '*' {
|
|
|
|
|
objName = fmt.Sprintf(`(%s)`, objName)
|
|
|
|
|
}
|
2018-08-16 21:33:47 +08:00
|
|
|
|
key := s.mergeBuildInNameToPattern(pattern, sname, mname, false)
|
|
|
|
|
m[key] = &handlerItem{
|
2018-10-17 10:09:42 +08:00
|
|
|
|
name : fmt.Sprintf(`%s.%s.%s`, pkgPath, objName, mname),
|
2018-08-19 11:25:15 +08:00
|
|
|
|
rtype : gROUTE_REGISTER_OBJECT,
|
2018-08-16 21:33:47 +08:00
|
|
|
|
ctype : nil,
|
|
|
|
|
fname : "",
|
2018-10-31 22:39:58 +08:00
|
|
|
|
faddr : faddr,
|
2018-08-17 19:01:49 +08:00
|
|
|
|
finit : finit,
|
|
|
|
|
fshut : fshut,
|
2018-07-31 21:05:02 +08:00
|
|
|
|
}
|
2018-08-16 21:33:47 +08:00
|
|
|
|
|
2018-07-31 21:05:02 +08:00
|
|
|
|
return s.bindHandlerByMap(m)
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// 绑定对象到URI请求处理中,会自动识别方法名称,并附加到对应的URI地址后面
|
|
|
|
|
// 需要注意对象方法的定义必须按照ghttp.HandlerFunc来定义
|
|
|
|
|
func (s *Server)BindObjectRest(pattern string, obj interface{}) error {
|
2018-08-17 19:01:49 +08:00
|
|
|
|
m := make(handlerMap)
|
|
|
|
|
v := reflect.ValueOf(obj)
|
|
|
|
|
t := v.Type()
|
|
|
|
|
finit := (func(*Request))(nil)
|
|
|
|
|
fshut := (func(*Request))(nil)
|
|
|
|
|
if v.MethodByName("Init").IsValid() {
|
|
|
|
|
finit = v.MethodByName("Init").Interface().(func(*Request))
|
|
|
|
|
}
|
|
|
|
|
if v.MethodByName("Shut").IsValid() {
|
|
|
|
|
fshut = v.MethodByName("Shut").Interface().(func(*Request))
|
|
|
|
|
}
|
2018-10-17 10:09:42 +08:00
|
|
|
|
pkgPath := t.Elem().PkgPath()
|
2018-07-31 21:05:02 +08:00
|
|
|
|
for i := 0; i < v.NumMethod(); i++ {
|
2018-10-17 10:09:42 +08:00
|
|
|
|
mname := t.Method(i).Name
|
|
|
|
|
method := strings.ToUpper(mname)
|
2018-07-31 21:05:02 +08:00
|
|
|
|
if _, ok := s.methodsMap[method]; !ok {
|
|
|
|
|
continue
|
|
|
|
|
}
|
2018-10-31 22:39:58 +08:00
|
|
|
|
faddr, ok := v.Method(i).Interface().(func(*Request))
|
|
|
|
|
if !ok {
|
|
|
|
|
s := fmt.Sprintf(`invalid medthod definition "%s", while "func()" is required`, v.Method(i).Type().String())
|
2018-11-01 08:37:23 +08:00
|
|
|
|
glog.Error(s)
|
2018-10-31 22:39:58 +08:00
|
|
|
|
return errors.New(s)
|
|
|
|
|
}
|
2018-10-17 10:09:42 +08:00
|
|
|
|
pkgName := gfile.Basename(pkgPath)
|
|
|
|
|
objName := gstr.Replace(t.String(), fmt.Sprintf(`%s.`, pkgName), "")
|
|
|
|
|
if objName[0] == '*' {
|
|
|
|
|
objName = fmt.Sprintf(`(%s)`, objName)
|
|
|
|
|
}
|
|
|
|
|
key := mname + ":" + pattern
|
2018-07-31 21:05:02 +08:00
|
|
|
|
m[key] = &handlerItem {
|
2018-10-17 10:09:42 +08:00
|
|
|
|
name : fmt.Sprintf(`%s.%s.%s`, pkgPath, objName, mname),
|
2018-08-19 11:25:15 +08:00
|
|
|
|
rtype : gROUTE_REGISTER_OBJECT,
|
2018-07-31 21:05:02 +08:00
|
|
|
|
ctype : nil,
|
|
|
|
|
fname : "",
|
2018-10-31 22:39:58 +08:00
|
|
|
|
faddr : faddr,
|
2018-08-17 19:01:49 +08:00
|
|
|
|
finit : finit,
|
|
|
|
|
fshut : fshut,
|
2018-07-31 21:05:02 +08:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
return s.bindHandlerByMap(m)
|
|
|
|
|
}
|