mirror of
https://github.com/mattermost/mattermost.git
synced 2025-02-25 18:55:24 -06:00
* allow `Wait()`ing on the supervisor In the event the plugin supervisor shuts down a plugin for crashing too many times, the new `Wait()` interface allows the `ActivatePlugin` to accept a callback function to trigger when `supervisor.Wait()` returns. If the supervisor shuts down normally, this callback is invoked with a nil error, otherwise any error reported by the supervisor is passed along. * improve plugin activation/deactivation logic Avoid triggering activation of previously failed-to-start plugins just becase something in the configuration changed. Now, intelligently compare the global enable bit as well as the each individual plugin's enabled bit. * expose store to manipulate PluginStatuses * expose API to fetch plugin statuses * keep track of whether or not plugin sandboxing is supported * transition plugin statuses * restore error on plugin activation if already active * don't initialize test plugins until successfully loaded * emit websocket events when plugin statuses change * skip pruning if already initialized * MM-8622: maintain plugin statuses in memory Switch away from persisting plugin statuses to the database, and maintain in memory instead. This will be followed by a cluster interface to query the in-memory status of plugin statuses from all cluster nodes. At the same time, rename `cluster_discovery_id` on the `PluginStatus` model object to `cluster_id`. * MM-8622: aggregate plugin statuses across cluster * fetch cluster plugin statuses when emitting websocket notification * address unit test fixes after rebasing * relax (poor) racey unit test re: supervisor.Wait() * make store-mocks
134 lines
2.8 KiB
Go
134 lines
2.8 KiB
Go
// Copyright (c) 2017-present Mattermost, Inc. All Rights Reserved.
|
|
// See License.txt for license information.
|
|
|
|
package model
|
|
|
|
import (
|
|
"encoding/json"
|
|
"io"
|
|
"net/http"
|
|
"os"
|
|
)
|
|
|
|
const (
|
|
CDS_OFFLINE_AFTER_MILLIS = 1000 * 60 * 30 // 30 minutes
|
|
CDS_TYPE_APP = "mattermost_app"
|
|
)
|
|
|
|
type ClusterDiscovery struct {
|
|
Id string `json:"id"`
|
|
Type string `json:"type"`
|
|
ClusterName string `json:"cluster_name"`
|
|
Hostname string `json:"hostname"`
|
|
GossipPort int32 `json:"gossip_port"`
|
|
Port int32 `json:"port"`
|
|
CreateAt int64 `json:"create_at"`
|
|
LastPingAt int64 `json:"last_ping_at"`
|
|
}
|
|
|
|
func (o *ClusterDiscovery) PreSave() {
|
|
if o.Id == "" {
|
|
o.Id = NewId()
|
|
}
|
|
|
|
if o.CreateAt == 0 {
|
|
o.CreateAt = GetMillis()
|
|
o.LastPingAt = o.CreateAt
|
|
}
|
|
}
|
|
|
|
func (o *ClusterDiscovery) AutoFillHostname() {
|
|
// attempt to set the hostname from the OS
|
|
if len(o.Hostname) == 0 {
|
|
if hn, err := os.Hostname(); err == nil {
|
|
o.Hostname = hn
|
|
}
|
|
}
|
|
}
|
|
|
|
func (o *ClusterDiscovery) AutoFillIpAddress() {
|
|
// attempt to set the hostname to the first non-local IP address
|
|
if len(o.Hostname) == 0 {
|
|
o.Hostname = GetServerIpAddress()
|
|
}
|
|
}
|
|
|
|
func (o *ClusterDiscovery) IsEqual(in *ClusterDiscovery) bool {
|
|
if in == nil {
|
|
return false
|
|
}
|
|
|
|
if o.Type != in.Type {
|
|
return false
|
|
}
|
|
|
|
if o.ClusterName != in.ClusterName {
|
|
return false
|
|
}
|
|
|
|
if o.Hostname != in.Hostname {
|
|
return false
|
|
}
|
|
|
|
return true
|
|
}
|
|
|
|
func FilterClusterDiscovery(vs []*ClusterDiscovery, f func(*ClusterDiscovery) bool) []*ClusterDiscovery {
|
|
copy := make([]*ClusterDiscovery, 0)
|
|
for _, v := range vs {
|
|
if f(v) {
|
|
copy = append(copy, v)
|
|
}
|
|
}
|
|
|
|
return copy
|
|
}
|
|
|
|
func (o *ClusterDiscovery) IsValid() *AppError {
|
|
if len(o.Id) != 26 {
|
|
return NewAppError("ClusterDiscovery.IsValid", "model.cluster.is_valid.id.app_error", nil, "", http.StatusBadRequest)
|
|
}
|
|
|
|
if len(o.ClusterName) == 0 {
|
|
return NewAppError("ClusterDiscovery.IsValid", "ClusterName must be set", nil, "", http.StatusBadRequest)
|
|
}
|
|
|
|
if len(o.Type) == 0 {
|
|
return NewAppError("ClusterDiscovery.IsValid", "Type must be set", nil, "", http.StatusBadRequest)
|
|
}
|
|
|
|
if len(o.Hostname) == 0 {
|
|
return NewAppError("ClusterDiscovery.IsValid", "Hostname must be set", nil, "", http.StatusBadRequest)
|
|
}
|
|
|
|
if o.CreateAt == 0 {
|
|
return NewAppError("ClusterDiscovery.IsValid", "CreateAt must be set", nil, "", http.StatusBadRequest)
|
|
}
|
|
|
|
if o.LastPingAt == 0 {
|
|
return NewAppError("ClusterDiscovery.IsValid", "LastPingAt must be set", nil, "", http.StatusBadRequest)
|
|
}
|
|
|
|
return nil
|
|
}
|
|
|
|
func (o *ClusterDiscovery) ToJson() string {
|
|
b, err := json.Marshal(o)
|
|
if err != nil {
|
|
return ""
|
|
}
|
|
|
|
return string(b)
|
|
}
|
|
|
|
func ClusterDiscoveryFromJson(data io.Reader) *ClusterDiscovery {
|
|
decoder := json.NewDecoder(data)
|
|
var me ClusterDiscovery
|
|
err := decoder.Decode(&me)
|
|
if err == nil {
|
|
return &me
|
|
}
|
|
|
|
return nil
|
|
}
|