Move Go module path to production Forgejo instance. Updates all imports, go.mod, go.sum, docs, and CI configs. Co-Authored-By: Claude Opus 4.6 <noreply@anthropic.com>
176 lines
5.1 KiB
Go
176 lines
5.1 KiB
Go
package process
|
|
|
|
import (
|
|
"context"
|
|
"testing"
|
|
|
|
"forge.lthn.ai/core/cli/pkg/framework"
|
|
"github.com/stretchr/testify/assert"
|
|
"github.com/stretchr/testify/require"
|
|
)
|
|
|
|
func newTestRunner(t *testing.T) *Runner {
|
|
t.Helper()
|
|
|
|
core, err := framework.New(
|
|
framework.WithName("process", NewService(Options{})),
|
|
)
|
|
require.NoError(t, err)
|
|
|
|
svc, err := framework.ServiceFor[*Service](core, "process")
|
|
require.NoError(t, err)
|
|
|
|
return NewRunner(svc)
|
|
}
|
|
|
|
func TestRunner_RunSequential(t *testing.T) {
|
|
t.Run("all pass", func(t *testing.T) {
|
|
runner := newTestRunner(t)
|
|
|
|
result, err := runner.RunSequential(context.Background(), []RunSpec{
|
|
{Name: "first", Command: "echo", Args: []string{"1"}},
|
|
{Name: "second", Command: "echo", Args: []string{"2"}},
|
|
{Name: "third", Command: "echo", Args: []string{"3"}},
|
|
})
|
|
require.NoError(t, err)
|
|
|
|
assert.True(t, result.Success())
|
|
assert.Equal(t, 3, result.Passed)
|
|
assert.Equal(t, 0, result.Failed)
|
|
assert.Equal(t, 0, result.Skipped)
|
|
})
|
|
|
|
t.Run("stops on failure", func(t *testing.T) {
|
|
runner := newTestRunner(t)
|
|
|
|
result, err := runner.RunSequential(context.Background(), []RunSpec{
|
|
{Name: "first", Command: "echo", Args: []string{"1"}},
|
|
{Name: "fails", Command: "sh", Args: []string{"-c", "exit 1"}},
|
|
{Name: "third", Command: "echo", Args: []string{"3"}},
|
|
})
|
|
require.NoError(t, err)
|
|
|
|
assert.False(t, result.Success())
|
|
assert.Equal(t, 1, result.Passed)
|
|
assert.Equal(t, 1, result.Failed)
|
|
assert.Equal(t, 1, result.Skipped)
|
|
})
|
|
|
|
t.Run("allow failure continues", func(t *testing.T) {
|
|
runner := newTestRunner(t)
|
|
|
|
result, err := runner.RunSequential(context.Background(), []RunSpec{
|
|
{Name: "first", Command: "echo", Args: []string{"1"}},
|
|
{Name: "fails", Command: "sh", Args: []string{"-c", "exit 1"}, AllowFailure: true},
|
|
{Name: "third", Command: "echo", Args: []string{"3"}},
|
|
})
|
|
require.NoError(t, err)
|
|
|
|
// Still counts as failed but pipeline continues
|
|
assert.Equal(t, 2, result.Passed)
|
|
assert.Equal(t, 1, result.Failed)
|
|
assert.Equal(t, 0, result.Skipped)
|
|
})
|
|
}
|
|
|
|
func TestRunner_RunParallel(t *testing.T) {
|
|
t.Run("all run concurrently", func(t *testing.T) {
|
|
runner := newTestRunner(t)
|
|
|
|
result, err := runner.RunParallel(context.Background(), []RunSpec{
|
|
{Name: "first", Command: "echo", Args: []string{"1"}},
|
|
{Name: "second", Command: "echo", Args: []string{"2"}},
|
|
{Name: "third", Command: "echo", Args: []string{"3"}},
|
|
})
|
|
require.NoError(t, err)
|
|
|
|
assert.True(t, result.Success())
|
|
assert.Equal(t, 3, result.Passed)
|
|
assert.Len(t, result.Results, 3)
|
|
})
|
|
|
|
t.Run("failure doesnt stop others", func(t *testing.T) {
|
|
runner := newTestRunner(t)
|
|
|
|
result, err := runner.RunParallel(context.Background(), []RunSpec{
|
|
{Name: "first", Command: "echo", Args: []string{"1"}},
|
|
{Name: "fails", Command: "sh", Args: []string{"-c", "exit 1"}},
|
|
{Name: "third", Command: "echo", Args: []string{"3"}},
|
|
})
|
|
require.NoError(t, err)
|
|
|
|
assert.False(t, result.Success())
|
|
assert.Equal(t, 2, result.Passed)
|
|
assert.Equal(t, 1, result.Failed)
|
|
})
|
|
}
|
|
|
|
func TestRunner_RunAll(t *testing.T) {
|
|
t.Run("respects dependencies", func(t *testing.T) {
|
|
runner := newTestRunner(t)
|
|
|
|
result, err := runner.RunAll(context.Background(), []RunSpec{
|
|
{Name: "third", Command: "echo", Args: []string{"3"}, After: []string{"second"}},
|
|
{Name: "first", Command: "echo", Args: []string{"1"}},
|
|
{Name: "second", Command: "echo", Args: []string{"2"}, After: []string{"first"}},
|
|
})
|
|
require.NoError(t, err)
|
|
|
|
assert.True(t, result.Success())
|
|
assert.Equal(t, 3, result.Passed)
|
|
})
|
|
|
|
t.Run("skips dependents on failure", func(t *testing.T) {
|
|
runner := newTestRunner(t)
|
|
|
|
result, err := runner.RunAll(context.Background(), []RunSpec{
|
|
{Name: "first", Command: "sh", Args: []string{"-c", "exit 1"}},
|
|
{Name: "second", Command: "echo", Args: []string{"2"}, After: []string{"first"}},
|
|
{Name: "third", Command: "echo", Args: []string{"3"}, After: []string{"second"}},
|
|
})
|
|
require.NoError(t, err)
|
|
|
|
assert.False(t, result.Success())
|
|
assert.Equal(t, 0, result.Passed)
|
|
assert.Equal(t, 1, result.Failed)
|
|
assert.Equal(t, 2, result.Skipped)
|
|
})
|
|
|
|
t.Run("parallel independent specs", func(t *testing.T) {
|
|
runner := newTestRunner(t)
|
|
|
|
// These should run in parallel since they have no dependencies
|
|
result, err := runner.RunAll(context.Background(), []RunSpec{
|
|
{Name: "a", Command: "echo", Args: []string{"a"}},
|
|
{Name: "b", Command: "echo", Args: []string{"b"}},
|
|
{Name: "c", Command: "echo", Args: []string{"c"}},
|
|
{Name: "final", Command: "echo", Args: []string{"done"}, After: []string{"a", "b", "c"}},
|
|
})
|
|
require.NoError(t, err)
|
|
|
|
assert.True(t, result.Success())
|
|
assert.Equal(t, 4, result.Passed)
|
|
})
|
|
}
|
|
|
|
func TestRunResult_Passed(t *testing.T) {
|
|
t.Run("success", func(t *testing.T) {
|
|
r := RunResult{ExitCode: 0}
|
|
assert.True(t, r.Passed())
|
|
})
|
|
|
|
t.Run("non-zero exit", func(t *testing.T) {
|
|
r := RunResult{ExitCode: 1}
|
|
assert.False(t, r.Passed())
|
|
})
|
|
|
|
t.Run("skipped", func(t *testing.T) {
|
|
r := RunResult{ExitCode: 0, Skipped: true}
|
|
assert.False(t, r.Passed())
|
|
})
|
|
|
|
t.Run("error", func(t *testing.T) {
|
|
r := RunResult{ExitCode: 0, Error: assert.AnError}
|
|
assert.False(t, r.Passed())
|
|
})
|
|
}
|