mirror of
https://github.com/mattermost/mattermost.git
synced 2026-02-03 20:40:00 -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>
241 lines
5.2 KiB
Go
241 lines
5.2 KiB
Go
// Copyright (c) 2015-present Mattermost, Inc. All Rights Reserved.
|
|
// See LICENSE.txt for license information.
|
|
|
|
package model
|
|
|
|
import (
|
|
"encoding/json"
|
|
"io"
|
|
"sort"
|
|
)
|
|
|
|
type PostList struct {
|
|
Order []string `json:"order"`
|
|
Posts map[string]*Post `json:"posts"`
|
|
NextPostId string `json:"next_post_id"`
|
|
PrevPostId string `json:"prev_post_id"`
|
|
// HasNext indicates whether there are more items to be fetched or not.
|
|
HasNext *bool `json:"has_next,omitempty"`
|
|
// If there are inaccessible posts, FirstInaccessiblePostTime is the time of the latest inaccessible post
|
|
FirstInaccessiblePostTime int64 `json:"first_inaccessible_post_time"`
|
|
// HasBurnOnRead indicates whether there are any burn on read posts in the list
|
|
// this is not sent to the client
|
|
BurnOnReadPosts map[string]*Post `json:"-"`
|
|
}
|
|
|
|
func NewPostList() *PostList {
|
|
return &PostList{
|
|
Order: make([]string, 0),
|
|
Posts: make(map[string]*Post),
|
|
NextPostId: "",
|
|
PrevPostId: "",
|
|
BurnOnReadPosts: make(map[string]*Post),
|
|
}
|
|
}
|
|
|
|
func (o *PostList) Clone() *PostList {
|
|
orderCopy := make([]string, len(o.Order))
|
|
postsCopy := make(map[string]*Post)
|
|
copy(orderCopy, o.Order)
|
|
for k, v := range o.Posts {
|
|
postsCopy[k] = v.Clone()
|
|
}
|
|
burnOnReadPostsCopy := make(map[string]*Post)
|
|
for k, v := range o.BurnOnReadPosts {
|
|
burnOnReadPostsCopy[k] = v.Clone()
|
|
}
|
|
return &PostList{
|
|
Order: orderCopy,
|
|
Posts: postsCopy,
|
|
NextPostId: o.NextPostId,
|
|
PrevPostId: o.PrevPostId,
|
|
HasNext: o.HasNext,
|
|
FirstInaccessiblePostTime: o.FirstInaccessiblePostTime,
|
|
BurnOnReadPosts: burnOnReadPostsCopy,
|
|
}
|
|
}
|
|
|
|
func (o *PostList) ForPlugin() *PostList {
|
|
plCopy := o.Clone()
|
|
for k, p := range plCopy.Posts {
|
|
plCopy.Posts[k] = p.ForPlugin()
|
|
}
|
|
return plCopy
|
|
}
|
|
|
|
func (o *PostList) ToSlice() []*Post {
|
|
var posts []*Post
|
|
|
|
if l := len(o.Posts); l > 0 {
|
|
posts = make([]*Post, 0, l)
|
|
}
|
|
|
|
for _, id := range o.Order {
|
|
posts = append(posts, o.Posts[id])
|
|
}
|
|
return posts
|
|
}
|
|
|
|
func (o *PostList) WithRewrittenImageURLs(f func(string) string) *PostList {
|
|
plCopy := *o
|
|
plCopy.Posts = make(map[string]*Post)
|
|
for id, post := range o.Posts {
|
|
plCopy.Posts[id] = post.WithRewrittenImageURLs(f)
|
|
}
|
|
return &plCopy
|
|
}
|
|
|
|
func (o *PostList) StripActionIntegrations() {
|
|
posts := o.Posts
|
|
o.Posts = make(map[string]*Post)
|
|
for id, post := range posts {
|
|
pcopy := post.Clone()
|
|
pcopy.StripActionIntegrations()
|
|
o.Posts[id] = pcopy
|
|
}
|
|
}
|
|
|
|
func (o *PostList) ToJSON() (string, error) {
|
|
plCopy := *o
|
|
plCopy.StripActionIntegrations()
|
|
b, err := json.Marshal(&plCopy)
|
|
return string(b), err
|
|
}
|
|
|
|
func (o *PostList) EncodeJSON(w io.Writer) error {
|
|
o.StripActionIntegrations()
|
|
return json.NewEncoder(w).Encode(o)
|
|
}
|
|
|
|
func (o *PostList) MakeNonNil() {
|
|
if o.Order == nil {
|
|
o.Order = make([]string, 0)
|
|
}
|
|
|
|
if o.Posts == nil {
|
|
o.Posts = make(map[string]*Post)
|
|
}
|
|
|
|
for _, v := range o.Posts {
|
|
v.MakeNonNil()
|
|
}
|
|
}
|
|
|
|
func (o *PostList) AddOrder(id string) {
|
|
if o.Order == nil {
|
|
o.Order = make([]string, 0, 128)
|
|
}
|
|
|
|
o.Order = append(o.Order, id)
|
|
}
|
|
|
|
func (o *PostList) AddPost(post *Post) {
|
|
if o.Posts == nil {
|
|
o.Posts = make(map[string]*Post)
|
|
}
|
|
|
|
if post.Type == PostTypeBurnOnRead {
|
|
o.BurnOnReadPosts[post.Id] = post
|
|
}
|
|
|
|
o.Posts[post.Id] = post
|
|
}
|
|
|
|
func (o *PostList) UniqueOrder() {
|
|
keys := make(map[string]bool)
|
|
order := []string{}
|
|
for _, postId := range o.Order {
|
|
if _, value := keys[postId]; !value {
|
|
keys[postId] = true
|
|
order = append(order, postId)
|
|
}
|
|
}
|
|
|
|
o.Order = order
|
|
}
|
|
|
|
func (o *PostList) Extend(other *PostList) {
|
|
for postId := range other.Posts {
|
|
o.AddPost(other.Posts[postId])
|
|
}
|
|
|
|
for _, postId := range other.Order {
|
|
o.AddOrder(postId)
|
|
}
|
|
|
|
o.UniqueOrder()
|
|
}
|
|
|
|
func (o *PostList) SortByCreateAt() {
|
|
sort.Slice(o.Order, func(i, j int) bool {
|
|
return o.Posts[o.Order[i]].CreateAt > o.Posts[o.Order[j]].CreateAt
|
|
})
|
|
}
|
|
|
|
func (o *PostList) Etag() string {
|
|
id := "0"
|
|
var t int64
|
|
|
|
for _, v := range o.Posts {
|
|
if v.UpdateAt > t {
|
|
t = v.UpdateAt
|
|
id = v.Id
|
|
} else if v.UpdateAt == t && v.Id > id {
|
|
t = v.UpdateAt
|
|
id = v.Id
|
|
}
|
|
}
|
|
|
|
orderId := ""
|
|
if len(o.Order) > 0 {
|
|
orderId = o.Order[0]
|
|
}
|
|
|
|
return Etag(orderId, id, t)
|
|
}
|
|
|
|
func (o *PostList) IsChannelId(channelId string) bool {
|
|
for _, v := range o.Posts {
|
|
if v.ChannelId != channelId {
|
|
return false
|
|
}
|
|
}
|
|
|
|
return true
|
|
}
|
|
|
|
func (o *PostList) BuildWranglerPostList() *WranglerPostList {
|
|
wpl := &WranglerPostList{}
|
|
|
|
o.UniqueOrder()
|
|
o.SortByCreateAt()
|
|
posts := o.ToSlice()
|
|
|
|
if len(posts) == 0 {
|
|
// Something was sorted wrong or an empty PostList was provided.
|
|
return wpl
|
|
}
|
|
|
|
// A separate ID key map to ensure no duplicates.
|
|
idKeys := make(map[string]bool)
|
|
|
|
for i := range posts {
|
|
p := posts[len(posts)-i-1]
|
|
|
|
// Add UserID to metadata if it's new.
|
|
if _, ok := idKeys[p.UserId]; !ok {
|
|
idKeys[p.UserId] = true
|
|
wpl.ThreadUserIDs = append(wpl.ThreadUserIDs, p.UserId)
|
|
}
|
|
|
|
wpl.FileAttachmentCount += int64(len(p.FileIds))
|
|
|
|
wpl.Posts = append(wpl.Posts, p)
|
|
}
|
|
|
|
// Set metadata for earliest and latest posts
|
|
wpl.EarlistPostTimestamp = wpl.RootPost().CreateAt
|
|
wpl.LatestPostTimestamp = wpl.Posts[wpl.NumPosts()-1].CreateAt
|
|
|
|
return wpl
|
|
}
|