mirror of
https://github.com/mattermost/mattermost.git
synced 2025-02-25 18:55:24 -06:00
* clean up plugins GoDoc: - eliminate plugin.NewBlankContext() as unnecessary - export ValidIdRegex as a string vs. the less readable var - add/update various documentation strings - hide everything by default, except where used by client plugins or the mattermost-server. The exception to this rule are the `*(Args|Returns)` structs which must be public for go-plugin, but are now prefixed with `Z_` with a warning not to use. - include a top-level example to get plugin authors started This is not a breaking change for existing plugins compiled against plugins-v2. * remove commented out ServeHTTPResponseWriter * update examples to match developer docs * add missing plugin/doc.go license header
272 lines
7.7 KiB
Go
272 lines
7.7 KiB
Go
// Copyright (c) 2015-present Mattermost, Inc. All Rights Reserved.
|
|
// See LICENSE.txt for license information.
|
|
|
|
package plugin
|
|
|
|
import (
|
|
"fmt"
|
|
"io/ioutil"
|
|
"path/filepath"
|
|
"sync"
|
|
|
|
"github.com/mattermost/mattermost-server/mlog"
|
|
"github.com/mattermost/mattermost-server/model"
|
|
"github.com/pkg/errors"
|
|
)
|
|
|
|
type apiImplCreatorFunc func(*model.Manifest) API
|
|
type supervisorCreatorFunc func(*model.BundleInfo, *mlog.Logger, API) (*supervisor, error)
|
|
|
|
// multiPluginHookRunnerFunc is a callback function to invoke as part of RunMultiPluginHook.
|
|
//
|
|
// Return false to stop the hook from iterating to subsequent plugins.
|
|
type multiPluginHookRunnerFunc func(hooks Hooks) bool
|
|
|
|
type activePlugin struct {
|
|
BundleInfo *model.BundleInfo
|
|
State int
|
|
|
|
supervisor *supervisor
|
|
}
|
|
|
|
// Environment represents the execution environment of active plugins.
|
|
//
|
|
// It is meant for use by the Mattermost server to manipulate, interact with and report on the set
|
|
// of active plugins.
|
|
type Environment struct {
|
|
activePlugins map[string]activePlugin
|
|
mutex sync.RWMutex
|
|
logger *mlog.Logger
|
|
newAPIImpl apiImplCreatorFunc
|
|
pluginDir string
|
|
webappPluginDir string
|
|
}
|
|
|
|
func NewEnvironment(newAPIImpl apiImplCreatorFunc, pluginDir string, webappPluginDir string, logger *mlog.Logger) (*Environment, error) {
|
|
return &Environment{
|
|
activePlugins: make(map[string]activePlugin),
|
|
logger: logger,
|
|
newAPIImpl: newAPIImpl,
|
|
pluginDir: pluginDir,
|
|
webappPluginDir: webappPluginDir,
|
|
}, nil
|
|
}
|
|
|
|
// Performs a full scan of the given path.
|
|
//
|
|
// This function will return info for all subdirectories that appear to be plugins (i.e. all
|
|
// subdirectories containing plugin manifest files, regardless of whether they could actually be
|
|
// parsed).
|
|
//
|
|
// Plugins are found non-recursively and paths beginning with a dot are always ignored.
|
|
func scanSearchPath(path string) ([]*model.BundleInfo, error) {
|
|
files, err := ioutil.ReadDir(path)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
var ret []*model.BundleInfo
|
|
for _, file := range files {
|
|
if !file.IsDir() || file.Name()[0] == '.' {
|
|
continue
|
|
}
|
|
if info := model.BundleInfoForPath(filepath.Join(path, file.Name())); info.ManifestPath != "" {
|
|
ret = append(ret, info)
|
|
}
|
|
}
|
|
return ret, nil
|
|
}
|
|
|
|
// Returns a list of all plugins within the environment.
|
|
func (env *Environment) Available() ([]*model.BundleInfo, error) {
|
|
return scanSearchPath(env.pluginDir)
|
|
}
|
|
|
|
// Returns a list of all currently active plugins within the environment.
|
|
func (env *Environment) Active() []*model.BundleInfo {
|
|
env.mutex.RLock()
|
|
defer env.mutex.RUnlock()
|
|
|
|
activePlugins := []*model.BundleInfo{}
|
|
for _, p := range env.activePlugins {
|
|
activePlugins = append(activePlugins, p.BundleInfo)
|
|
}
|
|
|
|
return activePlugins
|
|
}
|
|
|
|
// IsActive returns true if the plugin with the given id is active.
|
|
func (env *Environment) IsActive(id string) bool {
|
|
_, ok := env.activePlugins[id]
|
|
return ok
|
|
}
|
|
|
|
// Statuses returns a list of plugin statuses representing the state of every plugin
|
|
func (env *Environment) Statuses() (model.PluginStatuses, error) {
|
|
env.mutex.RLock()
|
|
defer env.mutex.RUnlock()
|
|
|
|
plugins, err := env.Available()
|
|
if err != nil {
|
|
return nil, errors.Wrap(err, "unable to get plugin statuses")
|
|
}
|
|
|
|
pluginStatuses := make(model.PluginStatuses, 0, len(plugins))
|
|
for _, plugin := range plugins {
|
|
// For now we don't handle bad manifests, we should
|
|
if plugin.Manifest == nil {
|
|
continue
|
|
}
|
|
|
|
pluginState := model.PluginStateNotRunning
|
|
if plugin, ok := env.activePlugins[plugin.Manifest.Id]; ok {
|
|
pluginState = plugin.State
|
|
}
|
|
|
|
status := &model.PluginStatus{
|
|
PluginId: plugin.Manifest.Id,
|
|
PluginPath: filepath.Dir(plugin.ManifestPath),
|
|
State: pluginState,
|
|
Name: plugin.Manifest.Name,
|
|
Description: plugin.Manifest.Description,
|
|
Version: plugin.Manifest.Version,
|
|
}
|
|
|
|
pluginStatuses = append(pluginStatuses, status)
|
|
}
|
|
|
|
return pluginStatuses, nil
|
|
}
|
|
|
|
// Activate activates the plugin with the given id.
|
|
func (env *Environment) Activate(id string) (reterr error) {
|
|
env.mutex.Lock()
|
|
defer env.mutex.Unlock()
|
|
|
|
// Check if we are already active
|
|
if _, ok := env.activePlugins[id]; ok {
|
|
return nil
|
|
}
|
|
|
|
plugins, err := env.Available()
|
|
if err != nil {
|
|
return err
|
|
}
|
|
var pluginInfo *model.BundleInfo
|
|
for _, p := range plugins {
|
|
if p.Manifest != nil && p.Manifest.Id == id {
|
|
if pluginInfo != nil {
|
|
return fmt.Errorf("multiple plugins found: %v", id)
|
|
}
|
|
pluginInfo = p
|
|
}
|
|
}
|
|
if pluginInfo == nil {
|
|
return fmt.Errorf("plugin not found: %v", id)
|
|
}
|
|
|
|
activePlugin := activePlugin{BundleInfo: pluginInfo}
|
|
defer func() {
|
|
if reterr == nil {
|
|
activePlugin.State = model.PluginStateRunning
|
|
} else {
|
|
activePlugin.State = model.PluginStateFailedToStart
|
|
}
|
|
env.activePlugins[pluginInfo.Manifest.Id] = activePlugin
|
|
}()
|
|
|
|
if pluginInfo.Manifest.Webapp != nil {
|
|
bundlePath := filepath.Clean(pluginInfo.Manifest.Webapp.BundlePath)
|
|
if bundlePath == "" || bundlePath[0] == '.' {
|
|
return fmt.Errorf("invalid webapp bundle path")
|
|
}
|
|
bundlePath = filepath.Join(env.pluginDir, id, bundlePath)
|
|
|
|
webappBundle, err := ioutil.ReadFile(bundlePath)
|
|
if err != nil {
|
|
return errors.Wrapf(err, "unable to read webapp bundle: %v", id)
|
|
}
|
|
|
|
err = ioutil.WriteFile(fmt.Sprintf("%s/%s_bundle.js", env.webappPluginDir, id), webappBundle, 0644)
|
|
if err != nil {
|
|
return errors.Wrapf(err, "unable to write webapp bundle: %v", id)
|
|
}
|
|
}
|
|
|
|
if pluginInfo.Manifest.Backend != nil {
|
|
supervisor, err := newSupervisor(pluginInfo, env.logger, env.newAPIImpl(pluginInfo.Manifest))
|
|
if err != nil {
|
|
return errors.Wrapf(err, "unable to start plugin: %v", id)
|
|
}
|
|
activePlugin.supervisor = supervisor
|
|
}
|
|
|
|
return nil
|
|
}
|
|
|
|
// Deactivates the plugin with the given id.
|
|
func (env *Environment) Deactivate(id string) {
|
|
env.mutex.Lock()
|
|
defer env.mutex.Unlock()
|
|
|
|
if activePlugin, ok := env.activePlugins[id]; !ok {
|
|
return
|
|
} else {
|
|
delete(env.activePlugins, id)
|
|
if activePlugin.supervisor != nil {
|
|
if err := activePlugin.supervisor.Hooks().OnDeactivate(); err != nil {
|
|
env.logger.Error("Plugin OnDeactivate() error", mlog.String("plugin_id", activePlugin.BundleInfo.Manifest.Id), mlog.Err(err))
|
|
}
|
|
activePlugin.supervisor.Shutdown()
|
|
}
|
|
}
|
|
}
|
|
|
|
// Shutdown deactivates all plugins and gracefully shuts down the environment.
|
|
func (env *Environment) Shutdown() {
|
|
env.mutex.Lock()
|
|
defer env.mutex.Unlock()
|
|
|
|
for _, activePlugin := range env.activePlugins {
|
|
if activePlugin.supervisor != nil {
|
|
if err := activePlugin.supervisor.Hooks().OnDeactivate(); err != nil {
|
|
env.logger.Error("Plugin OnDeactivate() error", mlog.String("plugin_id", activePlugin.BundleInfo.Manifest.Id), mlog.Err(err))
|
|
}
|
|
activePlugin.supervisor.Shutdown()
|
|
}
|
|
}
|
|
env.activePlugins = make(map[string]activePlugin)
|
|
return
|
|
}
|
|
|
|
// HooksForPlugin returns the hooks API for the plugin with the given id.
|
|
//
|
|
// Consider using RunMultiPluginHook instead.
|
|
func (env *Environment) HooksForPlugin(id string) (Hooks, error) {
|
|
env.mutex.RLock()
|
|
defer env.mutex.RUnlock()
|
|
|
|
if plug, ok := env.activePlugins[id]; ok && plug.supervisor != nil {
|
|
return plug.supervisor.Hooks(), nil
|
|
}
|
|
|
|
return nil, fmt.Errorf("plugin not found: %v", id)
|
|
}
|
|
|
|
// RunMultiPluginHook invokes hookRunnerFunc for each plugin that implements the given hookId.
|
|
//
|
|
// If hookRunnerFunc returns false, iteration will not continue. The iteration order among active
|
|
// plugins is not specified.
|
|
func (env *Environment) RunMultiPluginHook(hookRunnerFunc multiPluginHookRunnerFunc, hookId int) {
|
|
env.mutex.RLock()
|
|
defer env.mutex.RUnlock()
|
|
|
|
for _, activePlugin := range env.activePlugins {
|
|
if activePlugin.supervisor == nil || !activePlugin.supervisor.Implements(hookId) {
|
|
continue
|
|
}
|
|
if !hookRunnerFunc(activePlugin.supervisor.Hooks()) {
|
|
break
|
|
}
|
|
}
|
|
}
|