258 lines
7.0 KiB
Go
258 lines
7.0 KiB
Go
package cmd
|
|
|
|
import (
|
|
"context"
|
|
"fmt"
|
|
"net/http"
|
|
"net/http/httptest"
|
|
"strings"
|
|
"sync/atomic"
|
|
"testing"
|
|
"time"
|
|
|
|
"google.golang.org/api/gmail/v1"
|
|
"google.golang.org/api/option"
|
|
)
|
|
|
|
func TestFetchThreadDetails_Empty(t *testing.T) {
|
|
items, err := fetchThreadDetails(context.Background(), nil, nil, nil, false)
|
|
if err != nil {
|
|
t.Fatalf("unexpected error: %v", err)
|
|
}
|
|
if len(items) != 0 {
|
|
t.Errorf("expected empty items, got %d", len(items))
|
|
}
|
|
}
|
|
|
|
func TestFetchThreadDetails_Concurrent(t *testing.T) {
|
|
// Create a mock server that returns thread data
|
|
mux := http.NewServeMux()
|
|
|
|
// Track calls to verify concurrency (atomic for thread safety)
|
|
var callCount atomic.Int32
|
|
|
|
mux.HandleFunc("/gmail/v1/users/me/threads/", func(w http.ResponseWriter, r *http.Request) {
|
|
callCount.Add(1)
|
|
threadID := strings.TrimPrefix(r.URL.Path, "/gmail/v1/users/me/threads/")
|
|
|
|
response := fmt.Sprintf(`{
|
|
"id": "%s",
|
|
"messages": [{
|
|
"id": "msg_%s",
|
|
"labelIds": ["INBOX"],
|
|
"payload": {
|
|
"headers": [
|
|
{"name": "From", "value": "test@example.com"},
|
|
{"name": "Subject", "value": "Test Subject %s"},
|
|
{"name": "Date", "value": "Mon, 01 Jan 2024 10:00:00 +0000"}
|
|
]
|
|
}
|
|
}]
|
|
}`, threadID, threadID, threadID)
|
|
|
|
w.Header().Set("Content-Type", "application/json")
|
|
w.Write([]byte(response))
|
|
})
|
|
|
|
server := httptest.NewServer(mux)
|
|
defer server.Close()
|
|
|
|
// Create Gmail service pointing to mock server
|
|
svc, err := gmail.NewService(context.Background(),
|
|
option.WithEndpoint(server.URL),
|
|
option.WithHTTPClient(http.DefaultClient),
|
|
)
|
|
if err != nil {
|
|
t.Fatalf("failed to create service: %v", err)
|
|
}
|
|
|
|
threads := []*gmail.Thread{
|
|
{Id: "thread1"},
|
|
{Id: "thread2"},
|
|
{Id: "thread3"},
|
|
}
|
|
|
|
idToName := map[string]string{
|
|
"INBOX": "Inbox",
|
|
}
|
|
|
|
items, err := fetchThreadDetails(context.Background(), svc, threads, idToName, false)
|
|
if err != nil {
|
|
t.Fatalf("unexpected error: %v", err)
|
|
}
|
|
|
|
if len(items) != 3 {
|
|
t.Errorf("expected 3 items, got %d", len(items))
|
|
}
|
|
|
|
// Verify all threads were fetched
|
|
if callCount.Load() != 3 {
|
|
t.Errorf("expected 3 API calls, got %d", callCount.Load())
|
|
}
|
|
|
|
// Verify items have correct data (order should be preserved)
|
|
for i, item := range items {
|
|
expectedID := fmt.Sprintf("thread%d", i+1)
|
|
if item.ID != expectedID {
|
|
t.Errorf("item %d: expected ID %s, got %s", i, expectedID, item.ID)
|
|
}
|
|
if item.From != "test@example.com" {
|
|
t.Errorf("item %d: expected From test@example.com, got %s", i, item.From)
|
|
}
|
|
if !strings.Contains(item.Subject, "Test Subject") {
|
|
t.Errorf("item %d: expected Subject to contain 'Test Subject', got %s", i, item.Subject)
|
|
}
|
|
if len(item.Labels) != 1 || item.Labels[0] != "Inbox" {
|
|
t.Errorf("item %d: expected Labels [Inbox], got %v", i, item.Labels)
|
|
}
|
|
}
|
|
}
|
|
|
|
func TestFetchThreadDetails_DateSelection(t *testing.T) {
|
|
mux := http.NewServeMux()
|
|
older := time.Date(2024, 1, 1, 10, 0, 0, 0, time.UTC)
|
|
newer := time.Date(2024, 1, 2, 9, 0, 0, 0, time.UTC)
|
|
|
|
mux.HandleFunc("/gmail/v1/users/me/threads/", func(w http.ResponseWriter, r *http.Request) {
|
|
response := fmt.Sprintf(`{
|
|
"id": "thread1",
|
|
"messages": [{
|
|
"id": "msg_new",
|
|
"internalDate": "%d",
|
|
"payload": {
|
|
"headers": [
|
|
{"name": "From", "value": "new@example.com"},
|
|
{"name": "Subject", "value": "New Subject"},
|
|
{"name": "Date", "value": "%s"}
|
|
]
|
|
}
|
|
}, {
|
|
"id": "msg_old",
|
|
"internalDate": "%d",
|
|
"payload": {
|
|
"headers": [
|
|
{"name": "From", "value": "old@example.com"},
|
|
{"name": "Subject", "value": "Old Subject"},
|
|
{"name": "Date", "value": "%s"}
|
|
]
|
|
}
|
|
}]
|
|
}`, newer.UnixMilli(), newer.Format(time.RFC1123Z), older.UnixMilli(), older.Format(time.RFC1123Z))
|
|
|
|
w.Header().Set("Content-Type", "application/json")
|
|
w.Write([]byte(response))
|
|
})
|
|
|
|
server := httptest.NewServer(mux)
|
|
defer server.Close()
|
|
|
|
svc, err := gmail.NewService(context.Background(),
|
|
option.WithEndpoint(server.URL),
|
|
option.WithHTTPClient(http.DefaultClient),
|
|
)
|
|
if err != nil {
|
|
t.Fatalf("failed to create service: %v", err)
|
|
}
|
|
|
|
threads := []*gmail.Thread{{Id: "thread1"}}
|
|
|
|
itemsNewest, err := fetchThreadDetails(context.Background(), svc, threads, nil, false)
|
|
if err != nil {
|
|
t.Fatalf("unexpected error: %v", err)
|
|
}
|
|
if len(itemsNewest) != 1 {
|
|
t.Fatalf("expected 1 item, got %d", len(itemsNewest))
|
|
}
|
|
expectedNewest := formatGmailDate(newer.Format(time.RFC1123Z))
|
|
if itemsNewest[0].Date != expectedNewest {
|
|
t.Errorf("expected newest date %s, got %s", expectedNewest, itemsNewest[0].Date)
|
|
}
|
|
|
|
itemsOldest, err := fetchThreadDetails(context.Background(), svc, threads, nil, true)
|
|
if err != nil {
|
|
t.Fatalf("unexpected error: %v", err)
|
|
}
|
|
if len(itemsOldest) != 1 {
|
|
t.Fatalf("expected 1 item, got %d", len(itemsOldest))
|
|
}
|
|
expectedOldest := formatGmailDate(older.Format(time.RFC1123Z))
|
|
if itemsOldest[0].Date != expectedOldest {
|
|
t.Errorf("expected oldest date %s, got %s", expectedOldest, itemsOldest[0].Date)
|
|
}
|
|
}
|
|
|
|
func TestFetchThreadDetails_SkipsEmptyIDs(t *testing.T) {
|
|
mux := http.NewServeMux()
|
|
var callCount atomic.Int32
|
|
|
|
mux.HandleFunc("/gmail/v1/users/me/threads/", func(w http.ResponseWriter, r *http.Request) {
|
|
callCount.Add(1)
|
|
threadID := strings.TrimPrefix(r.URL.Path, "/gmail/v1/users/me/threads/")
|
|
response := fmt.Sprintf(`{"id": "%s", "messages": []}`, threadID)
|
|
w.Header().Set("Content-Type", "application/json")
|
|
w.Write([]byte(response))
|
|
})
|
|
|
|
server := httptest.NewServer(mux)
|
|
defer server.Close()
|
|
|
|
svc, _ := gmail.NewService(context.Background(),
|
|
option.WithEndpoint(server.URL),
|
|
option.WithHTTPClient(http.DefaultClient),
|
|
)
|
|
|
|
threads := []*gmail.Thread{
|
|
{Id: ""}, // Should be skipped
|
|
{Id: "thread1"}, // Should be processed
|
|
{Id: ""}, // Should be skipped
|
|
}
|
|
|
|
items, err := fetchThreadDetails(context.Background(), svc, threads, nil, false)
|
|
if err != nil {
|
|
t.Fatalf("unexpected error: %v", err)
|
|
}
|
|
|
|
// Only 1 API call should be made (for thread1)
|
|
if callCount.Load() != 1 {
|
|
t.Errorf("expected 1 API call (skipping empty IDs), got %d", callCount.Load())
|
|
}
|
|
|
|
if len(items) != 1 {
|
|
t.Errorf("expected 1 item, got %d", len(items))
|
|
}
|
|
}
|
|
|
|
func TestFetchThreadDetails_ContextCanceled(t *testing.T) {
|
|
mux := http.NewServeMux()
|
|
|
|
mux.HandleFunc("/gmail/v1/users/me/threads/", func(w http.ResponseWriter, r *http.Request) {
|
|
// Simulate slow response
|
|
select {
|
|
case <-r.Context().Done():
|
|
return
|
|
default:
|
|
response := `{"id": "thread1", "messages": []}`
|
|
w.Header().Set("Content-Type", "application/json")
|
|
w.Write([]byte(response))
|
|
}
|
|
})
|
|
|
|
server := httptest.NewServer(mux)
|
|
defer server.Close()
|
|
|
|
svc, _ := gmail.NewService(context.Background(),
|
|
option.WithEndpoint(server.URL),
|
|
option.WithHTTPClient(http.DefaultClient),
|
|
)
|
|
|
|
ctx, cancel := context.WithCancel(context.Background())
|
|
cancel() // Cancel immediately
|
|
|
|
threads := []*gmail.Thread{{Id: "thread1"}}
|
|
|
|
_, err := fetchThreadDetails(ctx, svc, threads, nil, false)
|
|
// Context was canceled, we may or may not get an error depending on timing.
|
|
// Either nil or context.Canceled is acceptable.
|
|
_ = err
|
|
}
|