mirror of
				https://github.com/go-gitea/gitea.git
				synced 2025-10-27 00:23:41 +09:00 
			
		
		
		
	Add workflow_run api + webhook (#33964)
Implements - https://docs.github.com/en/rest/actions/workflow-jobs?apiVersion=2022-11-28#list-jobs-for-a-workflow-run--code-samples - https://docs.github.com/en/rest/actions/workflow-jobs?apiVersion=2022-11-28#get-a-job-for-a-workflow-run--code-samples - https://docs.github.com/en/rest/actions/workflow-runs?apiVersion=2022-11-28#list-workflow-runs-for-a-repository - https://docs.github.com/en/rest/actions/workflow-runs?apiVersion=2022-11-28#get-a-workflow-run - `/actions/runs` for global + user + org (Gitea only) - `/actions/jobs` for global + user + org + repository (Gitea only) - workflow_run webhook + action trigger - limitations - workflow id is assigned to a string, this may result into problems in strongly typed clients Fixes - workflow_job webhook url to no longer contain the `runs/<run>` part to align with api - workflow instance does now use it's name inside the file instead of filename if set Refactoring - Moved a lot of logic from workflows/workflow_job into a shared module used by both webhook and api TODO - [x] Verify Keda Compatibility - [x] Edit Webhook API bug is resolved Closes https://github.com/go-gitea/gitea/issues/23670 Closes https://github.com/go-gitea/gitea/issues/23796 Closes https://github.com/go-gitea/gitea/issues/24898 Replaces https://github.com/go-gitea/gitea/pull/28047 and is much more complete --------- Co-authored-by: wxiaoguang <wxiaoguang@gmail.com>
This commit is contained in:
		| @@ -166,6 +166,17 @@ func (run *ActionRun) GetPullRequestEventPayload() (*api.PullRequestPayload, err | ||||
| 	return nil, fmt.Errorf("event %s is not a pull request event", run.Event) | ||||
| } | ||||
|  | ||||
| func (run *ActionRun) GetWorkflowRunEventPayload() (*api.WorkflowRunPayload, error) { | ||||
| 	if run.Event == webhook_module.HookEventWorkflowRun { | ||||
| 		var payload api.WorkflowRunPayload | ||||
| 		if err := json.Unmarshal([]byte(run.EventPayload), &payload); err != nil { | ||||
| 			return nil, err | ||||
| 		} | ||||
| 		return &payload, nil | ||||
| 	} | ||||
| 	return nil, fmt.Errorf("event %s is not a workflow run event", run.Event) | ||||
| } | ||||
|  | ||||
| func (run *ActionRun) IsSchedule() bool { | ||||
| 	return run.ScheduleID > 0 | ||||
| } | ||||
|   | ||||
| @@ -80,22 +80,31 @@ type FindRunJobOptions struct { | ||||
| func (opts FindRunJobOptions) ToConds() builder.Cond { | ||||
| 	cond := builder.NewCond() | ||||
| 	if opts.RunID > 0 { | ||||
| 		cond = cond.And(builder.Eq{"run_id": opts.RunID}) | ||||
| 		cond = cond.And(builder.Eq{"`action_run_job`.run_id": opts.RunID}) | ||||
| 	} | ||||
| 	if opts.RepoID > 0 { | ||||
| 		cond = cond.And(builder.Eq{"repo_id": opts.RepoID}) | ||||
| 	} | ||||
| 	if opts.OwnerID > 0 { | ||||
| 		cond = cond.And(builder.Eq{"owner_id": opts.OwnerID}) | ||||
| 		cond = cond.And(builder.Eq{"`action_run_job`.repo_id": opts.RepoID}) | ||||
| 	} | ||||
| 	if opts.CommitSHA != "" { | ||||
| 		cond = cond.And(builder.Eq{"commit_sha": opts.CommitSHA}) | ||||
| 		cond = cond.And(builder.Eq{"`action_run_job`.commit_sha": opts.CommitSHA}) | ||||
| 	} | ||||
| 	if len(opts.Statuses) > 0 { | ||||
| 		cond = cond.And(builder.In("status", opts.Statuses)) | ||||
| 		cond = cond.And(builder.In("`action_run_job`.status", opts.Statuses)) | ||||
| 	} | ||||
| 	if opts.UpdatedBefore > 0 { | ||||
| 		cond = cond.And(builder.Lt{"updated": opts.UpdatedBefore}) | ||||
| 		cond = cond.And(builder.Lt{"`action_run_job`.updated": opts.UpdatedBefore}) | ||||
| 	} | ||||
| 	return cond | ||||
| } | ||||
|  | ||||
| func (opts FindRunJobOptions) ToJoins() []db.JoinFunc { | ||||
| 	if opts.OwnerID > 0 { | ||||
| 		return []db.JoinFunc{ | ||||
| 			func(sess db.Engine) error { | ||||
| 				sess.Join("INNER", "repository", "repository.id = repo_id AND repository.owner_id = ?", opts.OwnerID) | ||||
| 				return nil | ||||
| 			}, | ||||
| 		} | ||||
| 	} | ||||
| 	return nil | ||||
| } | ||||
|   | ||||
| @@ -72,39 +72,50 @@ type FindRunOptions struct { | ||||
| 	TriggerEvent  webhook_module.HookEventType | ||||
| 	Approved      bool // not util.OptionalBool, it works only when it's true | ||||
| 	Status        []Status | ||||
| 	CommitSHA     string | ||||
| } | ||||
|  | ||||
| func (opts FindRunOptions) ToConds() builder.Cond { | ||||
| 	cond := builder.NewCond() | ||||
| 	if opts.RepoID > 0 { | ||||
| 		cond = cond.And(builder.Eq{"repo_id": opts.RepoID}) | ||||
| 	} | ||||
| 	if opts.OwnerID > 0 { | ||||
| 		cond = cond.And(builder.Eq{"owner_id": opts.OwnerID}) | ||||
| 		cond = cond.And(builder.Eq{"`action_run`.repo_id": opts.RepoID}) | ||||
| 	} | ||||
| 	if opts.WorkflowID != "" { | ||||
| 		cond = cond.And(builder.Eq{"workflow_id": opts.WorkflowID}) | ||||
| 		cond = cond.And(builder.Eq{"`action_run`.workflow_id": opts.WorkflowID}) | ||||
| 	} | ||||
| 	if opts.TriggerUserID > 0 { | ||||
| 		cond = cond.And(builder.Eq{"trigger_user_id": opts.TriggerUserID}) | ||||
| 		cond = cond.And(builder.Eq{"`action_run`.trigger_user_id": opts.TriggerUserID}) | ||||
| 	} | ||||
| 	if opts.Approved { | ||||
| 		cond = cond.And(builder.Gt{"approved_by": 0}) | ||||
| 		cond = cond.And(builder.Gt{"`action_run`.approved_by": 0}) | ||||
| 	} | ||||
| 	if len(opts.Status) > 0 { | ||||
| 		cond = cond.And(builder.In("status", opts.Status)) | ||||
| 		cond = cond.And(builder.In("`action_run`.status", opts.Status)) | ||||
| 	} | ||||
| 	if opts.Ref != "" { | ||||
| 		cond = cond.And(builder.Eq{"ref": opts.Ref}) | ||||
| 		cond = cond.And(builder.Eq{"`action_run`.ref": opts.Ref}) | ||||
| 	} | ||||
| 	if opts.TriggerEvent != "" { | ||||
| 		cond = cond.And(builder.Eq{"trigger_event": opts.TriggerEvent}) | ||||
| 		cond = cond.And(builder.Eq{"`action_run`.trigger_event": opts.TriggerEvent}) | ||||
| 	} | ||||
| 	if opts.CommitSHA != "" { | ||||
| 		cond = cond.And(builder.Eq{"`action_run`.commit_sha": opts.CommitSHA}) | ||||
| 	} | ||||
| 	return cond | ||||
| } | ||||
|  | ||||
| func (opts FindRunOptions) ToJoins() []db.JoinFunc { | ||||
| 	if opts.OwnerID > 0 { | ||||
| 		return []db.JoinFunc{func(sess db.Engine) error { | ||||
| 			sess.Join("INNER", "repository", "repository.id = repo_id AND repository.owner_id = ?", opts.OwnerID) | ||||
| 			return nil | ||||
| 		}} | ||||
| 	} | ||||
| 	return nil | ||||
| } | ||||
|  | ||||
| func (opts FindRunOptions) ToOrders() string { | ||||
| 	return "`id` DESC" | ||||
| 	return "`action_run`.`id` DESC" | ||||
| } | ||||
|  | ||||
| type StatusInfo struct { | ||||
|   | ||||
| @@ -9,6 +9,7 @@ | ||||
|   ref: "refs/heads/master" | ||||
|   commit_sha: "c2d72f548424103f01ee1dc02889c1e2bff816b0" | ||||
|   event: "push" | ||||
|   trigger_event: "push" | ||||
|   is_fork_pull_request: 0 | ||||
|   status: 1 | ||||
|   started: 1683636528 | ||||
| @@ -28,6 +29,7 @@ | ||||
|   ref: "refs/heads/master" | ||||
|   commit_sha: "c2d72f548424103f01ee1dc02889c1e2bff816b0" | ||||
|   event: "push" | ||||
|   trigger_event: "push" | ||||
|   is_fork_pull_request: 0 | ||||
|   status: 1 | ||||
|   started: 1683636528 | ||||
| @@ -47,6 +49,7 @@ | ||||
|   ref: "refs/heads/master" | ||||
|   commit_sha: "c2d72f548424103f01ee1dc02889c1e2bff816b0" | ||||
|   event: "push" | ||||
|   trigger_event: "push" | ||||
|   is_fork_pull_request: 0 | ||||
|   status: 6 # running | ||||
|   started: 1683636528 | ||||
| @@ -66,6 +69,47 @@ | ||||
|   ref: "refs/heads/test" | ||||
|   commit_sha: "c2d72f548424103f01ee1dc02889c1e2bff816b0" | ||||
|   event: "push" | ||||
|   trigger_event: "push" | ||||
|   is_fork_pull_request: 0 | ||||
|   status: 1 | ||||
|   started: 1683636528 | ||||
|   stopped: 1683636626 | ||||
|   created: 1683636108 | ||||
|   updated: 1683636626 | ||||
|   need_approval: 0 | ||||
|   approved_by: 0 | ||||
| - | ||||
|   id: 802 | ||||
|   title: "workflow run list" | ||||
|   repo_id: 5 | ||||
|   owner_id: 3 | ||||
|   workflow_id: "test.yaml" | ||||
|   index: 191 | ||||
|   trigger_user_id: 1 | ||||
|   ref: "refs/heads/test" | ||||
|   commit_sha: "c2d72f548424103f01ee1dc02889c1e2bff816b0" | ||||
|   event: "push" | ||||
|   trigger_event: "push" | ||||
|   is_fork_pull_request: 0 | ||||
|   status: 1 | ||||
|   started: 1683636528 | ||||
|   stopped: 1683636626 | ||||
|   created: 1683636108 | ||||
|   updated: 1683636626 | ||||
|   need_approval: 0 | ||||
|   approved_by: 0 | ||||
| - | ||||
|   id: 803 | ||||
|   title: "workflow run list for user" | ||||
|   repo_id: 2 | ||||
|   owner_id: 0 | ||||
|   workflow_id: "test.yaml" | ||||
|   index: 192 | ||||
|   trigger_user_id: 1 | ||||
|   ref: "refs/heads/test" | ||||
|   commit_sha: "c2d72f548424103f01ee1dc02889c1e2bff816b0" | ||||
|   event: "push" | ||||
|   trigger_event: "push" | ||||
|   is_fork_pull_request: 0 | ||||
|   status: 1 | ||||
|   started: 1683636528 | ||||
| @@ -86,6 +130,7 @@ | ||||
|   ref: "refs/heads/test" | ||||
|   commit_sha: "c2d72f548424103f01ee1dc02889c1e2bff816b0" | ||||
|   event: "push" | ||||
|   trigger_event: "push" | ||||
|   is_fork_pull_request: 0 | ||||
|   status: 2 | ||||
|   started: 1683636528 | ||||
|   | ||||
| @@ -99,3 +99,33 @@ | ||||
|   status: 2 | ||||
|   started: 1683636528 | ||||
|   stopped: 1683636626 | ||||
| - | ||||
|   id: 203 | ||||
|   run_id: 802 | ||||
|   repo_id: 5 | ||||
|   owner_id: 0 | ||||
|   commit_sha: c2d72f548424103f01ee1dc02889c1e2bff816b0 | ||||
|   is_fork_pull_request: 0 | ||||
|   name: job2 | ||||
|   attempt: 1 | ||||
|   job_id: job2 | ||||
|   needs: '["job1"]' | ||||
|   task_id: 51 | ||||
|   status: 5 | ||||
|   started: 1683636528 | ||||
|   stopped: 1683636626 | ||||
| - | ||||
|   id: 204 | ||||
|   run_id: 803 | ||||
|   repo_id: 2 | ||||
|   owner_id: 0 | ||||
|   commit_sha: c2d72f548424103f01ee1dc02889c1e2bff816b0 | ||||
|   is_fork_pull_request: 0 | ||||
|   name: job2 | ||||
|   attempt: 1 | ||||
|   job_id: job2 | ||||
|   needs: '["job1"]' | ||||
|   task_id: 51 | ||||
|   status: 5 | ||||
|   started: 1683636528 | ||||
|   stopped: 1683636626 | ||||
|   | ||||
| @@ -73,7 +73,7 @@ func TestWebhook_EventsArray(t *testing.T) { | ||||
| 		"pull_request", "pull_request_assign", "pull_request_label", "pull_request_milestone", | ||||
| 		"pull_request_comment", "pull_request_review_approved", "pull_request_review_rejected", | ||||
| 		"pull_request_review_comment", "pull_request_sync", "pull_request_review_request", "wiki", "repository", "release", | ||||
| 		"package", "status", "workflow_job", | ||||
| 		"package", "status", "workflow_run", "workflow_job", | ||||
| 	}, | ||||
| 		(&Webhook{ | ||||
| 			HookEvent: &webhook_module.HookEvent{SendEverything: true}, | ||||
|   | ||||
		Reference in New Issue
	
	Block a user