Optimize mock
This commit is contained in:
		
							parent
							
								
									32557769fc
								
							
						
					
					
						commit
						94019c1e1d
					
				| @ -12,7 +12,7 @@ | ||||
| // See the License for the specific language governing permissions and | ||||
| // limitations under the License. | ||||
| 
 | ||||
| package mock | ||||
| package httplib | ||||
| 
 | ||||
| import ( | ||||
| 	"bytes" | ||||
| @ -21,7 +21,8 @@ import ( | ||||
| 	"net/http" | ||||
| ) | ||||
| 
 | ||||
| // it will try to convert the data to json format | ||||
| // NewHttpResponseWithJsonBody will try to convert the data to json format | ||||
| // usually you only use this when you want to mock http Response | ||||
| func NewHttpResponseWithJsonBody(data interface{}) *http.Response { | ||||
| 	var body []byte | ||||
| 	if str, ok := data.(string); ok { | ||||
| @ -62,6 +62,7 @@ var defaultSetting = BeegoHTTPSettings{ | ||||
| 	ReadWriteTimeout: 60 * time.Second, | ||||
| 	Gzip:             true, | ||||
| 	DumpBody:         true, | ||||
| 	FilterChains: []FilterChain{mockFilter.FilterChain}, | ||||
| } | ||||
| 
 | ||||
| var defaultCookieJar http.CookieJar | ||||
|  | ||||
							
								
								
									
										71
									
								
								client/httplib/mock.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										71
									
								
								client/httplib/mock.go
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,71 @@ | ||||
| // Copyright 2020 beego | ||||
| // | ||||
| // Licensed under the Apache License, Version 2.0 (the "License"); | ||||
| // you may not use this file except in compliance with the License. | ||||
| // You may obtain a copy of the License at | ||||
| // | ||||
| // http://www.apache.org/licenses/LICENSE-2.0 | ||||
| // | ||||
| // Unless required by applicable law or agreed to in writing, software | ||||
| // distributed under the License is distributed on an "AS IS" BASIS, | ||||
| // WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. | ||||
| // See the License for the specific language governing permissions and | ||||
| // limitations under the License. | ||||
| 
 | ||||
| package httplib | ||||
| 
 | ||||
| import ( | ||||
| 	"context" | ||||
| 	"net/http" | ||||
| 
 | ||||
| 	"github.com/beego/beego/v2/core/logs" | ||||
| ) | ||||
| 
 | ||||
| const mockCtxKey = "beego-httplib-mock" | ||||
| 
 | ||||
| type Stub interface { | ||||
| 	Mock(cond RequestCondition, resp *http.Response, err error) | ||||
| 	Clear() | ||||
| 	MockByPath(path string, resp *http.Response, err error) | ||||
| } | ||||
| 
 | ||||
| var mockFilter = &MockResponseFilter{} | ||||
| 
 | ||||
| func StartMock() Stub { | ||||
| 	return mockFilter | ||||
| } | ||||
| 
 | ||||
| func CtxWithMock(ctx context.Context, mock... *Mock) context.Context { | ||||
| 	return context.WithValue(ctx, mockCtxKey, mock) | ||||
| } | ||||
| 
 | ||||
| func mockFromCtx(ctx context.Context) []*Mock { | ||||
| 	ms := ctx.Value(mockCtxKey) | ||||
| 	if ms != nil { | ||||
| 		if res, ok := ms.([]*Mock); ok { | ||||
| 			return res | ||||
| 		} | ||||
| 		logs.Error("mockCtxKey found in context, but value is not type []*Mock") | ||||
| 	} | ||||
| 	return nil | ||||
| } | ||||
| 
 | ||||
| type Mock struct { | ||||
| 	cond RequestCondition | ||||
| 	resp *http.Response | ||||
| 	err  error | ||||
| } | ||||
| 
 | ||||
| func NewMockByPath(path string, resp *http.Response, err error) *Mock { | ||||
| 	return NewMock(NewSimpleCondition(path), resp, err) | ||||
| } | ||||
| 
 | ||||
| func NewMock(con RequestCondition, resp *http.Response, err error) *Mock { | ||||
| 	return &Mock{ | ||||
| 		cond: con, | ||||
| 		resp: resp, | ||||
| 		err: err, | ||||
| 	} | ||||
| } | ||||
| 
 | ||||
| 
 | ||||
| @ -12,17 +12,19 @@ | ||||
| // See the License for the specific language governing permissions and | ||||
| // limitations under the License. | ||||
| 
 | ||||
| package mock | ||||
| package httplib | ||||
| 
 | ||||
| import ( | ||||
| 	"context" | ||||
| 	"encoding/json" | ||||
| 	"net/textproto" | ||||
| 	"regexp" | ||||
| 
 | ||||
| 	"github.com/beego/beego/v2/client/httplib" | ||||
| ) | ||||
| 
 | ||||
| type RequestCondition interface { | ||||
| 	Match(ctx context.Context, req *BeegoHTTPRequest) bool | ||||
| } | ||||
| 
 | ||||
| // reqCondition create condition | ||||
| // - path: same path | ||||
| // - pathReg: request path match pathReg | ||||
| @ -52,7 +54,7 @@ func NewSimpleCondition(path string, opts ...simpleConditionOption) *SimpleCondi | ||||
| 	return sc | ||||
| } | ||||
| 
 | ||||
| func (sc *SimpleCondition) Match(ctx context.Context, req *httplib.BeegoHTTPRequest) bool { | ||||
| func (sc *SimpleCondition) Match(ctx context.Context, req *BeegoHTTPRequest) bool { | ||||
| 	res := true | ||||
| 	if len(sc.path) > 0 { | ||||
| 		res = sc.matchPath(ctx, req) | ||||
| @ -68,12 +70,12 @@ func (sc *SimpleCondition) Match(ctx context.Context, req *httplib.BeegoHTTPRequ | ||||
| 		sc.matchBodyFields(ctx, req) | ||||
| } | ||||
| 
 | ||||
| func (sc *SimpleCondition) matchPath(ctx context.Context, req *httplib.BeegoHTTPRequest) bool { | ||||
| func (sc *SimpleCondition) matchPath(ctx context.Context, req *BeegoHTTPRequest) bool { | ||||
| 	path := req.GetRequest().URL.Path | ||||
| 	return path == sc.path | ||||
| } | ||||
| 
 | ||||
| func (sc *SimpleCondition) matchPathReg(ctx context.Context, req *httplib.BeegoHTTPRequest) bool { | ||||
| func (sc *SimpleCondition) matchPathReg(ctx context.Context, req *BeegoHTTPRequest) bool { | ||||
| 	path := req.GetRequest().URL.Path | ||||
| 	if b, err := regexp.Match(sc.pathReg, []byte(path)); err == nil { | ||||
| 		return b | ||||
| @ -81,7 +83,7 @@ func (sc *SimpleCondition) matchPathReg(ctx context.Context, req *httplib.BeegoH | ||||
| 	return false | ||||
| } | ||||
| 
 | ||||
| func (sc *SimpleCondition) matchQuery(ctx context.Context, req *httplib.BeegoHTTPRequest) bool { | ||||
| func (sc *SimpleCondition) matchQuery(ctx context.Context, req *BeegoHTTPRequest) bool { | ||||
| 	qs := req.GetRequest().URL.Query() | ||||
| 	for k, v := range sc.query { | ||||
| 		if uv, ok := qs[k]; !ok || uv[0] != v { | ||||
| @ -91,7 +93,7 @@ func (sc *SimpleCondition) matchQuery(ctx context.Context, req *httplib.BeegoHTT | ||||
| 	return true | ||||
| } | ||||
| 
 | ||||
| func (sc *SimpleCondition) matchHeader(ctx context.Context, req *httplib.BeegoHTTPRequest) bool { | ||||
| func (sc *SimpleCondition) matchHeader(ctx context.Context, req *BeegoHTTPRequest) bool { | ||||
| 	headers := req.GetRequest().Header | ||||
| 	for k, v := range sc.header { | ||||
| 		if uv, ok := headers[k]; !ok || uv[0] != v { | ||||
| @ -101,7 +103,7 @@ func (sc *SimpleCondition) matchHeader(ctx context.Context, req *httplib.BeegoHT | ||||
| 	return true | ||||
| } | ||||
| 
 | ||||
| func (sc *SimpleCondition) matchBodyFields(ctx context.Context, req *httplib.BeegoHTTPRequest) bool { | ||||
| func (sc *SimpleCondition) matchBodyFields(ctx context.Context, req *BeegoHTTPRequest) bool { | ||||
| 	if len(sc.body) == 0 { | ||||
| 		return true | ||||
| 	} | ||||
| @ -133,7 +135,7 @@ func (sc *SimpleCondition) matchBodyFields(ctx context.Context, req *httplib.Bee | ||||
| 	return true | ||||
| } | ||||
| 
 | ||||
| func (sc *SimpleCondition) matchMethod(ctx context.Context, req *httplib.BeegoHTTPRequest) bool { | ||||
| func (sc *SimpleCondition) matchMethod(ctx context.Context, req *BeegoHTTPRequest) bool { | ||||
| 	if len(sc.method) > 0 { | ||||
| 		return sc.method == req.GetRequest().Method | ||||
| 	} | ||||
| @ -12,7 +12,7 @@ | ||||
| // See the License for the specific language governing permissions and | ||||
| // limitations under the License. | ||||
| 
 | ||||
| package mock | ||||
| package httplib | ||||
| 
 | ||||
| import ( | ||||
| 	"context" | ||||
| @ -20,7 +20,6 @@ import ( | ||||
| 
 | ||||
| 	"github.com/stretchr/testify/assert" | ||||
| 
 | ||||
| 	"github.com/beego/beego/v2/client/httplib" | ||||
| ) | ||||
| 
 | ||||
| func init() { | ||||
| @ -29,37 +28,37 @@ func init() { | ||||
| 
 | ||||
| func TestSimpleCondition_MatchPath(t *testing.T) { | ||||
| 	sc := NewSimpleCondition("/abc/s") | ||||
| 	res := sc.Match(context.Background(), httplib.Get("http://localhost:8080/abc/s")) | ||||
| 	res := sc.Match(context.Background(), Get("http://localhost:8080/abc/s")) | ||||
| 	assert.True(t, res) | ||||
| } | ||||
| 
 | ||||
| func TestSimpleCondition_MatchQuery(t *testing.T) { | ||||
| 	k, v := "my-key", "my-value" | ||||
| 	sc := NewSimpleCondition("/abc/s") | ||||
| 	res := sc.Match(context.Background(), httplib.Get("http://localhost:8080/abc/s?my-key=my-value")) | ||||
| 	res := sc.Match(context.Background(), Get("http://localhost:8080/abc/s?my-key=my-value")) | ||||
| 	assert.True(t, res) | ||||
| 
 | ||||
| 	sc = NewSimpleCondition("/abc/s", WithQuery(k, v)) | ||||
| 	res = sc.Match(context.Background(), httplib.Get("http://localhost:8080/abc/s?my-key=my-value")) | ||||
| 	res = sc.Match(context.Background(), Get("http://localhost:8080/abc/s?my-key=my-value")) | ||||
| 	assert.True(t, res) | ||||
| 
 | ||||
| 	res = sc.Match(context.Background(), httplib.Get("http://localhost:8080/abc/s?my-key=my-valuesss")) | ||||
| 	res = sc.Match(context.Background(), Get("http://localhost:8080/abc/s?my-key=my-valuesss")) | ||||
| 	assert.False(t, res) | ||||
| 
 | ||||
| 	res = sc.Match(context.Background(), httplib.Get("http://localhost:8080/abc/s?my-key-a=my-value")) | ||||
| 	res = sc.Match(context.Background(), Get("http://localhost:8080/abc/s?my-key-a=my-value")) | ||||
| 	assert.False(t, res) | ||||
| 
 | ||||
| 	res = sc.Match(context.Background(), httplib.Get("http://localhost:8080/abc/s?my-key=my-value&abc=hello")) | ||||
| 	res = sc.Match(context.Background(), Get("http://localhost:8080/abc/s?my-key=my-value&abc=hello")) | ||||
| 	assert.True(t, res) | ||||
| } | ||||
| 
 | ||||
| func TestSimpleCondition_MatchHeader(t *testing.T) { | ||||
| 	k, v := "my-header", "my-header-value" | ||||
| 	sc := NewSimpleCondition("/abc/s") | ||||
| 	req := httplib.Get("http://localhost:8080/abc/s") | ||||
| 	req := Get("http://localhost:8080/abc/s") | ||||
| 	assert.True(t, sc.Match(context.Background(), req)) | ||||
| 
 | ||||
| 	req = httplib.Get("http://localhost:8080/abc/s") | ||||
| 	req = Get("http://localhost:8080/abc/s") | ||||
| 	req.Header(k, v) | ||||
| 	assert.True(t, sc.Match(context.Background(), req)) | ||||
| 
 | ||||
| @ -74,7 +73,7 @@ func TestSimpleCondition_MatchHeader(t *testing.T) { | ||||
| func TestSimpleCondition_MatchBodyField(t *testing.T) { | ||||
| 
 | ||||
| 	sc := NewSimpleCondition("/abc/s") | ||||
| 	req := httplib.Post("http://localhost:8080/abc/s") | ||||
| 	req := Post("http://localhost:8080/abc/s") | ||||
| 
 | ||||
| 	assert.True(t, sc.Match(context.Background(), req)) | ||||
| 
 | ||||
| @ -103,7 +102,7 @@ func TestSimpleCondition_MatchBodyField(t *testing.T) { | ||||
| 
 | ||||
| func TestSimpleCondition_Match(t *testing.T) { | ||||
| 	sc := NewSimpleCondition("/abc/s") | ||||
| 	req := httplib.Post("http://localhost:8080/abc/s") | ||||
| 	req := Post("http://localhost:8080/abc/s") | ||||
| 
 | ||||
| 	assert.True(t, sc.Match(context.Background(), req)) | ||||
| 
 | ||||
| @ -116,9 +115,9 @@ func TestSimpleCondition_Match(t *testing.T) { | ||||
| 
 | ||||
| func TestSimpleCondition_MatchPathReg(t *testing.T) { | ||||
| 	sc := NewSimpleCondition("", WithPathReg(`\/abc\/.*`)) | ||||
| 	req := httplib.Post("http://localhost:8080/abc/s") | ||||
| 	req := Post("http://localhost:8080/abc/s") | ||||
| 	assert.True(t, sc.Match(context.Background(), req)) | ||||
| 
 | ||||
| 	req = httplib.Post("http://localhost:8080/abcd/s") | ||||
| 	req = Post("http://localhost:8080/abcd/s") | ||||
| 	assert.False(t, sc.Match(context.Background(), req)) | ||||
| } | ||||
| @ -12,16 +12,16 @@ | ||||
| // See the License for the specific language governing permissions and | ||||
| // limitations under the License. | ||||
| 
 | ||||
| package mock | ||||
| package httplib | ||||
| 
 | ||||
| import ( | ||||
| 	"context" | ||||
| 	"fmt" | ||||
| 	"net/http" | ||||
| 
 | ||||
| 	"github.com/beego/beego/v2/client/httplib" | ||||
| ) | ||||
| 
 | ||||
| // MockResponse will return mock response if find any suitable mock data | ||||
| // if you want to test your code using httplib, you need this. | ||||
| type MockResponseFilter struct { | ||||
| 	ms []*Mock | ||||
| } | ||||
| @ -32,9 +32,14 @@ func NewMockResponseFilter() *MockResponseFilter { | ||||
| 	} | ||||
| } | ||||
| 
 | ||||
| func (m *MockResponseFilter) FilterChain(next httplib.Filter) httplib.Filter { | ||||
| 	return func(ctx context.Context, req *httplib.BeegoHTTPRequest) (*http.Response, error) { | ||||
| 		for _, mock := range m.ms { | ||||
| func (m *MockResponseFilter) FilterChain(next Filter) Filter { | ||||
| 	return func(ctx context.Context, req *BeegoHTTPRequest) (*http.Response, error) { | ||||
| 
 | ||||
| 		ms := mockFromCtx(ctx) | ||||
| 		ms = append(ms, m.ms...) | ||||
| 
 | ||||
| 		fmt.Printf("url: %s, mock: %d \n", req.url, len(ms)) | ||||
| 		for _, mock := range ms { | ||||
| 			if mock.cond.Match(ctx, req) { | ||||
| 				return mock.resp, mock.err | ||||
| 			} | ||||
| @ -43,22 +48,16 @@ func (m *MockResponseFilter) FilterChain(next httplib.Filter) httplib.Filter { | ||||
| 	} | ||||
| } | ||||
| 
 | ||||
| func (m *MockResponseFilter) MockByPath(path string, resp *http.Response, err error) { | ||||
| 	m.Mock(NewSimpleCondition(path), resp, err) | ||||
| } | ||||
| 
 | ||||
| func (m *MockResponseFilter) Clear() { | ||||
| 	m.ms = make([]*Mock, 0, 1) | ||||
| } | ||||
| 
 | ||||
| // Mock add mock data | ||||
| // If the cond.Match(...) = true, the resp and err will be returned | ||||
| func (m *MockResponseFilter) Mock(cond RequestCondition, resp *http.Response, err error) { | ||||
| 	m.ms = append(m.ms, &Mock{ | ||||
| 		cond: cond, | ||||
| 		resp: resp, | ||||
| 		err:  err, | ||||
| 	}) | ||||
| } | ||||
| 
 | ||||
| type Mock struct { | ||||
| 	cond RequestCondition | ||||
| 	resp *http.Response | ||||
| 	err  error | ||||
| } | ||||
| 
 | ||||
| type RequestCondition interface { | ||||
| 	Match(ctx context.Context, req *httplib.BeegoHTTPRequest) bool | ||||
| 	m.ms = append(m.ms, NewMock(cond, resp, err)) | ||||
| } | ||||
| @ -12,19 +12,17 @@ | ||||
| // See the License for the specific language governing permissions and | ||||
| // limitations under the License. | ||||
| 
 | ||||
| package mock | ||||
| package httplib | ||||
| 
 | ||||
| import ( | ||||
| 	"errors" | ||||
| 	"testing" | ||||
| 
 | ||||
| 	"github.com/stretchr/testify/assert" | ||||
| 
 | ||||
| 	"github.com/beego/beego/v2/client/httplib" | ||||
| ) | ||||
| 
 | ||||
| func TestMockResponseFilter_FilterChain(t *testing.T) { | ||||
| 	req := httplib.Get("http://localhost:8080/abc/s") | ||||
| 	req := Get("http://localhost:8080/abc/s") | ||||
| 	ft := NewMockResponseFilter() | ||||
| 
 | ||||
| 	expectedResp := NewHttpResponseWithJsonBody(`{}`) | ||||
| @ -37,15 +35,14 @@ func TestMockResponseFilter_FilterChain(t *testing.T) { | ||||
| 	assert.Equal(t, expectedErr, err) | ||||
| 	assert.Equal(t, expectedResp, resp) | ||||
| 
 | ||||
| 	req = httplib.Get("http://localhost:8080/abcd/s") | ||||
| 	req = Get("http://localhost:8080/abcd/s") | ||||
| 	req.AddFilters(ft.FilterChain) | ||||
| 
 | ||||
| 	resp, err = req.DoRequest() | ||||
| 	assert.NotEqual(t, expectedErr, err) | ||||
| 	assert.NotEqual(t, expectedResp, resp) | ||||
| 
 | ||||
| 
 | ||||
| 	req = httplib.Get("http://localhost:8080/abc/s") | ||||
| 	req = Get("http://localhost:8080/abc/s") | ||||
| 	req.AddFilters(ft.FilterChain) | ||||
| 	expectedResp1 := NewHttpResponseWithJsonBody(map[string]string{}) | ||||
| 	expectedErr1 := errors.New("expected error") | ||||
| @ -55,7 +52,7 @@ func TestMockResponseFilter_FilterChain(t *testing.T) { | ||||
| 	assert.Equal(t, expectedErr, err) | ||||
| 	assert.Equal(t, expectedResp, resp) | ||||
| 
 | ||||
| 	req = httplib.Get("http://localhost:8080/abc/abs/bbc") | ||||
| 	req = Get("http://localhost:8080/abc/abs/bbc") | ||||
| 	req.AddFilters(ft.FilterChain) | ||||
| 	ft.Mock(NewSimpleCondition("/abc/abs/bbc"), expectedResp1, expectedErr1) | ||||
| 	resp, err = req.DoRequest() | ||||
							
								
								
									
										75
									
								
								client/httplib/mock_test.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										75
									
								
								client/httplib/mock_test.go
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,75 @@ | ||||
| // Copyright 2020 beego | ||||
| // | ||||
| // Licensed under the Apache License, Version 2.0 (the "License"); | ||||
| // you may not use this file except in compliance with the License. | ||||
| // You may obtain a copy of the License at | ||||
| // | ||||
| // http://www.apache.org/licenses/LICENSE-2.0 | ||||
| // | ||||
| // Unless required by applicable law or agreed to in writing, software | ||||
| // distributed under the License is distributed on an "AS IS" BASIS, | ||||
| // WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. | ||||
| // See the License for the specific language governing permissions and | ||||
| // limitations under the License. | ||||
| 
 | ||||
| package httplib | ||||
| 
 | ||||
| import ( | ||||
| 	"context" | ||||
| 	"errors" | ||||
| 	"net/http" | ||||
| 	"testing" | ||||
| 
 | ||||
| 	"github.com/stretchr/testify/assert" | ||||
| ) | ||||
| 
 | ||||
| func TestStartMock(t *testing.T) { | ||||
| 
 | ||||
| 	defaultSetting.FilterChains = []FilterChain{mockFilter.FilterChain} | ||||
| 
 | ||||
| 	stub := StartMock() | ||||
| 	// defer stub.Clear() | ||||
| 
 | ||||
| 	expectedResp := NewHttpResponseWithJsonBody([]byte(`{}`)) | ||||
| 	expectedErr := errors.New("expected err") | ||||
| 
 | ||||
| 	stub.Mock(NewSimpleCondition("/abc"), expectedResp, expectedErr) | ||||
| 
 | ||||
| 	resp, err := OriginalCodeUsingHttplib() | ||||
| 
 | ||||
| 	assert.Equal(t, expectedErr, err) | ||||
| 	assert.Equal(t, expectedResp, resp) | ||||
| 
 | ||||
| } | ||||
| 
 | ||||
| // TestStartMock_Isolation Test StartMock that | ||||
| // mock only work for this request | ||||
| func TestStartMock_Isolation(t *testing.T) { | ||||
| 	defaultSetting.FilterChains = []FilterChain{mockFilter.FilterChain} | ||||
| 	// setup global stub | ||||
| 	stub := StartMock() | ||||
| 	globalMockResp := NewHttpResponseWithJsonBody([]byte(`{}`)) | ||||
| 	globalMockErr := errors.New("expected err") | ||||
| 	stub.Mock(NewSimpleCondition("/abc"), globalMockResp, globalMockErr) | ||||
| 
 | ||||
| 	expectedResp := NewHttpResponseWithJsonBody(struct { | ||||
| 		A string `json:"a"` | ||||
| 	}{ | ||||
| 		A: "aaa", | ||||
| 	}) | ||||
| 	expectedErr := errors.New("expected err aa") | ||||
| 	m := NewMockByPath("/abc", expectedResp, expectedErr) | ||||
| 	ctx := CtxWithMock(context.Background(), m) | ||||
| 
 | ||||
| 	resp, err := OriginnalCodeUsingHttplibPassCtx(ctx) | ||||
| 	assert.Equal(t, expectedErr, err) | ||||
| 	assert.Equal(t, expectedResp, resp) | ||||
| } | ||||
| 
 | ||||
| func OriginnalCodeUsingHttplibPassCtx(ctx context.Context) (*http.Response, error) { | ||||
| 	return Get("http://localhost:7777/abc").DoRequestWithCtx(ctx) | ||||
| } | ||||
| 
 | ||||
| func OriginalCodeUsingHttplib() (*http.Response, error){ | ||||
| 	return Get("http://localhost:7777/abc").DoRequest() | ||||
| } | ||||
		Loading…
	
	
			
			x
			
			
		
	
		Reference in New Issue
	
	Block a user