Discovered a regression caused by #9017. Steps to reproduce: - Disable the forgejo-runner that will pick up a workflow - Trigger any workflow to run - Through the Actions list, click on the new workflow that is pending a runner to fetch it - You'll be redirected to /user/repo/actions/runs/73/jobs/0/attempt/0 (attempt = 0) - The UI will appear normal with the job "Waiting"... - Startup the forgejo-runner to pick up the workflow - The UI will begin to have errors: - JavaScript promise rejection: JSON.parse: unexpected keyword at line 1 column 1 of the JSON data. Open browser console to see more details. (5) The cause is that the redirect to `/attempt/0` occurs for a job that hasn't been started, but once the job is started attempt 0 is not a valid attempt and errors will occur when polling for data. This fix corrects the problem by redirecting to the attempt that will be present (attempt 1) when the job is fetched. Reviewed-on: https://codeberg.org/forgejo/forgejo/pulls/9293 Reviewed-by: Earl Warren <earl-warren@noreply.codeberg.org> Co-authored-by: Mathieu Fenniak <mathieu@fenniak.net> Co-committed-by: Mathieu Fenniak <mathieu@fenniak.net>
512 lines
14 KiB
Go
512 lines
14 KiB
Go
// Copyright 2025 The Forgejo Authors. All rights reserved.
|
|
// SPDX-License-Identifier: GPL-3.0-or-later
|
|
|
|
package actions
|
|
|
|
import (
|
|
"fmt"
|
|
"html/template"
|
|
"net/http"
|
|
"testing"
|
|
|
|
actions_model "forgejo.org/models/actions"
|
|
repo_model "forgejo.org/models/repo"
|
|
unittest "forgejo.org/models/unittest"
|
|
"forgejo.org/modules/json"
|
|
"forgejo.org/modules/web"
|
|
"forgejo.org/services/contexttest"
|
|
|
|
"github.com/stretchr/testify/assert"
|
|
"github.com/stretchr/testify/require"
|
|
)
|
|
|
|
func Test_getRunByID(t *testing.T) {
|
|
unittest.PrepareTestEnv(t)
|
|
|
|
repo := unittest.AssertExistsAndLoadBean(t, &repo_model.Repository{OwnerID: 5, ID: 4})
|
|
|
|
for _, testCase := range []struct {
|
|
name string
|
|
runID int64
|
|
err string
|
|
}{
|
|
{
|
|
name: "Found",
|
|
runID: 792,
|
|
},
|
|
{
|
|
name: "NotFound",
|
|
runID: 24344,
|
|
err: "no such run",
|
|
},
|
|
{
|
|
name: "ZeroNotFound",
|
|
runID: 0,
|
|
err: "zero is not a valid run ID",
|
|
},
|
|
} {
|
|
t.Run(testCase.name, func(t *testing.T) {
|
|
ctx, resp := contexttest.MockContext(t, fmt.Sprintf("user5/repo4/actions/runs/%v/artifacts/some-name", testCase.runID))
|
|
ctx.Repo.Repository = repo
|
|
run := getRunByID(ctx, testCase.runID)
|
|
if testCase.err == "" {
|
|
assert.NotNil(t, run)
|
|
assert.False(t, ctx.Written(), resp.Body.String())
|
|
} else {
|
|
assert.Nil(t, run)
|
|
assert.True(t, ctx.Written())
|
|
assert.Contains(t, resp.Body.String(), testCase.err)
|
|
}
|
|
})
|
|
}
|
|
}
|
|
|
|
func Test_artifactsFind(t *testing.T) {
|
|
unittest.PrepareTestEnv(t)
|
|
|
|
for _, testCase := range []struct {
|
|
name string
|
|
artifactName string
|
|
count int
|
|
}{
|
|
{
|
|
name: "Found",
|
|
artifactName: "artifact-v4-download",
|
|
count: 1,
|
|
},
|
|
{
|
|
name: "NotFound",
|
|
artifactName: "notexist",
|
|
count: 0,
|
|
},
|
|
} {
|
|
t.Run(testCase.name, func(t *testing.T) {
|
|
runID := int64(792)
|
|
ctx, _ := contexttest.MockContext(t, fmt.Sprintf("user5/repo4/actions/runs/%v/artifacts/%v", runID, testCase.artifactName))
|
|
artifacts := artifactsFind(ctx, actions_model.FindArtifactsOptions{
|
|
RunID: runID,
|
|
ArtifactName: testCase.artifactName,
|
|
})
|
|
assert.False(t, ctx.Written())
|
|
assert.Len(t, artifacts, testCase.count)
|
|
})
|
|
}
|
|
}
|
|
|
|
func Test_artifactsFindByNameOrID(t *testing.T) {
|
|
unittest.PrepareTestEnv(t)
|
|
|
|
for _, testCase := range []struct {
|
|
name string
|
|
nameOrID string
|
|
err string
|
|
}{
|
|
{
|
|
name: "NameFound",
|
|
nameOrID: "artifact-v4-download",
|
|
},
|
|
{
|
|
name: "NameNotFound",
|
|
nameOrID: "notexist",
|
|
err: "artifact name not found",
|
|
},
|
|
{
|
|
name: "IDFound",
|
|
nameOrID: "22",
|
|
},
|
|
{
|
|
name: "IDNotFound",
|
|
nameOrID: "666",
|
|
err: "artifact ID not found",
|
|
},
|
|
{
|
|
name: "IDZeroNotFound",
|
|
nameOrID: "0",
|
|
err: "artifact name not found",
|
|
},
|
|
} {
|
|
t.Run(testCase.name, func(t *testing.T) {
|
|
runID := int64(792)
|
|
ctx, resp := contexttest.MockContext(t, fmt.Sprintf("user5/repo4/actions/runs/%v/artifacts/%v", runID, testCase.nameOrID))
|
|
artifacts := artifactsFindByNameOrID(ctx, runID, testCase.nameOrID)
|
|
if testCase.err == "" {
|
|
assert.NotEmpty(t, artifacts)
|
|
assert.False(t, ctx.Written(), resp.Body.String())
|
|
} else {
|
|
assert.Empty(t, artifacts)
|
|
assert.True(t, ctx.Written())
|
|
assert.Contains(t, resp.Body.String(), testCase.err)
|
|
}
|
|
})
|
|
}
|
|
}
|
|
|
|
func baseExpectedViewResponse() *ViewResponse {
|
|
return &ViewResponse{
|
|
State: ViewState{
|
|
Run: ViewRunInfo{
|
|
Link: "/user5/repo4/actions/runs/187",
|
|
Title: "update actions",
|
|
TitleHTML: template.HTML("update actions"),
|
|
Status: "success",
|
|
CanCancel: false,
|
|
CanApprove: false,
|
|
CanRerun: false,
|
|
CanDeleteArtifact: false,
|
|
Done: true,
|
|
Jobs: []*ViewJob{
|
|
{
|
|
ID: 192,
|
|
Name: "job_2",
|
|
Status: "success",
|
|
CanRerun: false,
|
|
Duration: "1m38s",
|
|
},
|
|
},
|
|
Commit: ViewCommit{
|
|
LocaleCommit: "actions.runs.commit",
|
|
LocalePushedBy: "actions.runs.pushed_by",
|
|
LocaleWorkflow: "actions.runs.workflow",
|
|
ShortSha: "c2d72f5484",
|
|
Link: "/user5/repo4/commit/c2d72f548424103f01ee1dc02889c1e2bff816b0",
|
|
Pusher: ViewUser{
|
|
DisplayName: "user1",
|
|
Link: "/user1",
|
|
},
|
|
Branch: ViewBranch{
|
|
Name: "master",
|
|
Link: "/user5/repo4/src/branch/master",
|
|
IsDeleted: false,
|
|
},
|
|
},
|
|
},
|
|
CurrentJob: ViewCurrentJob{
|
|
Title: "job_2",
|
|
Detail: "actions.status.success",
|
|
Steps: []*ViewJobStep{
|
|
{
|
|
Summary: "Set up job",
|
|
Status: "running",
|
|
},
|
|
{
|
|
Summary: "Complete job",
|
|
Status: "waiting",
|
|
},
|
|
},
|
|
AllAttempts: []*TaskAttempt{
|
|
{
|
|
Number: 3,
|
|
Started: template.HTML("<relative-time prefix=\"\" tense=\"past\" datetime=\"2023-05-09T12:48:48Z\" data-tooltip-content data-tooltip-interactive=\"true\">2023-05-09 12:48:48 +00:00</relative-time>"),
|
|
Status: "running",
|
|
},
|
|
{
|
|
Number: 2,
|
|
Started: template.HTML("<relative-time prefix=\"\" tense=\"past\" datetime=\"2023-05-09T12:48:48Z\" data-tooltip-content data-tooltip-interactive=\"true\">2023-05-09 12:48:48 +00:00</relative-time>"),
|
|
Status: "success",
|
|
},
|
|
{
|
|
Number: 1,
|
|
Started: template.HTML("<relative-time prefix=\"\" tense=\"past\" datetime=\"2023-05-09T12:48:48Z\" data-tooltip-content data-tooltip-interactive=\"true\">2023-05-09 12:48:48 +00:00</relative-time>"),
|
|
Status: "success",
|
|
},
|
|
},
|
|
},
|
|
},
|
|
Logs: ViewLogs{
|
|
StepsLog: []*ViewStepLog{},
|
|
},
|
|
}
|
|
}
|
|
|
|
func TestActionsViewViewPost(t *testing.T) {
|
|
unittest.PrepareTestEnv(t)
|
|
|
|
tests := []struct {
|
|
name string
|
|
runIndex int64
|
|
jobIndex int64
|
|
attemptNumber int64
|
|
expected *ViewResponse
|
|
expectedTweaks func(*ViewResponse)
|
|
}{
|
|
{
|
|
name: "base case",
|
|
runIndex: 187,
|
|
jobIndex: 0,
|
|
attemptNumber: 1,
|
|
expected: baseExpectedViewResponse(),
|
|
expectedTweaks: func(resp *ViewResponse) {
|
|
resp.State.CurrentJob.Steps[0].Status = "success"
|
|
resp.State.CurrentJob.Steps[1].Status = "success"
|
|
},
|
|
},
|
|
{
|
|
name: "run with waiting jobs",
|
|
runIndex: 189,
|
|
jobIndex: 0,
|
|
attemptNumber: 1,
|
|
expected: baseExpectedViewResponse(),
|
|
expectedTweaks: func(resp *ViewResponse) {
|
|
// Variations from runIndex 187 -> runIndex 189 that are not the subject of this test...
|
|
resp.State.Run.Link = "/user5/repo4/actions/runs/189"
|
|
resp.State.Run.Title = "job output"
|
|
resp.State.Run.TitleHTML = "job output"
|
|
resp.State.Run.Jobs = []*ViewJob{
|
|
{
|
|
ID: 194,
|
|
Name: "job1 (1)",
|
|
Status: "success",
|
|
},
|
|
{
|
|
ID: 195,
|
|
Name: "job1 (2)",
|
|
Status: "success",
|
|
},
|
|
{
|
|
ID: 196,
|
|
Name: "job2",
|
|
Status: "waiting",
|
|
},
|
|
}
|
|
resp.State.CurrentJob.Title = "job1 (1)"
|
|
resp.State.CurrentJob.Steps = []*ViewJobStep{
|
|
{
|
|
Summary: "Set up job",
|
|
Status: "success",
|
|
},
|
|
{
|
|
Summary: "Complete job",
|
|
Status: "success",
|
|
},
|
|
}
|
|
resp.State.CurrentJob.AllAttempts = []*TaskAttempt{
|
|
{
|
|
Number: 1,
|
|
Started: template.HTML("<relative-time prefix=\"\" tense=\"past\" datetime=\"2023-05-09T12:48:48Z\" data-tooltip-content data-tooltip-interactive=\"true\">2023-05-09 12:48:48 +00:00</relative-time>"),
|
|
Status: "success",
|
|
},
|
|
}
|
|
|
|
// Under test in this case: verify that Done is set to false; in the fixture data, job.ID=195 is status
|
|
// Success, but job.ID=196 is status Waiting, and so we expect to signal Done=false to indicate to the
|
|
// UI to continue refreshing the page.
|
|
resp.State.Run.Done = false
|
|
},
|
|
},
|
|
{
|
|
name: "attempt 3",
|
|
runIndex: 187,
|
|
jobIndex: 0,
|
|
attemptNumber: 3,
|
|
expected: baseExpectedViewResponse(),
|
|
expectedTweaks: func(resp *ViewResponse) {
|
|
resp.State.CurrentJob.Steps[0].Status = "running"
|
|
resp.State.CurrentJob.Steps[1].Status = "waiting"
|
|
},
|
|
},
|
|
{
|
|
// This ActionRunJob has TaskID: null, which allows us to access out-of-range attempts without errors and
|
|
// with just some stub data for the UI to start waiting around on.
|
|
name: "attempt out-of-bounds on non-picked task",
|
|
runIndex: 190,
|
|
jobIndex: 0,
|
|
attemptNumber: 100,
|
|
expected: baseExpectedViewResponse(),
|
|
expectedTweaks: func(resp *ViewResponse) {
|
|
// Variations from runIndex 187 -> runIndex 190 that are not the subject of this test...
|
|
resp.State.Run.Link = "/user5/repo4/actions/runs/190"
|
|
resp.State.Run.Title = "job output"
|
|
resp.State.Run.TitleHTML = "job output"
|
|
resp.State.Run.Done = false
|
|
resp.State.Run.Jobs = []*ViewJob{
|
|
{
|
|
ID: 396,
|
|
Name: "job_2",
|
|
Status: "waiting",
|
|
},
|
|
}
|
|
resp.State.Run.Commit.Branch = ViewBranch{
|
|
Name: "test",
|
|
Link: "/user5/repo4/src/branch/test",
|
|
IsDeleted: true,
|
|
}
|
|
|
|
// Expected blank data in the response because this job isn't picked by a runner yet. Keep details here
|
|
// in-sync with the RepoActionView 'view non-picked action run job' test.
|
|
resp.State.CurrentJob.Detail = "actions.status.waiting"
|
|
resp.State.CurrentJob.Steps = []*ViewJobStep{}
|
|
resp.State.CurrentJob.AllAttempts = nil
|
|
},
|
|
},
|
|
}
|
|
|
|
for _, tt := range tests {
|
|
t.Run(tt.name, func(t *testing.T) {
|
|
ctx, resp := contexttest.MockContext(t, "user2/repo1/actions/runs/0")
|
|
contexttest.LoadUser(t, ctx, 2)
|
|
contexttest.LoadRepo(t, ctx, 4)
|
|
ctx.SetParams(":run", fmt.Sprintf("%d", tt.runIndex))
|
|
ctx.SetParams(":job", fmt.Sprintf("%d", tt.jobIndex))
|
|
ctx.SetParams(":attempt", fmt.Sprintf("%d", tt.attemptNumber))
|
|
web.SetForm(ctx, &ViewRequest{})
|
|
|
|
ViewPost(ctx)
|
|
require.Equal(t, http.StatusOK, resp.Result().StatusCode, "failure in ViewPost(): %q", resp.Body.String())
|
|
|
|
var actual ViewResponse
|
|
err := json.Unmarshal(resp.Body.Bytes(), &actual)
|
|
require.NoError(t, err)
|
|
|
|
// `Duration` field is dynamic based upon current time, so eliminate it from comparison -- but check that it
|
|
// has the right format at least.
|
|
zeroDurations := func(vr *ViewResponse) {
|
|
for _, job := range vr.State.Run.Jobs {
|
|
assert.Regexp(t, `^(\d+[hms]){1,3}$`, job.Duration)
|
|
job.Duration = ""
|
|
}
|
|
for _, step := range vr.State.CurrentJob.Steps {
|
|
step.Duration = ""
|
|
}
|
|
}
|
|
zeroDurations(&actual)
|
|
zeroDurations(tt.expected)
|
|
tt.expectedTweaks(tt.expected)
|
|
|
|
assert.Equal(t, *tt.expected, actual)
|
|
})
|
|
}
|
|
}
|
|
|
|
func TestActionsViewRedirectToLatestAttempt(t *testing.T) {
|
|
unittest.PrepareTestEnv(t)
|
|
|
|
tests := []struct {
|
|
name string
|
|
runIndex int64
|
|
jobIndex int64
|
|
expectedCode int
|
|
expectedURL string
|
|
userID int64
|
|
repoID int64
|
|
}{
|
|
{
|
|
name: "no job index",
|
|
runIndex: 187,
|
|
jobIndex: -1,
|
|
expectedURL: "https://try.gitea.io/user2/repo1/actions/runs/187/jobs/0/attempt/1",
|
|
},
|
|
{
|
|
name: "job w/ 1 attempt",
|
|
runIndex: 187,
|
|
jobIndex: 0,
|
|
expectedURL: "https://try.gitea.io/user2/repo1/actions/runs/187/jobs/0/attempt/1",
|
|
},
|
|
{
|
|
name: "job w/ multiple attempts",
|
|
runIndex: 187,
|
|
jobIndex: 2,
|
|
expectedURL: "https://try.gitea.io/user2/repo1/actions/runs/187/jobs/2/attempt/2",
|
|
},
|
|
{
|
|
name: "run out-of-range",
|
|
runIndex: 5000,
|
|
jobIndex: -1,
|
|
expectedCode: http.StatusNotFound,
|
|
},
|
|
// Odd behavior with an out-of-bound jobIndex -- defaults to the first job. This is existing behavior
|
|
// documented in the getRunJobs internal helper which... seems not perfect for the redirect... but it's high
|
|
// risk to change and it's an OK user outcome to be redirected to something valid in the requested run.
|
|
{
|
|
name: "job out-of-range",
|
|
runIndex: 187,
|
|
jobIndex: 500,
|
|
expectedURL: "https://try.gitea.io/user2/repo1/actions/runs/187/jobs/0/attempt/1",
|
|
},
|
|
// This ActionRunJob has Attempt: 0 and TaskID: null, which indicates its first run is pending pickup by a
|
|
// runner. Should redirect to the attempt/1 since that's what it will be when it is running.
|
|
{
|
|
name: "redirect to non-picked task",
|
|
userID: 2,
|
|
repoID: 4,
|
|
runIndex: 190,
|
|
jobIndex: 0,
|
|
expectedURL: "https://try.gitea.io/user5/repo4/actions/runs/190/jobs/0/attempt/1",
|
|
},
|
|
}
|
|
|
|
for _, tt := range tests {
|
|
t.Run(tt.name, func(t *testing.T) {
|
|
ctx, resp := contexttest.MockContext(t, "user2/repo1/actions/runs/0")
|
|
if tt.userID == 0 {
|
|
contexttest.LoadUser(t, ctx, 2)
|
|
} else {
|
|
contexttest.LoadUser(t, ctx, tt.userID)
|
|
}
|
|
if tt.repoID == 0 {
|
|
contexttest.LoadRepo(t, ctx, 1)
|
|
} else {
|
|
contexttest.LoadRepo(t, ctx, tt.repoID)
|
|
}
|
|
ctx.SetParams(":run", fmt.Sprintf("%d", tt.runIndex))
|
|
if tt.jobIndex != -1 {
|
|
ctx.SetParams(":job", fmt.Sprintf("%d", tt.jobIndex))
|
|
}
|
|
|
|
RedirectToLatestAttempt(ctx)
|
|
if tt.expectedCode == 0 {
|
|
assert.Equal(t, http.StatusTemporaryRedirect, resp.Code)
|
|
url, err := resp.Result().Location()
|
|
require.NoError(t, err)
|
|
assert.Equal(t, tt.expectedURL, url.String())
|
|
} else {
|
|
assert.Equal(t, tt.expectedCode, resp.Code)
|
|
}
|
|
})
|
|
}
|
|
}
|
|
|
|
func TestActionsRerun(t *testing.T) {
|
|
tests := []struct {
|
|
name string
|
|
runIndex int64
|
|
jobIndex int64
|
|
expectedCode int
|
|
expectedURL string
|
|
}{
|
|
{
|
|
name: "rerun all",
|
|
runIndex: 187,
|
|
jobIndex: -1,
|
|
expectedURL: "https://try.gitea.io/user2/repo1/actions/runs/187/jobs/0/attempt/2",
|
|
},
|
|
{
|
|
name: "rerun job",
|
|
runIndex: 187,
|
|
jobIndex: 2,
|
|
expectedURL: "https://try.gitea.io/user2/repo1/actions/runs/187/jobs/2/attempt/3",
|
|
},
|
|
}
|
|
for _, tt := range tests {
|
|
t.Run(tt.name, func(t *testing.T) {
|
|
ctx, resp := contexttest.MockContext(t, "user2/repo1/actions/runs/187/rerun")
|
|
contexttest.LoadUser(t, ctx, 2)
|
|
contexttest.LoadRepo(t, ctx, 1)
|
|
ctx.SetParams(":run", fmt.Sprintf("%d", tt.runIndex))
|
|
if tt.jobIndex != -1 {
|
|
ctx.SetParams(":job", fmt.Sprintf("%d", tt.jobIndex))
|
|
}
|
|
|
|
Rerun(ctx)
|
|
require.Equal(t, http.StatusOK, resp.Result().StatusCode, "failure in Rerun(): %q", resp.Body.String())
|
|
|
|
var actual redirectObject
|
|
err := json.Unmarshal(resp.Body.Bytes(), &actual)
|
|
require.NoError(t, err)
|
|
|
|
// Note: this test isn't doing any functional testing of the Rerun handler's actual ability to set up a job
|
|
// rerun. This test was added when the redirect to the correct `attempt` was added and only covers that
|
|
// addition at this time.
|
|
assert.Equal(t, redirectObject{Redirect: tt.expectedURL}, actual)
|
|
})
|
|
}
|
|
}
|