diff --git a/.github/workflows/release.yml b/.github/workflows/release.yml index 7af7f66..54f8a17 100644 --- a/.github/workflows/release.yml +++ b/.github/workflows/release.yml @@ -34,6 +34,12 @@ jobs: steps: - uses: actions/checkout@v3 + - name: Set up QEMU + uses: docker/setup-qemu-action@v3 + + - name: Set up Docker Buildx + uses: docker/setup-buildx-action@v3 + - name: Login to DockerHub uses: docker/login-action@v3 with: @@ -53,5 +59,6 @@ jobs: uses: docker/build-push-action@v5 with: context: . + platforms: linux/amd64,linux/arm64 push: true tags: ${{ steps.meta.outputs.tags }} \ No newline at end of file diff --git a/Dockerfile b/Dockerfile index 857fb20..85e5ad7 100644 --- a/Dockerfile +++ b/Dockerfile @@ -2,13 +2,16 @@ FROM golang:1.21-alpine AS build LABEL MAINTAINER = 'Friends of Go (it@friendsofgo.tech)' +ARG TARGETOS=linux +ARG TARGETARCH=amd64 + RUN apk add --update git RUN apk add ca-certificates WORKDIR /go/src/github.com/friendsofgo/killgrave COPY . . RUN go mod tidy && TAG=$(git describe --tags --abbrev=0) \ && LDFLAGS=$(echo "-s -w -X github.com/friendsofgo/killgrave/internal/app/cmd._version="docker-$TAG) \ - && CGO_ENABLED=0 GOOS=linux GOARCH=amd64 go build -a -installsuffix cgo -o /go/bin/killgrave -ldflags "$LDFLAGS" cmd/killgrave/main.go + && CGO_ENABLED=0 GOOS="${TARGETOS}" GOARCH="${TARGETARCH}" go build -a -installsuffix cgo -o /go/bin/killgrave -ldflags "$LDFLAGS" cmd/killgrave/main.go # Building image with the binary FROM scratch diff --git a/README.md b/README.md index a1792d1..8b12570 100644 --- a/README.md +++ b/README.md @@ -9,13 +9,8 @@ Killgrave is a simulator for HTTP-based APIs, in simple words a **Mock Server**, ![Github actions](https://github.com/friendsofgo/killgrave/actions/workflows/main.yaml/badge.svg?branch=main) [![Version](https://img.shields.io/github/release/friendsofgo/killgrave.svg?style=flat-square)](https://github.com/friendsofgo/killgrave/releases/latest) [![Go Report Card](https://goreportcard.com/badge/github.com/friendsofgo/killgrave)](https://goreportcard.com/report/github.com/friendsofgo/killgrave) -[![Total alerts](https://img.shields.io/lgtm/alerts/g/friendsofgo/killgrave.svg?logo=lgtm&logoWidth=18)](https://lgtm.com/projects/g/friendsofgo/killgrave/alerts/) [![FriendsOfGo](https://img.shields.io/badge/powered%20by-Friends%20of%20Go-73D7E2.svg)](https://friendsofgo.tech) -
- # Table of Content - [Overview](#overview) - [Concepts](#concepts) @@ -91,13 +86,13 @@ $ brew install friendsofgo/tap/killgrave The application is also available through [Docker](https://hub.docker.com/r/friendsofgo/killgrave). ```bash -docker run -it --rm -p 3000:3000 -v $PWD/:/home -w /home friendsofgo/killgrave -host 0.0.0.0 +docker run -it --rm -p 3000:3000 -v $PWD/:/home -w /home friendsofgo/killgrave --host 0.0.0.0 ``` `-p 3000:3000` [publishes](https://docs.docker.com/engine/reference/run/#expose-incoming-ports) port 3000 (Killgrave's default port) inside the container to port 3000 on the host machine. -`-host 0.0.0.0` is necessary to allow Killgrave to listen and respond to requests from outside the container (the default, +`--host 0.0.0.0` is necessary to allow Killgrave to listen and respond to requests from outside the container (the default, `localhost`, will not capture requests from the host network). ### Compile by yourself diff --git a/internal/server/http/handler.go b/internal/server/http/handler.go index 83ee85a..5108d6c 100644 --- a/internal/server/http/handler.go +++ b/internal/server/http/handler.go @@ -9,32 +9,33 @@ import ( ) // ImposterHandler create specific handler for the received imposter -func ImposterHandler(imposter Imposter) http.HandlerFunc { +func ImposterHandler(i Imposter) http.HandlerFunc { return func(w http.ResponseWriter, r *http.Request) { - if imposter.Delay() > 0 { - time.Sleep(imposter.Delay()) + res := i.NextResponse() + if res.Delay.Delay() > 0 { + time.Sleep(res.Delay.Delay()) } - writeHeaders(imposter, w) - w.WriteHeader(imposter.Response.Status) - writeBody(imposter, w) + writeHeaders(res, w) + w.WriteHeader(res.Status) + writeBody(i, res, w) } } -func writeHeaders(imposter Imposter, w http.ResponseWriter) { - if imposter.Response.Headers == nil { +func writeHeaders(r Response, w http.ResponseWriter) { + if r.Headers == nil { return } - for key, val := range *imposter.Response.Headers { + for key, val := range *r.Headers { w.Header().Set(key, val) } } -func writeBody(imposter Imposter, w http.ResponseWriter) { - wb := []byte(imposter.Response.Body) +func writeBody(i Imposter, r Response, w http.ResponseWriter) { + wb := []byte(r.Body) - if imposter.Response.BodyFile != nil { - bodyFile := imposter.CalculateFilePath(*imposter.Response.BodyFile) + if r.BodyFile != nil { + bodyFile := i.CalculateFilePath(*r.BodyFile) wb = fetchBodyFromFile(bodyFile) } w.Write(wb) diff --git a/internal/server/http/handler_test.go b/internal/server/http/handler_test.go index c9be466..395748e 100644 --- a/internal/server/http/handler_test.go +++ b/internal/server/http/handler_test.go @@ -9,6 +9,7 @@ import ( "testing" "github.com/stretchr/testify/assert" + "github.com/stretchr/testify/require" ) func TestImposterHandler(t *testing.T) { @@ -47,9 +48,9 @@ func TestImposterHandler(t *testing.T) { expectedBody string statusCode int }{ - {"valid imposter with body", Imposter{Request: validRequest, Response: Response{Status: http.StatusOK, Headers: &headers, Body: body}}, body, http.StatusOK}, - {"valid imposter with bodyFile", Imposter{Request: validRequest, Response: Response{Status: http.StatusOK, Headers: &headers, BodyFile: &bodyFile}}, string(expectedBodyFileData), http.StatusOK}, - {"valid imposter with not exists bodyFile", Imposter{Request: validRequest, Response: Response{Status: http.StatusOK, Headers: &headers, BodyFile: &bodyFileFake}}, "", http.StatusOK}, + {"valid imposter with body", Imposter{Request: validRequest, Response: Responses{{Status: http.StatusOK, Headers: &headers, Body: body}}}, body, http.StatusOK}, + {"valid imposter with bodyFile", Imposter{Request: validRequest, Response: Responses{{Status: http.StatusOK, Headers: &headers, BodyFile: &bodyFile}}}, string(expectedBodyFileData), http.StatusOK}, + {"valid imposter with not exists bodyFile", Imposter{Request: validRequest, Response: Responses{{Status: http.StatusOK, Headers: &headers, BodyFile: &bodyFileFake}}}, "", http.StatusOK}, } for _, tt := range dataTest { @@ -58,7 +59,7 @@ func TestImposterHandler(t *testing.T) { assert.NoError(t, err) rec := httptest.NewRecorder() - handler := http.HandlerFunc(ImposterHandler(tt.imposter)) + handler := ImposterHandler(tt.imposter) handler.ServeHTTP(rec, req) assert.Equal(t, rec.Code, tt.statusCode) @@ -85,7 +86,7 @@ func TestInvalidRequestWithSchema(t *testing.T) { statusCode int request []byte }{ - {"valid request no schema", Imposter{Request: Request{Method: "POST", Endpoint: "/gophers"}, Response: Response{Status: http.StatusOK, Body: "test ok"}}, http.StatusOK, validRequest}, + {"valid request no schema", Imposter{Request: Request{Method: "POST", Endpoint: "/gophers"}, Response: Responses{{Status: http.StatusOK, Body: "test ok"}}}, http.StatusOK, validRequest}, } for _, tt := range dataTest { @@ -94,7 +95,7 @@ func TestInvalidRequestWithSchema(t *testing.T) { req, err := http.NewRequest("POST", "/gophers", bytes.NewBuffer(tt.request)) assert.Nil(t, err) rec := httptest.NewRecorder() - handler := http.HandlerFunc(ImposterHandler(tt.imposter)) + handler := ImposterHandler(tt.imposter) handler.ServeHTTP(rec, req) @@ -102,3 +103,61 @@ func TestInvalidRequestWithSchema(t *testing.T) { }) } } + +func TestImposterHandler_MultipleRequests(t *testing.T) { + req, err := http.NewRequest("POST", "/gophers", bytes.NewBuffer([]byte(`{ + "data": { + "type": "gophers", + "attributes": { + "name": "Zebediah", + "color": "Purple" + } + } + }`))) + require.NoError(t, err) + + t.Run("created then conflict", func(t *testing.T) { + imp := Imposter{ + Request: Request{Method: "POST", Endpoint: "/gophers"}, + Response: Responses{ + {Status: http.StatusCreated, Body: "Created"}, + {Status: http.StatusConflict, Body: "Conflict"}, + }, + } + + handler := ImposterHandler(imp) + + // First request + rec := httptest.NewRecorder() + handler.ServeHTTP(rec, req) + assert.Equal(t, http.StatusCreated, rec.Code) + assert.Equal(t, "Created", rec.Body.String()) + + // Second request + rec = httptest.NewRecorder() + handler.ServeHTTP(rec, req) + assert.Equal(t, http.StatusConflict, rec.Code) + assert.Equal(t, "Conflict", rec.Body.String()) + }) + + t.Run("idempotent", func(t *testing.T) { + handler := ImposterHandler(Imposter{ + Request: Request{Method: "POST", Endpoint: "/gophers"}, + Response: Responses{ + {Status: http.StatusAccepted, Body: "Accepted"}, + }, + }) + + // First request + rec := httptest.NewRecorder() + handler.ServeHTTP(rec, req) + assert.Equal(t, http.StatusAccepted, rec.Code) + assert.Equal(t, "Accepted", rec.Body.String()) + + // Second request + rec = httptest.NewRecorder() + handler.ServeHTTP(rec, req) + assert.Equal(t, http.StatusAccepted, rec.Code) + assert.Equal(t, "Accepted", rec.Body.String()) + }) +} diff --git a/internal/server/http/imposter.go b/internal/server/http/imposter.go index bfd99b5..0fe0a95 100644 --- a/internal/server/http/imposter.go +++ b/internal/server/http/imposter.go @@ -3,12 +3,12 @@ package http import ( "encoding/json" "fmt" + "io" "io/fs" "os" "path" "path/filepath" "strings" - "time" "gopkg.in/yaml.v2" ) @@ -37,18 +37,22 @@ type ImposterConfig struct { // Imposter define an imposter structure type Imposter struct { - BasePath string `json:"-" yaml:"-"` - Path string `json:"-" yaml:"-"` - Request Request `json:"request"` - Response Response `json:"response"` + BasePath string `json:"-" yaml:"-"` + Path string `json:"-" yaml:"-"` + Request Request `json:"request"` + Response Responses `json:"response"` + resIdx int } -// Delay returns delay for response that user can specify in imposter config -func (i *Imposter) Delay() time.Duration { - return i.Response.Delay.Delay() +// NextResponse returns the imposter's response. +// If there are multiple responses, it will return them sequentially. +func (i *Imposter) NextResponse() Response { + r := i.Response[i.resIdx] + i.resIdx = (i.resIdx + 1) % len(i.Response) + return r } -// CalculateFilePath calculate file path based on basePath of imposter directory +// CalculateFilePath calculate file path based on basePath of imposter's directory func (i *Imposter) CalculateFilePath(filePath string) string { return path.Join(i.BasePath, filePath) } @@ -71,6 +75,58 @@ type Response struct { Delay ResponseDelay `json:"delay" yaml:"delay"` } +// Responses is a wrapper for Response, to allow the use of either a single +// response or an array of responses, while keeping backwards compatibility. +type Responses []Response + +func (rr *Responses) MarshalJSON() ([]byte, error) { + if len(*rr) == 1 { + return json.Marshal((*rr)[0]) + } + return json.Marshal(*rr) +} + +func (rr *Responses) UnmarshalJSON(data []byte) error { + if string(data) == "null" { + *rr = nil + return nil + } + + if data[0] == '[' { + return json.Unmarshal(data, (*[]Response)(rr)) + } + + var r Response + if err := json.Unmarshal(data, &r); err != nil { + return err + } + + *rr = Responses{r} + return nil +} + +func (rr *Responses) MarshalYAML() (interface{}, error) { + if len(*rr) == 1 { + return (*rr)[0], nil + } + return *rr, nil +} + +func (rr *Responses) UnmarshalYAML(unmarshal func(interface{}) error) error { + var r Response + if err := unmarshal(&r); err == nil { + *rr = Responses{r} + return nil + } + + var tmp []Response + if err := unmarshal(&tmp); err != nil { + return err + } + *rr = tmp + return nil +} + type ImposterFs struct { path string fs fs.FS @@ -117,8 +173,10 @@ func (i ImposterFs) FindImposters(impostersCh chan []Imposter) error { } func (i ImposterFs) unmarshalImposters(imposterConfig ImposterConfig) ([]Imposter, error) { - // TODO: Error handling? - bytes, _ := fs.ReadFile(i.fs, imposterConfig.FilePath) + imposterFile, _ := i.fs.Open(imposterConfig.FilePath) + defer imposterFile.Close() + + bytes, _ := io.ReadAll(imposterFile) var parseError error var imposters []Imposter diff --git a/internal/server/http/imposter_test.go b/internal/server/http/imposter_test.go index 4e8f3d9..c757597 100644 --- a/internal/server/http/imposter_test.go +++ b/internal/server/http/imposter_test.go @@ -3,7 +3,10 @@ package http import ( "testing" + "encoding/json" "github.com/stretchr/testify/assert" + "github.com/stretchr/testify/require" + "gopkg.in/yaml.v2" ) func TestNewImposterFS(t *testing.T) { @@ -18,3 +21,138 @@ func TestNewImposterFS(t *testing.T) { assert.NoError(t, err) }) } + +func TestResponses_MarshalJSON(t *testing.T) { + tcs := map[string]struct { + rr *Responses + exp string + }{ + "single response": { + rr: &Responses{{Status: 200, Body: "OK"}}, + exp: `{"status":200,"body":"OK","bodyFile":null,"headers":null,"delay":{}}`, + }, + "multiple response": { + rr: &Responses{{Status: 200, Body: "OK"}, {Status: 404, Body: "Not Found"}}, + exp: `[{"status":200,"body":"OK","bodyFile":null,"headers":null,"delay":{}},{"status":404,"body":"Not Found","bodyFile":null,"headers":null,"delay":{}}]`, + }, + "empty array": { + rr: &Responses{}, + exp: `[]`, + }, + "null array": { + rr: nil, + exp: `null`, + }, + } + for name, tc := range tcs { + t.Run(name, func(t *testing.T) { + got, err := json.Marshal(tc.rr) + assert.NoError(t, err) + assert.Equal(t, tc.exp, string(got)) + }) + } +} + +func TestResponses_UnmarshalJSON(t *testing.T) { + tcs := map[string]struct { + data string + exp Imposter + }{ + "single response": { + data: `{"response": {"status":200,"body":"OK"}}`, + exp: Imposter{Response: Responses{{Status: 200, Body: "OK"}}}, + }, + "single array response": { + data: `{"response": [{"status":200,"body":"OK"}]}`, + exp: Imposter{Response: Responses{{Status: 200, Body: "OK"}}}, + }, + "multiple array response": { + data: `{"response": [{"status":200,"body":"OK"}, {"status":404,"body":"Not Found"}]}`, + exp: Imposter{Response: Responses{{Status: 200, Body: "OK"}, {Status: 404, Body: "Not Found"}}}, + }, + "empty array": { + data: `{"response": []}`, + exp: Imposter{Response: Responses{}}, + }, + "null array": { + data: `{"response": null}`, + exp: Imposter{Response: nil}, + }} + + for name, tc := range tcs { + t.Run(name, func(t *testing.T) { + var got Imposter + err := json.Unmarshal([]byte(tc.data), &got) + require.NoError(t, err) + assert.Equal(t, tc.exp, got) + }) + } +} + +func TestResponses_MarshalYAML(t *testing.T) { + tcs := map[string]struct { + rr *Responses + exp string + }{ + "single response": { + rr: &Responses{{Status: 200, Body: "OK"}}, + exp: "status: 200\nbody: OK\nbodyFile: null\nheaders: null\ndelay: {}\n", + }, + "multiple response": { + rr: &Responses{{Status: 200, Body: "OK"}, {Status: 404, Body: "Not Found"}}, + exp: "- status: 200\n body: OK\n bodyFile: null\n headers: null\n delay: {}\n- status: 404\n body: Not Found\n bodyFile: null\n headers: null\n delay: {}\n", + }, + "empty array": { + rr: &Responses{}, + exp: "[]\n", + }, + "null array": { + rr: nil, + exp: "null\n", + }, + } + + for name, tc := range tcs { + t.Run(name, func(t *testing.T) { + got, err := yaml.Marshal(tc.rr) + assert.NoError(t, err) + assert.Equal(t, tc.exp, string(got)) + }) + } +} + +func TestResponses_UnmarshalYAML(t *testing.T) { + tcs := map[string]struct { + data string + exp Imposter + }{ + "single response": { + data: "response:\n status: 200\n body: OK\n", + exp: Imposter{Response: Responses{{Status: 200, Body: "OK"}}}, + }, + "single array response": { + data: "response:\n- status: 200\n body: OK\n", + exp: Imposter{Response: Responses{{Status: 200, Body: "OK"}}}, + }, + "multiple array response": { + data: "response:\n- status: 200\n body: OK\n- status: 404\n body: Not Found\n", + exp: Imposter{Response: Responses{{Status: 200, Body: "OK"}, {Status: 404, Body: "Not Found"}}}, + }, + "empty array": { + data: "response: []\n", + exp: Imposter{Response: Responses{}}, + }, + "null array": { + data: "response: \n", + exp: Imposter{Response: nil}, + }} + + for name, tc := range tcs { + t.Run(name, func(t *testing.T) { + var got Imposter + err := yaml.Unmarshal([]byte(tc.data), &got) + require.NoError(t, err) + assert.Equal(t, tc.exp, got) + }) + } +} diff --git a/internal/server/http/route_matchers_test.go b/internal/server/http/route_matchers_test.go index 664394b..8a03253 100644 --- a/internal/server/http/route_matchers_test.go +++ b/internal/server/http/route_matchers_test.go @@ -47,7 +47,7 @@ func TestMatcherBySchema(t *testing.T) { httpRequestA := &http.Request{Body: bodyA} httpRequestB := &http.Request{Body: bodyB} - okResponse := Response{Status: http.StatusOK} + okResponse := Responses{{Status: http.StatusOK}} var matcherData = map[string]struct { fn mux.MatcherFunc @@ -56,7 +56,7 @@ func TestMatcherBySchema(t *testing.T) { }{ "correct request schema": {MatcherBySchema(Imposter{Request: requestWithSchema, Response: okResponse}), httpRequestA, true}, "imposter without request schema": {MatcherBySchema(Imposter{Request: requestWithoutSchema, Response: okResponse}), httpRequestA, true}, - "malformatted schema file": {MatcherBySchema(Imposter{Request: requestWithWrongSchema, Response: okResponse}), httpRequestA, false}, + "malformed schema file": {MatcherBySchema(Imposter{Request: requestWithWrongSchema, Response: okResponse}), httpRequestA, false}, "incorrect request schema": {MatcherBySchema(Imposter{Request: requestWithSchema, Response: okResponse}), httpRequestB, false}, "non-existing schema file": {MatcherBySchema(Imposter{Request: requestWithNonExistingSchema, Response: okResponse}), httpRequestB, false}, "empty body with required schema file": {MatcherBySchema(Imposter{Request: requestWithSchema, Response: okResponse}), &http.Request{Body: emptyBody}, false}, diff --git a/internal/server/http/server_test.go b/internal/server/http/server_test.go index 42c2d4b..26ea559 100644 --- a/internal/server/http/server_test.go +++ b/internal/server/http/server_test.go @@ -3,7 +3,6 @@ package http import ( "crypto/tls" "io" - "io/ioutil" "log" "net/http" "net/http/httptest" @@ -18,7 +17,7 @@ import ( ) func TestMain(m *testing.M) { - log.SetOutput(ioutil.Discard) + log.SetOutput(io.Discard) os.Exit(m.Run()) } @@ -201,7 +200,7 @@ func TestBuildSecureMode(t *testing.T) { defer response.Body.Close() - body, err := ioutil.ReadAll(response.Body) + body, err := io.ReadAll(response.Body) if err != nil { return false }