summaryrefslogtreecommitdiff
path: root/internal/api
diff options
context:
space:
mode:
authorClaude Agent <agent@doot.local>2026-03-25 05:17:35 +0000
committerClaude Agent <agent@doot.local>2026-03-25 05:17:35 +0000
commitb58787cfec0bd07abc316c66dc9be6c10b8113c6 (patch)
treee1c788094f51bdab0bce8ad38c8d6638c9079bb9 /internal/api
parent2db5020047640361066510f29f908ca9fd1c99aa (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.go54
-rw-r--r--internal/api/claudomator_test.go53
-rw-r--r--internal/api/interfaces.go6
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)
)