Simplified API

This commit is contained in:
Ian Gulliver
2025-07-12 15:32:03 -07:00
parent 7d50eda620
commit a96b350b28
3 changed files with 175 additions and 179 deletions

14
mcp.go
View File

@@ -113,7 +113,12 @@ func handleSetTaskSuccess(s *Service, ctx context.Context, args setTaskSuccessAr
return nil, fmt.Errorf("failed to get project: %w", err) return nil, fmt.Errorf("failed to get project: %w", err)
} }
nextTask, err := project.SetTaskSuccess(args.TaskID, args.Result, args.Notes) task, err := project.GetRunningTask(args.TaskID)
if err != nil {
return nil, fmt.Errorf("failed to get task: %w", err)
}
nextTask, err := task.SetSuccess(args.Result, args.Notes)
if err != nil { if err != nil {
return nil, fmt.Errorf("completion callback error: %w", err) return nil, fmt.Errorf("completion callback error: %w", err)
} }
@@ -131,7 +136,12 @@ func handleSetTaskFailure(s *Service, ctx context.Context, args setTaskFailureAr
return nil, fmt.Errorf("failed to get project: %w", err) return nil, fmt.Errorf("failed to get project: %w", err)
} }
nextTask, err := project.SetTaskFailure(args.TaskID, args.Error, args.Notes) task, err := project.GetRunningTask(args.TaskID)
if err != nil {
return nil, fmt.Errorf("failed to get task: %w", err)
}
nextTask, err := task.SetFailure(args.Error, args.Notes)
if err != nil { if err != nil {
return nil, fmt.Errorf("completion callback error: %w", err) return nil, fmt.Errorf("completion callback error: %w", err)
} }

209
taskcp.go
View File

@@ -3,7 +3,6 @@ package taskcp
import ( import (
"encoding/json" "encoding/json"
"fmt" "fmt"
"iter"
"strings" "strings"
) )
@@ -14,23 +13,16 @@ type Service struct {
type Project struct { type Project struct {
ID int ID int
Tasks []*Task PendingTasks []*Task
nextTaskID int RunningTasks []*Task
SuccessTasks []*Task
FailureTasks []*Task
mcpService string mcpService string
nextTaskID int
} }
type TaskState string
const (
TaskStatePending TaskState = "pending"
TaskStateRunning TaskState = "running"
TaskStateSuccess TaskState = "success"
TaskStateFailure TaskState = "failure"
)
type Task struct { type Task struct {
ID int `json:"id"` ID int `json:"id"`
State TaskState `json:"-"`
Title string `json:"title"` Title string `json:"title"`
Instructions string `json:"instructions"` Instructions string `json:"instructions"`
Data map[string]any `json:"data,omitempty"` Data map[string]any `json:"data,omitempty"`
@@ -38,21 +30,21 @@ type Task struct {
Error string `json:"-"` Error string `json:"-"`
Notes string `json:"-"` Notes string `json:"-"`
NextTaskID int `json:"-"` completionCallback func(task *Task) error
project *Project project *Project
completionCallback func(project *Project, task *Task) error
} }
type TaskSummary struct { type TaskSummary struct {
Title string `json:"title"` Title string `json:"title"`
State TaskState `json:"state"`
Error string `json:"error,omitempty"` Error string `json:"error,omitempty"`
Notes string `json:"notes,omitempty"` Notes string `json:"notes,omitempty"`
} }
type ProjectSummary struct { type ProjectSummary struct {
Tasks []TaskSummary `json:"tasks"` PendingTasks []*TaskSummary `json:"pending_tasks"`
RunningTasks []*TaskSummary `json:"running_tasks"`
SuccessTasks []*TaskSummary `json:"success_tasks"`
FailureTasks []*TaskSummary `json:"failure_tasks"`
} }
func New(mcpService string) *Service { func New(mcpService string) *Service {
@@ -64,9 +56,12 @@ func New(mcpService string) *Service {
func (s *Service) AddProject() *Project { func (s *Service) AddProject() *Project {
project := &Project{ project := &Project{
ID: len(s.projects), ID: len(s.projects),
Tasks: []*Task{}, PendingTasks: []*Task{},
nextTaskID: -1, RunningTasks: []*Task{},
SuccessTasks: []*Task{},
FailureTasks: []*Task{},
mcpService: s.mcpService, mcpService: s.mcpService,
nextTaskID: 0,
} }
s.projects = append(s.projects, project) s.projects = append(s.projects, project)
return project return project
@@ -80,106 +75,75 @@ func (s *Service) GetProject(id int) (*Project, error) {
return s.projects[id], nil return s.projects[id], nil
} }
func (p *Project) InsertTaskBefore(beforeID int) *Task { func (p *Project) AddNextTask() *Task {
newTask := p.newTask(beforeID) t := p.newTask()
p.PendingTasks = append([]*Task{t}, p.PendingTasks...)
if p.nextTaskID == -1 && beforeID == -1 { return t
p.nextTaskID = newTask.ID
} else {
for t := range p.tasks() {
if t.NextTaskID == beforeID {
t.NextTaskID = newTask.ID
break
}
}
} }
return newTask func (p *Project) AddLastTask() *Task {
t := p.newTask()
p.PendingTasks = append(p.PendingTasks, t)
return t
} }
func (p *Project) GetNextTask() *Task { func (p *Project) newTask() *Task {
if p.nextTaskID == -1 {
return nil
}
task := p.Tasks[p.nextTaskID]
task.State = TaskStateRunning
return task
}
func (p *Project) SetTaskSuccess(id int, result string, notes string) (*Task, error) {
task := p.Tasks[id]
task.State = TaskStateSuccess
task.Result = result
task.Notes = notes
if task.completionCallback != nil {
err := task.completionCallback(task.project, task)
if err != nil {
return nil, err
}
}
p.nextTaskID = task.NextTaskID
return p.GetNextTask(), nil
}
func (p *Project) SetTaskFailure(id int, error string, notes string) (*Task, error) {
task := p.Tasks[id]
task.State = TaskStateFailure
task.Error = error
task.Notes = notes
if task.completionCallback != nil {
err := task.completionCallback(task.project, task)
if err != nil {
return nil, err
}
}
p.nextTaskID = task.NextTaskID
return p.GetNextTask(), nil
}
func (p *Project) newTask(nextTaskID int) *Task {
task := &Task{ task := &Task{
ID: len(p.Tasks), ID: p.nextTaskID,
State: TaskStatePending,
NextTaskID: nextTaskID,
Data: map[string]any{}, Data: map[string]any{},
project: p, project: p,
} }
p.Tasks = append(p.Tasks, task) p.nextTaskID++
return task return task
} }
func (p *Project) tasks() iter.Seq[*Task] { func (p *Project) PopNextTask() (*Task, error) {
return func(yield func(*Task) bool) { if len(p.PendingTasks) == 0 {
for tid := p.nextTaskID; tid != -1; tid = p.Tasks[tid].NextTaskID { return nil, nil
t := p.Tasks[tid]
if !yield(t) {
return
} }
task := p.PendingTasks[0]
p.PendingTasks = p.PendingTasks[1:]
return task, nil
} }
func (p *Project) GetRunningTask(id int) (*Task, error) {
for _, task := range p.RunningTasks {
if task.ID == id {
return task, nil
} }
} }
func (p *Project) Summary() ProjectSummary { return nil, fmt.Errorf("task not found: %d", id)
var tasks []TaskSummary
for _, task := range p.Tasks {
if task.State != TaskStatePending {
tasks = append(tasks, TaskSummary{
Title: task.Title,
State: task.State,
Error: task.Error,
Notes: task.Notes,
})
} }
func (p *Project) Summary() *ProjectSummary {
s := &ProjectSummary{
PendingTasks: []*TaskSummary{},
RunningTasks: []*TaskSummary{},
SuccessTasks: []*TaskSummary{},
FailureTasks: []*TaskSummary{},
} }
return ProjectSummary{Tasks: tasks}
for _, task := range p.PendingTasks {
s.PendingTasks = append(s.PendingTasks, task.AsSummary())
}
for _, task := range p.RunningTasks {
s.RunningTasks = append(s.RunningTasks, task.AsSummary())
}
for _, task := range p.SuccessTasks {
s.SuccessTasks = append(s.SuccessTasks, task.AsSummary())
}
for _, task := range p.FailureTasks {
s.FailureTasks = append(s.FailureTasks, task.AsSummary())
}
return s
} }
func (t *Task) WithTitle(title string) *Task { func (t *Task) WithTitle(title string) *Task {
@@ -199,9 +163,40 @@ func (t *Task) WithData(key string, value any) *Task {
return t return t
} }
func (t *Task) Then(completionCallback func(project *Project, task *Task) error) *Task { func (t *Task) Then(completionCallback func(task *Task) error) {
t.completionCallback = completionCallback t.completionCallback = completionCallback
return t }
func (t *Task) SetSuccess(result string, notes string) (*Task, error) {
t.Result = result
t.Notes = notes
t.project.SuccessTasks = append(t.project.SuccessTasks, t)
if t.completionCallback != nil {
err := t.completionCallback(t)
if err != nil {
return nil, err
}
}
return t.project.PopNextTask()
}
func (t *Task) SetFailure(error string, notes string) (*Task, error) {
t.Error = error
t.Notes = notes
t.project.FailureTasks = append(t.project.FailureTasks, t)
if t.completionCallback != nil {
err := t.completionCallback(t)
if err != nil {
return nil, err
}
}
return t.project.PopNextTask()
} }
func (t *Task) SuccessPrompt() string { func (t *Task) SuccessPrompt() string {
@@ -225,6 +220,14 @@ func (t *Task) String() string {
return string(json) return string(json)
} }
func (t *Task) AsSummary() *TaskSummary {
return &TaskSummary{
Title: t.Title,
Error: t.Error,
Notes: t.Notes,
}
}
func (ps ProjectSummary) String() string { func (ps ProjectSummary) String() string {
json, err := json.MarshalIndent(ps, "", " ") json, err := json.MarshalIndent(ps, "", " ")
if err != nil { if err != nil {

View File

@@ -8,113 +8,96 @@ import (
"github.com/stretchr/testify/require" "github.com/stretchr/testify/require"
) )
func TestTaskPrompts(t *testing.T) {
service := taskcp.New("my_service")
project := service.AddProject()
task := project.InsertTaskBefore(-1).
WithTitle("Write unit tests").
WithInstructions("This is a test task.").
Then(func(project *taskcp.Project, task *taskcp.Task) error {
return nil
})
successPrompt := task.SuccessPrompt()
require.Contains(t, successPrompt, "my_service.set_task_success")
require.Contains(t, successPrompt, fmt.Sprintf(`project_id=%d`, project.ID))
require.Contains(t, successPrompt, fmt.Sprintf(`task_id=%d`, task.ID))
failurePrompt := task.FailurePrompt()
require.Contains(t, failurePrompt, "my_service.set_task_failure")
require.Contains(t, failurePrompt, fmt.Sprintf(`project_id=%d`, project.ID))
require.Contains(t, failurePrompt, fmt.Sprintf(`task_id=%d`, task.ID))
}
func TestPlaceholderExpansion(t *testing.T) { func TestPlaceholderExpansion(t *testing.T) {
service := taskcp.New("my_service") service := taskcp.New("my_service")
project := service.AddProject() p := service.AddProject()
task1 := project.InsertTaskBefore(-1). p.AddLastTask().
WithTitle("Please complete this task."). WithTitle("Please complete this task.").
WithInstructions("{SUCCESS_PROMPT}"). WithInstructions("{SUCCESS_PROMPT}").
Then(func(project *taskcp.Project, task *taskcp.Task) error { Then(func(task *taskcp.Task) error {
return nil return nil
}) })
require.Contains(t, task1.Instructions, "my_service.set_task_success")
require.NotContains(t, task1.Instructions, "{SUCCESS_PROMPT}")
task2 := project.InsertTaskBefore(-1). task, err := p.PopNextTask()
require.NoError(t, err)
require.NotNil(t, task)
require.Contains(t, task.Instructions, "my_service.set_task_success")
require.NotContains(t, task.Instructions, "{SUCCESS_PROMPT}")
p.AddLastTask().
WithTitle("Try this risky operation."). WithTitle("Try this risky operation.").
WithInstructions("{FAILURE_PROMPT}"). WithInstructions("{FAILURE_PROMPT}").
Then(func(project *taskcp.Project, task *taskcp.Task) error { Then(func(task *taskcp.Task) error {
return nil return nil
}) })
require.Contains(t, task2.Instructions, "my_service.set_task_failure")
require.NotContains(t, task2.Instructions, "{FAILURE_PROMPT}") task, err = p.PopNextTask()
require.NoError(t, err)
require.NotNil(t, task)
require.Contains(t, task.Instructions, "my_service.set_task_failure")
require.NotContains(t, task.Instructions, "{FAILURE_PROMPT}")
} }
func TestTaskFlow(t *testing.T) { func TestTaskFlow(t *testing.T) {
service := taskcp.New("test_service") service := taskcp.New("test_service")
project := service.AddProject() p := service.AddProject()
var completed []int var completed []string
task1 := project.InsertTaskBefore(-1). p.AddLastTask().
WithTitle("First task"). WithTitle("First task").
Then(func(project *taskcp.Project, task *taskcp.Task) error { Then(func(task *taskcp.Task) error {
completed = append(completed, task.ID) completed = append(completed, task.Title)
return nil return nil
}) })
task2 := project.InsertTaskBefore(-1). p.AddLastTask().
WithTitle("Second task"). WithTitle("Second task").
Then(func(project *taskcp.Project, task *taskcp.Task) error { Then(func(task *taskcp.Task) error {
completed = append(completed, task.ID) completed = append(completed, task.Title)
return nil return nil
}) })
current := project.GetNextTask() task1, err := p.PopNextTask()
require.NotNil(t, current)
require.Equal(t, task1.ID, current.ID)
next, err := project.SetTaskSuccess(current.ID, "Task 1 done", "")
require.NoError(t, err) require.NoError(t, err)
require.NotNil(t, next) require.NotNil(t, task1)
require.Equal(t, task2.ID, next.ID) require.Equal(t, "First task", task1.Title)
require.Equal(t, taskcp.TaskStateRunning, next.State)
next2, err := project.SetTaskFailure(next.ID, "Task 2 failed", "Error details") task2, err := task1.SetSuccess("Task 1 done", "")
require.NoError(t, err) require.NoError(t, err)
require.Nil(t, next2) require.NotNil(t, task2)
require.Equal(t, "Second task", task2.Title)
require.Equal(t, []int{task1.ID, task2.ID}, completed) task3, err := task2.SetFailure("Task 2 failed", "Error details")
require.Equal(t, taskcp.TaskStateSuccess, project.Tasks[task1.ID].State) require.NoError(t, err)
require.Equal(t, taskcp.TaskStateFailure, project.Tasks[task2.ID].State) require.Nil(t, task3)
require.Equal(t, []string{"First task", "Second task"}, completed)
require.Equal(t, "Task 1 done", task1.Result)
require.Equal(t, "Task 2 failed", task2.Error)
} }
func TestCallbackError(t *testing.T) { func TestCallbackError(t *testing.T) {
service := taskcp.New("test_service") service := taskcp.New("test_service")
project := service.AddProject() p := service.AddProject()
expectedErr := fmt.Errorf("callback error") expectedErr := fmt.Errorf("callback error")
task := project.InsertTaskBefore(-1). p.AddLastTask().
WithTitle("Task with error callback"). WithTitle("Task with error callback").
WithInstructions("This is a test task."). WithInstructions("This is a test task.").
WithData("key", "value"). WithData("key", "value").
Then(func(project *taskcp.Project, task *taskcp.Task) error { Then(func(task *taskcp.Task) error {
return expectedErr return expectedErr
}) })
current := project.GetNextTask() task, err := p.PopNextTask()
require.NotNil(t, current) require.NoError(t, err)
require.Equal(t, task.ID, current.ID) require.NotNil(t, task)
require.Equal(t, "Task with error callback", task.Title)
_, err := project.SetTaskSuccess(current.ID, "Result", "") _, err = task.SetSuccess("Result", "")
require.Error(t, err)
require.Equal(t, expectedErr, err)
_, err = project.SetTaskFailure(current.ID, "Task failed", "")
require.Error(t, err) require.Error(t, err)
require.Equal(t, expectedErr, err) require.Equal(t, expectedErr, err)
} }