2025-07-03 23:12:44 -07:00
|
|
|
package taskcp
|
|
|
|
|
|
|
|
|
|
import (
|
2025-07-05 21:58:34 -07:00
|
|
|
"encoding/json"
|
2025-07-03 23:12:44 -07:00
|
|
|
"fmt"
|
|
|
|
|
"iter"
|
2025-07-05 14:42:26 -07:00
|
|
|
"strings"
|
2025-07-03 23:12:44 -07:00
|
|
|
|
|
|
|
|
"github.com/google/uuid"
|
|
|
|
|
)
|
|
|
|
|
|
|
|
|
|
type Service struct {
|
2025-07-05 21:34:49 -07:00
|
|
|
Projects map[string]*Project
|
|
|
|
|
mcpService string
|
2025-07-03 23:12:44 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
type Project struct {
|
|
|
|
|
ID string
|
|
|
|
|
Tasks map[string]*Task
|
2025-07-05 14:42:26 -07:00
|
|
|
nextTaskID string
|
|
|
|
|
mcpService string
|
2025-07-03 23:12:44 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
type TaskState string
|
|
|
|
|
|
|
|
|
|
const (
|
|
|
|
|
TaskStatePending TaskState = "pending"
|
|
|
|
|
TaskStateRunning TaskState = "running"
|
|
|
|
|
TaskStateSuccess TaskState = "success"
|
|
|
|
|
TaskStateFailure TaskState = "failure"
|
|
|
|
|
)
|
|
|
|
|
|
|
|
|
|
type Task struct {
|
2025-07-05 21:58:34 -07:00
|
|
|
ID string `json:"id"`
|
|
|
|
|
State TaskState `json:"-"`
|
2025-07-10 22:07:51 -07:00
|
|
|
Title string `json:"title"`
|
2025-07-05 21:58:34 -07:00
|
|
|
Instructions string `json:"instructions"`
|
|
|
|
|
Data map[string]any `json:"data,omitempty"`
|
|
|
|
|
Result string `json:"-"`
|
|
|
|
|
Error string `json:"-"`
|
|
|
|
|
Notes string `json:"-"`
|
2025-07-05 14:24:50 -07:00
|
|
|
|
2025-07-05 23:37:13 -07:00
|
|
|
NextTaskID string `json:"-"`
|
|
|
|
|
|
2025-07-10 22:07:51 -07:00
|
|
|
project *Project
|
|
|
|
|
completionCallback func(project *Project, task *Task) error
|
2025-07-03 23:12:44 -07:00
|
|
|
}
|
|
|
|
|
|
2025-07-11 21:14:33 -07:00
|
|
|
type TaskSummary struct {
|
|
|
|
|
Title string `json:"title"`
|
|
|
|
|
State TaskState `json:"state"`
|
|
|
|
|
Error string `json:"error,omitempty"`
|
|
|
|
|
Notes string `json:"notes,omitempty"`
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
type ProjectSummary struct {
|
|
|
|
|
Tasks []TaskSummary `json:"tasks"`
|
|
|
|
|
}
|
|
|
|
|
|
2025-07-05 14:42:26 -07:00
|
|
|
func New(mcpService string) *Service {
|
2025-07-03 23:12:44 -07:00
|
|
|
return &Service{
|
2025-07-05 14:42:26 -07:00
|
|
|
Projects: map[string]*Project{},
|
|
|
|
|
mcpService: mcpService,
|
2025-07-03 23:12:44 -07:00
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
func (s *Service) AddProject() *Project {
|
|
|
|
|
project := &Project{
|
|
|
|
|
ID: uuid.New().String(),
|
|
|
|
|
Tasks: map[string]*Task{},
|
2025-07-05 14:42:26 -07:00
|
|
|
nextTaskID: "",
|
|
|
|
|
mcpService: s.mcpService,
|
2025-07-03 23:12:44 -07:00
|
|
|
}
|
|
|
|
|
s.Projects[project.ID] = project
|
|
|
|
|
return project
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
func (s *Service) GetProject(id string) (*Project, error) {
|
|
|
|
|
project, ok := s.Projects[id]
|
|
|
|
|
if !ok {
|
|
|
|
|
return nil, fmt.Errorf("project not found")
|
|
|
|
|
}
|
2025-07-04 22:52:18 -07:00
|
|
|
|
2025-07-03 23:12:44 -07:00
|
|
|
return project, nil
|
|
|
|
|
}
|
|
|
|
|
|
2025-07-10 22:07:51 -07:00
|
|
|
func (p *Project) InsertTaskBefore(beforeID string, title string, instructions string, completionCallback func(project *Project, task *Task) error) *Task {
|
|
|
|
|
task := p.newTask(title, instructions, completionCallback, beforeID)
|
2025-07-03 23:12:44 -07:00
|
|
|
|
2025-07-05 14:42:26 -07:00
|
|
|
if p.nextTaskID == "" && beforeID == "" {
|
|
|
|
|
p.nextTaskID = task.ID
|
|
|
|
|
} else {
|
|
|
|
|
for t := range p.tasks() {
|
2025-07-05 23:37:13 -07:00
|
|
|
if t.NextTaskID == beforeID {
|
|
|
|
|
t.NextTaskID = task.ID
|
2025-07-05 14:42:26 -07:00
|
|
|
break
|
|
|
|
|
}
|
2025-07-03 23:12:44 -07:00
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return task
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
func (p *Project) GetNextTask() *Task {
|
2025-07-05 14:42:26 -07:00
|
|
|
if p.nextTaskID == "" {
|
2025-07-03 23:15:46 -07:00
|
|
|
return nil
|
|
|
|
|
}
|
|
|
|
|
|
2025-07-05 14:42:26 -07:00
|
|
|
task := p.Tasks[p.nextTaskID]
|
2025-07-04 22:52:18 -07:00
|
|
|
task.State = TaskStateRunning
|
|
|
|
|
return task
|
2025-07-03 23:12:44 -07:00
|
|
|
}
|
|
|
|
|
|
2025-07-05 21:34:49 -07:00
|
|
|
func (p *Project) SetTaskSuccess(id string, result string, notes string) (*Task, error) {
|
2025-07-03 23:12:44 -07:00
|
|
|
task := p.Tasks[id]
|
|
|
|
|
task.State = TaskStateSuccess
|
|
|
|
|
task.Result = result
|
|
|
|
|
task.Notes = notes
|
2025-07-05 21:34:49 -07:00
|
|
|
|
2025-07-10 22:07:51 -07:00
|
|
|
if task.completionCallback != nil {
|
|
|
|
|
err := task.completionCallback(task.project, task)
|
|
|
|
|
if err != nil {
|
|
|
|
|
return nil, err
|
|
|
|
|
}
|
2025-07-05 21:34:49 -07:00
|
|
|
}
|
|
|
|
|
|
2025-07-05 23:37:13 -07:00
|
|
|
p.nextTaskID = task.NextTaskID
|
2025-07-03 23:15:46 -07:00
|
|
|
|
2025-07-05 21:34:49 -07:00
|
|
|
return p.GetNextTask(), nil
|
2025-07-03 23:12:44 -07:00
|
|
|
}
|
|
|
|
|
|
2025-07-05 21:34:49 -07:00
|
|
|
func (p *Project) SetTaskFailure(id string, error string, notes string) (*Task, error) {
|
2025-07-03 23:12:44 -07:00
|
|
|
task := p.Tasks[id]
|
|
|
|
|
task.State = TaskStateFailure
|
|
|
|
|
task.Error = error
|
|
|
|
|
task.Notes = notes
|
2025-07-05 21:34:49 -07:00
|
|
|
|
2025-07-10 22:07:51 -07:00
|
|
|
if task.completionCallback != nil {
|
|
|
|
|
err := task.completionCallback(task.project, task)
|
|
|
|
|
if err != nil {
|
|
|
|
|
return nil, err
|
|
|
|
|
}
|
2025-07-05 21:34:49 -07:00
|
|
|
}
|
|
|
|
|
|
2025-07-05 23:37:13 -07:00
|
|
|
p.nextTaskID = task.NextTaskID
|
2025-07-04 22:52:18 -07:00
|
|
|
|
2025-07-05 21:34:49 -07:00
|
|
|
return p.GetNextTask(), nil
|
2025-07-03 23:12:44 -07:00
|
|
|
}
|
|
|
|
|
|
2025-07-10 22:07:51 -07:00
|
|
|
func (p *Project) newTask(title string, instructions string, completionCallback func(project *Project, task *Task) error, nextTaskID string) *Task {
|
2025-07-03 23:12:44 -07:00
|
|
|
task := &Task{
|
2025-07-04 22:52:18 -07:00
|
|
|
ID: uuid.New().String(),
|
|
|
|
|
State: TaskStatePending,
|
2025-07-05 23:37:13 -07:00
|
|
|
NextTaskID: nextTaskID,
|
2025-07-10 22:07:51 -07:00
|
|
|
Title: title,
|
2025-07-04 22:52:18 -07:00
|
|
|
Instructions: instructions,
|
2025-07-05 22:02:24 -07:00
|
|
|
Data: map[string]any{},
|
2025-07-05 14:24:50 -07:00
|
|
|
completionCallback: completionCallback,
|
2025-07-10 22:07:51 -07:00
|
|
|
project: p,
|
2025-07-03 23:12:44 -07:00
|
|
|
}
|
2025-07-05 21:34:49 -07:00
|
|
|
|
2025-07-05 14:42:26 -07:00
|
|
|
task.Instructions = strings.ReplaceAll(task.Instructions, "{SUCCESS_PROMPT}", task.SuccessPrompt())
|
|
|
|
|
task.Instructions = strings.ReplaceAll(task.Instructions, "{FAILURE_PROMPT}", task.FailurePrompt())
|
2025-07-05 21:34:49 -07:00
|
|
|
|
2025-07-03 23:12:44 -07:00
|
|
|
p.Tasks[task.ID] = task
|
|
|
|
|
return task
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
func (p *Project) tasks() iter.Seq[*Task] {
|
|
|
|
|
return func(yield func(*Task) bool) {
|
2025-07-05 23:37:13 -07:00
|
|
|
for tid := p.nextTaskID; tid != ""; tid = p.Tasks[tid].NextTaskID {
|
2025-07-03 23:12:44 -07:00
|
|
|
t := p.Tasks[tid]
|
|
|
|
|
if !yield(t) {
|
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
2025-07-05 14:42:26 -07:00
|
|
|
|
2025-07-11 21:14:33 -07:00
|
|
|
func (p *Project) Summary() ProjectSummary {
|
|
|
|
|
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,
|
|
|
|
|
})
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
return ProjectSummary{Tasks: tasks}
|
|
|
|
|
}
|
|
|
|
|
|
2025-07-05 14:42:26 -07:00
|
|
|
func (t *Task) SuccessPrompt() string {
|
|
|
|
|
return fmt.Sprintf(`To mark this task as successful, use the MCP tool:
|
2025-07-05 21:34:49 -07:00
|
|
|
%s.set_task_success(project_id="%s", task_id="%s", result="<your result>", notes="<optional notes>")`,
|
2025-07-10 22:07:51 -07:00
|
|
|
t.project.mcpService, t.project.ID, t.ID)
|
2025-07-05 14:42:26 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
func (t *Task) FailurePrompt() string {
|
|
|
|
|
return fmt.Sprintf(`To mark this task as failed, use the MCP tool:
|
2025-07-05 21:34:49 -07:00
|
|
|
%s.set_task_failure(project_id="%s", task_id="%s", error="<error message>", notes="<optional notes>")`,
|
2025-07-10 22:07:51 -07:00
|
|
|
t.project.mcpService, t.project.ID, t.ID)
|
2025-07-05 14:42:26 -07:00
|
|
|
}
|
2025-07-05 21:58:34 -07:00
|
|
|
|
|
|
|
|
func (t *Task) String() string {
|
|
|
|
|
json, err := json.MarshalIndent(t, "", " ")
|
|
|
|
|
if err != nil {
|
|
|
|
|
panic(err)
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return string(json)
|
|
|
|
|
}
|
2025-07-11 21:14:33 -07:00
|
|
|
|
|
|
|
|
func (ps ProjectSummary) String() string {
|
|
|
|
|
json, err := json.MarshalIndent(ps, "", " ")
|
|
|
|
|
if err != nil {
|
|
|
|
|
panic(err)
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return string(json)
|
|
|
|
|
}
|