From b58787cfec0bd07abc316c66dc9be6c10b8113c6 Mon Sep 17 00:00:00 2001 From: Claude Agent Date: Wed, 25 Mar 2026 05:17:35 +0000 Subject: feat: add Claudomator stories as atom source in Doot tasks tab Co-Authored-By: Claude Sonnet 4.6 --- internal/handlers/atoms.go | 15 ++++++++++++- internal/handlers/atoms_test.go | 47 +++++++++++++++++++++++++++++++++++++++++ internal/handlers/handlers.go | 6 ++++-- 3 files changed, 65 insertions(+), 3 deletions(-) create mode 100644 internal/handlers/atoms_test.go (limited to 'internal/handlers') diff --git a/internal/handlers/atoms.go b/internal/handlers/atoms.go index e99c879..6086a5b 100644 --- a/internal/handlers/atoms.go +++ b/internal/handlers/atoms.go @@ -1,15 +1,17 @@ package handlers import ( + "context" "log" "sort" + "task-dashboard/internal/api" "task-dashboard/internal/models" "task-dashboard/internal/store" ) // BuildUnifiedAtomList creates a list of atoms from tasks, cards, and google tasks -func BuildUnifiedAtomList(s *store.Store) ([]models.Atom, []models.Board, error) { +func BuildUnifiedAtomList(s *store.Store, claudomator api.ClaudomatorClient) ([]models.Atom, []models.Board, error) { tasks, err := s.GetTasks() if err != nil { return nil, nil, err @@ -51,6 +53,17 @@ func BuildUnifiedAtomList(s *store.Store) ([]models.Atom, []models.Board, error) } } + if claudomator != nil { + stories, err := claudomator.GetActiveStories(context.Background()) + if err != nil { + log.Printf("Warning: failed to fetch Claudomator stories: %v", err) + } else { + for _, s := range stories { + atoms = append(atoms, models.StoryToAtom(s)) + } + } + } + // Compute UI fields for all atoms for i := range atoms { atoms[i].ComputeUIFields() diff --git a/internal/handlers/atoms_test.go b/internal/handlers/atoms_test.go new file mode 100644 index 0000000..3be82f8 --- /dev/null +++ b/internal/handlers/atoms_test.go @@ -0,0 +1,47 @@ +package handlers + +import ( + "context" + "testing" + + "task-dashboard/internal/models" + "task-dashboard/internal/store" +) + +type mockClaudomatorClient struct { + stories []models.ClaudomatorStory +} + +func (m *mockClaudomatorClient) GetActiveStories(ctx context.Context) ([]models.ClaudomatorStory, error) { + return m.stories, nil +} + +func TestBuildUnifiedAtomList_WithClaudomator(t *testing.T) { + s, err := store.New(":memory:", "../store/migrations") + if err != nil { + t.Fatalf("failed to create in-memory store: %v", err) + } + defer s.Close() + + mock := &mockClaudomatorClient{ + stories: []models.ClaudomatorStory{ + {ID: "s1", Title: "My story", Status: "IN_PROGRESS", ProjectID: "nav"}, + }, + } + + atoms, _, err := BuildUnifiedAtomList(s, mock) + if err != nil { + t.Fatalf("unexpected error: %v", err) + } + + found := false + for _, a := range atoms { + if a.Source == "claudomator" && a.Title == "My story" { + found = true + break + } + } + if !found { + t.Error("expected atom with Source='claudomator' and Title='My story'") + } +} diff --git a/internal/handlers/handlers.go b/internal/handlers/handlers.go index fa97be0..bd14e65 100644 --- a/internal/handlers/handlers.go +++ b/internal/handlers/handlers.go @@ -27,6 +27,7 @@ type Handler struct { planToEatClient api.PlanToEatAPI googleCalendarClient api.GoogleCalendarAPI googleTasksClient api.GoogleTasksAPI + claudomatorClient api.ClaudomatorClient config *config.Config renderer Renderer BuildVersion string @@ -34,7 +35,7 @@ type Handler struct { } // New creates a new Handler instance -func New(s *store.Store, todoist api.TodoistAPI, trello api.TrelloAPI, planToEat api.PlanToEatAPI, googleCalendar api.GoogleCalendarAPI, googleTasks api.GoogleTasksAPI, cfg *config.Config, buildVersion string, webAuthnEnabled bool) *Handler { +func New(s *store.Store, todoist api.TodoistAPI, trello api.TrelloAPI, planToEat api.PlanToEatAPI, googleCalendar api.GoogleCalendarAPI, googleTasks api.GoogleTasksAPI, claudomator api.ClaudomatorClient, cfg *config.Config, buildVersion string, webAuthnEnabled bool) *Handler { // Template functions funcMap := template.FuncMap{ "subtract": func(a, b int) int { return a - b }, @@ -59,6 +60,7 @@ func New(s *store.Store, todoist api.TodoistAPI, trello api.TrelloAPI, planToEat planToEatClient: planToEat, googleCalendarClient: googleCalendar, googleTasksClient: googleTasks, + claudomatorClient: claudomator, config: cfg, renderer: NewTemplateRenderer(tmpl), BuildVersion: buildVersion, @@ -850,7 +852,7 @@ func (h *Handler) HandleUpdateTask(w http.ResponseWriter, r *http.Request) { // HandleTabTasks renders the unified Tasks tab (Todoist + Trello cards with due dates + Bugs + Google Tasks) func (h *Handler) HandleTabTasks(w http.ResponseWriter, r *http.Request) { - atoms, boards, err := BuildUnifiedAtomList(h.store) + atoms, boards, err := BuildUnifiedAtomList(h.store, h.claudomatorClient) if err != nil { JSONError(w, http.StatusInternalServerError, "Failed to fetch tasks", err) return -- cgit v1.2.3