visedriver/request/http/server_test.go

234 lines
5.9 KiB
Go
Raw Normal View History

2025-01-04 08:29:22 +01:00
package http
import (
"bytes"
2025-01-06 06:50:53 +01:00
"context"
2025-01-04 08:29:22 +01:00
"errors"
"net/http"
"net/http/httptest"
"testing"
"git.defalsify.org/vise.git/engine"
2025-01-11 22:47:57 +01:00
viseerrors "git.grassecon.net/grassrootseconomics/visedriver/errors"
"git.grassecon.net/grassrootseconomics/visedriver/testutil/mocks/httpmocks"
"git.grassecon.net/grassrootseconomics/visedriver/request"
2025-01-04 08:29:22 +01:00
)
// invalidRequestType is a custom type to test invalid request scenarios
type invalidRequestType struct{}
// errorReader is a helper type that always returns an error when Read is called
type errorReader struct{}
func (e *errorReader) Read(p []byte) (n int, err error) {
return 0, errors.New("read error")
}
func TestRequestHandler_ServeHTTP(t *testing.T) {
2025-01-04 08:29:22 +01:00
tests := []struct {
name string
sessionID string
input []byte
parserErr error
processErr error
outputErr error
resetErr error
expectedStatus int
}{
{
name: "Success",
sessionID: "123",
input: []byte("test input"),
expectedStatus: http.StatusOK,
},
{
name: "Missing Session ID",
sessionID: "",
2025-01-11 22:47:57 +01:00
parserErr: viseerrors.ErrSessionMissing,
2025-01-04 08:29:22 +01:00
expectedStatus: http.StatusBadRequest,
},
{
name: "Process Error",
sessionID: "123",
input: []byte("test input"),
2025-01-11 22:47:57 +01:00
processErr: viseerrors.ErrStorage,
2025-01-04 08:29:22 +01:00
expectedStatus: http.StatusInternalServerError,
},
{
name: "Output Error",
sessionID: "123",
input: []byte("test input"),
outputErr: errors.New("output error"),
expectedStatus: http.StatusOK,
},
{
name: "Reset Error",
sessionID: "123",
input: []byte("test input"),
resetErr: errors.New("reset error"),
expectedStatus: http.StatusOK,
},
}
for _, tt := range tests {
t.Run(tt.name, func(t *testing.T) {
mockRequestParser := &httpmocks.MockRequestParser{
GetSessionIdFunc: func(any) (string, error) {
return tt.sessionID, tt.parserErr
},
GetInputFunc: func(any) ([]byte, error) {
return tt.input, nil
},
}
mockRequestHandler := &httpmocks.MockRequestHandler{
2025-01-04 23:27:46 +01:00
ProcessFunc: func(rs request.RequestSession) (request.RequestSession, error) {
2025-01-04 08:29:22 +01:00
return rs, tt.processErr
},
2025-01-04 23:27:46 +01:00
OutputFunc: func(rs request.RequestSession) (request.RequestSession, error) {
2025-01-04 08:29:22 +01:00
return rs, tt.outputErr
},
2025-01-04 23:27:46 +01:00
ResetFunc: func(rs request.RequestSession) (request.RequestSession, error) {
2025-01-04 08:29:22 +01:00
return rs, tt.resetErr
},
2025-01-04 23:27:46 +01:00
GetRequestParserFunc: func() request.RequestParser {
2025-01-04 08:29:22 +01:00
return mockRequestParser
},
GetConfigFunc: func() engine.Config {
return engine.Config{}
},
}
sessionHandler := &HTTPRequestHandler{
2025-01-05 10:54:19 +01:00
RequestHandler: mockRequestHandler,
}
2025-01-04 08:29:22 +01:00
req := httptest.NewRequest(http.MethodPost, "/", bytes.NewBuffer(tt.input))
req.Header.Set("X-Vise-Session", tt.sessionID)
rr := httptest.NewRecorder()
sessionHandler.ServeHTTP(rr, req)
if status := rr.Code; status != tt.expectedStatus {
t.Errorf("handler returned wrong status code: got %v want %v",
status, tt.expectedStatus)
}
})
}
}
func TestRequestHandler_WriteError(t *testing.T) {
handler := &HTTPRequestHandler{}
2025-01-04 08:29:22 +01:00
mockWriter := &httpmocks.MockWriter{}
err := errors.New("test error")
handler.WriteError(mockWriter, http.StatusBadRequest, err)
if mockWriter.WrittenString != "" {
t.Errorf("Expected empty body, got %s", mockWriter.WrittenString)
}
}
func TestDefaultRequestParser_GetSessionId(t *testing.T) {
tests := []struct {
name string
request any
expectedID string
expectedError error
}{
{
name: "Valid Session ID",
request: func() *http.Request {
req := httptest.NewRequest(http.MethodPost, "/", nil)
req.Header.Set("X-Vise-Session", "123456")
return req
}(),
expectedID: "123456",
expectedError: nil,
},
{
name: "Missing Session ID",
request: httptest.NewRequest(http.MethodPost, "/", nil),
expectedID: "",
2025-01-11 22:47:57 +01:00
expectedError: viseerrors.ErrSessionMissing,
2025-01-04 08:29:22 +01:00
},
{
name: "Invalid Request Type",
request: invalidRequestType{},
expectedID: "",
2025-01-11 22:47:57 +01:00
expectedError: viseerrors.ErrInvalidRequest,
2025-01-04 08:29:22 +01:00
},
}
parser := &DefaultRequestParser{}
for _, tt := range tests {
t.Run(tt.name, func(t *testing.T) {
2025-01-06 06:50:53 +01:00
id, err := parser.GetSessionId(context.Background(),tt.request)
2025-01-04 08:29:22 +01:00
if id != tt.expectedID {
t.Errorf("Expected session ID %s, got %s", tt.expectedID, id)
}
if err != tt.expectedError {
t.Errorf("Expected error %v, got %v", tt.expectedError, err)
}
})
}
}
func TestDefaultRequestParser_GetInput(t *testing.T) {
tests := []struct {
name string
request any
expectedInput []byte
expectedError error
}{
{
name: "Valid Input",
request: func() *http.Request {
return httptest.NewRequest(http.MethodPost, "/", bytes.NewBufferString("test input"))
}(),
expectedInput: []byte("test input"),
expectedError: nil,
},
{
name: "Empty Input",
request: httptest.NewRequest(http.MethodPost, "/", nil),
expectedInput: []byte{},
expectedError: nil,
},
{
name: "Invalid Request Type",
request: invalidRequestType{},
expectedInput: nil,
2025-01-11 22:47:57 +01:00
expectedError: viseerrors.ErrInvalidRequest,
2025-01-04 08:29:22 +01:00
},
{
name: "Read Error",
request: func() *http.Request {
return httptest.NewRequest(http.MethodPost, "/", &errorReader{})
}(),
expectedInput: nil,
expectedError: errors.New("read error"),
},
}
parser := &DefaultRequestParser{}
for _, tt := range tests {
t.Run(tt.name, func(t *testing.T) {
input, err := parser.GetInput(tt.request)
if !bytes.Equal(input, tt.expectedInput) {
t.Errorf("Expected input %s, got %s", tt.expectedInput, input)
}
if err != tt.expectedError && (err == nil || err.Error() != tt.expectedError.Error()) {
t.Errorf("Expected error %v, got %v", tt.expectedError, err)
}
})
}
}