2024-09-18 07:09:25 +02:00
|
|
|
package service
|
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-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-09 07:56:01 +02:00
|
|
|
items := []Item{
|
2024-09-11 07:33:22 +02:00
|
|
|
{ID: "id-0", Kind: KindEvent, Updated: now.Add(-10 * time.Minute)},
|
|
|
|
{ID: "id-1", Kind: KindEvent, Updated: now.Add(-5 * time.Minute)},
|
|
|
|
{ID: "id-2", Kind: 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-09 07:56:01 +02:00
|
|
|
expItems []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-09 07:56:01 +02:00
|
|
|
expItems: []Item{items[1], items[2]},
|
2024-08-28 07:21:02 +02:00
|
|
|
},
|
2024-09-11 07:33:22 +02:00
|
|
|
{
|
|
|
|
name: "kind",
|
|
|
|
ks: []string{string(KindTask)},
|
|
|
|
expStatus: http.StatusOK,
|
|
|
|
expItems: []Item{items[2]},
|
|
|
|
},
|
|
|
|
{
|
|
|
|
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-09 07:56:01 +02:00
|
|
|
var actItems []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-09 07:56:01 +02:00
|
|
|
expItems []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-09 07:56:01 +02:00
|
|
|
expItems: []Item{
|
2024-09-11 07:33:22 +02:00
|
|
|
{ID: "id-1", Kind: KindEvent, Updated: time.Date(2024, 9, 6, 8, 0, 0, 0, time.UTC)},
|
|
|
|
{ID: "id-2", Kind: 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-11 07:33:22 +02:00
|
|
|
actItems, err := mem.Updated([]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)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
})
|
|
|
|
}
|
|
|
|
}
|