summaryrefslogtreecommitdiffstats
path: root/plugin/pluginenv/environment.go
blob: 36a8c6e7691a3ed92db9652ef29b719afc9d8cfd (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
// Package pluginenv provides high level functionality for discovering and launching plugins.
package pluginenv

import (
	"fmt"

	"github.com/pkg/errors"

	"github.com/mattermost/platform/plugin"
)

type APIProviderFunc func(*plugin.Manifest) (plugin.API, error)
type SupervisorProviderFunc func(*plugin.BundleInfo) (plugin.Supervisor, error)

// Environment represents an environment that plugins are discovered and launched in.
type Environment struct {
	searchPath         string
	apiProvider        APIProviderFunc
	supervisorProvider SupervisorProviderFunc
	activePlugins      map[string]plugin.Supervisor
}

type Option func(*Environment)

// Creates a new environment. At a minimum, the APIProvider and SearchPath options are required.
func New(options ...Option) (*Environment, error) {
	env := &Environment{
		activePlugins: make(map[string]plugin.Supervisor),
	}
	for _, opt := range options {
		opt(env)
	}
	if env.supervisorProvider == nil {
		env.supervisorProvider = DefaultSupervisorProvider
	}
	if env.searchPath == "" {
		return nil, fmt.Errorf("a search path must be provided")
	} else if env.apiProvider == nil {
		return nil, fmt.Errorf("an api provider must be provided")
	}
	return env, nil
}

// Returns a list of all plugins found within the environment.
func (env *Environment) Plugins() ([]*plugin.BundleInfo, error) {
	return ScanSearchPath(env.searchPath)
}

// Returns the ids of the currently active plugins.
func (env *Environment) ActivePluginIds() (ids []string) {
	for id := range env.activePlugins {
		ids = append(ids, id)
	}
	return
}

// Activates the plugin with the given id.
func (env *Environment) ActivatePlugin(id string) error {
	if _, ok := env.activePlugins[id]; ok {
		return fmt.Errorf("plugin already active: %v", id)
	}
	plugins, err := ScanSearchPath(env.searchPath)
	if err != nil {
		return err
	}
	var plugin *plugin.BundleInfo
	for _, p := range plugins {
		if p.Manifest != nil && p.Manifest.Id == id {
			if plugin != nil {
				return fmt.Errorf("multiple plugins found: %v", id)
			}
			plugin = p
		}
	}
	if plugin == nil {
		return fmt.Errorf("plugin not found: %v", id)
	}
	supervisor, err := env.supervisorProvider(plugin)
	if err != nil {
		return errors.Wrapf(err, "unable to create supervisor for plugin: %v", id)
	}
	api, err := env.apiProvider(plugin.Manifest)
	if err != nil {
		return errors.Wrapf(err, "unable to get api for plugin: %v", id)
	}
	if err := supervisor.Start(); err != nil {
		return errors.Wrapf(err, "unable to start plugin: %v", id)
	}
	if err := supervisor.Hooks().OnActivate(api); err != nil {
		supervisor.Stop()
		return errors.Wrapf(err, "unable to activate plugin: %v", id)
	}
	env.activePlugins[id] = supervisor
	return nil
}

// Deactivates the plugin with the given id.
func (env *Environment) DeactivatePlugin(id string) error {
	if supervisor, ok := env.activePlugins[id]; !ok {
		return fmt.Errorf("plugin not active: %v", id)
	} else {
		delete(env.activePlugins, id)
		err := supervisor.Hooks().OnDeactivate()
		if serr := supervisor.Stop(); err == nil {
			err = serr
		}
		return err
	}
}

// Deactivates all plugins and gracefully shuts down the environment.
func (env *Environment) Shutdown() (errs []error) {
	for _, supervisor := range env.activePlugins {
		if err := supervisor.Hooks().OnDeactivate(); err != nil {
			errs = append(errs, err)
		}
		if err := supervisor.Stop(); err != nil {
			errs = append(errs, err)
		}
	}
	env.activePlugins = make(map[string]plugin.Supervisor)
	return
}