mirror of
				https://github.com/etcd-io/etcd.git
				synced 2024-09-27 06:25:44 +00:00 
			
		
		
		
	 2a151c8982
			
		
	
	
		2a151c8982
		
			
		
	
	
	
	
		
			
			The io/ioutil package has been deprecated as of Go 1.16, see https://golang.org/doc/go1.16#ioutil. This commit replaces the existing io/ioutil functions with their new definitions in io and os packages. Signed-off-by: Eng Zer Jun <engzerjun@gmail.com>
		
			
				
	
	
		
			138 lines
		
	
	
		
			3.4 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
			
		
		
	
	
			138 lines
		
	
	
		
			3.4 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
| package etcdhttp
 | |
| 
 | |
| import (
 | |
| 	"context"
 | |
| 	"encoding/json"
 | |
| 	"fmt"
 | |
| 	"io"
 | |
| 	"net/http"
 | |
| 	"net/http/httptest"
 | |
| 	"testing"
 | |
| 
 | |
| 	pb "go.etcd.io/etcd/api/v3/etcdserverpb"
 | |
| 	"go.etcd.io/etcd/client/pkg/v3/testutil"
 | |
| 	"go.etcd.io/etcd/client/pkg/v3/types"
 | |
| 	"go.etcd.io/etcd/raft/v3"
 | |
| 	"go.etcd.io/etcd/server/v3/etcdserver"
 | |
| 	stats "go.etcd.io/etcd/server/v3/etcdserver/api/v2stats"
 | |
| 	"go.uber.org/zap"
 | |
| )
 | |
| 
 | |
| type fakeStats struct{}
 | |
| 
 | |
| func (s *fakeStats) SelfStats() []byte   { return nil }
 | |
| func (s *fakeStats) LeaderStats() []byte { return nil }
 | |
| func (s *fakeStats) StoreStats() []byte  { return nil }
 | |
| 
 | |
| type fakeServerV2 struct {
 | |
| 	fakeServer
 | |
| 	stats.Stats
 | |
| 	health string
 | |
| }
 | |
| 
 | |
| func (s *fakeServerV2) Leader() types.ID {
 | |
| 	if s.health == "true" {
 | |
| 		return 1
 | |
| 	}
 | |
| 	return types.ID(raft.None)
 | |
| }
 | |
| func (s *fakeServerV2) Do(ctx context.Context, r pb.Request) (etcdserver.Response, error) {
 | |
| 	if s.health == "true" {
 | |
| 		return etcdserver.Response{}, nil
 | |
| 	}
 | |
| 	return etcdserver.Response{}, fmt.Errorf("fail health check")
 | |
| }
 | |
| func (s *fakeServerV2) ClientCertAuthEnabled() bool { return false }
 | |
| 
 | |
| func TestHealthHandler(t *testing.T) {
 | |
| 	// define the input and expected output
 | |
| 	// input: alarms, and healthCheckURL
 | |
| 	tests := []struct {
 | |
| 		alarms         []*pb.AlarmMember
 | |
| 		healthCheckURL string
 | |
| 		statusCode     int
 | |
| 		health         string
 | |
| 	}{
 | |
| 		{
 | |
| 			[]*pb.AlarmMember{},
 | |
| 			"/health",
 | |
| 			http.StatusOK,
 | |
| 			"true",
 | |
| 		},
 | |
| 		{
 | |
| 			[]*pb.AlarmMember{{MemberID: uint64(0), Alarm: pb.AlarmType_NOSPACE}},
 | |
| 			"/health",
 | |
| 			http.StatusServiceUnavailable,
 | |
| 			"false",
 | |
| 		},
 | |
| 		{
 | |
| 			[]*pb.AlarmMember{{MemberID: uint64(0), Alarm: pb.AlarmType_NOSPACE}},
 | |
| 			"/health?exclude=NOSPACE",
 | |
| 			http.StatusOK,
 | |
| 			"true",
 | |
| 		},
 | |
| 		{
 | |
| 			[]*pb.AlarmMember{},
 | |
| 			"/health?exclude=NOSPACE",
 | |
| 			http.StatusOK,
 | |
| 			"true",
 | |
| 		},
 | |
| 		{
 | |
| 			[]*pb.AlarmMember{{MemberID: uint64(0), Alarm: pb.AlarmType_NOSPACE}, {MemberID: uint64(1), Alarm: pb.AlarmType_CORRUPT}},
 | |
| 			"/health?exclude=NOSPACE",
 | |
| 			http.StatusServiceUnavailable,
 | |
| 			"false",
 | |
| 		},
 | |
| 		{
 | |
| 			[]*pb.AlarmMember{{MemberID: uint64(0), Alarm: pb.AlarmType_NOSPACE}, {MemberID: uint64(1), Alarm: pb.AlarmType_CORRUPT}},
 | |
| 			"/health?exclude=NOSPACE&exclude=CORRUPT",
 | |
| 			http.StatusOK,
 | |
| 			"true",
 | |
| 		},
 | |
| 	}
 | |
| 
 | |
| 	for i, tt := range tests {
 | |
| 		func() {
 | |
| 			mux := http.NewServeMux()
 | |
| 			HandleMetricsHealth(zap.NewExample(), mux, &fakeServerV2{
 | |
| 				fakeServer: fakeServer{alarms: tt.alarms},
 | |
| 				Stats:      &fakeStats{},
 | |
| 				health:     tt.health,
 | |
| 			})
 | |
| 			ts := httptest.NewServer(mux)
 | |
| 			defer ts.Close()
 | |
| 
 | |
| 			res, err := ts.Client().Do(&http.Request{Method: http.MethodGet, URL: testutil.MustNewURL(t, ts.URL+tt.healthCheckURL)})
 | |
| 			if err != nil {
 | |
| 				t.Errorf("fail serve http request %s %v in test case #%d", tt.healthCheckURL, err, i+1)
 | |
| 			}
 | |
| 			if res == nil {
 | |
| 				t.Errorf("got nil http response with http request %s in test case #%d", tt.healthCheckURL, i+1)
 | |
| 				return
 | |
| 			}
 | |
| 			if res.StatusCode != tt.statusCode {
 | |
| 				t.Errorf("want statusCode %d but got %d in test case #%d", tt.statusCode, res.StatusCode, i+1)
 | |
| 			}
 | |
| 			health, err := parseHealthOutput(res.Body)
 | |
| 			if err != nil {
 | |
| 				t.Errorf("fail parse health check output %v", err)
 | |
| 			}
 | |
| 			if health.Health != tt.health {
 | |
| 				t.Errorf("want health %s but got %s", tt.health, health.Health)
 | |
| 			}
 | |
| 		}()
 | |
| 	}
 | |
| }
 | |
| 
 | |
| func parseHealthOutput(body io.Reader) (Health, error) {
 | |
| 	obj := Health{}
 | |
| 	d, derr := io.ReadAll(body)
 | |
| 	if derr != nil {
 | |
| 		return obj, derr
 | |
| 	}
 | |
| 	if err := json.Unmarshal(d, &obj); err != nil {
 | |
| 		return obj, err
 | |
| 	}
 | |
| 	return obj, nil
 | |
| }
 |