mirror of
https://github.com/gogf/gf.git
synced 2025-04-05 03:05:05 +08:00
This commit is contained in:
parent
3b1d1d33b8
commit
8e075fba53
@ -26,12 +26,6 @@ type implementer struct {
|
||||
}
|
||||
|
||||
func New(config httpclient.Config) IClient {
|
||||
if !gstr.HasPrefix(config.URL, "http") {
|
||||
config.URL = fmt.Sprintf("http://%s", config.URL)
|
||||
}
|
||||
if config.Logger == nil {
|
||||
config.Logger = g.Log()
|
||||
}
|
||||
return &implementer{
|
||||
config: config,
|
||||
}
|
||||
|
@ -9,11 +9,10 @@ package httpclient
|
||||
|
||||
import (
|
||||
"context"
|
||||
"encoding/json"
|
||||
"fmt"
|
||||
"net/http"
|
||||
|
||||
"github.com/gogf/gf/v2/encoding/gurl"
|
||||
"github.com/gogf/gf/v2/errors/gcode"
|
||||
"github.com/gogf/gf/v2/errors/gerror"
|
||||
"github.com/gogf/gf/v2/net/gclient"
|
||||
"github.com/gogf/gf/v2/net/ghttp"
|
||||
@ -24,50 +23,31 @@ import (
|
||||
"github.com/gogf/gf/v2/util/gtag"
|
||||
)
|
||||
|
||||
// Client is an http client for SDK.
|
||||
// Client is a http client for SDK.
|
||||
type Client struct {
|
||||
*gclient.Client
|
||||
config Config
|
||||
Handler
|
||||
}
|
||||
|
||||
// New creates and returns an http client for SDK.
|
||||
// New creates and returns a http client for SDK.
|
||||
func New(config Config) *Client {
|
||||
client := config.Client
|
||||
if client == nil {
|
||||
client = gclient.New()
|
||||
}
|
||||
handler := config.Handler
|
||||
if handler == nil {
|
||||
handler = NewDefaultHandler(config.Logger, config.RawDump)
|
||||
}
|
||||
if !gstr.HasPrefix(config.URL, "http") {
|
||||
config.URL = fmt.Sprintf("http://%s", config.URL)
|
||||
}
|
||||
return &Client{
|
||||
Client: client,
|
||||
config: config,
|
||||
Client: client.Prefix(config.URL),
|
||||
Handler: handler,
|
||||
}
|
||||
}
|
||||
|
||||
func (c *Client) handleResponse(ctx context.Context, res *gclient.Response, out interface{}) error {
|
||||
if c.config.RawDump {
|
||||
c.config.Logger.Debugf(ctx, "raw request&response:\n%s", res.Raw())
|
||||
}
|
||||
|
||||
var (
|
||||
responseBytes = res.ReadAll()
|
||||
result = ghttp.DefaultHandlerResponse{
|
||||
Data: out,
|
||||
}
|
||||
)
|
||||
if !json.Valid(responseBytes) {
|
||||
return gerror.Newf(`invalid response content: %s`, responseBytes)
|
||||
}
|
||||
if err := json.Unmarshal(responseBytes, &result); err != nil {
|
||||
return gerror.Wrapf(err, `json.Unmarshal failed with content:%s`, responseBytes)
|
||||
}
|
||||
if result.Code != gcode.CodeOK.Code() {
|
||||
return gerror.NewCode(
|
||||
gcode.New(result.Code, result.Message, nil),
|
||||
result.Message,
|
||||
)
|
||||
}
|
||||
return nil
|
||||
}
|
||||
|
||||
// Request sends request to service by struct object `req`, and receives response to struct object `res`.
|
||||
func (c *Client) Request(ctx context.Context, req, res interface{}) error {
|
||||
var (
|
||||
@ -83,20 +63,21 @@ func (c *Client) Request(ctx context.Context, req, res interface{}) error {
|
||||
if err != nil {
|
||||
return err
|
||||
}
|
||||
return c.handleResponse(ctx, result, res)
|
||||
return c.HandleResponse(ctx, result, res)
|
||||
}
|
||||
}
|
||||
|
||||
// Get sends a request using GET method.
|
||||
func (c *Client) Get(ctx context.Context, path string, in, out interface{}) error {
|
||||
if urlParams := ghttp.BuildParams(in); urlParams != "" {
|
||||
path += "?" + ghttp.BuildParams(in)
|
||||
// TODO: Path params will also be built in urlParams, not graceful now.
|
||||
if urlParams := ghttp.BuildParams(in); urlParams != "" && urlParams != "{}" {
|
||||
path += "?" + urlParams
|
||||
}
|
||||
res, err := c.ContentJson().Get(ctx, c.handlePath(path, in))
|
||||
if err != nil {
|
||||
return gerror.Wrap(err, `http request failed`)
|
||||
}
|
||||
return c.handleResponse(ctx, res, out)
|
||||
return c.HandleResponse(ctx, res, out)
|
||||
}
|
||||
|
||||
func (c *Client) handlePath(path string, in interface{}) string {
|
||||
|
@ -15,6 +15,7 @@ import (
|
||||
type Config struct {
|
||||
URL string `v:"required"` // Service address. Eg: user.svc.local, http://user.svc.local
|
||||
Client *gclient.Client // Custom underlying client.
|
||||
Handler Handler // Custom response handler.
|
||||
Logger *glog.Logger // Custom logger.
|
||||
RawDump bool // Whether auto dump request&response in stdout.
|
||||
}
|
||||
|
68
contrib/sdk/httpclient/httpclient_handler.go
Normal file
68
contrib/sdk/httpclient/httpclient_handler.go
Normal file
@ -0,0 +1,68 @@
|
||||
// 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.
|
||||
|
||||
package httpclient
|
||||
|
||||
import (
|
||||
"context"
|
||||
"encoding/json"
|
||||
|
||||
"github.com/gogf/gf/v2/errors/gcode"
|
||||
"github.com/gogf/gf/v2/errors/gerror"
|
||||
"github.com/gogf/gf/v2/frame/g"
|
||||
"github.com/gogf/gf/v2/net/gclient"
|
||||
"github.com/gogf/gf/v2/net/ghttp"
|
||||
"github.com/gogf/gf/v2/os/glog"
|
||||
)
|
||||
|
||||
// Handler is the interface for http response handling.
|
||||
type Handler interface {
|
||||
// HandleResponse handles the http response and transforms its body to the specified object.
|
||||
// The parameter `out` specifies the object that the response body is transformed to.
|
||||
HandleResponse(ctx context.Context, res *gclient.Response, out interface{}) error
|
||||
}
|
||||
|
||||
// DefaultHandler handle ghttp.DefaultHandlerResponse of json format.
|
||||
type DefaultHandler struct {
|
||||
Logger *glog.Logger
|
||||
RawDump bool
|
||||
}
|
||||
|
||||
func NewDefaultHandler(logger *glog.Logger, rawRump bool) *DefaultHandler {
|
||||
if rawRump && logger == nil {
|
||||
logger = g.Log()
|
||||
}
|
||||
return &DefaultHandler{
|
||||
Logger: logger,
|
||||
RawDump: rawRump,
|
||||
}
|
||||
}
|
||||
|
||||
func (h DefaultHandler) HandleResponse(ctx context.Context, res *gclient.Response, out interface{}) error {
|
||||
defer res.Close()
|
||||
if h.RawDump {
|
||||
h.Logger.Debugf(ctx, "raw request&response:\n%s", res.Raw())
|
||||
}
|
||||
var (
|
||||
responseBytes = res.ReadAll()
|
||||
result = ghttp.DefaultHandlerResponse{
|
||||
Data: out,
|
||||
}
|
||||
)
|
||||
if !json.Valid(responseBytes) {
|
||||
return gerror.Newf(`invalid response content: %s`, responseBytes)
|
||||
}
|
||||
if err := json.Unmarshal(responseBytes, &result); err != nil {
|
||||
return gerror.Wrapf(err, `json.Unmarshal failed with content:%s`, responseBytes)
|
||||
}
|
||||
if result.Code != gcode.CodeOK.Code() {
|
||||
return gerror.NewCode(
|
||||
gcode.New(result.Code, result.Message, nil),
|
||||
result.Message,
|
||||
)
|
||||
}
|
||||
return nil
|
||||
}
|
109
contrib/sdk/httpclient/httpclient_z_unit_feature_handler_test.go
Normal file
109
contrib/sdk/httpclient/httpclient_z_unit_feature_handler_test.go
Normal file
@ -0,0 +1,109 @@
|
||||
// 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.
|
||||
|
||||
package httpclient_test
|
||||
|
||||
import (
|
||||
"context"
|
||||
"fmt"
|
||||
"testing"
|
||||
"time"
|
||||
|
||||
"github.com/gogf/gf/contrib/sdk/httpclient/v2"
|
||||
"github.com/gogf/gf/v2/errors/gcode"
|
||||
"github.com/gogf/gf/v2/errors/gerror"
|
||||
"github.com/gogf/gf/v2/frame/g"
|
||||
"github.com/gogf/gf/v2/net/gclient"
|
||||
"github.com/gogf/gf/v2/net/ghttp"
|
||||
"github.com/gogf/gf/v2/os/gctx"
|
||||
"github.com/gogf/gf/v2/test/gtest"
|
||||
"github.com/gogf/gf/v2/util/guid"
|
||||
)
|
||||
|
||||
func Test_HttpClient_With_Default_Handler(t *testing.T) {
|
||||
type Req struct {
|
||||
g.Meta `path:"/get" method:"get"`
|
||||
}
|
||||
type Res struct {
|
||||
Uid int
|
||||
Name string
|
||||
}
|
||||
|
||||
s := g.Server(guid.S())
|
||||
s.BindHandler("/get", func(r *ghttp.Request) {
|
||||
res := ghttp.DefaultHandlerResponse{
|
||||
Data: Res{
|
||||
Uid: 1,
|
||||
Name: "test",
|
||||
},
|
||||
}
|
||||
r.Response.WriteJson(res)
|
||||
})
|
||||
s.SetDumpRouterMap(false)
|
||||
s.Start()
|
||||
defer s.Shutdown()
|
||||
|
||||
time.Sleep(100 * time.Millisecond)
|
||||
|
||||
gtest.C(t, func(t *gtest.T) {
|
||||
client := httpclient.New(httpclient.Config{
|
||||
URL: fmt.Sprintf("http://127.0.0.1:%d", s.GetListenedPort()),
|
||||
})
|
||||
var (
|
||||
req = &Req{}
|
||||
res = &Res{}
|
||||
)
|
||||
err := client.Request(gctx.New(), req, res)
|
||||
t.AssertNil(err)
|
||||
t.AssertEQ(res.Uid, 1)
|
||||
t.AssertEQ(res.Name, "test")
|
||||
})
|
||||
}
|
||||
|
||||
type CustomHandler struct{}
|
||||
|
||||
func (c CustomHandler) HandleResponse(ctx context.Context, res *gclient.Response, out interface{}) error {
|
||||
defer res.Close()
|
||||
if pointer, ok := out.(*string); ok {
|
||||
*pointer = res.ReadAllString()
|
||||
} else {
|
||||
return gerror.NewCodef(gcode.CodeInvalidParameter, "[CustomHandler] expectedType:'*string', but realType:'%T'", out)
|
||||
}
|
||||
return nil
|
||||
}
|
||||
|
||||
func Test_HttpClient_With_Custom_Handler(t *testing.T) {
|
||||
type Req struct {
|
||||
g.Meta `path:"/get" method:"get"`
|
||||
}
|
||||
|
||||
s := g.Server(guid.S())
|
||||
s.BindHandler("/get", func(r *ghttp.Request) {
|
||||
r.Response.WriteExit("It is a test.")
|
||||
})
|
||||
s.SetDumpRouterMap(false)
|
||||
s.Start()
|
||||
defer s.Shutdown()
|
||||
|
||||
time.Sleep(100 * time.Millisecond)
|
||||
|
||||
client := httpclient.New(httpclient.Config{
|
||||
URL: fmt.Sprintf("127.0.0.1:%d", s.GetListenedPort()),
|
||||
Handler: CustomHandler{},
|
||||
})
|
||||
req := &Req{}
|
||||
gtest.C(t, func(t *gtest.T) {
|
||||
var res = new(string)
|
||||
err := client.Request(gctx.New(), req, res)
|
||||
t.AssertNil(err)
|
||||
t.AssertEQ(*res, "It is a test.")
|
||||
})
|
||||
gtest.C(t, func(t *gtest.T) {
|
||||
var res string
|
||||
err := client.Request(gctx.New(), req, res)
|
||||
t.AssertEQ(err, gerror.NewCodef(gcode.CodeInvalidParameter, "[CustomHandler] expectedType:'*string', but realType:'%T'", res))
|
||||
})
|
||||
}
|
Loading…
x
Reference in New Issue
Block a user