mattermost/server/channels/app/post_helpers_test.go
Harshil Sharma 9cb5f15c79
Some checks failed
API / build (push) Waiting to run
Server CI / Compute Go Version (push) Waiting to run
Server CI / Check mocks (push) Blocked by required conditions
Server CI / Check go mod tidy (push) Blocked by required conditions
Server CI / check-style (push) Blocked by required conditions
Server CI / Check serialization methods for hot structs (push) Blocked by required conditions
Server CI / Vet API (push) Blocked by required conditions
Server CI / Check migration files (push) Blocked by required conditions
Server CI / Generate email templates (push) Blocked by required conditions
Server CI / Check store layers (push) Blocked by required conditions
Server CI / Check mmctl docs (push) Blocked by required conditions
Server CI / Postgres with binary parameters (push) Blocked by required conditions
Server CI / Postgres (push) Blocked by required conditions
Server CI / Postgres (FIPS) (push) Blocked by required conditions
Server CI / Generate Test Coverage (push) Blocked by required conditions
Server CI / Run mmctl tests (push) Blocked by required conditions
Server CI / Run mmctl tests (FIPS) (push) Blocked by required conditions
Server CI / Build mattermost server app (push) Blocked by required conditions
Web App CI / check-lint (push) Waiting to run
Web App CI / check-i18n (push) Blocked by required conditions
Web App CI / check-types (push) Blocked by required conditions
Web App CI / test (platform) (push) Blocked by required conditions
Web App CI / test (mattermost-redux) (push) Blocked by required conditions
Web App CI / test (channels shard 1/4) (push) Blocked by required conditions
Web App CI / test (channels shard 2/4) (push) Blocked by required conditions
Web App CI / test (channels shard 3/4) (push) Blocked by required conditions
Web App CI / test (channels shard 4/4) (push) Blocked by required conditions
Web App CI / upload-coverage (push) Blocked by required conditions
Web App CI / build (push) Blocked by required conditions
Opensearch Docker Image / build-image (push) Has been cancelled
Changes for BoR post soft-deletion (#35100)
* Draft changes for BoR post soft-deletion

* Handled the case for author's BoR post read receipt

* lint fix

* Updated text

* Updated tests

* review fixes

* review fixes

* Paginated and batched temperory post deletion

* Updated test

* unmocked store

* logged instead of erroring out

* i18n fix

* review fixes
2026-02-12 05:25:49 -05:00

463 lines
15 KiB
Go

// Copyright (c) 2015-present Mattermost, Inc. All Rights Reserved.
// See LICENSE.txt for license information.
package app
import (
"testing"
"github.com/stretchr/testify/assert"
"github.com/stretchr/testify/require"
"github.com/mattermost/mattermost/server/public/model"
)
func TestGetTimeSortedPostAccessibleBounds(t *testing.T) {
mainHelper.Parallel(t)
postFromCreateAt := func(at int64) *model.Post {
return &model.Post{CreateAt: at}
}
getPostListCreateAtFunc := func(pl *model.PostList) func(i int) int64 {
return func(i int) int64 {
return pl.Posts[pl.Order[i]].CreateAt
}
}
t.Run("empty posts returns all accessible posts", func(t *testing.T) {
pl := &model.PostList{
Posts: map[string]*model.Post{},
Order: []string{},
}
bounds := getTimeSortedPostAccessibleBounds(0, len(pl.Posts), getPostListCreateAtFunc(pl))
require.True(t, bounds.allAccessible(len(pl.Posts)))
})
t.Run("one accessible post returns all accessible posts", func(t *testing.T) {
pl := &model.PostList{
Posts: map[string]*model.Post{
"post_a": postFromCreateAt(1),
},
Order: []string{"post_a"},
}
bounds := getTimeSortedPostAccessibleBounds(0, len(pl.Posts), getPostListCreateAtFunc(pl))
require.True(t, bounds.allAccessible(len(pl.Posts)))
})
t.Run("one inaccessible post returns no accessible posts", func(t *testing.T) {
pl := &model.PostList{
Posts: map[string]*model.Post{
"post_a": postFromCreateAt(0),
},
Order: []string{"post_a"},
}
bounds := getTimeSortedPostAccessibleBounds(1, len(pl.Posts), getPostListCreateAtFunc(pl))
require.True(t, bounds.noAccessible())
})
t.Run("all accessible posts returns all accessible posts", func(t *testing.T) {
pl := &model.PostList{
Posts: map[string]*model.Post{
"post_a": postFromCreateAt(1),
"post_b": postFromCreateAt(2),
"post_c": postFromCreateAt(3),
"post_d": postFromCreateAt(4),
"post_e": postFromCreateAt(5),
"post_f": postFromCreateAt(6),
},
Order: []string{"post_a", "post_b", "post_c", "post_d", "post_e", "post_f"},
}
bounds := getTimeSortedPostAccessibleBounds(0, len(pl.Posts), getPostListCreateAtFunc(pl))
require.True(t, bounds.allAccessible(len(pl.Posts)))
})
t.Run("all inaccessible posts returns all inaccessible posts", func(t *testing.T) {
pl := &model.PostList{
Posts: map[string]*model.Post{
"post_a": postFromCreateAt(1),
"post_b": postFromCreateAt(2),
"post_c": postFromCreateAt(3),
"post_d": postFromCreateAt(4),
"post_e": postFromCreateAt(5),
"post_f": postFromCreateAt(6),
},
Order: []string{"post_a", "post_b", "post_c", "post_d", "post_e", "post_f"},
}
bounds := getTimeSortedPostAccessibleBounds(7, len(pl.Posts), getPostListCreateAtFunc(pl))
require.True(t, bounds.noAccessible())
})
t.Run("all accessible posts returns all accessible posts, descending ordered", func(t *testing.T) {
pl := &model.PostList{
Posts: map[string]*model.Post{
"post_a": postFromCreateAt(1),
"post_b": postFromCreateAt(2),
"post_c": postFromCreateAt(3),
"post_d": postFromCreateAt(4),
"post_e": postFromCreateAt(5),
"post_f": postFromCreateAt(6),
},
Order: []string{"post_f", "post_e", "post_d", "post_c", "post_b", "post_a"},
}
bounds := getTimeSortedPostAccessibleBounds(0, len(pl.Posts), getPostListCreateAtFunc(pl))
require.True(t, bounds.allAccessible(len(pl.Posts)))
})
t.Run("all inaccessible posts returns all inaccessible posts, descending ordered", func(t *testing.T) {
pl := &model.PostList{
Posts: map[string]*model.Post{
"post_a": postFromCreateAt(1),
"post_b": postFromCreateAt(2),
"post_c": postFromCreateAt(3),
"post_d": postFromCreateAt(4),
"post_e": postFromCreateAt(5),
"post_f": postFromCreateAt(6),
},
Order: []string{"post_f", "post_e", "post_d", "post_c", "post_b", "post_a"},
}
bounds := getTimeSortedPostAccessibleBounds(7, len(pl.Posts), getPostListCreateAtFunc(pl))
require.True(t, bounds.noAccessible())
})
t.Run("two posts, first accessible", func(t *testing.T) {
pl := &model.PostList{
Posts: map[string]*model.Post{
"post_a": postFromCreateAt(1),
"post_b": postFromCreateAt(0),
},
Order: []string{"post_a", "post_b"},
}
bounds := getTimeSortedPostAccessibleBounds(1, len(pl.Posts), getPostListCreateAtFunc(pl))
require.Equal(t, accessibleBounds{start: 0, end: 0}, bounds)
})
t.Run("two posts, second accessible", func(t *testing.T) {
pl := &model.PostList{
Posts: map[string]*model.Post{
"post_a": postFromCreateAt(0),
"post_b": postFromCreateAt(1),
},
Order: []string{"post_a", "post_b"},
}
bounds := getTimeSortedPostAccessibleBounds(1, len(pl.Posts), getPostListCreateAtFunc(pl))
require.Equal(t, accessibleBounds{start: 1, end: 1}, bounds)
})
t.Run("picks the left most post for boundaries when there are time ties", func(t *testing.T) {
pl := &model.PostList{
Posts: map[string]*model.Post{
"post_a": postFromCreateAt(0),
"post_b": postFromCreateAt(1),
"post_c": postFromCreateAt(1),
"post_d": postFromCreateAt(2),
},
Order: []string{"post_a", "post_b", "post_c", "post_d"},
}
bounds := getTimeSortedPostAccessibleBounds(1, len(pl.Posts), getPostListCreateAtFunc(pl))
require.Equal(t, accessibleBounds{start: 1, end: len(pl.Posts) - 1}, bounds)
})
t.Run("picks the right most post for boundaries when there are time ties, descending ordered", func(t *testing.T) {
pl := &model.PostList{
Posts: map[string]*model.Post{
"post_a": postFromCreateAt(0),
"post_b": postFromCreateAt(1),
"post_c": postFromCreateAt(1),
"post_d": postFromCreateAt(2),
},
Order: []string{"post_d", "post_c", "post_b", "post_a"},
}
bounds := getTimeSortedPostAccessibleBounds(1, len(pl.Posts), getPostListCreateAtFunc(pl))
require.Equal(t, accessibleBounds{start: 0, end: 2}, bounds)
})
t.Run("odd number of posts and reverse time selects right boundaries", func(t *testing.T) {
pl := &model.PostList{
Posts: map[string]*model.Post{
"post_a": postFromCreateAt(0),
"post_b": postFromCreateAt(1),
"post_c": postFromCreateAt(2),
"post_d": postFromCreateAt(3),
"post_e": postFromCreateAt(4),
},
Order: []string{"post_e", "post_d", "post_c", "post_b", "post_a"},
}
bounds := getTimeSortedPostAccessibleBounds(2, len(pl.Posts), getPostListCreateAtFunc(pl))
require.Equal(t, accessibleBounds{start: 0, end: 2}, bounds)
})
t.Run("posts-slice: odd number of posts and reverse time selects right boundaries", func(t *testing.T) {
posts := []*model.Post{postFromCreateAt(4), postFromCreateAt(3), postFromCreateAt(2), postFromCreateAt(1), postFromCreateAt(0)}
bounds := getTimeSortedPostAccessibleBounds(2, len(posts), func(i int) int64 { return posts[i].CreateAt })
require.Equal(t, accessibleBounds{start: 0, end: 2}, bounds)
})
}
func TestFilterInaccessiblePosts(t *testing.T) {
mainHelper.Parallel(t)
th := Setup(t)
// Set up license with PostHistory limits to enable post filtering
cloudLicenseWithLimits := model.NewTestLicense("cloud")
cloudLicenseWithLimits.Limits = &model.LicenseLimits{PostHistory: 100}
th.App.Srv().SetLicense(cloudLicenseWithLimits)
err := th.App.Srv().Store().System().Save(&model.System{
Name: model.SystemLastAccessiblePostTime,
Value: "2",
})
require.NoError(t, err)
postFromCreateAt := func(at int64) *model.Post {
return &model.Post{CreateAt: at}
}
t.Run("ascending order returns correct posts", func(t *testing.T) {
postList := &model.PostList{
Posts: map[string]*model.Post{
"post_a": postFromCreateAt(0),
"post_b": postFromCreateAt(1),
"post_c": postFromCreateAt(2),
"post_d": postFromCreateAt(3),
"post_e": postFromCreateAt(4),
},
Order: []string{"post_a", "post_b", "post_c", "post_d", "post_e"},
}
appErr := th.App.filterInaccessiblePosts(postList, filterPostOptions{assumeSortedCreatedAt: true})
require.Nil(t, appErr)
assert.Equal(t, map[string]*model.Post{
"post_c": postFromCreateAt(2),
"post_d": postFromCreateAt(3),
"post_e": postFromCreateAt(4),
}, postList.Posts)
assert.Equal(t, []string{
"post_c",
"post_d",
"post_e",
}, postList.Order)
assert.Equal(t, int64(1), postList.FirstInaccessiblePostTime)
})
t.Run("descending order returns correct posts", func(t *testing.T) {
postList := &model.PostList{
Posts: map[string]*model.Post{
"post_a": postFromCreateAt(0),
"post_b": postFromCreateAt(1),
"post_c": postFromCreateAt(2),
"post_d": postFromCreateAt(3),
"post_e": postFromCreateAt(4),
},
Order: []string{"post_e", "post_d", "post_c", "post_b", "post_a"},
}
appErr := th.App.filterInaccessiblePosts(postList, filterPostOptions{assumeSortedCreatedAt: true})
require.Nil(t, appErr)
assert.Equal(t, map[string]*model.Post{
"post_c": postFromCreateAt(2),
"post_d": postFromCreateAt(3),
"post_e": postFromCreateAt(4),
}, postList.Posts)
assert.Equal(t, []string{
"post_e",
"post_d",
"post_c",
}, postList.Order)
assert.Equal(t, int64(1), postList.FirstInaccessiblePostTime)
})
t.Run("handles mixed create at ordering correctly if correct options given", func(t *testing.T) {
postList := &model.PostList{
Posts: map[string]*model.Post{
"post_a": postFromCreateAt(0),
"post_b": postFromCreateAt(1),
"post_c": postFromCreateAt(2),
"post_d": postFromCreateAt(3),
"post_e": postFromCreateAt(4),
},
Order: []string{"post_e", "post_b", "post_a", "post_d", "post_c"},
}
appErr := th.App.filterInaccessiblePosts(postList, filterPostOptions{assumeSortedCreatedAt: false})
require.Nil(t, appErr)
assert.Equal(t, map[string]*model.Post{
"post_c": postFromCreateAt(2),
"post_d": postFromCreateAt(3),
"post_e": postFromCreateAt(4),
}, postList.Posts)
assert.Equal(t, []string{
"post_e",
"post_d",
"post_c",
}, postList.Order)
})
t.Run("handles posts missing from order when doing linear search", func(t *testing.T) {
postList := &model.PostList{
Posts: map[string]*model.Post{
"post_a": postFromCreateAt(0),
"post_b": postFromCreateAt(1),
"post_c": postFromCreateAt(1),
"post_d": postFromCreateAt(3),
"post_e": postFromCreateAt(4),
},
Order: []string{"post_e", "post_a", "post_d", "post_b"},
}
appErr := th.App.filterInaccessiblePosts(postList, filterPostOptions{assumeSortedCreatedAt: false})
require.Nil(t, appErr)
assert.Equal(t, map[string]*model.Post{
"post_d": postFromCreateAt(3),
"post_e": postFromCreateAt(4),
}, postList.Posts)
assert.Equal(t, []string{
"post_e",
"post_d",
}, postList.Order)
})
}
func TestGetFilteredAccessiblePosts(t *testing.T) {
mainHelper.Parallel(t)
th := Setup(t)
entryLicenseWithLimits := model.NewTestLicenseSKU(model.LicenseShortSkuMattermostEntry)
entryLicenseWithLimits.Limits = &model.LicenseLimits{PostHistory: 100}
th.App.Srv().SetLicense(entryLicenseWithLimits)
err := th.App.Srv().Store().System().Save(&model.System{
Name: model.SystemLastAccessiblePostTime,
Value: "2",
})
require.NoError(t, err)
postFromCreateAt := func(at int64) *model.Post {
return &model.Post{CreateAt: at}
}
t.Run("ascending order returns correct posts", func(t *testing.T) {
posts := []*model.Post{postFromCreateAt(0), postFromCreateAt(1), postFromCreateAt(2), postFromCreateAt(3), postFromCreateAt(4)}
filteredPosts, firstInaccessiblePostTime, appErr := th.App.getFilteredAccessiblePosts(posts, filterPostOptions{assumeSortedCreatedAt: true})
assert.Nil(t, appErr)
assert.Equal(t, []*model.Post{postFromCreateAt(2), postFromCreateAt(3), postFromCreateAt(4)}, filteredPosts)
assert.Equal(t, int64(1), firstInaccessiblePostTime)
})
t.Run("descending order returns correct posts", func(t *testing.T) {
posts := []*model.Post{postFromCreateAt(4), postFromCreateAt(3), postFromCreateAt(2), postFromCreateAt(1), postFromCreateAt(0)}
filteredPosts, firstInaccessiblePostTime, appErr := th.App.getFilteredAccessiblePosts(posts, filterPostOptions{assumeSortedCreatedAt: true})
assert.Nil(t, appErr)
assert.Equal(t, []*model.Post{postFromCreateAt(4), postFromCreateAt(3), postFromCreateAt(2)}, filteredPosts)
assert.Equal(t, int64(1), firstInaccessiblePostTime)
})
t.Run("handles mixed create at ordering correctly if correct options given", func(t *testing.T) {
posts := []*model.Post{postFromCreateAt(4), postFromCreateAt(1), postFromCreateAt(0), postFromCreateAt(3), postFromCreateAt(2)}
filteredPosts, _, appErr := th.App.getFilteredAccessiblePosts(posts, filterPostOptions{assumeSortedCreatedAt: false})
assert.Nil(t, appErr)
assert.Equal(t, []*model.Post{postFromCreateAt(4), postFromCreateAt(3), postFromCreateAt(2)}, filteredPosts)
})
}
func TestIsInaccessiblePost(t *testing.T) {
mainHelper.Parallel(t)
th := Setup(t)
// Set up license with PostHistory limits to enable post filtering
entryLicenseWithLimits := model.NewTestLicenseSKU(model.LicenseShortSkuMattermostEntry)
entryLicenseWithLimits.Limits = &model.LicenseLimits{PostHistory: 100}
th.App.Srv().SetLicense(entryLicenseWithLimits)
err := th.App.Srv().Store().System().Save(&model.System{
Name: model.SystemLastAccessiblePostTime,
Value: "2",
})
require.NoError(t, err)
post := &model.Post{CreateAt: 3}
firstInaccessiblePostTime, appErr := th.App.isInaccessiblePost(post)
assert.Nil(t, appErr)
assert.Equal(t, int64(0), firstInaccessiblePostTime)
post = &model.Post{CreateAt: 1}
firstInaccessiblePostTime, appErr = th.App.isInaccessiblePost(post)
assert.Nil(t, appErr)
assert.Equal(t, int64(1), firstInaccessiblePostTime)
}
func Test_getInaccessibleRange(t *testing.T) {
mainHelper.Parallel(t)
type test struct {
label string
bounds accessibleBounds
listLength int
expectedStart int
expectedEnd int
}
tests := []test{
{
label: "inaccessible at end",
bounds: accessibleBounds{start: 0, end: 3},
listLength: 6,
expectedStart: 4,
expectedEnd: 5,
},
}
for _, test := range tests {
t.Run(test.label, func(t *testing.T) {
start, end := test.bounds.getInaccessibleRange(test.listLength)
assert.Equal(t, test.expectedStart, start)
assert.Equal(t, test.expectedEnd, end)
})
}
}
func TestRevealBurnOnReadPostsForUser(t *testing.T) {
th := Setup(t).InitBasic(t)
// Enable BurnOnRead feature
th.App.Srv().SetLicense(model.NewTestLicenseSKU(model.LicenseShortSkuEnterpriseAdvanced))
th.App.UpdateConfig(func(cfg *model.Config) {
cfg.FeatureFlags.BurnOnRead = true
cfg.ServiceSettings.EnableBurnOnRead = model.NewPointer(true)
})
t.Run("skips deleted burn-on-read post", func(t *testing.T) {
deletedPost := &model.Post{
Id: model.NewId(),
UserId: th.BasicUser.Id,
ChannelId: th.BasicChannel.Id,
Message: "deleted burn on read message",
Type: model.PostTypeBurnOnRead,
DeleteAt: model.GetMillis(),
CreateAt: model.GetMillis(),
}
postList := model.NewPostList()
postList.AddPost(deletedPost)
resultList, appErr := th.App.revealBurnOnReadPostsForUser(th.Context, postList, th.BasicUser2.Id)
require.Nil(t, appErr)
require.NotNil(t, resultList)
// The deleted post should remain in BurnOnReadPosts but not be processed
assert.Contains(t, resultList.BurnOnReadPosts, deletedPost.Id)
// Verify the post was not modified (still has DeleteAt set)
assert.Equal(t, deletedPost.DeleteAt, resultList.BurnOnReadPosts[deletedPost.Id].DeleteAt)
assert.Equal(t, deletedPost.Message, resultList.BurnOnReadPosts[deletedPost.Id].Message)
})
}