2025-07-03 23:12:44 -07:00
|
|
|
package taskcp_test
|
|
|
|
|
|
|
|
|
|
import (
|
2025-07-05 21:34:49 -07:00
|
|
|
"fmt"
|
2025-07-03 23:12:44 -07:00
|
|
|
"testing"
|
|
|
|
|
|
|
|
|
|
"github.com/gopatchy/taskcp"
|
|
|
|
|
"github.com/stretchr/testify/require"
|
|
|
|
|
)
|
|
|
|
|
|
2025-07-05 14:42:26 -07:00
|
|
|
func TestTaskPrompts(t *testing.T) {
|
|
|
|
|
service := taskcp.New("my_service")
|
|
|
|
|
project := service.AddProject()
|
2025-07-10 22:07:51 -07:00
|
|
|
|
2025-07-12 14:52:47 -07:00
|
|
|
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
|
|
|
|
|
})
|
2025-07-10 22:07:51 -07:00
|
|
|
|
2025-07-05 14:42:26 -07:00
|
|
|
successPrompt := task.SuccessPrompt()
|
|
|
|
|
require.Contains(t, successPrompt, "my_service.set_task_success")
|
2025-07-12 14:43:18 -07:00
|
|
|
require.Contains(t, successPrompt, fmt.Sprintf(`project_id=%d`, project.ID))
|
|
|
|
|
require.Contains(t, successPrompt, fmt.Sprintf(`task_id=%d`, task.ID))
|
2025-07-10 22:07:51 -07:00
|
|
|
|
2025-07-05 14:42:26 -07:00
|
|
|
failurePrompt := task.FailurePrompt()
|
|
|
|
|
require.Contains(t, failurePrompt, "my_service.set_task_failure")
|
2025-07-12 14:43:18 -07:00
|
|
|
require.Contains(t, failurePrompt, fmt.Sprintf(`project_id=%d`, project.ID))
|
|
|
|
|
require.Contains(t, failurePrompt, fmt.Sprintf(`task_id=%d`, task.ID))
|
2025-07-05 14:42:26 -07:00
|
|
|
}
|
2025-07-03 23:12:44 -07:00
|
|
|
|
2025-07-05 14:42:26 -07:00
|
|
|
func TestPlaceholderExpansion(t *testing.T) {
|
|
|
|
|
service := taskcp.New("my_service")
|
|
|
|
|
project := service.AddProject()
|
2025-07-10 22:07:51 -07:00
|
|
|
|
2025-07-12 14:52:47 -07:00
|
|
|
task1 := project.InsertTaskBefore(-1).
|
|
|
|
|
WithTitle("Please complete this task.").
|
|
|
|
|
WithInstructions("{SUCCESS_PROMPT}").
|
|
|
|
|
Then(func(project *taskcp.Project, task *taskcp.Task) error {
|
|
|
|
|
return nil
|
|
|
|
|
})
|
2025-07-05 14:42:26 -07:00
|
|
|
require.Contains(t, task1.Instructions, "my_service.set_task_success")
|
|
|
|
|
require.NotContains(t, task1.Instructions, "{SUCCESS_PROMPT}")
|
2025-07-10 22:07:51 -07:00
|
|
|
|
2025-07-12 14:52:47 -07:00
|
|
|
task2 := project.InsertTaskBefore(-1).
|
|
|
|
|
WithTitle("Try this risky operation.").
|
|
|
|
|
WithInstructions("{FAILURE_PROMPT}").
|
|
|
|
|
Then(func(project *taskcp.Project, task *taskcp.Task) error {
|
|
|
|
|
return nil
|
|
|
|
|
})
|
2025-07-05 14:42:26 -07:00
|
|
|
require.Contains(t, task2.Instructions, "my_service.set_task_failure")
|
|
|
|
|
require.NotContains(t, task2.Instructions, "{FAILURE_PROMPT}")
|
2025-07-03 23:12:44 -07:00
|
|
|
}
|
2025-07-05 14:42:26 -07:00
|
|
|
|
|
|
|
|
func TestTaskFlow(t *testing.T) {
|
|
|
|
|
service := taskcp.New("test_service")
|
|
|
|
|
project := service.AddProject()
|
2025-07-10 22:07:51 -07:00
|
|
|
|
2025-07-12 14:43:18 -07:00
|
|
|
var completed []int
|
2025-07-10 22:07:51 -07:00
|
|
|
|
2025-07-12 14:52:47 -07:00
|
|
|
task1 := project.InsertTaskBefore(-1).
|
|
|
|
|
WithTitle("First task").
|
|
|
|
|
Then(func(project *taskcp.Project, task *taskcp.Task) error {
|
|
|
|
|
completed = append(completed, task.ID)
|
|
|
|
|
return nil
|
|
|
|
|
})
|
2025-07-10 22:07:51 -07:00
|
|
|
|
2025-07-12 14:52:47 -07:00
|
|
|
task2 := project.InsertTaskBefore(-1).
|
|
|
|
|
WithTitle("Second task").
|
|
|
|
|
Then(func(project *taskcp.Project, task *taskcp.Task) error {
|
|
|
|
|
completed = append(completed, task.ID)
|
|
|
|
|
return nil
|
|
|
|
|
})
|
2025-07-10 22:07:51 -07:00
|
|
|
|
2025-07-05 14:42:26 -07:00
|
|
|
current := project.GetNextTask()
|
|
|
|
|
require.NotNil(t, current)
|
|
|
|
|
require.Equal(t, task1.ID, current.ID)
|
2025-07-10 22:07:51 -07:00
|
|
|
|
2025-07-05 21:34:49 -07:00
|
|
|
next, err := project.SetTaskSuccess(current.ID, "Task 1 done", "")
|
|
|
|
|
require.NoError(t, err)
|
2025-07-05 14:42:26 -07:00
|
|
|
require.NotNil(t, next)
|
|
|
|
|
require.Equal(t, task2.ID, next.ID)
|
|
|
|
|
require.Equal(t, taskcp.TaskStateRunning, next.State)
|
2025-07-10 22:07:51 -07:00
|
|
|
|
2025-07-05 21:34:49 -07:00
|
|
|
next2, err := project.SetTaskFailure(next.ID, "Task 2 failed", "Error details")
|
|
|
|
|
require.NoError(t, err)
|
2025-07-05 14:42:26 -07:00
|
|
|
require.Nil(t, next2)
|
2025-07-10 22:07:51 -07:00
|
|
|
|
2025-07-12 14:43:18 -07:00
|
|
|
require.Equal(t, []int{task1.ID, task2.ID}, completed)
|
2025-07-05 14:42:26 -07:00
|
|
|
require.Equal(t, taskcp.TaskStateSuccess, project.Tasks[task1.ID].State)
|
|
|
|
|
require.Equal(t, taskcp.TaskStateFailure, project.Tasks[task2.ID].State)
|
2025-07-05 21:34:49 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
func TestCallbackError(t *testing.T) {
|
|
|
|
|
service := taskcp.New("test_service")
|
|
|
|
|
project := service.AddProject()
|
2025-07-10 22:07:51 -07:00
|
|
|
|
2025-07-05 21:34:49 -07:00
|
|
|
expectedErr := fmt.Errorf("callback error")
|
2025-07-10 22:07:51 -07:00
|
|
|
|
2025-07-12 14:52:47 -07:00
|
|
|
task := project.InsertTaskBefore(-1).
|
|
|
|
|
WithTitle("Task with error callback").
|
|
|
|
|
WithInstructions("This is a test task.").
|
|
|
|
|
WithData("key", "value").
|
|
|
|
|
Then(func(project *taskcp.Project, task *taskcp.Task) error {
|
|
|
|
|
return expectedErr
|
|
|
|
|
})
|
2025-07-10 22:07:51 -07:00
|
|
|
|
2025-07-05 21:34:49 -07:00
|
|
|
current := project.GetNextTask()
|
|
|
|
|
require.NotNil(t, current)
|
|
|
|
|
require.Equal(t, task.ID, current.ID)
|
2025-07-10 22:07:51 -07:00
|
|
|
|
2025-07-05 21:34:49 -07:00
|
|
|
_, err := project.SetTaskSuccess(current.ID, "Result", "")
|
|
|
|
|
require.Error(t, err)
|
|
|
|
|
require.Equal(t, expectedErr, err)
|
2025-07-10 22:07:51 -07:00
|
|
|
|
2025-07-05 21:34:49 -07:00
|
|
|
_, err = project.SetTaskFailure(current.ID, "Task failed", "")
|
|
|
|
|
require.Error(t, err)
|
|
|
|
|
require.Equal(t, expectedErr, err)
|
2025-07-10 22:07:51 -07:00
|
|
|
}
|