Skip to content

Commit 9a8af92

Browse files
authored
Fix the bug when getting files changed for pull_request_target event (#26320)
Follow #25229 Copy from #26290 (comment) The bug is that we cannot get changed files for the `pull_request_target` event. This event runs in the context of the base branch, so we won't get any changes if we call `GetFilesChangedSinceCommit` with `PullRequest.Base.Ref`.
1 parent 5db4c8d commit 9a8af92

File tree

4 files changed

+77
-13
lines changed

4 files changed

+77
-13
lines changed

modules/actions/workflows.go

+21-9
Original file line numberDiff line numberDiff line change
@@ -95,7 +95,7 @@ func GetEventsFromContent(content []byte) ([]*jobparser.Event, error) {
9595
return events, nil
9696
}
9797

98-
func DetectWorkflows(commit *git.Commit, triggedEvent webhook_module.HookEventType, payload api.Payloader) ([]*DetectedWorkflow, error) {
98+
func DetectWorkflows(gitRepo *git.Repository, commit *git.Commit, triggedEvent webhook_module.HookEventType, payload api.Payloader) ([]*DetectedWorkflow, error) {
9999
entries, err := ListWorkflows(commit)
100100
if err != nil {
101101
return nil, err
@@ -114,7 +114,7 @@ func DetectWorkflows(commit *git.Commit, triggedEvent webhook_module.HookEventTy
114114
}
115115
for _, evt := range events {
116116
log.Trace("detect workflow %q for event %#v matching %q", entry.Name(), evt, triggedEvent)
117-
if detectMatched(commit, triggedEvent, payload, evt) {
117+
if detectMatched(gitRepo, commit, triggedEvent, payload, evt) {
118118
dwf := &DetectedWorkflow{
119119
EntryName: entry.Name(),
120120
TriggerEvent: evt.Name,
@@ -128,7 +128,7 @@ func DetectWorkflows(commit *git.Commit, triggedEvent webhook_module.HookEventTy
128128
return workflows, nil
129129
}
130130

131-
func detectMatched(commit *git.Commit, triggedEvent webhook_module.HookEventType, payload api.Payloader, evt *jobparser.Event) bool {
131+
func detectMatched(gitRepo *git.Repository, commit *git.Commit, triggedEvent webhook_module.HookEventType, payload api.Payloader, evt *jobparser.Event) bool {
132132
if !canGithubEventMatch(evt.Name, triggedEvent) {
133133
return false
134134
}
@@ -168,7 +168,7 @@ func detectMatched(commit *git.Commit, triggedEvent webhook_module.HookEventType
168168
webhook_module.HookEventPullRequestSync,
169169
webhook_module.HookEventPullRequestAssign,
170170
webhook_module.HookEventPullRequestLabel:
171-
return matchPullRequestEvent(commit, payload.(*api.PullRequestPayload), evt)
171+
return matchPullRequestEvent(gitRepo, commit, payload.(*api.PullRequestPayload), evt)
172172

173173
case // pull_request_review
174174
webhook_module.HookEventPullRequestReviewApproved,
@@ -331,7 +331,7 @@ func matchIssuesEvent(commit *git.Commit, issuePayload *api.IssuePayload, evt *j
331331
return matchTimes == len(evt.Acts())
332332
}
333333

334-
func matchPullRequestEvent(commit *git.Commit, prPayload *api.PullRequestPayload, evt *jobparser.Event) bool {
334+
func matchPullRequestEvent(gitRepo *git.Repository, commit *git.Commit, prPayload *api.PullRequestPayload, evt *jobparser.Event) bool {
335335
acts := evt.Acts()
336336
activityTypeMatched := false
337337
matchTimes := 0
@@ -370,6 +370,18 @@ func matchPullRequestEvent(commit *git.Commit, prPayload *api.PullRequestPayload
370370
}
371371
}
372372

373+
var (
374+
headCommit = commit
375+
err error
376+
)
377+
if evt.Name == GithubEventPullRequestTarget && (len(acts["paths"]) > 0 || len(acts["paths-ignore"]) > 0) {
378+
headCommit, err = gitRepo.GetCommit(prPayload.PullRequest.Head.Sha)
379+
if err != nil {
380+
log.Error("GetCommit [ref: %s]: %v", prPayload.PullRequest.Head.Sha, err)
381+
return false
382+
}
383+
}
384+
373385
// all acts conditions should be satisfied
374386
for cond, vals := range acts {
375387
switch cond {
@@ -392,9 +404,9 @@ func matchPullRequestEvent(commit *git.Commit, prPayload *api.PullRequestPayload
392404
matchTimes++
393405
}
394406
case "paths":
395-
filesChanged, err := commit.GetFilesChangedSinceCommit(prPayload.PullRequest.Base.Ref)
407+
filesChanged, err := headCommit.GetFilesChangedSinceCommit(prPayload.PullRequest.Base.Ref)
396408
if err != nil {
397-
log.Error("GetFilesChangedSinceCommit [commit_sha1: %s]: %v", commit.ID.String(), err)
409+
log.Error("GetFilesChangedSinceCommit [commit_sha1: %s]: %v", headCommit.ID.String(), err)
398410
} else {
399411
patterns, err := workflowpattern.CompilePatterns(vals...)
400412
if err != nil {
@@ -405,9 +417,9 @@ func matchPullRequestEvent(commit *git.Commit, prPayload *api.PullRequestPayload
405417
}
406418
}
407419
case "paths-ignore":
408-
filesChanged, err := commit.GetFilesChangedSinceCommit(prPayload.PullRequest.Base.Ref)
420+
filesChanged, err := headCommit.GetFilesChangedSinceCommit(prPayload.PullRequest.Base.Ref)
409421
if err != nil {
410-
log.Error("GetFilesChangedSinceCommit [commit_sha1: %s]: %v", commit.ID.String(), err)
422+
log.Error("GetFilesChangedSinceCommit [commit_sha1: %s]: %v", headCommit.ID.String(), err)
411423
} else {
412424
patterns, err := workflowpattern.CompilePatterns(vals...)
413425
if err != nil {

modules/actions/workflows_test.go

+1-1
Original file line numberDiff line numberDiff line change
@@ -125,7 +125,7 @@ func TestDetectMatched(t *testing.T) {
125125
evts, err := GetEventsFromContent([]byte(tc.yamlOn))
126126
assert.NoError(t, err)
127127
assert.Len(t, evts, 1)
128-
assert.Equal(t, tc.expected, detectMatched(tc.commit, tc.triggedEvent, tc.payload, evts[0]))
128+
assert.Equal(t, tc.expected, detectMatched(nil, tc.commit, tc.triggedEvent, tc.payload, evts[0]))
129129
})
130130
}
131131
}

services/actions/notifier_helper.go

+2-2
Original file line numberDiff line numberDiff line change
@@ -143,7 +143,7 @@ func notify(ctx context.Context, input *notifyInput) error {
143143
}
144144

145145
var detectedWorkflows []*actions_module.DetectedWorkflow
146-
workflows, err := actions_module.DetectWorkflows(commit, input.Event, input.Payload)
146+
workflows, err := actions_module.DetectWorkflows(gitRepo, commit, input.Event, input.Payload)
147147
if err != nil {
148148
return fmt.Errorf("DetectWorkflows: %w", err)
149149
}
@@ -164,7 +164,7 @@ func notify(ctx context.Context, input *notifyInput) error {
164164
if err != nil {
165165
return fmt.Errorf("gitRepo.GetCommit: %w", err)
166166
}
167-
baseWorkflows, err := actions_module.DetectWorkflows(baseCommit, input.Event, input.Payload)
167+
baseWorkflows, err := actions_module.DetectWorkflows(gitRepo, baseCommit, input.Event, input.Payload)
168168
if err != nil {
169169
return fmt.Errorf("DetectWorkflows: %w", err)
170170
}

tests/integration/actions_trigger_test.go

+53-1
Original file line numberDiff line numberDiff line change
@@ -67,7 +67,7 @@ func TestPullRequestTargetEvent(t *testing.T) {
6767
{
6868
Operation: "create",
6969
TreePath: ".gitea/workflows/pr.yml",
70-
ContentReader: strings.NewReader("name: test\non: pull_request_target\njobs:\n test:\n runs-on: ubuntu-latest\n steps:\n - run: echo helloworld\n"),
70+
ContentReader: strings.NewReader("name: test\non:\n pull_request_target:\n paths:\n - 'file_*.txt'\njobs:\n test:\n runs-on: ubuntu-latest\n steps:\n - run: echo helloworld\n"),
7171
},
7272
},
7373
Message: "add workflow",
@@ -138,8 +138,60 @@ func TestPullRequestTargetEvent(t *testing.T) {
138138
assert.NoError(t, err)
139139

140140
// load and compare ActionRun
141+
assert.Equal(t, 1, unittest.GetCount(t, &actions_model.ActionRun{RepoID: baseRepo.ID}))
141142
actionRun := unittest.AssertExistsAndLoadBean(t, &actions_model.ActionRun{RepoID: baseRepo.ID})
142143
assert.Equal(t, addFileToForkedResp.Commit.SHA, actionRun.CommitSHA)
143144
assert.Equal(t, actions_module.GithubEventPullRequestTarget, actionRun.TriggerEvent)
145+
146+
// add another file whose name cannot match the specified path
147+
addFileToForkedResp, err = files_service.ChangeRepoFiles(git.DefaultContext, forkedRepo, user3, &files_service.ChangeRepoFilesOptions{
148+
Files: []*files_service.ChangeRepoFile{
149+
{
150+
Operation: "create",
151+
TreePath: "foo.txt",
152+
ContentReader: strings.NewReader("foo"),
153+
},
154+
},
155+
Message: "add foo.txt",
156+
OldBranch: "main",
157+
NewBranch: "fork-branch-2",
158+
Author: &files_service.IdentityOptions{
159+
Name: user3.Name,
160+
Email: user3.Email,
161+
},
162+
Committer: &files_service.IdentityOptions{
163+
Name: user3.Name,
164+
Email: user3.Email,
165+
},
166+
Dates: &files_service.CommitDateOptions{
167+
Author: time.Now(),
168+
Committer: time.Now(),
169+
},
170+
})
171+
assert.NoError(t, err)
172+
assert.NotEmpty(t, addFileToForkedResp)
173+
174+
// create Pull
175+
pullIssue = &issues_model.Issue{
176+
RepoID: baseRepo.ID,
177+
Title: "A mismatched path cannot trigger pull-request-target-event",
178+
PosterID: user3.ID,
179+
Poster: user3,
180+
IsPull: true,
181+
}
182+
pullRequest = &issues_model.PullRequest{
183+
HeadRepoID: forkedRepo.ID,
184+
BaseRepoID: baseRepo.ID,
185+
HeadBranch: "fork-branch-2",
186+
BaseBranch: "main",
187+
HeadRepo: forkedRepo,
188+
BaseRepo: baseRepo,
189+
Type: issues_model.PullRequestGitea,
190+
}
191+
err = pull_service.NewPullRequest(git.DefaultContext, baseRepo, pullIssue, nil, nil, pullRequest, nil)
192+
assert.NoError(t, err)
193+
194+
// the new pull request cannot trigger actions, so there is still only 1 record
195+
assert.Equal(t, 1, unittest.GetCount(t, &actions_model.ActionRun{RepoID: baseRepo.ID}))
144196
})
145197
}

0 commit comments

Comments
 (0)