mirror of
https://github.com/go-gitea/gitea.git
synced 2026-05-06 04:01:05 +09:00
This PR introduces a new `ActionRunAttempt` model and makes Actions
execution attempt-scoped.
**Main Changes**
- Each workflow run trigger generates a new `ActionRunAttempt`. The
triggered jobs are then associated with this new `ActionRunAttempt`
record.
- Each rerun now creates:
- a new `ActionRunAttempt` record for the workflow run
- a full new set of `ActionRunJob` records for the new
`ActionRunAttempt`
- For jobs that need to be rerun, the new job records are created as
runnable jobs in the new attempt.
- For jobs that do not need to be rerun, new job records are still
created in the new attempt, but they reuse the result of the previous
attempt instead of executing again.
- Introduce `rerunPlan` to manage each rerun and refactored rerun flow
into a two-phase plan-based model:
- `buildRerunPlan`
- `execRerunPlan`
- `RerunFailedWorkflowRun` and `RerunFailed` no longer directly derives
all jobs that need to be rerun; this step is now handled by
`buildRerunPlan`.
- Converted artifacts from run-scoped to attempt-scoped:
- uploads are now associated with `RunAttemptID`
- listing, download, and deletion resolve against the current attempt
- Added attempt-aware web Actions views:
- the default run page shows the latest attempt
(`/actions/runs/{run_id}`)
- previous attempt pages show jobs and artifacts for that attempt
(`/actions/runs/{run_id}/attempts/{attempt_num}`)
- New APIs:
- `/repos/{owner}/{repo}/actions/runs/{run}/attempts/{attempt}`
- `/repos/{owner}/{repo}/actions/runs/{run}/attempts/{attempt}/jobs`
- New configuration `MAX_RERUN_ATTEMPTS`
- https://gitea.com/gitea/docs/pulls/383
**Compatibility**
- Existing legacy runs use `LatestAttemptID = 0` and legacy jobs use
`RunAttemptID = 0`. Therefore, these fields can be used to identify
legacy runs and jobs and provide backward compatibility.
- If a legacy run is rerun, an `ActionRunAttempt` with `attempt=1` will
be created to represent the original execution. Then a new
`ActionRunAttempt` with `attempt=2` will be created for the real rerun.
- Existing artifact records are not backfilled; legacy artifacts
continue to use `RunAttemptID = 0`.
**Improvements**
- It is now easier to inspect and download logs from previous attempts.
-
[`run_attempt`](https://docs.github.com/en/actions/reference/workflows-and-actions/contexts#github-context)
semantics are now aligned with GitHub.
- > A unique number for each attempt of a particular workflow run in a
repository. This number begins at 1 for the workflow run's first
attempt, and increments with each re-run.
- Rerun behavior is now clearer and more explicit.
- Instead of mutating the status of previous jobs in place, each rerun
creates a new attempt with a full new set of job records.
- Artifacts produced by different reruns can now be listed separately.
Signed-off-by: Zettat123 <zettat123@gmail.com>
Co-authored-by: silverwind <me@silverwind.io>
Co-authored-by: Claude Opus 4.7 <noreply@anthropic.com>
Co-authored-by: wxiaoguang <wxiaoguang@gmail.com>
Co-authored-by: Giteabot <teabot@gitea.io>
229 lines
8.2 KiB
Go
229 lines
8.2 KiB
Go
// Copyright 2023 The Gitea Authors. All rights reserved.
|
|
// SPDX-License-Identifier: MIT
|
|
|
|
package structs
|
|
|
|
import (
|
|
"time"
|
|
)
|
|
|
|
// ActionTask represents a ActionTask
|
|
type ActionTask struct {
|
|
// ID is the unique identifier for the action task
|
|
ID int64 `json:"id"`
|
|
// Name is the name of the workflow
|
|
Name string `json:"name"`
|
|
// HeadBranch is the branch that triggered the workflow
|
|
HeadBranch string `json:"head_branch"`
|
|
// HeadSHA is the commit SHA that triggered the workflow
|
|
HeadSHA string `json:"head_sha"`
|
|
// RunNumber is the sequential number of the workflow run
|
|
RunNumber int64 `json:"run_number"`
|
|
// Event is the type of event that triggered the workflow
|
|
Event string `json:"event"`
|
|
// DisplayTitle is the display title for the workflow run
|
|
DisplayTitle string `json:"display_title"`
|
|
// Status indicates the current status of the workflow run
|
|
Status string `json:"status"`
|
|
// WorkflowID is the identifier of the workflow
|
|
WorkflowID string `json:"workflow_id"`
|
|
// URL is the API URL for this workflow run
|
|
URL string `json:"url"`
|
|
// swagger:strfmt date-time
|
|
CreatedAt time.Time `json:"created_at"`
|
|
// swagger:strfmt date-time
|
|
UpdatedAt time.Time `json:"updated_at"`
|
|
// swagger:strfmt date-time
|
|
RunStartedAt time.Time `json:"run_started_at"`
|
|
}
|
|
|
|
// ActionTaskResponse returns a ActionTask
|
|
type ActionTaskResponse struct {
|
|
// Entries contains the list of workflow runs
|
|
Entries []*ActionTask `json:"workflow_runs"`
|
|
// TotalCount is the total number of workflow runs
|
|
TotalCount int64 `json:"total_count"`
|
|
}
|
|
|
|
// CreateActionWorkflowDispatch represents the payload for triggering a workflow dispatch event
|
|
// swagger:model
|
|
type CreateActionWorkflowDispatch struct {
|
|
// required: true
|
|
// example: refs/heads/main
|
|
Ref string `json:"ref" binding:"Required"`
|
|
// required: false
|
|
Inputs map[string]string `json:"inputs,omitempty"`
|
|
}
|
|
|
|
// ActionWorkflow represents a ActionWorkflow
|
|
type ActionWorkflow struct {
|
|
// ID is the unique identifier for the workflow
|
|
ID string `json:"id"`
|
|
// Name is the name of the workflow
|
|
Name string `json:"name"`
|
|
// Path is the file path of the workflow
|
|
Path string `json:"path"`
|
|
// State indicates if the workflow is active or disabled
|
|
State string `json:"state"`
|
|
// swagger:strfmt date-time
|
|
CreatedAt time.Time `json:"created_at"`
|
|
// swagger:strfmt date-time
|
|
UpdatedAt time.Time `json:"updated_at"`
|
|
// URL is the API URL for this workflow
|
|
URL string `json:"url"`
|
|
// HTMLURL is the web URL for viewing the workflow
|
|
HTMLURL string `json:"html_url"`
|
|
// BadgeURL is the URL for the workflow badge
|
|
BadgeURL string `json:"badge_url"`
|
|
// swagger:strfmt date-time
|
|
DeletedAt time.Time `json:"deleted_at"`
|
|
}
|
|
|
|
// ActionWorkflowResponse returns a ActionWorkflow
|
|
type ActionWorkflowResponse struct {
|
|
Workflows []*ActionWorkflow `json:"workflows"`
|
|
TotalCount int64 `json:"total_count"`
|
|
}
|
|
|
|
// ActionArtifact represents a ActionArtifact
|
|
type ActionArtifact struct {
|
|
ID int64 `json:"id"`
|
|
Name string `json:"name"`
|
|
SizeInBytes int64 `json:"size_in_bytes"`
|
|
URL string `json:"url"`
|
|
ArchiveDownloadURL string `json:"archive_download_url"`
|
|
Expired bool `json:"expired"`
|
|
WorkflowRun *ActionWorkflowRun `json:"workflow_run"`
|
|
|
|
// swagger:strfmt date-time
|
|
CreatedAt time.Time `json:"created_at"`
|
|
// swagger:strfmt date-time
|
|
UpdatedAt time.Time `json:"updated_at"`
|
|
// swagger:strfmt date-time
|
|
ExpiresAt time.Time `json:"expires_at"`
|
|
}
|
|
|
|
// ActionWorkflowRun represents a WorkflowRun
|
|
type ActionWorkflowRun struct {
|
|
ID int64 `json:"id"`
|
|
URL string `json:"url"`
|
|
// PreviousAttemptURL is the API URL of the previous attempt of this run, e.g. ".../actions/runs/{run_id}/attempts/{attempt-1}".
|
|
// It is set only when the current attempt is > 1 (i.e. a rerun). For the first attempt, or for legacy runs that pre-date ActionRunAttempt, it is null.
|
|
PreviousAttemptURL *string `json:"previous_attempt_url"`
|
|
HTMLURL string `json:"html_url"`
|
|
DisplayTitle string `json:"display_title"`
|
|
Path string `json:"path"`
|
|
Event string `json:"event"`
|
|
// RunAttempt is 1-based for runs created after ActionRunAttempt was introduced.
|
|
// A value of 0 is a legacy-only sentinel for runs created before attempts existed
|
|
// and indicates no corresponding /attempts/{n} resource is available.
|
|
RunAttempt int64 `json:"run_attempt"`
|
|
RunNumber int64 `json:"run_number"`
|
|
RepositoryID int64 `json:"repository_id,omitempty"`
|
|
HeadSha string `json:"head_sha"`
|
|
HeadBranch string `json:"head_branch,omitempty"`
|
|
Status string `json:"status"`
|
|
Actor *User `json:"actor,omitempty"`
|
|
TriggerActor *User `json:"trigger_actor,omitempty"`
|
|
Repository *Repository `json:"repository,omitempty"`
|
|
HeadRepository *Repository `json:"head_repository,omitempty"`
|
|
Conclusion string `json:"conclusion,omitempty"`
|
|
// swagger:strfmt date-time
|
|
StartedAt time.Time `json:"started_at"`
|
|
// swagger:strfmt date-time
|
|
CompletedAt time.Time `json:"completed_at"`
|
|
}
|
|
|
|
// ActionWorkflowRunsResponse returns ActionWorkflowRuns
|
|
type ActionWorkflowRunsResponse struct {
|
|
Entries []*ActionWorkflowRun `json:"workflow_runs"`
|
|
TotalCount int64 `json:"total_count"`
|
|
}
|
|
|
|
// ActionWorkflowJobsResponse returns ActionWorkflowJobs
|
|
type ActionWorkflowJobsResponse struct {
|
|
Entries []*ActionWorkflowJob `json:"jobs"`
|
|
TotalCount int64 `json:"total_count"`
|
|
}
|
|
|
|
// ActionArtifactsResponse returns ActionArtifacts
|
|
type ActionArtifactsResponse struct {
|
|
Entries []*ActionArtifact `json:"artifacts"`
|
|
TotalCount int64 `json:"total_count"`
|
|
}
|
|
|
|
// ActionWorkflowStep represents a step of a WorkflowJob
|
|
type ActionWorkflowStep struct {
|
|
Name string `json:"name"`
|
|
Number int64 `json:"number"`
|
|
Status string `json:"status"`
|
|
Conclusion string `json:"conclusion,omitempty"`
|
|
// swagger:strfmt date-time
|
|
StartedAt time.Time `json:"started_at"`
|
|
// swagger:strfmt date-time
|
|
CompletedAt time.Time `json:"completed_at"`
|
|
}
|
|
|
|
// ActionWorkflowJob represents a WorkflowJob
|
|
type ActionWorkflowJob struct {
|
|
ID int64 `json:"id"`
|
|
URL string `json:"url"`
|
|
HTMLURL string `json:"html_url"`
|
|
RunID int64 `json:"run_id"`
|
|
RunURL string `json:"run_url"`
|
|
Name string `json:"name"`
|
|
Labels []string `json:"labels"`
|
|
RunAttempt int64 `json:"run_attempt"`
|
|
HeadSha string `json:"head_sha"`
|
|
HeadBranch string `json:"head_branch,omitempty"`
|
|
Status string `json:"status"`
|
|
Conclusion string `json:"conclusion,omitempty"`
|
|
RunnerID int64 `json:"runner_id,omitempty"`
|
|
RunnerName string `json:"runner_name,omitempty"`
|
|
Steps []*ActionWorkflowStep `json:"steps"`
|
|
// swagger:strfmt date-time
|
|
CreatedAt time.Time `json:"created_at"`
|
|
// swagger:strfmt date-time
|
|
StartedAt time.Time `json:"started_at"`
|
|
// swagger:strfmt date-time
|
|
CompletedAt time.Time `json:"completed_at"`
|
|
}
|
|
|
|
// ActionRunnerLabel represents a Runner Label
|
|
type ActionRunnerLabel struct {
|
|
ID int64 `json:"id"`
|
|
Name string `json:"name"`
|
|
Type string `json:"type"`
|
|
}
|
|
|
|
// ActionRunner represents a Runner
|
|
type ActionRunner struct {
|
|
ID int64 `json:"id"`
|
|
Name string `json:"name"`
|
|
Status string `json:"status"`
|
|
Busy bool `json:"busy"`
|
|
Disabled bool `json:"disabled"`
|
|
Ephemeral bool `json:"ephemeral"`
|
|
Labels []*ActionRunnerLabel `json:"labels"`
|
|
}
|
|
|
|
// EditActionRunnerOption represents the editable fields for a runner.
|
|
// swagger:model
|
|
type EditActionRunnerOption struct {
|
|
// required: true
|
|
Disabled *bool `json:"disabled"`
|
|
}
|
|
|
|
// ActionRunnersResponse returns Runners
|
|
type ActionRunnersResponse struct {
|
|
Entries []*ActionRunner `json:"runners"`
|
|
TotalCount int64 `json:"total_count"`
|
|
}
|
|
|
|
// RunDetails returns workflow_dispatch runid and url
|
|
type RunDetails struct {
|
|
WorkflowRunID int64 `json:"workflow_run_id"`
|
|
RunURL string `json:"run_url"`
|
|
HTMLURL string `json:"html_url"`
|
|
}
|