* Add LastActivity calculation for convoy dashboard (hq-x2xy) Adds internal/activity package with color-coded activity tracking: - Green: <2 minutes (active) - Yellow: 2-5 minutes (stale) - Red: >5 minutes (stuck) Features: - Calculate() function returns Info with formatted age and color class - Helper methods: IsActive(), IsStale(), IsStuck() - Handles edge cases: zero time, future time (clock skew) Tests: 8 test functions with 25 sub-tests covering all thresholds. 🤖 Generated with [Claude Code](https://claude.com/claude-code) Co-Authored-By: Claude Opus 4.5 <noreply@anthropic.com> * Add convoy dashboard HTML template with Last Activity (hq-fq1g) Adds internal/web package with convoy dashboard template: - convoy.html with Last Activity column and color coding - Green (<2min), Yellow (2-5min), Red (>5min) activity indicators - htmx auto-refresh every 30 seconds - Progress bars for convoy completion - Status indicators for open/closed convoys - Empty state when no convoys Also includes internal/activity package (dependency from hq-x2xy): - Calculate() returns Info with formatted age and color class - Helper methods: IsActive(), IsStale(), IsStuck() Tests: 6 template tests + 8 activity tests, all passing. 🤖 Generated with [Claude Code](https://claude.com/claude-code) Co-Authored-By: Claude Opus 4.5 <noreply@anthropic.com> * Add convoy list handler with activity data (hq-3edt) Adds HTTP handler that wires convoy dashboard template to real data: - ConvoyHandler: HTTP handler for GET / rendering convoy dashboard - LiveConvoyFetcher: Fetches convoys from beads with activity data - ConvoyFetcher interface: Enables mocking for tests Features: - Fetches open convoys from town beads - Calculates progress (completed/total) from tracked issues - Gets Last Activity from worker agent beads - Color codes activity: Green (<2min), Yellow (2-5min), Red (>5min) Includes dependencies (not yet merged): - internal/activity: Activity calculation (hq-x2xy) - internal/web/templates: HTML template (hq-fq1g) Tests: 5 handler tests + 6 template tests + 8 activity tests = 19 total 🤖 Generated with [Claude Code](https://claude.com/claude-code) Co-Authored-By: Claude Opus 4.5 <noreply@anthropic.com> * Add 'gt dashboard' CLI command (hq-s1bg) Add dashboard command to start the convoy tracking web server. Usage: gt dashboard [--port=8080] [--open] Features: - --port: Configurable HTTP port (default 8080) - --open: Auto-open browser on start - Cross-platform browser launch (darwin/linux/windows) - Graceful workspace detection 🤖 Generated with [Claude Code](https://claude.com/claude-code) Co-Authored-By: Claude Opus 4.5 <noreply@anthropic.com> --------- Co-authored-by: Claude Opus 4.5 <noreply@anthropic.com>
51 lines
1.2 KiB
Go
51 lines
1.2 KiB
Go
package web
|
|
|
|
import (
|
|
"html/template"
|
|
"net/http"
|
|
)
|
|
|
|
// ConvoyFetcher defines the interface for fetching convoy data.
|
|
type ConvoyFetcher interface {
|
|
FetchConvoys() ([]ConvoyRow, error)
|
|
}
|
|
|
|
// ConvoyHandler handles HTTP requests for the convoy dashboard.
|
|
type ConvoyHandler struct {
|
|
fetcher ConvoyFetcher
|
|
template *template.Template
|
|
}
|
|
|
|
// NewConvoyHandler creates a new convoy handler with the given fetcher.
|
|
func NewConvoyHandler(fetcher ConvoyFetcher) (*ConvoyHandler, error) {
|
|
tmpl, err := LoadTemplates()
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
return &ConvoyHandler{
|
|
fetcher: fetcher,
|
|
template: tmpl,
|
|
}, nil
|
|
}
|
|
|
|
// ServeHTTP handles GET / requests and renders the convoy dashboard.
|
|
func (h *ConvoyHandler) ServeHTTP(w http.ResponseWriter, r *http.Request) {
|
|
convoys, err := h.fetcher.FetchConvoys()
|
|
if err != nil {
|
|
http.Error(w, "Failed to fetch convoys", http.StatusInternalServerError)
|
|
return
|
|
}
|
|
|
|
data := ConvoyData{
|
|
Convoys: convoys,
|
|
}
|
|
|
|
w.Header().Set("Content-Type", "text/html; charset=utf-8")
|
|
|
|
if err := h.template.ExecuteTemplate(w, "convoy.html", data); err != nil {
|
|
http.Error(w, "Failed to render template", http.StatusInternalServerError)
|
|
return
|
|
}
|
|
}
|