// Copyright 2009 The Go Authors. All rights reserved. // Copyright 2012 The Gorilla Authors. All rights reserved. // Use of this source code is governed by a BSD-style // license that can be found in the LICENSE file. // Copyright 2021 MinIO, Inc. All rights reserved. // forked from https://github.com/gorilla/rpc/v2 // modified to be used with MinIO under GNU Affero General // Public License 3.0 license that can be found in // the LICENSE file. package rpc import ( "errors" "net/http" "strconv" "testing" ) type Service1Request struct { A int B int } type Service1Response struct { Result int } type Service1 struct { } func (t *Service1) Multiply(r *http.Request, req *Service1Request, res *Service1Response) error { res.Result = req.A * req.B return nil } type Service2 struct { } func TestRegisterService(t *testing.T) { var err error s := NewServer() service1 := new(Service1) service2 := new(Service2) // Inferred name. err = s.RegisterService(service1, "") if err != nil || !s.HasMethod("Service1.Multiply") { t.Errorf("Expected to be registered: Service1.Multiply") } // Provided name. err = s.RegisterService(service1, "Foo") if err != nil || !s.HasMethod("Foo.Multiply") { t.Errorf("Expected to be registered: Foo.Multiply") } // No methods. err = s.RegisterService(service2, "") if err == nil { t.Errorf("Expected error on service2") } } // MockCodec decodes to Service1.Multiply. type MockCodec struct { A, B int } func (c MockCodec) NewRequest(*http.Request) CodecRequest { return MockCodecRequest{c.A, c.B} } type MockCodecRequest struct { A, B int } func (r MockCodecRequest) Method() (string, error) { return "Service1.Multiply", nil } func (r MockCodecRequest) ReadRequest(args interface{}) error { req := args.(*Service1Request) req.A, req.B = r.A, r.B return nil } func (r MockCodecRequest) WriteResponse(w http.ResponseWriter, reply interface{}) { res := reply.(*Service1Response) w.Write([]byte(strconv.Itoa(res.Result))) } func (r MockCodecRequest) WriteError(w http.ResponseWriter, status int, err error) { w.WriteHeader(status) w.Write([]byte(err.Error())) } type MockResponseWriter struct { header http.Header Status int Body string } func NewMockResponseWriter() *MockResponseWriter { header := make(http.Header) return &MockResponseWriter{header: header} } func (w *MockResponseWriter) Header() http.Header { return w.header } func (w *MockResponseWriter) Write(p []byte) (int, error) { w.Body = string(p) if w.Status == 0 { w.Status = 200 } return len(p), nil } func (w *MockResponseWriter) WriteHeader(status int) { w.Status = status } func TestServeHTTP(t *testing.T) { const ( A = 2 B = 3 ) expected := A * B s := NewServer() s.RegisterService(new(Service1), "") s.RegisterCodec(MockCodec{A, B}, "mock") r, err := http.NewRequest("POST", "", nil) if err != nil { t.Fatal(err) } r.Header.Set("Content-Type", "mock; dummy") w := NewMockResponseWriter() s.ServeHTTP(w, r) if w.Status != 200 { t.Errorf("Status was %d, should be 200.", w.Status) } if w.Body != strconv.Itoa(expected) { t.Errorf("Response body was %s, should be %s.", w.Body, strconv.Itoa(expected)) } // Test wrong Content-Type r.Header.Set("Content-Type", "invalid") w = NewMockResponseWriter() s.ServeHTTP(w, r) if w.Status != 415 { t.Errorf("Status was %d, should be 415.", w.Status) } if w.Body != "rpc: unrecognized Content-Type: invalid" { t.Errorf("Wrong response body.") } // Test omitted Content-Type; codec should default to the sole registered one. r.Header.Del("Content-Type") w = NewMockResponseWriter() s.ServeHTTP(w, r) if w.Status != 200 { t.Errorf("Status was %d, should be 200.", w.Status) } if w.Body != strconv.Itoa(expected) { t.Errorf("Response body was %s, should be %s.", w.Body, strconv.Itoa(expected)) } } func TestInterception(t *testing.T) { const ( A = 2 B = 3 ) expected := A * B r2, err := http.NewRequest("POST", "mocked/request", nil) if err != nil { t.Fatal(err) } s := NewServer() s.RegisterService(new(Service1), "") s.RegisterCodec(MockCodec{A, B}, "mock") s.RegisterInterceptFunc(func(i *RequestInfo) *http.Request { return r2 }) s.RegisterValidateRequestFunc(func(info *RequestInfo, v interface{}) error { return nil }) s.RegisterAfterFunc(func(i *RequestInfo) { if i.Request != r2 { t.Errorf("Request was %v, should be %v.", i.Request, r2) } }) r, err := http.NewRequest("POST", "", nil) if err != nil { t.Fatal(err) } r.Header.Set("Content-Type", "mock; dummy") w := NewMockResponseWriter() s.ServeHTTP(w, r) if w.Status != 200 { t.Errorf("Status was %d, should be 200.", w.Status) } if w.Body != strconv.Itoa(expected) { t.Errorf("Response body was %s, should be %s.", w.Body, strconv.Itoa(expected)) } } func TestValidationSuccessful(t *testing.T) { const ( A = 2 B = 3 expected = A * B ) validate := func(info *RequestInfo, v interface{}) error { return nil } s := NewServer() s.RegisterService(new(Service1), "") s.RegisterCodec(MockCodec{A, B}, "mock") s.RegisterValidateRequestFunc(validate) r, err := http.NewRequest("POST", "", nil) if err != nil { t.Fatal(err) } r.Header.Set("Content-Type", "mock; dummy") w := NewMockResponseWriter() s.ServeHTTP(w, r) if w.Status != 200 { t.Errorf("Status was %d, should be 200.", w.Status) } if w.Body != strconv.Itoa(expected) { t.Errorf("Response body was %s, should be %s.", w.Body, strconv.Itoa(expected)) } } func TestValidationFails(t *testing.T) { const expected = "this instance only supports zero values" validate := func(r *RequestInfo, v interface{}) error { req := v.(*Service1Request) if req.A != 0 || req.B != 0 { return errors.New(expected) } return nil } s := NewServer() s.RegisterService(new(Service1), "") s.RegisterCodec(MockCodec{1, 2}, "mock") s.RegisterValidateRequestFunc(validate) r, err := http.NewRequest("POST", "", nil) if err != nil { t.Fatal(err) } r.Header.Set("Content-Type", "mock; dummy") w := NewMockResponseWriter() s.ServeHTTP(w, r) if w.Status != 400 { t.Errorf("Status was %d, should be 200.", w.Status) } if w.Body != expected { t.Errorf("Response body was %s, should be %s.", w.Body, expected) } }