goc/pkg/cover/client.go

186 lines
5.6 KiB
Go
Raw Normal View History

/*
2020-05-25 16:19:20 +00:00
Copyright 2020 Qiniu Cloud (qiniu.com)
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 cover
import (
2020-09-04 13:16:42 +00:00
"bytes"
"encoding/json"
"fmt"
"io"
"io/ioutil"
"net"
"net/http"
2020-06-12 11:07:21 +00:00
"net/url"
"strings"
2020-08-05 09:28:11 +00:00
log "github.com/sirupsen/logrus"
)
2020-06-12 06:08:15 +00:00
// Action provides methods to contact with the covered service under test
type Action interface {
2020-07-09 12:55:07 +00:00
Profile(param ProfileParam) ([]byte, error)
2020-09-11 07:45:45 +00:00
Clear(param ProfileParam) ([]byte, error)
2020-09-20 12:29:17 +00:00
Remove(param ProfileParam) ([]byte, error)
2020-06-12 11:07:21 +00:00
InitSystem() ([]byte, error)
ListServices() ([]byte, error)
RegisterService(svr ServiceUnderTest) ([]byte, error)
}
2020-06-12 06:08:15 +00:00
const (
//CoverInitSystemAPI prepare a new round of testing
CoverInitSystemAPI = "/v1/cover/init"
//CoverProfileAPI is provided by the covered service to get profiles
CoverProfileAPI = "/v1/cover/profile"
//CoverProfileClearAPI is provided by the covered service to clear profiles
CoverProfileClearAPI = "/v1/cover/clear"
//CoverServicesListAPI list all the registered services
CoverServicesListAPI = "/v1/cover/list"
2020-06-12 11:07:21 +00:00
//CoverRegisterServiceAPI register a service into service center
CoverRegisterServiceAPI = "/v1/cover/register"
2020-09-20 12:29:17 +00:00
//CoverServicesRemoveAPI remove one services from the service center
CoverServicesRemoveAPI = "/v1/cover/remove"
2020-06-12 06:08:15 +00:00
)
type client struct {
2020-06-12 11:07:21 +00:00
Host string
client *http.Client
}
// NewWorker creates a worker to contact with service
2020-06-12 11:07:21 +00:00
func NewWorker(host string) Action {
_, err := url.ParseRequestURI(host)
if err != nil {
log.Fatalf("Parse url %s failed, err: %v", host, err)
}
return &client{
2020-06-12 11:07:21 +00:00
Host: host,
client: http.DefaultClient,
}
}
func (c *client) RegisterService(srv ServiceUnderTest) ([]byte, error) {
2020-06-12 11:07:21 +00:00
if _, err := url.ParseRequestURI(srv.Address); err != nil {
return nil, err
}
if strings.TrimSpace(srv.Name) == "" {
return nil, fmt.Errorf("invalid service name")
}
u := fmt.Sprintf("%s%s?name=%s&address=%s", c.Host, CoverRegisterServiceAPI, srv.Name, srv.Address)
2020-09-04 13:16:42 +00:00
_, res, err := c.do("POST", u, "", nil)
2020-06-12 11:07:21 +00:00
return res, err
}
func (c *client) ListServices() ([]byte, error) {
u := fmt.Sprintf("%s%s", c.Host, CoverServicesListAPI)
2020-09-04 13:16:42 +00:00
_, services, err := c.do("GET", u, "", nil)
2020-06-12 06:08:15 +00:00
if err != nil && isNetworkError(err) {
2020-09-04 13:16:42 +00:00
_, services, err = c.do("GET", u, "", nil)
2020-06-12 06:08:15 +00:00
}
return services, err
}
2020-07-09 12:55:07 +00:00
func (c *client) Profile(param ProfileParam) ([]byte, error) {
2020-09-04 13:16:42 +00:00
u := fmt.Sprintf("%s%s", c.Host, CoverProfileAPI)
2020-07-16 04:19:26 +00:00
if len(param.Service) != 0 && len(param.Address) != 0 {
return nil, fmt.Errorf("use 'service' flag and 'address' flag at the same time may cause ambiguity, please use them separately")
2020-07-16 04:19:26 +00:00
}
2020-09-11 14:28:53 +00:00
// the json.Marshal function can return two types of errors: UnsupportedTypeError or UnsupportedValueError
// so no need to check here
body, _ := json.Marshal(param)
2020-09-04 13:16:42 +00:00
res, profile, err := c.do("POST", u, "application/json", bytes.NewReader(body))
if err != nil && isNetworkError(err) {
2020-09-04 13:16:42 +00:00
res, profile, err = c.do("POST", u, "application/json", bytes.NewReader(body))
2020-07-30 09:30:50 +00:00
}
2020-07-30 09:30:50 +00:00
if err == nil && res.StatusCode != 200 {
err = fmt.Errorf(string(profile))
}
return profile, err
}
2020-09-11 07:45:45 +00:00
func (c *client) Clear(param ProfileParam) ([]byte, error) {
2020-06-12 11:07:21 +00:00
u := fmt.Sprintf("%s%s", c.Host, CoverProfileClearAPI)
2020-09-11 07:45:45 +00:00
if len(param.Service) != 0 && len(param.Address) != 0 {
return nil, fmt.Errorf("use 'service' flag and 'address' flag at the same time may cause ambiguity, please use them separately")
}
2020-09-11 14:28:53 +00:00
// the json.Marshal function can return two types of errors: UnsupportedTypeError or UnsupportedValueError
// so no need to check here
2020-09-20 12:29:17 +00:00
body, _ := json.Marshal(param)
_, resp, err := c.do("POST", u, "application/json", bytes.NewReader(body))
if err != nil && isNetworkError(err) {
_, resp, err = c.do("POST", u, "application/json", bytes.NewReader(body))
}
return resp, err
}
func (c *client) Remove(param ProfileParam) ([]byte, error) {
u := fmt.Sprintf("%s%s", c.Host, CoverServicesRemoveAPI)
if len(param.Service) != 0 && len(param.Address) != 0 {
return nil, fmt.Errorf("use 'service' flag and 'address' flag at the same time may cause ambiguity, please use them separately")
}
// the json.Marshal function can return two types of errors: UnsupportedTypeError or UnsupportedValueError
// so no need to check here
2020-09-11 14:28:53 +00:00
body, _ := json.Marshal(param)
2020-09-11 07:45:45 +00:00
_, resp, err := c.do("POST", u, "application/json", bytes.NewReader(body))
if err != nil && isNetworkError(err) {
2020-09-11 07:45:45 +00:00
_, resp, err = c.do("POST", u, "application/json", bytes.NewReader(body))
}
return resp, err
}
2020-06-12 11:07:21 +00:00
func (c *client) InitSystem() ([]byte, error) {
u := fmt.Sprintf("%s%s", c.Host, CoverInitSystemAPI)
2020-09-04 13:16:42 +00:00
_, body, err := c.do("POST", u, "", nil)
2020-07-30 09:30:50 +00:00
return body, err
}
2020-09-04 13:16:42 +00:00
func (c *client) do(method, url, contentType string, body io.Reader) (*http.Response, []byte, error) {
req, err := http.NewRequest(method, url, body)
if err != nil {
2020-07-30 09:30:50 +00:00
return nil, nil, err
}
2020-09-04 13:16:42 +00:00
if contentType != "" {
req.Header.Set("Content-Type", contentType)
}
2020-06-12 06:08:15 +00:00
res, err := c.client.Do(req)
if err != nil {
2020-07-30 09:30:50 +00:00
return nil, nil, err
}
defer res.Body.Close()
2020-09-04 13:16:42 +00:00
responseBody, err := ioutil.ReadAll(res.Body)
if err != nil {
2020-07-30 09:30:50 +00:00
return res, nil, err
}
2020-07-30 09:30:50 +00:00
return res, responseBody, nil
}
func isNetworkError(err error) bool {
if err == io.EOF {
return true
}
_, ok := err.(net.Error)
return ok
}