summaryrefslogtreecommitdiff
path: root/internal/handlers
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/handlers
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/handlers')
-rw-r--r--internal/handlers/atoms.go15
-rw-r--r--internal/handlers/atoms_test.go47
-rw-r--r--internal/handlers/handlers.go6
3 files changed, 65 insertions, 3 deletions
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