mirror of
https://github.com/etcd-io/etcd.git
synced 2024-09-27 06:25:44 +00:00
547 lines
13 KiB
Go
547 lines
13 KiB
Go
// Copyright 2015 CoreOS, Inc.
|
|
//
|
|
// 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 client
|
|
|
|
import (
|
|
"errors"
|
|
"io/ioutil"
|
|
"net/http"
|
|
"net/url"
|
|
"reflect"
|
|
"strings"
|
|
"testing"
|
|
"time"
|
|
|
|
"github.com/coreos/etcd/Godeps/_workspace/src/golang.org/x/net/context"
|
|
)
|
|
|
|
type actionAssertingHTTPClient struct {
|
|
t *testing.T
|
|
num int
|
|
act httpAction
|
|
|
|
resp http.Response
|
|
body []byte
|
|
err error
|
|
}
|
|
|
|
func (a *actionAssertingHTTPClient) Do(_ context.Context, act httpAction) (*http.Response, []byte, error) {
|
|
if !reflect.DeepEqual(a.act, act) {
|
|
a.t.Errorf("#%d: unexpected httpAction: want=%#v got=%#v", a.num, a.act, act)
|
|
}
|
|
|
|
return &a.resp, a.body, a.err
|
|
}
|
|
|
|
type staticHTTPClient struct {
|
|
resp http.Response
|
|
body []byte
|
|
err error
|
|
}
|
|
|
|
func (s *staticHTTPClient) Do(context.Context, httpAction) (*http.Response, []byte, error) {
|
|
return &s.resp, s.body, s.err
|
|
}
|
|
|
|
type staticHTTPAction struct {
|
|
request http.Request
|
|
}
|
|
|
|
func (s *staticHTTPAction) HTTPRequest(url.URL) *http.Request {
|
|
return &s.request
|
|
}
|
|
|
|
type staticHTTPResponse struct {
|
|
resp http.Response
|
|
err error
|
|
}
|
|
|
|
type multiStaticHTTPClient struct {
|
|
responses []staticHTTPResponse
|
|
cur int
|
|
}
|
|
|
|
func (s *multiStaticHTTPClient) Do(context.Context, httpAction) (*http.Response, []byte, error) {
|
|
r := s.responses[s.cur]
|
|
s.cur++
|
|
return &r.resp, nil, r.err
|
|
}
|
|
|
|
func newStaticHTTPClientFactory(responses []staticHTTPResponse) httpClientFactory {
|
|
var cur int
|
|
return func(url.URL) httpClient {
|
|
r := responses[cur]
|
|
cur++
|
|
return &staticHTTPClient{resp: r.resp, err: r.err}
|
|
}
|
|
}
|
|
|
|
type fakeTransport struct {
|
|
respchan chan *http.Response
|
|
errchan chan error
|
|
startCancel chan struct{}
|
|
finishCancel chan struct{}
|
|
}
|
|
|
|
func newFakeTransport() *fakeTransport {
|
|
return &fakeTransport{
|
|
respchan: make(chan *http.Response, 1),
|
|
errchan: make(chan error, 1),
|
|
startCancel: make(chan struct{}, 1),
|
|
finishCancel: make(chan struct{}, 1),
|
|
}
|
|
}
|
|
|
|
func (t *fakeTransport) RoundTrip(*http.Request) (*http.Response, error) {
|
|
select {
|
|
case resp := <-t.respchan:
|
|
return resp, nil
|
|
case err := <-t.errchan:
|
|
return nil, err
|
|
case <-t.startCancel:
|
|
// wait on finishCancel to simulate taking some amount of
|
|
// time while calling CancelRequest
|
|
<-t.finishCancel
|
|
return nil, errors.New("cancelled")
|
|
}
|
|
}
|
|
|
|
func (t *fakeTransport) CancelRequest(*http.Request) {
|
|
t.startCancel <- struct{}{}
|
|
}
|
|
|
|
type fakeAction struct{}
|
|
|
|
func (a *fakeAction) HTTPRequest(url.URL) *http.Request {
|
|
return &http.Request{}
|
|
}
|
|
|
|
func TestSimpleHTTPClientDoSuccess(t *testing.T) {
|
|
tr := newFakeTransport()
|
|
c := &simpleHTTPClient{transport: tr}
|
|
|
|
tr.respchan <- &http.Response{
|
|
StatusCode: http.StatusTeapot,
|
|
Body: ioutil.NopCloser(strings.NewReader("foo")),
|
|
}
|
|
|
|
resp, body, err := c.Do(context.Background(), &fakeAction{})
|
|
if err != nil {
|
|
t.Fatalf("incorrect error value: want=nil got=%v", err)
|
|
}
|
|
|
|
wantCode := http.StatusTeapot
|
|
if wantCode != resp.StatusCode {
|
|
t.Fatalf("invalid response code: want=%d got=%d", wantCode, resp.StatusCode)
|
|
}
|
|
|
|
wantBody := []byte("foo")
|
|
if !reflect.DeepEqual(wantBody, body) {
|
|
t.Fatalf("invalid response body: want=%q got=%q", wantBody, body)
|
|
}
|
|
}
|
|
|
|
func TestSimpleHTTPClientDoError(t *testing.T) {
|
|
tr := newFakeTransport()
|
|
c := &simpleHTTPClient{transport: tr}
|
|
|
|
tr.errchan <- errors.New("fixture")
|
|
|
|
_, _, err := c.Do(context.Background(), &fakeAction{})
|
|
if err == nil {
|
|
t.Fatalf("expected non-nil error, got nil")
|
|
}
|
|
}
|
|
|
|
func TestSimpleHTTPClientDoCancelContext(t *testing.T) {
|
|
tr := newFakeTransport()
|
|
c := &simpleHTTPClient{transport: tr}
|
|
|
|
tr.startCancel <- struct{}{}
|
|
tr.finishCancel <- struct{}{}
|
|
|
|
_, _, err := c.Do(context.Background(), &fakeAction{})
|
|
if err == nil {
|
|
t.Fatalf("expected non-nil error, got nil")
|
|
}
|
|
}
|
|
|
|
func TestSimpleHTTPClientDoCancelContextWaitForRoundTrip(t *testing.T) {
|
|
tr := newFakeTransport()
|
|
c := &simpleHTTPClient{transport: tr}
|
|
|
|
donechan := make(chan struct{})
|
|
ctx, cancel := context.WithCancel(context.Background())
|
|
go func() {
|
|
c.Do(ctx, &fakeAction{})
|
|
close(donechan)
|
|
}()
|
|
|
|
// This should call CancelRequest and begin the cancellation process
|
|
cancel()
|
|
|
|
select {
|
|
case <-donechan:
|
|
t.Fatalf("simpleHTTPClient.Do should not have exited yet")
|
|
default:
|
|
}
|
|
|
|
tr.finishCancel <- struct{}{}
|
|
|
|
select {
|
|
case <-donechan:
|
|
//expected behavior
|
|
return
|
|
case <-time.After(time.Second):
|
|
t.Fatalf("simpleHTTPClient.Do did not exit within 1s")
|
|
}
|
|
}
|
|
|
|
func TestHTTPClusterClientDo(t *testing.T) {
|
|
fakeErr := errors.New("fake!")
|
|
fakeURL := url.URL{}
|
|
tests := []struct {
|
|
client *httpClusterClient
|
|
wantCode int
|
|
wantErr error
|
|
}{
|
|
// first good response short-circuits Do
|
|
{
|
|
client: &httpClusterClient{
|
|
endpoints: []url.URL{fakeURL, fakeURL},
|
|
clientFactory: newStaticHTTPClientFactory(
|
|
[]staticHTTPResponse{
|
|
staticHTTPResponse{resp: http.Response{StatusCode: http.StatusTeapot}},
|
|
staticHTTPResponse{err: fakeErr},
|
|
},
|
|
),
|
|
},
|
|
wantCode: http.StatusTeapot,
|
|
},
|
|
|
|
// fall through to good endpoint if err is arbitrary
|
|
{
|
|
client: &httpClusterClient{
|
|
endpoints: []url.URL{fakeURL, fakeURL},
|
|
clientFactory: newStaticHTTPClientFactory(
|
|
[]staticHTTPResponse{
|
|
staticHTTPResponse{err: fakeErr},
|
|
staticHTTPResponse{resp: http.Response{StatusCode: http.StatusTeapot}},
|
|
},
|
|
),
|
|
},
|
|
wantCode: http.StatusTeapot,
|
|
},
|
|
|
|
// context.DeadlineExceeded short-circuits Do
|
|
{
|
|
client: &httpClusterClient{
|
|
endpoints: []url.URL{fakeURL, fakeURL},
|
|
clientFactory: newStaticHTTPClientFactory(
|
|
[]staticHTTPResponse{
|
|
staticHTTPResponse{err: context.DeadlineExceeded},
|
|
staticHTTPResponse{resp: http.Response{StatusCode: http.StatusTeapot}},
|
|
},
|
|
),
|
|
},
|
|
wantErr: context.DeadlineExceeded,
|
|
},
|
|
|
|
// context.Canceled short-circuits Do
|
|
{
|
|
client: &httpClusterClient{
|
|
endpoints: []url.URL{fakeURL, fakeURL},
|
|
clientFactory: newStaticHTTPClientFactory(
|
|
[]staticHTTPResponse{
|
|
staticHTTPResponse{err: context.Canceled},
|
|
staticHTTPResponse{resp: http.Response{StatusCode: http.StatusTeapot}},
|
|
},
|
|
),
|
|
},
|
|
wantErr: context.Canceled,
|
|
},
|
|
|
|
// return err if there are no endpoints
|
|
{
|
|
client: &httpClusterClient{
|
|
endpoints: []url.URL{},
|
|
clientFactory: newHTTPClientFactory(nil, nil),
|
|
},
|
|
wantErr: ErrNoEndpoints,
|
|
},
|
|
|
|
// return err if all endpoints return arbitrary errors
|
|
{
|
|
client: &httpClusterClient{
|
|
endpoints: []url.URL{fakeURL, fakeURL},
|
|
clientFactory: newStaticHTTPClientFactory(
|
|
[]staticHTTPResponse{
|
|
staticHTTPResponse{err: fakeErr},
|
|
staticHTTPResponse{err: fakeErr},
|
|
},
|
|
),
|
|
},
|
|
wantErr: fakeErr,
|
|
},
|
|
|
|
// 500-level errors cause Do to fallthrough to next endpoint
|
|
{
|
|
client: &httpClusterClient{
|
|
endpoints: []url.URL{fakeURL, fakeURL},
|
|
clientFactory: newStaticHTTPClientFactory(
|
|
[]staticHTTPResponse{
|
|
staticHTTPResponse{resp: http.Response{StatusCode: http.StatusBadGateway}},
|
|
staticHTTPResponse{resp: http.Response{StatusCode: http.StatusTeapot}},
|
|
},
|
|
),
|
|
},
|
|
wantCode: http.StatusTeapot,
|
|
},
|
|
}
|
|
|
|
for i, tt := range tests {
|
|
resp, _, err := tt.client.Do(context.Background(), nil)
|
|
if !reflect.DeepEqual(tt.wantErr, err) {
|
|
t.Errorf("#%d: got err=%v, want=%v", i, err, tt.wantErr)
|
|
continue
|
|
}
|
|
|
|
if resp == nil {
|
|
if tt.wantCode != 0 {
|
|
t.Errorf("#%d: resp is nil, want=%d", i, tt.wantCode)
|
|
}
|
|
continue
|
|
}
|
|
|
|
if resp.StatusCode != tt.wantCode {
|
|
t.Errorf("#%d: resp code=%d, want=%d", i, resp.StatusCode, tt.wantCode)
|
|
continue
|
|
}
|
|
}
|
|
}
|
|
|
|
func TestRedirectedHTTPAction(t *testing.T) {
|
|
act := &redirectedHTTPAction{
|
|
action: &staticHTTPAction{
|
|
request: http.Request{
|
|
Method: "DELETE",
|
|
URL: &url.URL{
|
|
Scheme: "https",
|
|
Host: "foo.example.com",
|
|
Path: "/ping",
|
|
},
|
|
},
|
|
},
|
|
location: url.URL{
|
|
Scheme: "https",
|
|
Host: "bar.example.com",
|
|
Path: "/pong",
|
|
},
|
|
}
|
|
|
|
want := &http.Request{
|
|
Method: "DELETE",
|
|
URL: &url.URL{
|
|
Scheme: "https",
|
|
Host: "bar.example.com",
|
|
Path: "/pong",
|
|
},
|
|
}
|
|
got := act.HTTPRequest(url.URL{Scheme: "http", Host: "baz.example.com", Path: "/pang"})
|
|
|
|
if !reflect.DeepEqual(want, got) {
|
|
t.Fatalf("HTTPRequest is %#v, want %#v", want, got)
|
|
}
|
|
}
|
|
|
|
func TestRedirectFollowingHTTPClient(t *testing.T) {
|
|
tests := []struct {
|
|
checkRedirect CheckRedirectFunc
|
|
client httpClient
|
|
wantCode int
|
|
wantErr error
|
|
}{
|
|
// errors bubbled up
|
|
{
|
|
checkRedirect: func(int) error { return ErrTooManyRedirects },
|
|
client: &multiStaticHTTPClient{
|
|
responses: []staticHTTPResponse{
|
|
staticHTTPResponse{
|
|
err: errors.New("fail!"),
|
|
},
|
|
},
|
|
},
|
|
wantErr: errors.New("fail!"),
|
|
},
|
|
|
|
// no need to follow redirect if none given
|
|
{
|
|
checkRedirect: func(int) error { return ErrTooManyRedirects },
|
|
client: &multiStaticHTTPClient{
|
|
responses: []staticHTTPResponse{
|
|
staticHTTPResponse{
|
|
resp: http.Response{
|
|
StatusCode: http.StatusTeapot,
|
|
},
|
|
},
|
|
},
|
|
},
|
|
wantCode: http.StatusTeapot,
|
|
},
|
|
|
|
// redirects if less than max
|
|
{
|
|
checkRedirect: func(via int) error {
|
|
if via >= 2 {
|
|
return ErrTooManyRedirects
|
|
}
|
|
return nil
|
|
},
|
|
client: &multiStaticHTTPClient{
|
|
responses: []staticHTTPResponse{
|
|
staticHTTPResponse{
|
|
resp: http.Response{
|
|
StatusCode: http.StatusTemporaryRedirect,
|
|
Header: http.Header{"Location": []string{"http://example.com"}},
|
|
},
|
|
},
|
|
staticHTTPResponse{
|
|
resp: http.Response{
|
|
StatusCode: http.StatusTeapot,
|
|
},
|
|
},
|
|
},
|
|
},
|
|
wantCode: http.StatusTeapot,
|
|
},
|
|
|
|
// succeed after reaching max redirects
|
|
{
|
|
checkRedirect: func(via int) error {
|
|
if via >= 3 {
|
|
return ErrTooManyRedirects
|
|
}
|
|
return nil
|
|
},
|
|
client: &multiStaticHTTPClient{
|
|
responses: []staticHTTPResponse{
|
|
staticHTTPResponse{
|
|
resp: http.Response{
|
|
StatusCode: http.StatusTemporaryRedirect,
|
|
Header: http.Header{"Location": []string{"http://example.com"}},
|
|
},
|
|
},
|
|
staticHTTPResponse{
|
|
resp: http.Response{
|
|
StatusCode: http.StatusTemporaryRedirect,
|
|
Header: http.Header{"Location": []string{"http://example.com"}},
|
|
},
|
|
},
|
|
staticHTTPResponse{
|
|
resp: http.Response{
|
|
StatusCode: http.StatusTeapot,
|
|
},
|
|
},
|
|
},
|
|
},
|
|
wantCode: http.StatusTeapot,
|
|
},
|
|
|
|
// fail if too many redirects
|
|
{
|
|
checkRedirect: func(via int) error {
|
|
if via >= 2 {
|
|
return ErrTooManyRedirects
|
|
}
|
|
return nil
|
|
},
|
|
client: &multiStaticHTTPClient{
|
|
responses: []staticHTTPResponse{
|
|
staticHTTPResponse{
|
|
resp: http.Response{
|
|
StatusCode: http.StatusTemporaryRedirect,
|
|
Header: http.Header{"Location": []string{"http://example.com"}},
|
|
},
|
|
},
|
|
staticHTTPResponse{
|
|
resp: http.Response{
|
|
StatusCode: http.StatusTemporaryRedirect,
|
|
Header: http.Header{"Location": []string{"http://example.com"}},
|
|
},
|
|
},
|
|
staticHTTPResponse{
|
|
resp: http.Response{
|
|
StatusCode: http.StatusTeapot,
|
|
},
|
|
},
|
|
},
|
|
},
|
|
wantErr: ErrTooManyRedirects,
|
|
},
|
|
|
|
// fail if Location header not set
|
|
{
|
|
checkRedirect: func(int) error { return ErrTooManyRedirects },
|
|
client: &multiStaticHTTPClient{
|
|
responses: []staticHTTPResponse{
|
|
staticHTTPResponse{
|
|
resp: http.Response{
|
|
StatusCode: http.StatusTemporaryRedirect,
|
|
},
|
|
},
|
|
},
|
|
},
|
|
wantErr: errors.New("Location header not set"),
|
|
},
|
|
|
|
// fail if Location header is invalid
|
|
{
|
|
checkRedirect: func(int) error { return ErrTooManyRedirects },
|
|
client: &multiStaticHTTPClient{
|
|
responses: []staticHTTPResponse{
|
|
staticHTTPResponse{
|
|
resp: http.Response{
|
|
StatusCode: http.StatusTemporaryRedirect,
|
|
Header: http.Header{"Location": []string{":"}},
|
|
},
|
|
},
|
|
},
|
|
},
|
|
wantErr: errors.New("Location header not valid URL: :"),
|
|
},
|
|
}
|
|
|
|
for i, tt := range tests {
|
|
client := &redirectFollowingHTTPClient{client: tt.client, checkRedirect: tt.checkRedirect}
|
|
resp, _, err := client.Do(context.Background(), nil)
|
|
if !reflect.DeepEqual(tt.wantErr, err) {
|
|
t.Errorf("#%d: got err=%v, want=%v", i, err, tt.wantErr)
|
|
continue
|
|
}
|
|
|
|
if resp == nil {
|
|
if tt.wantCode != 0 {
|
|
t.Errorf("#%d: resp is nil, want=%d", i, tt.wantCode)
|
|
}
|
|
continue
|
|
}
|
|
|
|
if resp.StatusCode != tt.wantCode {
|
|
t.Errorf("#%d: resp code=%d, want=%d", i, resp.StatusCode, tt.wantCode)
|
|
continue
|
|
}
|
|
}
|
|
}
|