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>
120 lines
3.9 KiB
Go
120 lines
3.9 KiB
Go
// Copyright (c) 2015-present Mattermost, Inc. All Rights Reserved.
|
|
// See LICENSE.txt for license information.
|
|
|
|
package model
|
|
|
|
import (
|
|
"encoding/json"
|
|
"net/http"
|
|
"unicode/utf8"
|
|
|
|
"github.com/pkg/errors"
|
|
)
|
|
|
|
const (
|
|
PropertyValueTargetIDMaxRunes = 255
|
|
PropertyValueTargetTypeMaxRunes = 255
|
|
|
|
PropertyValueTargetTypePost = "post"
|
|
PropertyValueTargetTypeUser = "user"
|
|
)
|
|
|
|
type PropertyValue struct {
|
|
ID string `json:"id"`
|
|
TargetID string `json:"target_id"`
|
|
TargetType string `json:"target_type"`
|
|
GroupID string `json:"group_id"`
|
|
FieldID string `json:"field_id"`
|
|
Value json.RawMessage `json:"value"`
|
|
CreateAt int64 `json:"create_at"`
|
|
UpdateAt int64 `json:"update_at"`
|
|
DeleteAt int64 `json:"delete_at"`
|
|
}
|
|
|
|
func (pv *PropertyValue) PreSave() {
|
|
if pv.ID == "" {
|
|
pv.ID = NewId()
|
|
}
|
|
|
|
if pv.CreateAt == 0 {
|
|
pv.CreateAt = GetMillis()
|
|
}
|
|
pv.UpdateAt = pv.CreateAt
|
|
}
|
|
|
|
func (pv *PropertyValue) IsValid() error {
|
|
if !IsValidId(pv.ID) {
|
|
return NewAppError("PropertyValue.IsValid", "model.property_value.is_valid.app_error", map[string]any{"FieldName": "id", "Reason": "invalid id"}, "", http.StatusBadRequest)
|
|
}
|
|
|
|
if !IsValidId(pv.TargetID) {
|
|
return NewAppError("PropertyValue.IsValid", "model.property_value.is_valid.app_error", map[string]any{"FieldName": "target_id", "Reason": "invalid id"}, "id="+pv.ID, http.StatusBadRequest)
|
|
}
|
|
|
|
if pv.TargetType == "" {
|
|
return NewAppError("PropertyValue.IsValid", "model.property_value.is_valid.app_error", map[string]any{"FieldName": "target_type", "Reason": "value cannot be empty"}, "id="+pv.ID, http.StatusBadRequest)
|
|
}
|
|
|
|
if utf8.RuneCountInString(pv.TargetType) > PropertyValueTargetTypeMaxRunes {
|
|
return NewAppError("PropertyValue.IsValid", "model.property_value.is_valid.app_error", map[string]any{"FieldName": "target_type", "Reason": "value exceeds maximum length"}, "id="+pv.ID, http.StatusBadRequest)
|
|
}
|
|
|
|
if utf8.RuneCountInString(pv.TargetID) > PropertyValueTargetIDMaxRunes {
|
|
return NewAppError("PropertyValue.IsValid", "model.property_value.is_valid.app_error", map[string]any{"FieldName": "target_id", "Reason": "value exceeds maximum length"}, "id="+pv.ID, http.StatusBadRequest)
|
|
}
|
|
|
|
if !IsValidId(pv.GroupID) {
|
|
return NewAppError("PropertyValue.IsValid", "model.property_value.is_valid.app_error", map[string]any{"FieldName": "group_id", "Reason": "invalid id"}, "id="+pv.ID, http.StatusBadRequest)
|
|
}
|
|
|
|
if !IsValidId(pv.FieldID) {
|
|
return NewAppError("PropertyValue.IsValid", "model.property_value.is_valid.app_error", map[string]any{"FieldName": "field_id", "Reason": "invalid id"}, "id="+pv.ID, http.StatusBadRequest)
|
|
}
|
|
|
|
if pv.CreateAt == 0 {
|
|
return NewAppError("PropertyValue.IsValid", "model.property_value.is_valid.app_error", map[string]any{"FieldName": "create_at", "Reason": "value cannot be zero"}, "id="+pv.ID, http.StatusBadRequest)
|
|
}
|
|
|
|
if pv.UpdateAt == 0 {
|
|
return NewAppError("PropertyValue.IsValid", "model.property_value.is_valid.app_error", map[string]any{"FieldName": "update_at", "Reason": "value cannot be zero"}, "id="+pv.ID, http.StatusBadRequest)
|
|
}
|
|
|
|
return nil
|
|
}
|
|
|
|
type PropertyValueSearchCursor struct {
|
|
PropertyValueID string
|
|
CreateAt int64
|
|
}
|
|
|
|
func (p PropertyValueSearchCursor) IsEmpty() bool {
|
|
return p.PropertyValueID == "" && p.CreateAt == 0
|
|
}
|
|
|
|
func (p PropertyValueSearchCursor) IsValid() error {
|
|
if p.IsEmpty() {
|
|
return nil
|
|
}
|
|
|
|
if p.CreateAt <= 0 {
|
|
return errors.New("create at cannot be negative or zero")
|
|
}
|
|
|
|
if !IsValidId(p.PropertyValueID) {
|
|
return errors.New("property field id is invalid")
|
|
}
|
|
return nil
|
|
}
|
|
|
|
type PropertyValueSearchOpts struct {
|
|
GroupID string
|
|
TargetType string
|
|
TargetIDs []string
|
|
FieldID string
|
|
SinceUpdateAt int64 // UpdateAt after which to send the items
|
|
IncludeDeleted bool
|
|
Cursor PropertyValueSearchCursor
|
|
PerPage int
|
|
Value json.RawMessage
|
|
}
|