mirror of
https://github.com/mattermost/mattermost.git
synced 2026-03-06 23:31:18 -05:00
Some checks are pending
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
* Add read receipt store for burn on read message types * update mocks * fix invalidation target * have consistent case on index creation * Add temporary posts table * add mock * add transaction support * reflect review comments * wip: Add reveal endpoint * user check error id instead * wip: Add ws events and cleanup for burn on read posts * add burn endpoint for explicitly burning messages * add translations * Added logic to associate files of BoR post with the post * Added test * fixes * disable pinning posts and review comments * MM-66594 - Burn on read UI integration (#34647) * MM-66244 - add BoR visual components to message editor * MM-66246 - BoR visual indicator for sender and receiver * MM-66607 - bor - add timer countdown and autodeletion * add the system console max time to live config * use the max expire at and create global scheduler to register bor messages * use seconds for BoR config values in BE * implement the read by text shown in the tooltip logic * unestack the posts from same receiver and BoR and fix styling * avoid opening reply RHS * remove unused dispatchers * persis the BoR label in the drafts * move expiration value to metadata * adjust unit tests to metadata insted of props * code clean up and some performance improvements; add period grace for deletion too * adjust migration serie number * hide bor messages when config is off * performance improvements on post component and code clean up * keep bor existing post functionality if config is disabled * Add read receipt store for burn on read message types * Add temporary posts table * add transaction support * reflect review comments * wip: Add reveal endpoint * user check error id instead * wip: Add ws events and cleanup for burn on read posts * avoid reacting to unrevealed bor messages * adjust migration number * Add read receipt store for burn on read message types * have consistent case on index creation * Add temporary posts table * add mock * add transaction support * reflect review comments * wip: Add reveal endpoint * user check error id instead * wip: Add ws events and cleanup for burn on read posts * add burn endpoint for explicitly burning messages * adjust post reveal and type with backend changes * use real config values, adjust icon usage and style * adjust the delete from from sender and receiver * improve self deleting logic by placing in badge, use burn endpoint * adjust websocket events handling for the read by sender label information * adjust styling for concealed and error state * update burn-on-read post event handling for improved recipient tracking and multi-device sync * replace burn_on_read with type in database migrations and model * remove burn_on_read metadata from PostMetadata and related structures * Added logic to associate files of BoR post with the post * Added test * adjust migration name and fix linter * Add read receipt store for burn on read message types * update mocks * have consistent case on index creation * Add temporary posts table * add mock * add transaction support * reflect review comments * wip: Add reveal endpoint * user check error id instead * wip: Add ws events and cleanup for burn on read posts * add burn endpoint for explicitly burning messages * Added logic to associate files of BoR post with the post * Added test * disable pinning posts and review comments * show attachment on bor reveal * remove unused translation * Enhance burn-on-read post handling and refine previous post ID retrieval logic * adjust the returning chunk to work with bor messages * read temp post from master db * read from master * show the copy link button to the sender * revert unnecessary check * restore correct json tag * remove unused error handling and clarify burn-on-read comment * improve type safety and use proper selectors * eliminate code duplication in deletion handler * optimize performance and add documentation * delete bor message for sender once all receivers reveal it * add burn on read to scheduled posts * add feature enable check * use master to avoid all read recipients race condition --------- Co-authored-by: Mattermost Build <build@mattermost.com> Co-authored-by: Ibrahim Serdar Acikgoz <serdaracikgoz86@gmail.com> Co-authored-by: Harshil Sharma <harshilsharma63@gmail.com> * squash migrations into single file * add configuration for the scheduler * don't run messagehasbeenposted hook * remove parallel tests on burn on read * add clean up for closing opened modals from previous tests * simplify delete menu item rendering * add cleanup step to close open modals after each test to prevent pollution * streamline delete button visibility logic for Burn on Read posts * improve reliability of closing post menu and modals by using body ESC key --------- Co-authored-by: Harshil Sharma <harshilsharma63@gmail.com> Co-authored-by: Pablo Vélez <pablovv2012@gmail.com> Co-authored-by: Mattermost Build <build@mattermost.com>
207 lines
7 KiB
Go
207 lines
7 KiB
Go
// Copyright (c) 2015-present Mattermost, Inc. All Rights Reserved.
|
|
// See LICENSE.txt for license information.
|
|
|
|
package searchlayer
|
|
|
|
import (
|
|
"github.com/pkg/errors"
|
|
|
|
"github.com/mattermost/mattermost/server/public/model"
|
|
"github.com/mattermost/mattermost/server/public/shared/mlog"
|
|
"github.com/mattermost/mattermost/server/public/shared/request"
|
|
"github.com/mattermost/mattermost/server/v8/channels/store"
|
|
"github.com/mattermost/mattermost/server/v8/platform/services/searchengine"
|
|
)
|
|
|
|
type SearchPostStore struct {
|
|
store.PostStore
|
|
rootStore *SearchStore
|
|
}
|
|
|
|
func (s SearchPostStore) indexPost(rctx request.CTX, post *model.Post) {
|
|
for _, engine := range s.rootStore.searchEngine.GetActiveEngines() {
|
|
if engine.IsIndexingEnabled() {
|
|
runIndexFn(rctx, engine, func(engineCopy searchengine.SearchEngineInterface) {
|
|
if post.Type == model.PostTypeBurnOnRead {
|
|
return
|
|
}
|
|
channel, chanErr := s.rootStore.Channel().Get(post.ChannelId, true)
|
|
if chanErr != nil {
|
|
rctx.Logger().Error("Couldn't get channel for post for SearchEngine indexing.", mlog.String("channel_id", post.ChannelId), mlog.String("search_engine", engineCopy.GetName()), mlog.String("post_id", post.Id), mlog.Err(chanErr))
|
|
return
|
|
}
|
|
if err := engineCopy.IndexPost(post, channel.TeamId); err != nil {
|
|
rctx.Logger().Warn("Encountered error indexing post", mlog.String("post_id", post.Id), mlog.String("search_engine", engineCopy.GetName()), mlog.Err(err))
|
|
return
|
|
}
|
|
})
|
|
}
|
|
}
|
|
}
|
|
|
|
func (s SearchPostStore) deletePostIndex(rctx request.CTX, post *model.Post) {
|
|
for _, engine := range s.rootStore.searchEngine.GetActiveEngines() {
|
|
if engine.IsIndexingEnabled() {
|
|
runIndexFn(rctx, engine, func(engineCopy searchengine.SearchEngineInterface) {
|
|
if err := engineCopy.DeletePost(post); err != nil {
|
|
rctx.Logger().Warn("Encountered error deleting post", mlog.String("post_id", post.Id), mlog.String("search_engine", engineCopy.GetName()), mlog.Err(err))
|
|
return
|
|
}
|
|
})
|
|
}
|
|
}
|
|
}
|
|
|
|
func (s SearchPostStore) deleteChannelPostsIndex(rctx request.CTX, channelID string) {
|
|
for _, engine := range s.rootStore.searchEngine.GetActiveEngines() {
|
|
if engine.IsIndexingEnabled() {
|
|
runIndexFn(rctx, engine, func(engineCopy searchengine.SearchEngineInterface) {
|
|
if err := engineCopy.DeleteChannelPosts(rctx, channelID); err != nil {
|
|
rctx.Logger().Warn("Encountered error deleting channel posts", mlog.String("channel_id", channelID), mlog.String("search_engine", engineCopy.GetName()), mlog.Err(err))
|
|
return
|
|
}
|
|
rctx.Logger().Debug("Removed all channel posts from the index in search engine", mlog.String("channel_id", channelID), mlog.String("search_engine", engineCopy.GetName()))
|
|
})
|
|
}
|
|
}
|
|
}
|
|
|
|
func (s SearchPostStore) deleteUserPostsIndex(rctx request.CTX, userID string) {
|
|
for _, engine := range s.rootStore.searchEngine.GetActiveEngines() {
|
|
if engine.IsIndexingEnabled() {
|
|
runIndexFn(rctx, engine, func(engineCopy searchengine.SearchEngineInterface) {
|
|
if err := engineCopy.DeleteUserPosts(rctx, userID); err != nil {
|
|
rctx.Logger().Warn("Encountered error deleting user posts", mlog.String("user_id", userID), mlog.String("search_engine", engineCopy.GetName()), mlog.Err(err))
|
|
return
|
|
}
|
|
rctx.Logger().Debug("Removed all user posts from the index in search engine", mlog.String("user_id", userID), mlog.String("search_engine", engineCopy.GetName()))
|
|
})
|
|
}
|
|
}
|
|
}
|
|
|
|
func (s SearchPostStore) Update(rctx request.CTX, newPost, oldPost *model.Post) (*model.Post, error) {
|
|
post, err := s.PostStore.Update(rctx, newPost, oldPost)
|
|
|
|
if err == nil {
|
|
s.indexPost(rctx, post)
|
|
}
|
|
return post, err
|
|
}
|
|
|
|
func (s *SearchPostStore) Overwrite(rctx request.CTX, post *model.Post) (*model.Post, error) {
|
|
post, err := s.PostStore.Overwrite(rctx, post)
|
|
if err == nil {
|
|
s.indexPost(rctx, post)
|
|
}
|
|
return post, err
|
|
}
|
|
|
|
func (s SearchPostStore) Save(rctx request.CTX, post *model.Post) (*model.Post, error) {
|
|
npost, err := s.PostStore.Save(rctx, post)
|
|
|
|
if err == nil {
|
|
s.indexPost(rctx, npost)
|
|
}
|
|
return npost, err
|
|
}
|
|
|
|
func (s SearchPostStore) Delete(rctx request.CTX, postId string, date int64, deletedByID string) error {
|
|
err := s.PostStore.Delete(rctx, postId, date, deletedByID)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
post, err := s.PostStore.GetSingle(rctx, postId, true)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
s.deletePostIndex(rctx, post)
|
|
return nil
|
|
}
|
|
|
|
func (s SearchPostStore) PermanentDelete(rctx request.CTX, postID string) error {
|
|
// Get full post struct for later
|
|
post, err := s.PostStore.GetSingle(rctx, postID, true)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
err = s.PostStore.PermanentDelete(rctx, postID)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
s.deletePostIndex(rctx, post)
|
|
return nil
|
|
}
|
|
|
|
func (s SearchPostStore) PermanentDeleteByUser(rctx request.CTX, userID string) error {
|
|
err := s.PostStore.PermanentDeleteByUser(rctx, userID)
|
|
if err == nil {
|
|
s.deleteUserPostsIndex(rctx, userID)
|
|
}
|
|
return err
|
|
}
|
|
|
|
func (s SearchPostStore) PermanentDeleteByChannel(rctx request.CTX, channelID string) error {
|
|
err := s.PostStore.PermanentDeleteByChannel(rctx, channelID)
|
|
if err == nil {
|
|
s.deleteChannelPostsIndex(rctx, channelID)
|
|
}
|
|
return err
|
|
}
|
|
|
|
func (s SearchPostStore) searchPostsForUserByEngine(engine searchengine.SearchEngineInterface, paramsList []*model.SearchParams, userId, teamId string, page, perPage int) (*model.PostSearchResults, error) {
|
|
if err := model.IsSearchParamsListValid(paramsList); err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
// We only allow the user to search in channels they are a member of.
|
|
userChannels, err2 := s.rootStore.Channel().GetChannels(teamId, userId,
|
|
&model.ChannelSearchOpts{
|
|
IncludeDeleted: paramsList[0].IncludeDeletedChannels,
|
|
LastDeleteAt: 0,
|
|
})
|
|
if err2 != nil {
|
|
return nil, errors.Wrap(err2, "error getting channel for user")
|
|
}
|
|
|
|
postIds, matches, err := engine.SearchPosts(userChannels, paramsList, page, perPage)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
// Get the posts
|
|
postList := model.NewPostList()
|
|
if len(postIds) > 0 {
|
|
posts, err := s.PostStore.GetPostsByIds(postIds)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
for _, p := range posts {
|
|
if p.DeleteAt == 0 {
|
|
postList.AddPost(p)
|
|
postList.AddOrder(p.Id)
|
|
}
|
|
}
|
|
}
|
|
|
|
return model.MakePostSearchResults(postList, matches), nil
|
|
}
|
|
|
|
func (s SearchPostStore) SearchPostsForUser(rctx request.CTX, paramsList []*model.SearchParams, userId, teamId string, page, perPage int) (*model.PostSearchResults, error) {
|
|
for _, engine := range s.rootStore.searchEngine.GetActiveEngines() {
|
|
if engine.IsSearchEnabled() {
|
|
results, err := s.searchPostsForUserByEngine(engine, paramsList, userId, teamId, page, perPage)
|
|
if err != nil {
|
|
rctx.Logger().Warn("Encountered error on SearchPostsInTeamForUser.", mlog.String("search_engine", engine.GetName()), mlog.Err(err))
|
|
continue
|
|
}
|
|
return results, err
|
|
}
|
|
}
|
|
|
|
if *s.rootStore.getConfig().SqlSettings.DisableDatabaseSearch {
|
|
return &model.PostSearchResults{PostList: model.NewPostList(), Matches: model.PostSearchMatches{}}, nil
|
|
}
|
|
|
|
return s.PostStore.SearchPostsForUser(rctx, paramsList, userId, teamId, page, perPage)
|
|
}
|