mirror of
https://github.com/mattermost/mattermost.git
synced 2025-02-25 18:55:24 -06:00
* begin backend plugin wip * flesh out rpcplugin. everything done except for minor supervisor stubs * done with basic plugin infrastructure * simplify tests * remove unused test lines
292 lines
6.8 KiB
Go
292 lines
6.8 KiB
Go
package pluginenv
|
|
|
|
import (
|
|
"fmt"
|
|
"io/ioutil"
|
|
"os"
|
|
"path/filepath"
|
|
"testing"
|
|
|
|
"github.com/stretchr/testify/assert"
|
|
"github.com/stretchr/testify/mock"
|
|
"github.com/stretchr/testify/require"
|
|
|
|
"github.com/mattermost/platform/plugin"
|
|
"github.com/mattermost/platform/plugin/plugintest"
|
|
)
|
|
|
|
type MockProvider struct {
|
|
mock.Mock
|
|
}
|
|
|
|
func (m *MockProvider) API(manifest *plugin.Manifest) (plugin.API, error) {
|
|
ret := m.Called()
|
|
if ret.Get(0) == nil {
|
|
return nil, ret.Error(1)
|
|
}
|
|
return ret.Get(0).(plugin.API), ret.Error(1)
|
|
}
|
|
|
|
func (m *MockProvider) Supervisor(bundle *plugin.BundleInfo) (plugin.Supervisor, error) {
|
|
ret := m.Called()
|
|
if ret.Get(0) == nil {
|
|
return nil, ret.Error(1)
|
|
}
|
|
return ret.Get(0).(plugin.Supervisor), ret.Error(1)
|
|
}
|
|
|
|
type MockSupervisor struct {
|
|
mock.Mock
|
|
}
|
|
|
|
func (m *MockSupervisor) Start() error {
|
|
return m.Called().Error(0)
|
|
}
|
|
|
|
func (m *MockSupervisor) Stop() error {
|
|
return m.Called().Error(0)
|
|
}
|
|
|
|
func (m *MockSupervisor) Hooks() plugin.Hooks {
|
|
return m.Called().Get(0).(plugin.Hooks)
|
|
}
|
|
|
|
func initTmpDir(t *testing.T, files map[string]string) string {
|
|
success := false
|
|
dir, err := ioutil.TempDir("", "mm-plugin-test")
|
|
require.NoError(t, err)
|
|
defer func() {
|
|
if !success {
|
|
os.RemoveAll(dir)
|
|
}
|
|
}()
|
|
|
|
for name, contents := range files {
|
|
path := filepath.Join(dir, name)
|
|
parent := filepath.Dir(path)
|
|
require.NoError(t, os.MkdirAll(parent, 0700))
|
|
f, err := os.Create(path)
|
|
require.NoError(t, err)
|
|
_, err = f.WriteString(contents)
|
|
f.Close()
|
|
require.NoError(t, err)
|
|
}
|
|
|
|
success = true
|
|
return dir
|
|
}
|
|
|
|
func TestNew_MissingOptions(t *testing.T) {
|
|
dir := initTmpDir(t, map[string]string{
|
|
"foo/plugin.json": `{"id": "foo"}`,
|
|
})
|
|
defer os.RemoveAll(dir)
|
|
|
|
var provider MockProvider
|
|
defer provider.AssertExpectations(t)
|
|
|
|
env, err := New(
|
|
APIProvider(provider.API),
|
|
)
|
|
assert.Nil(t, env)
|
|
assert.Error(t, err)
|
|
|
|
env, err = New(
|
|
SearchPath(dir),
|
|
)
|
|
assert.Nil(t, env)
|
|
assert.Error(t, err)
|
|
}
|
|
|
|
func TestEnvironment(t *testing.T) {
|
|
dir := initTmpDir(t, map[string]string{
|
|
".foo/plugin.json": `{"id": "foo"}`,
|
|
"foo/bar": "asdf",
|
|
"foo/plugin.json": `{"id": "foo"}`,
|
|
"bar/zxc": "qwer",
|
|
"baz/plugin.yaml": "id: baz",
|
|
"bad/plugin.json": "asd",
|
|
"qwe": "asd",
|
|
})
|
|
defer os.RemoveAll(dir)
|
|
|
|
var provider MockProvider
|
|
defer provider.AssertExpectations(t)
|
|
|
|
env, err := New(
|
|
SearchPath(dir),
|
|
APIProvider(provider.API),
|
|
SupervisorProvider(provider.Supervisor),
|
|
)
|
|
require.NoError(t, err)
|
|
defer env.Shutdown()
|
|
|
|
plugins, err := env.Plugins()
|
|
assert.NoError(t, err)
|
|
assert.Len(t, plugins, 3)
|
|
|
|
assert.Error(t, env.ActivatePlugin("x"))
|
|
|
|
var api struct{ plugin.API }
|
|
var supervisor MockSupervisor
|
|
defer supervisor.AssertExpectations(t)
|
|
var hooks plugintest.Hooks
|
|
defer hooks.AssertExpectations(t)
|
|
|
|
provider.On("API").Return(&api, nil)
|
|
provider.On("Supervisor").Return(&supervisor, nil)
|
|
|
|
supervisor.On("Start").Return(nil)
|
|
supervisor.On("Stop").Return(nil)
|
|
supervisor.On("Hooks").Return(&hooks)
|
|
|
|
hooks.On("OnActivate", &api).Return(nil)
|
|
|
|
assert.NoError(t, env.ActivatePlugin("foo"))
|
|
assert.Equal(t, env.ActivePluginIds(), []string{"foo"})
|
|
assert.Error(t, env.ActivatePlugin("foo"))
|
|
|
|
hooks.On("OnDeactivate").Return(nil)
|
|
assert.NoError(t, env.DeactivatePlugin("foo"))
|
|
assert.Error(t, env.DeactivatePlugin("foo"))
|
|
|
|
assert.NoError(t, env.ActivatePlugin("foo"))
|
|
assert.Equal(t, env.ActivePluginIds(), []string{"foo"})
|
|
assert.Empty(t, env.Shutdown())
|
|
}
|
|
|
|
func TestEnvironment_DuplicatePluginError(t *testing.T) {
|
|
dir := initTmpDir(t, map[string]string{
|
|
"foo/plugin.json": `{"id": "foo"}`,
|
|
"foo2/plugin.json": `{"id": "foo"}`,
|
|
})
|
|
defer os.RemoveAll(dir)
|
|
|
|
var provider MockProvider
|
|
defer provider.AssertExpectations(t)
|
|
|
|
env, err := New(
|
|
SearchPath(dir),
|
|
APIProvider(provider.API),
|
|
SupervisorProvider(provider.Supervisor),
|
|
)
|
|
require.NoError(t, err)
|
|
defer env.Shutdown()
|
|
|
|
assert.Error(t, env.ActivatePlugin("foo"))
|
|
assert.Empty(t, env.ActivePluginIds())
|
|
}
|
|
|
|
func TestEnvironment_BadSearchPathError(t *testing.T) {
|
|
var provider MockProvider
|
|
defer provider.AssertExpectations(t)
|
|
|
|
env, err := New(
|
|
SearchPath("thissearchpathshouldnotexist!"),
|
|
APIProvider(provider.API),
|
|
SupervisorProvider(provider.Supervisor),
|
|
)
|
|
require.NoError(t, err)
|
|
defer env.Shutdown()
|
|
|
|
assert.Error(t, env.ActivatePlugin("foo"))
|
|
assert.Empty(t, env.ActivePluginIds())
|
|
}
|
|
|
|
func TestEnvironment_ActivatePluginErrors(t *testing.T) {
|
|
dir := initTmpDir(t, map[string]string{
|
|
"foo/plugin.json": `{"id": "foo"}`,
|
|
})
|
|
defer os.RemoveAll(dir)
|
|
|
|
var provider MockProvider
|
|
|
|
env, err := New(
|
|
SearchPath(dir),
|
|
APIProvider(provider.API),
|
|
SupervisorProvider(provider.Supervisor),
|
|
)
|
|
require.NoError(t, err)
|
|
defer env.Shutdown()
|
|
|
|
var api struct{ plugin.API }
|
|
var supervisor MockSupervisor
|
|
var hooks plugintest.Hooks
|
|
|
|
for name, setup := range map[string]func(){
|
|
"SupervisorProviderError": func() {
|
|
provider.On("Supervisor").Return(nil, fmt.Errorf("test error"))
|
|
},
|
|
"APIProviderError": func() {
|
|
provider.On("API").Return(plugin.API(nil), fmt.Errorf("test error"))
|
|
provider.On("Supervisor").Return(&supervisor, nil)
|
|
},
|
|
"SupervisorError": func() {
|
|
provider.On("API").Return(&api, nil)
|
|
provider.On("Supervisor").Return(&supervisor, nil)
|
|
|
|
supervisor.On("Start").Return(fmt.Errorf("test error"))
|
|
},
|
|
"HooksError": func() {
|
|
provider.On("API").Return(&api, nil)
|
|
provider.On("Supervisor").Return(&supervisor, nil)
|
|
|
|
supervisor.On("Start").Return(nil)
|
|
supervisor.On("Stop").Return(nil)
|
|
supervisor.On("Hooks").Return(&hooks)
|
|
|
|
hooks.On("OnActivate", &api).Return(fmt.Errorf("test error"))
|
|
},
|
|
} {
|
|
t.Run(name, func(t *testing.T) {
|
|
supervisor.Mock = mock.Mock{}
|
|
hooks.Mock = mock.Mock{}
|
|
provider.Mock = mock.Mock{}
|
|
setup()
|
|
assert.Error(t, env.ActivatePlugin("foo"))
|
|
assert.Empty(t, env.ActivePluginIds())
|
|
supervisor.AssertExpectations(t)
|
|
hooks.AssertExpectations(t)
|
|
provider.AssertExpectations(t)
|
|
})
|
|
}
|
|
}
|
|
|
|
func TestEnvironment_ShutdownError(t *testing.T) {
|
|
dir := initTmpDir(t, map[string]string{
|
|
"foo/plugin.json": `{"id": "foo"}`,
|
|
})
|
|
defer os.RemoveAll(dir)
|
|
|
|
var provider MockProvider
|
|
defer provider.AssertExpectations(t)
|
|
|
|
env, err := New(
|
|
SearchPath(dir),
|
|
APIProvider(provider.API),
|
|
SupervisorProvider(provider.Supervisor),
|
|
)
|
|
require.NoError(t, err)
|
|
defer env.Shutdown()
|
|
|
|
var api struct{ plugin.API }
|
|
var supervisor MockSupervisor
|
|
defer supervisor.AssertExpectations(t)
|
|
var hooks plugintest.Hooks
|
|
defer hooks.AssertExpectations(t)
|
|
|
|
provider.On("API").Return(&api, nil)
|
|
provider.On("Supervisor").Return(&supervisor, nil)
|
|
|
|
supervisor.On("Start").Return(nil)
|
|
supervisor.On("Stop").Return(fmt.Errorf("test error"))
|
|
supervisor.On("Hooks").Return(&hooks)
|
|
|
|
hooks.On("OnActivate", &api).Return(nil)
|
|
hooks.On("OnDeactivate").Return(fmt.Errorf("test error"))
|
|
|
|
assert.NoError(t, env.ActivatePlugin("foo"))
|
|
assert.Equal(t, env.ActivePluginIds(), []string{"foo"})
|
|
assert.Len(t, env.Shutdown(), 2)
|
|
}
|