diff options
| author | Claude Agent <agent@doot.local> | 2026-03-25 05:17:35 +0000 |
|---|---|---|
| committer | Claude Agent <agent@doot.local> | 2026-03-25 05:17:35 +0000 |
| commit | b58787cfec0bd07abc316c66dc9be6c10b8113c6 (patch) | |
| tree | e1c788094f51bdab0bce8ad38c8d6638c9079bb9 /internal/api | |
| parent | 2db5020047640361066510f29f908ca9fd1c99aa (diff) | |
feat: add Claudomator stories as atom source in Doot tasks tab
Co-Authored-By: Claude Sonnet 4.6 <noreply@anthropic.com>
Diffstat (limited to 'internal/api')
| -rw-r--r-- | internal/api/claudomator.go | 54 | ||||
| -rw-r--r-- | internal/api/claudomator_test.go | 53 | ||||
| -rw-r--r-- | internal/api/interfaces.go | 6 |
3 files changed, 113 insertions, 0 deletions
diff --git a/internal/api/claudomator.go b/internal/api/claudomator.go new file mode 100644 index 0000000..3be4812 --- /dev/null +++ b/internal/api/claudomator.go @@ -0,0 +1,54 @@ +package api + +import ( + "context" + "encoding/json" + "fmt" + "net/http" + "time" + + "task-dashboard/internal/models" +) + +type ClaudomatorHTTPClient struct { + BaseURL string + HTTPClient *http.Client +} + +func NewClaudomatorClient(baseURL string) *ClaudomatorHTTPClient { + return &ClaudomatorHTTPClient{ + BaseURL: baseURL, + HTTPClient: &http.Client{Timeout: 5 * time.Second}, + } +} + +func (c *ClaudomatorHTTPClient) GetActiveStories(ctx context.Context) ([]models.ClaudomatorStory, error) { + req, err := http.NewRequestWithContext(ctx, "GET", c.BaseURL+"/api/stories", nil) + if err != nil { + return nil, err + } + httpClient := c.HTTPClient + if httpClient == nil { + httpClient = http.DefaultClient + } + resp, err := httpClient.Do(req) + if err != nil { + return nil, fmt.Errorf("claudomator: %w", err) + } + defer resp.Body.Close() + if resp.StatusCode != http.StatusOK { + return nil, fmt.Errorf("claudomator: unexpected status %d", resp.StatusCode) + } + var all []models.ClaudomatorStory + if err := json.NewDecoder(resp.Body).Decode(&all); err != nil { + return nil, err + } + active := make([]models.ClaudomatorStory, 0, len(all)) + for _, s := range all { + switch s.Status { + case "IN_PROGRESS", "REVIEW_READY", "NEEDS_FIX": + active = append(active, s) + } + } + return active, nil +} diff --git a/internal/api/claudomator_test.go b/internal/api/claudomator_test.go new file mode 100644 index 0000000..4a2cb00 --- /dev/null +++ b/internal/api/claudomator_test.go @@ -0,0 +1,53 @@ +package api + +import ( + "context" + "encoding/json" + "net/http" + "net/http/httptest" + "testing" + + "task-dashboard/internal/models" +) + +func TestClaudomatorClient_GetActiveStories(t *testing.T) { + stories := []models.ClaudomatorStory{ + {ID: "1", Title: "Story One", Status: "IN_PROGRESS"}, + {ID: "2", Title: "Story Two", Status: "REVIEW_READY"}, + {ID: "3", Title: "Story Three", Status: "DRAFT"}, + } + + server := httptest.NewServer(http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) { + if r.URL.Path != "/api/stories" { + http.NotFound(w, r) + return + } + w.Header().Set("Content-Type", "application/json") + _ = json.NewEncoder(w).Encode(stories) + })) + defer server.Close() + + client := ClaudomatorHTTPClient{BaseURL: server.URL} + result, err := client.GetActiveStories(context.Background()) + if err != nil { + t.Fatalf("unexpected error: %v", err) + } + + if len(result) != 2 { + t.Fatalf("expected 2 active stories, got %d", len(result)) + } + + statuses := map[string]bool{} + for _, s := range result { + statuses[s.Status] = true + } + if !statuses["IN_PROGRESS"] { + t.Error("expected IN_PROGRESS story in results") + } + if !statuses["REVIEW_READY"] { + t.Error("expected REVIEW_READY story in results") + } + if statuses["DRAFT"] { + t.Error("DRAFT story should be excluded from results") + } +} diff --git a/internal/api/interfaces.go b/internal/api/interfaces.go index 99701a1..0bd67b6 100644 --- a/internal/api/interfaces.go +++ b/internal/api/interfaces.go @@ -52,6 +52,11 @@ type GoogleTasksAPI interface { SetTaskListID(id string) } +// ClaudomatorClient defines the interface for Claudomator operations +type ClaudomatorClient interface { + GetActiveStories(ctx context.Context) ([]models.ClaudomatorStory, error) +} + // Ensure concrete types implement interfaces var ( _ TodoistAPI = (*TodoistClient)(nil) @@ -59,4 +64,5 @@ var ( _ PlanToEatAPI = (*PlanToEatClient)(nil) _ GoogleCalendarAPI = (*GoogleCalendarClient)(nil) _ GoogleTasksAPI = (*GoogleTasksClient)(nil) + _ ClaudomatorClient = (*ClaudomatorHTTPClient)(nil) ) |
