Skip to content

Commit f4f4e18

Browse files
authored
Filter out duplicate action(activity) items for a repository (#30957)
Fix #20986
1 parent b1d8f13 commit f4f4e18

File tree

2 files changed

+31
-1
lines changed

2 files changed

+31
-1
lines changed

models/activities/action.go

+10-1
Original file line numberDiff line numberDiff line change
@@ -524,7 +524,12 @@ func activityQueryCondition(ctx context.Context, opts GetFeedsOptions) (builder.
524524
}
525525

526526
if opts.RequestedRepo != nil {
527-
cond = cond.And(builder.Eq{"repo_id": opts.RequestedRepo.ID})
527+
// repo's actions could have duplicate items, see the comment of NotifyWatchers
528+
// so here we only filter the "original items", aka: user_id == act_user_id
529+
cond = cond.And(
530+
builder.Eq{"`action`.repo_id": opts.RequestedRepo.ID},
531+
builder.Expr("`action`.user_id = `action`.act_user_id"),
532+
)
528533
}
529534

530535
if opts.RequestedTeam != nil {
@@ -577,6 +582,10 @@ func DeleteOldActions(ctx context.Context, olderThan time.Duration) (err error)
577582
}
578583

579584
// NotifyWatchers creates batch of actions for every watcher.
585+
// It could insert duplicate actions for a repository action, like this:
586+
// * Original action: UserID=1 (the real actor), ActUserID=1
587+
// * Organization action: UserID=100 (the repo's org), ActUserID=1
588+
// * Watcher action: UserID=20 (a user who is watching a repo), ActUserID=1
580589
func NotifyWatchers(ctx context.Context, actions ...*Action) error {
581590
var watchers []*repo_model.Watch
582591
var repo *repo_model.Repository

models/activities/action_test.go

+21
Original file line numberDiff line numberDiff line change
@@ -318,3 +318,24 @@ func TestDeleteIssueActions(t *testing.T) {
318318
assert.NoError(t, activities_model.DeleteIssueActions(db.DefaultContext, issue.RepoID, issue.ID, issue.Index))
319319
unittest.AssertCount(t, &activities_model.Action{}, 0)
320320
}
321+
322+
func TestRepoActions(t *testing.T) {
323+
assert.NoError(t, unittest.PrepareTestDatabase())
324+
repo := unittest.AssertExistsAndLoadBean(t, &repo_model.Repository{ID: 1})
325+
_ = db.TruncateBeans(db.DefaultContext, &activities_model.Action{})
326+
for i := 0; i < 3; i++ {
327+
_ = db.Insert(db.DefaultContext, &activities_model.Action{
328+
UserID: 2 + int64(i),
329+
ActUserID: 2,
330+
RepoID: repo.ID,
331+
OpType: activities_model.ActionCommentIssue,
332+
})
333+
}
334+
count, _ := db.Count[activities_model.Action](db.DefaultContext, &db.ListOptions{})
335+
assert.EqualValues(t, 3, count)
336+
actions, _, err := activities_model.GetFeeds(db.DefaultContext, activities_model.GetFeedsOptions{
337+
RequestedRepo: repo,
338+
})
339+
assert.NoError(t, err)
340+
assert.Len(t, actions, 1)
341+
}

0 commit comments

Comments
 (0)