planner/sync/service/handler_test.go

224 lines
5.4 KiB
Go
Raw Normal View History

2024-09-18 07:40:50 +02:00
package main
2024-08-28 07:21:02 +02:00
import (
"bytes"
"encoding/json"
"fmt"
"io"
"log/slog"
"net/http"
"net/http/httptest"
"net/url"
"os"
"sort"
2024-09-11 07:33:22 +02:00
"strings"
2024-08-28 07:21:02 +02:00
"testing"
"time"
2024-09-18 07:55:14 +02:00
"go-mod.ewintr.nl/planner/sync/item"
2024-08-28 07:21:02 +02:00
)
2024-09-07 12:10:48 +02:00
func TestServerServeHTTP(t *testing.T) {
t.Parallel()
apiKey := "test"
2024-09-08 10:09:54 +02:00
srv := NewServer(NewMemory(), apiKey, slog.New(slog.NewJSONHandler(os.Stdout, nil)))
2024-09-07 12:10:48 +02:00
for _, tc := range []struct {
name string
key string
url string
method string
expStatus int
}{
{
name: "index always visible",
url: "/",
method: http.MethodGet,
expStatus: http.StatusOK,
},
} {
t.Run(tc.name, func(t *testing.T) {
req, err := http.NewRequest(tc.method, tc.url, nil)
if err != nil {
t.Errorf("exp nil, got %v", err)
}
res := httptest.NewRecorder()
srv.ServeHTTP(res, req)
if res.Result().StatusCode != tc.expStatus {
t.Errorf("exp %v, got %v", tc.expStatus, res.Result().StatusCode)
}
})
}
}
2024-08-28 07:21:02 +02:00
func TestSyncGet(t *testing.T) {
t.Parallel()
now := time.Now()
2024-09-08 10:09:54 +02:00
mem := NewMemory()
2024-08-28 07:21:02 +02:00
2024-09-18 07:55:14 +02:00
items := []item.Item{
{ID: "id-0", Kind: item.KindEvent, Updated: now.Add(-10 * time.Minute)},
{ID: "id-1", Kind: item.KindEvent, Updated: now.Add(-5 * time.Minute)},
{ID: "id-2", Kind: item.KindTask, Updated: now.Add(time.Minute)},
2024-08-28 07:21:02 +02:00
}
for _, item := range items {
if err := mem.Update(item); err != nil {
t.Errorf("exp nil, got %v", err)
}
}
2024-09-07 12:10:48 +02:00
apiKey := "test"
2024-09-08 10:09:54 +02:00
srv := NewServer(mem, apiKey, slog.New(slog.NewJSONHandler(os.Stdout, nil)))
2024-08-28 07:21:02 +02:00
for _, tc := range []struct {
name string
ts time.Time
2024-09-11 07:33:22 +02:00
ks []string
2024-08-28 07:21:02 +02:00
expStatus int
2024-09-18 07:55:14 +02:00
expItems []item.Item
2024-08-28 07:21:02 +02:00
}{
{
name: "full",
expStatus: http.StatusOK,
expItems: items,
},
{
2024-09-11 07:33:22 +02:00
name: "new",
2024-08-28 07:21:02 +02:00
ts: now.Add(-6 * time.Minute),
expStatus: http.StatusOK,
2024-09-18 07:55:14 +02:00
expItems: []item.Item{items[1], items[2]},
2024-08-28 07:21:02 +02:00
},
2024-09-11 07:33:22 +02:00
{
name: "kind",
2024-09-18 07:55:14 +02:00
ks: []string{string(item.KindTask)},
2024-09-11 07:33:22 +02:00
expStatus: http.StatusOK,
2024-09-18 07:55:14 +02:00
expItems: []item.Item{items[2]},
2024-09-11 07:33:22 +02:00
},
{
name: "unknown kind",
ks: []string{"test"},
expStatus: http.StatusBadRequest,
},
2024-08-28 07:21:02 +02:00
} {
t.Run(tc.name, func(t *testing.T) {
url := fmt.Sprintf("/sync?ts=%s", url.QueryEscape(tc.ts.Format(time.RFC3339)))
2024-09-11 07:33:22 +02:00
if len(tc.ks) > 0 {
url = fmt.Sprintf("%s&ks=%s", url, strings.Join(tc.ks, ","))
}
2024-08-28 07:21:02 +02:00
req, err := http.NewRequest(http.MethodGet, url, nil)
if err != nil {
t.Errorf("exp nil, got %v", err)
}
2024-09-07 12:10:48 +02:00
req.Header.Set("Authorization", fmt.Sprintf("Bearer %s", apiKey))
2024-08-28 07:21:02 +02:00
res := httptest.NewRecorder()
srv.ServeHTTP(res, req)
if res.Result().StatusCode != tc.expStatus {
t.Errorf("exp %v, got %v", tc.expStatus, res.Result().StatusCode)
}
2024-09-11 07:33:22 +02:00
if tc.expStatus != http.StatusOK {
return
}
2024-09-18 07:55:14 +02:00
var actItems []item.Item
2024-08-28 07:21:02 +02:00
actBody, err := io.ReadAll(res.Result().Body)
if err != nil {
t.Errorf("exp nil, got %v", err)
}
defer res.Result().Body.Close()
if err := json.Unmarshal(actBody, &actItems); err != nil {
t.Errorf("exp nil, got %v", err)
}
if len(actItems) != len(tc.expItems) {
t.Errorf("exp %d, got %d", len(tc.expItems), len(actItems))
}
sort.Slice(actItems, func(i, j int) bool {
return actItems[i].ID < actItems[j].ID
})
for i := range actItems {
if actItems[i].ID != tc.expItems[i].ID {
t.Errorf("exp %v, got %v", tc.expItems[i].ID, actItems[i].ID)
}
}
})
}
}
func TestSyncPost(t *testing.T) {
t.Parallel()
2024-09-07 12:10:48 +02:00
apiKey := "test"
2024-08-28 07:21:02 +02:00
for _, tc := range []struct {
name string
reqBody []byte
expStatus int
2024-09-18 07:55:14 +02:00
expItems []item.Item
2024-08-28 07:21:02 +02:00
}{
{
name: "empty",
expStatus: http.StatusBadRequest,
},
{
2024-09-09 07:56:01 +02:00
name: "invalid json",
2024-08-28 07:21:02 +02:00
reqBody: []byte(`{"fail}`),
expStatus: http.StatusBadRequest,
},
2024-09-09 07:56:01 +02:00
{
name: "invalid item",
reqBody: []byte(`[
2024-09-11 07:33:22 +02:00
{"id":"id-1","kind":"event","updated":"2024-09-06T08:00:00Z"},
2024-09-09 07:56:01 +02:00
]`),
expStatus: http.StatusBadRequest,
},
2024-08-28 07:21:02 +02:00
{
name: "normal",
reqBody: []byte(`[
2024-09-11 07:33:22 +02:00
{"id":"id-1","kind":"event","updated":"2024-09-06T08:00:00Z","deleted":false,"body":"item"},
{"id":"id-2","kind":"event","updated":"2024-09-06T08:12:00Z","deleted":false,"body":"item2"}
2024-08-28 07:21:02 +02:00
]`),
expStatus: http.StatusNoContent,
2024-09-18 07:55:14 +02:00
expItems: []item.Item{
{ID: "id-1", Kind: item.KindEvent, Updated: time.Date(2024, 9, 6, 8, 0, 0, 0, time.UTC)},
{ID: "id-2", Kind: item.KindEvent, Updated: time.Date(2024, 9, 6, 12, 0, 0, 0, time.UTC)},
2024-08-28 07:21:02 +02:00
},
},
} {
t.Run(tc.name, func(t *testing.T) {
2024-09-08 10:09:54 +02:00
mem := NewMemory()
srv := NewServer(mem, apiKey, slog.New(slog.NewJSONHandler(os.Stdout, nil)))
2024-08-28 07:21:02 +02:00
req, err := http.NewRequest(http.MethodPost, "/sync", bytes.NewBuffer(tc.reqBody))
if err != nil {
t.Errorf("exp nil, got %v", err)
}
2024-09-07 12:10:48 +02:00
req.Header.Set("Authorization", fmt.Sprintf("Bearer %s", apiKey))
2024-08-28 07:21:02 +02:00
res := httptest.NewRecorder()
srv.ServeHTTP(res, req)
if res.Result().StatusCode != tc.expStatus {
t.Errorf("exp %v, got %v", tc.expStatus, res.Result().StatusCode)
}
2024-09-18 07:55:14 +02:00
actItems, err := mem.Updated([]item.Kind{}, time.Time{})
2024-08-28 07:21:02 +02:00
if err != nil {
t.Errorf("exp nil, git %v", err)
}
if len(actItems) != len(tc.expItems) {
t.Errorf("exp %d, got %d", len(tc.expItems), len(actItems))
}
sort.Slice(actItems, func(i, j int) bool {
return actItems[i].ID < actItems[j].ID
})
for i := range actItems {
if actItems[i].ID != tc.expItems[i].ID {
t.Errorf("exp %v, got %v", tc.expItems[i].ID, actItems[i].ID)
}
}
})
}
}