rdev/cmd/sdlc/cmd_query.go
jordan 425ef0f806 feat: add SDLC orchestration - library, CLI, and API integration
Implements deterministic feature lifecycle management for agent-driven
development. Agents use the CLI in pods; operators control via REST API.

Library (internal/sdlc/):
- Feature lifecycle with 10 phases (draft → released)
- Classifier engine with priority-ordered rules
- Artifact tracking with approval workflow
- Task management within features
- YAML-based state persistence

CLI (cmd/sdlc/):
- init, state, next, feature, artifact, task, query commands
- --json flag for machine-readable output
- Runs inside project pods

API (21 endpoints under /projects/{id}/sdlc/):
- State: GET /state, GET /next
- Features: CRUD + transition/block/unblock
- Artifacts: approve/reject per type
- Tasks: add/start/complete/block
- Queries: blocked/ready/needs-approval

Architecture:
- Port: SDLCExecutor interface (internal/port/)
- Adapter: kubectl exec into pods (internal/adapter/kubernetes/)
- Service: pod resolution + logging (internal/service/)
- Handlers: 5 files under 500-line limit (internal/handlers/)

Also includes template upgrades (chassis framework, UI components,
OpenAPI helpers, backend/frontend guides) and component improvements.

Co-Authored-By: Claude Opus 4.5 <noreply@anthropic.com>
2026-02-02 09:57:05 -07:00

203 lines
3.8 KiB
Go

package main
import (
"fmt"
"github.com/orchard9/rdev/internal/sdlc"
"github.com/spf13/cobra"
)
var queryCmd = &cobra.Command{
Use: "query",
Short: "Query SDLC state",
}
var queryBlockedCmd = &cobra.Command{
Use: "blocked",
Short: "List all blocked items",
RunE: func(_ *cobra.Command, _ []string) error {
root := mustResolveRoot()
features, err := sdlc.ListFeatures(root)
if err != nil {
return err
}
type blockedInfo struct {
Slug string `json:"slug"`
Phase string `json:"phase"`
Blockers []string `json:"blockers"`
}
var blocked []blockedInfo
for _, f := range features {
if f.IsBlocked() {
blocked = append(blocked, blockedInfo{
Slug: f.Slug,
Phase: string(f.Phase),
Blockers: f.Blockers,
})
}
}
if jsonOutput {
printJSON(blocked)
return nil
}
if len(blocked) == 0 {
fmt.Println("No blocked items.")
return nil
}
fmt.Println("Blocked Items:")
for _, b := range blocked {
fmt.Printf(" %s [%s]:\n", b.Slug, b.Phase)
for _, reason := range b.Blockers {
fmt.Printf(" - %s\n", reason)
}
}
return nil
},
}
var queryReadyCmd = &cobra.Command{
Use: "ready",
Short: "List items ready for work",
RunE: func(_ *cobra.Command, _ []string) error {
root := mustResolveRoot()
state, err := sdlc.LoadState(root)
if err != nil {
return err
}
cfg, err := sdlc.LoadConfig(root)
if err != nil {
return err
}
features, err := sdlc.ListFeatures(root)
if err != nil {
return err
}
classifier := sdlc.NewClassifier()
type readyInfo struct {
Slug string `json:"slug"`
Phase string `json:"phase"`
Action string `json:"action"`
}
var ready []readyInfo
for _, f := range features {
if f.IsBlocked() {
continue
}
cl := classifier.Classify(&sdlc.EvalContext{
State: state,
Feature: f,
Config: cfg,
Root: root,
})
if cl.Action != sdlc.ActionIdle && cl.Action != sdlc.ActionBlocked && cl.Action != sdlc.ActionAwaitApproval {
ready = append(ready, readyInfo{
Slug: f.Slug,
Phase: string(f.Phase),
Action: string(cl.Action),
})
}
}
if jsonOutput {
printJSON(ready)
return nil
}
if len(ready) == 0 {
fmt.Println("No items ready for work.")
return nil
}
fmt.Println("Ready for Work:")
for _, r := range ready {
fmt.Printf(" %-20s [%-15s] -> %s\n", r.Slug, r.Phase, r.Action)
}
return nil
},
}
var queryNeedsApprovalCmd = &cobra.Command{
Use: "needs-approval",
Short: "List items awaiting approval",
RunE: func(_ *cobra.Command, _ []string) error {
root := mustResolveRoot()
state, err := sdlc.LoadState(root)
if err != nil {
return err
}
cfg, err := sdlc.LoadConfig(root)
if err != nil {
return err
}
features, err := sdlc.ListFeatures(root)
if err != nil {
return err
}
classifier := sdlc.NewClassifier()
type approvalInfo struct {
Slug string `json:"slug"`
Phase string `json:"phase"`
Message string `json:"message"`
}
var pending []approvalInfo
for _, f := range features {
cl := classifier.Classify(&sdlc.EvalContext{
State: state,
Feature: f,
Config: cfg,
Root: root,
})
if cl.Action == sdlc.ActionAwaitApproval {
pending = append(pending, approvalInfo{
Slug: f.Slug,
Phase: string(f.Phase),
Message: cl.Message,
})
}
}
if jsonOutput {
printJSON(pending)
return nil
}
if len(pending) == 0 {
fmt.Println("No items awaiting approval.")
return nil
}
fmt.Println("Awaiting Approval:")
for _, p := range pending {
fmt.Printf(" %-20s [%-15s] %s\n", p.Slug, p.Phase, p.Message)
}
return nil
},
}
func init() {
queryCmd.AddCommand(
queryBlockedCmd,
queryReadyCmd,
queryNeedsApprovalCmd,
)
rootCmd.AddCommand(queryCmd)
}