Full Mattermost server source with integrated Community Enterprise features. Includes vendor directory for offline/air-gapped builds. Structure: - enterprise-impl/: Enterprise feature implementations - enterprise-community/: Init files that register implementations - enterprise/: Bridge imports (community_imports.go) - vendor/: All dependencies for offline builds Build (online): go build ./cmd/mattermost Build (offline/air-gapped): go build -mod=vendor ./cmd/mattermost 🤖 Generated with [Claude Code](https://claude.com/claude-code) Co-Authored-By: Claude Opus 4.5 <noreply@anthropic.com>
229 lines
4.7 KiB
Go
229 lines
4.7 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"`
|
|
}
|
|
|
|
func NewPostList() *PostList {
|
|
return &PostList{
|
|
Order: make([]string, 0),
|
|
Posts: make(map[string]*Post),
|
|
NextPostId: "",
|
|
PrevPostId: "",
|
|
}
|
|
}
|
|
|
|
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()
|
|
}
|
|
return &PostList{
|
|
Order: orderCopy,
|
|
Posts: postsCopy,
|
|
NextPostId: o.NextPostId,
|
|
PrevPostId: o.PrevPostId,
|
|
HasNext: o.HasNext,
|
|
FirstInaccessiblePostTime: o.FirstInaccessiblePostTime,
|
|
}
|
|
}
|
|
|
|
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)
|
|
}
|
|
|
|
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
|
|
}
|