1
0
mirror of https://github.com/gogf/gf.git synced 2025-04-05 03:05:05 +08:00

feat(net/ghttp): add GetMetaTag function to retrieve metadata value for HandlerItem (#4206)

This commit is contained in:
John Guo 2025-03-17 09:21:00 +08:00 committed by GitHub
parent bc1e1019c5
commit 07696fc779
No known key found for this signature in database
GPG Key ID: B5690EEEBB952194
4 changed files with 89 additions and 44 deletions

View File

@ -64,11 +64,11 @@ type (
Handler *HandlerItem // The handler. Handler *HandlerItem // The handler.
Server string // Server name. Server string // Server name.
Address string // Listening address. Address string // Listening address.
Domain string // Bound domain. Domain string // Bound domain, eg: example.com
Type HandlerType // Route handler type. Type HandlerType // Route handler type.
Middleware string // Bound middleware. Middleware string // Bound middleware.
Method string // Handler method name. Method string // Handler method name, eg: get, post.
Route string // Route URI. Route string // Route URI, eg: /api/v1/user/{id}.
Priority int // Just for reference. Priority int // Just for reference.
IsServiceHandler bool // Is a service handler. IsServiceHandler bool // Is a service handler.
} }

View File

@ -6,8 +6,6 @@
package ghttp package ghttp
import "github.com/gogf/gf/v2/util/gmeta"
// GetHandlerResponse retrieves and returns the handler response object and its error. // GetHandlerResponse retrieves and returns the handler response object and its error.
func (r *Request) GetHandlerResponse() interface{} { func (r *Request) GetHandlerResponse() interface{} {
return r.handlerResponse return r.handlerResponse
@ -17,25 +15,3 @@ func (r *Request) GetHandlerResponse() interface{} {
func (r *Request) GetServeHandler() *HandlerItemParsed { func (r *Request) GetServeHandler() *HandlerItemParsed {
return r.serveHandler return r.serveHandler
} }
// GetMetaTag retrieves and returns the metadata value associated with the given key from the request struct.
// The meta value is from struct tags from g.Meta/gmeta.Meta type.
// For example:
//
// type GetMetaTagReq struct {
// g.Meta `path:"/test" method:"post" summary:"meta_tag" tags:"meta"`
// // ...
// }
//
// r.GetServeHandler().GetMetaTag("summary") // returns "meta_tag"
// r.GetServeHandler().GetMetaTag("method") // returns "post"
func (h *HandlerItemParsed) GetMetaTag(key string) string {
if h == nil || h.Handler == nil {
return ""
}
metaValue := gmeta.Get(h.Handler.Info.Type.In(1), key)
if metaValue != nil {
return metaValue.String()
}
return ""
}

View File

@ -18,6 +18,7 @@ import (
"github.com/gogf/gf/v2/internal/intlog" "github.com/gogf/gf/v2/internal/intlog"
"github.com/gogf/gf/v2/internal/json" "github.com/gogf/gf/v2/internal/json"
"github.com/gogf/gf/v2/text/gregex" "github.com/gogf/gf/v2/text/gregex"
"github.com/gogf/gf/v2/util/gmeta"
) )
// handlerCacheItem is an item just for internal router searching cache. // handlerCacheItem is an item just for internal router searching cache.
@ -252,34 +253,34 @@ func (s *Server) searchHandlers(method, path, domain string) (parsedItems []*Han
} }
// MarshalJSON implements the interface MarshalJSON for json.Marshal. // MarshalJSON implements the interface MarshalJSON for json.Marshal.
func (item HandlerItem) MarshalJSON() ([]byte, error) { func (h *HandlerItem) MarshalJSON() ([]byte, error) {
switch item.Type { switch h.Type {
case HandlerTypeHook: case HandlerTypeHook:
return json.Marshal( return json.Marshal(
fmt.Sprintf( fmt.Sprintf(
`%s %s:%s (%s)`, `%s %s:%s (%s)`,
item.Router.Uri, h.Router.Uri,
item.Router.Domain, h.Router.Domain,
item.Router.Method, h.Router.Method,
item.HookName, h.HookName,
), ),
) )
case HandlerTypeMiddleware: case HandlerTypeMiddleware:
return json.Marshal( return json.Marshal(
fmt.Sprintf( fmt.Sprintf(
`%s %s:%s (MIDDLEWARE)`, `%s %s:%s (MIDDLEWARE)`,
item.Router.Uri, h.Router.Uri,
item.Router.Domain, h.Router.Domain,
item.Router.Method, h.Router.Method,
), ),
) )
default: default:
return json.Marshal( return json.Marshal(
fmt.Sprintf( fmt.Sprintf(
`%s %s:%s`, `%s %s:%s`,
item.Router.Uri, h.Router.Uri,
item.Router.Domain, h.Router.Domain,
item.Router.Method, h.Router.Method,
), ),
) )
} }
@ -289,3 +290,34 @@ func (item HandlerItem) MarshalJSON() ([]byte, error) {
func (h *HandlerItemParsed) MarshalJSON() ([]byte, error) { func (h *HandlerItemParsed) MarshalJSON() ([]byte, error) {
return json.Marshal(h.Handler) return json.Marshal(h.Handler)
} }
// GetMetaTag retrieves and returns the metadata value associated with the given key from the request struct.
// The meta value is from struct tags from g.Meta/gmeta.Meta type.
func (h *HandlerItem) GetMetaTag(key string) string {
if h == nil {
return ""
}
metaValue := gmeta.Get(h.Info.Type.In(1), key)
if metaValue != nil {
return metaValue.String()
}
return ""
}
// GetMetaTag retrieves and returns the metadata value associated with the given key from the request struct.
// The meta value is from struct tags from g.Meta/gmeta.Meta type.
// For example:
//
// type GetMetaTagReq struct {
// g.Meta `path:"/test" method:"post" summary:"meta_tag" tags:"meta"`
// // ...
// }
//
// r.GetServeHandler().GetMetaTag("summary") // returns "meta_tag"
// r.GetServeHandler().GetMetaTag("method") // returns "post"
func (h *HandlerItemParsed) GetMetaTag(key string) string {
if h == nil || h.Handler == nil {
return ""
}
return h.Handler.GetMetaTag(key)
}

View File

@ -21,7 +21,7 @@ import (
"github.com/gogf/gf/v2/util/guid" "github.com/gogf/gf/v2/util/guid"
) )
func Test_Router_Handler_Strict_WithObject(t *testing.T) { func Test_Router_Handler_Standard_WithObject(t *testing.T) {
type TestReq struct { type TestReq struct {
Age int Age int
Name string Name string
@ -137,7 +137,7 @@ func (ControllerForHandlerWithObjectAndMeta2) Test4(ctx context.Context, req *Te
}, nil }, nil
} }
func Test_Router_Handler_Strict_WithObjectAndMeta(t *testing.T) { func Test_Router_Handler_Standard_WithObjectAndMeta(t *testing.T) {
s := g.Server(guid.S()) s := g.Server(guid.S())
s.Use(ghttp.MiddlewareHandlerResponse) s.Use(ghttp.MiddlewareHandlerResponse)
s.Group("/", func(group *ghttp.RouterGroup) { s.Group("/", func(group *ghttp.RouterGroup) {
@ -159,7 +159,7 @@ func Test_Router_Handler_Strict_WithObjectAndMeta(t *testing.T) {
}) })
} }
func Test_Router_Handler_Strict_Group_Bind(t *testing.T) { func Test_Router_Handler_Standard_Group_Bind(t *testing.T) {
s := g.Server(guid.S()) s := g.Server(guid.S())
s.Use(ghttp.MiddlewareHandlerResponse) s.Use(ghttp.MiddlewareHandlerResponse)
s.Group("/api/v1", func(group *ghttp.RouterGroup) { s.Group("/api/v1", func(group *ghttp.RouterGroup) {
@ -300,7 +300,7 @@ func Test_Custom_Slice_Type_Attribute(t *testing.T) {
}) })
} }
func Test_Router_Handler_Strict_WithGeneric(t *testing.T) { func Test_Router_Handler_Standard_WithGeneric(t *testing.T) {
type TestReq struct { type TestReq struct {
Age int Age int
} }
@ -397,7 +397,7 @@ func (c *ParameterCaseSensitiveController) Path(
return &ParameterCaseSensitiveControllerPathRes{Path: req.Path}, nil return &ParameterCaseSensitiveControllerPathRes{Path: req.Path}, nil
} }
func Test_Router_Handler_Strict_ParameterCaseSensitive(t *testing.T) { func Test_Router_Handler_Standard_ParameterCaseSensitive(t *testing.T) {
s := g.Server(guid.S()) s := g.Server(guid.S())
s.Use(ghttp.MiddlewareHandlerResponse) s.Use(ghttp.MiddlewareHandlerResponse)
s.Group("/", func(group *ghttp.RouterGroup) { s.Group("/", func(group *ghttp.RouterGroup) {
@ -534,3 +534,40 @@ func Test_NullString_Issue3465(t *testing.T) {
}) })
} }
type testHandlerItemGetMetaTagReq struct {
g.Meta `path:"/test" method:"get" sm:"hello" tags:"示例"`
}
type testHandlerItemGetMetaTagRes struct{}
type testHandlerItemGetMetaTag struct {
}
func (t *testHandlerItemGetMetaTag) Test(ctx context.Context, req *testHandlerItemGetMetaTagReq) (res *testHandlerItemGetMetaTagRes, err error) {
return nil, nil
}
func TestHandlerItem_GetMetaTag(t *testing.T) {
s := g.Server(guid.S())
s.Use(ghttp.MiddlewareHandlerResponse)
s.Group("/", func(group *ghttp.RouterGroup) {
group.Bind(new(testHandlerItemGetMetaTag))
})
s.SetDumpRouterMap(false)
s.Start()
defer s.Shutdown()
time.Sleep(100 * time.Millisecond)
gtest.C(t, func(t *gtest.T) {
routes := s.GetRoutes()
for _, route := range routes {
if !route.IsServiceHandler {
continue
}
t.Assert(route.Handler.GetMetaTag("path"), "/test")
t.Assert(route.Handler.GetMetaTag("method"), "get")
t.Assert(route.Handler.GetMetaTag("sm"), "hello")
t.Assert(route.Handler.GetMetaTag("tags"), "示例")
}
})
}