mirror of
https://github.com/42wim/matterbridge.git
synced 2025-01-09 17:09:02 -08:00
4dd8bae5c9
* Update dependencies * Update module to go 1.17
571 lines
17 KiB
Go
571 lines
17 KiB
Go
// Copyright (c) 2015-present Mattermost, Inc. All Rights Reserved.
|
|
// See LICENSE.txt for license information.
|
|
|
|
package model
|
|
|
|
import (
|
|
"encoding/json"
|
|
"fmt"
|
|
"io"
|
|
"io/ioutil"
|
|
"os"
|
|
"path/filepath"
|
|
"strings"
|
|
|
|
"github.com/blang/semver"
|
|
"github.com/pkg/errors"
|
|
"gopkg.in/yaml.v2"
|
|
)
|
|
|
|
type PluginOption struct {
|
|
// The display name for the option.
|
|
DisplayName string `json:"display_name" yaml:"display_name"`
|
|
|
|
// The string value for the option.
|
|
Value string `json:"value" yaml:"value"`
|
|
}
|
|
|
|
type PluginSettingType int
|
|
|
|
const (
|
|
Bool PluginSettingType = iota
|
|
Dropdown
|
|
Generated
|
|
Radio
|
|
Text
|
|
LongText
|
|
Number
|
|
Username
|
|
Custom
|
|
)
|
|
|
|
type PluginSetting struct {
|
|
// The key that the setting will be assigned to in the configuration file.
|
|
Key string `json:"key" yaml:"key"`
|
|
|
|
// The display name for the setting.
|
|
DisplayName string `json:"display_name" yaml:"display_name"`
|
|
|
|
// The type of the setting.
|
|
//
|
|
// "bool" will result in a boolean true or false setting.
|
|
//
|
|
// "dropdown" will result in a string setting that allows the user to select from a list of
|
|
// pre-defined options.
|
|
//
|
|
// "generated" will result in a string setting that is set to a random, cryptographically secure
|
|
// string.
|
|
//
|
|
// "radio" will result in a string setting that allows the user to select from a short selection
|
|
// of pre-defined options.
|
|
//
|
|
// "text" will result in a string setting that can be typed in manually.
|
|
//
|
|
// "longtext" will result in a multi line string that can be typed in manually.
|
|
//
|
|
// "number" will result in in integer setting that can be typed in manually.
|
|
//
|
|
// "username" will result in a text setting that will autocomplete to a username.
|
|
//
|
|
// "custom" will result in a custom defined setting and will load the custom component registered for the Web App System Console.
|
|
Type string `json:"type" yaml:"type"`
|
|
|
|
// The help text to display to the user. Supports Markdown formatting.
|
|
HelpText string `json:"help_text" yaml:"help_text"`
|
|
|
|
// The help text to display alongside the "Regenerate" button for settings of the "generated" type.
|
|
RegenerateHelpText string `json:"regenerate_help_text,omitempty" yaml:"regenerate_help_text,omitempty"`
|
|
|
|
// The placeholder to display for "generated", "text", "longtext", "number" and "username" types when blank.
|
|
Placeholder string `json:"placeholder" yaml:"placeholder"`
|
|
|
|
// The default value of the setting.
|
|
Default interface{} `json:"default" yaml:"default"`
|
|
|
|
// For "radio" or "dropdown" settings, this is the list of pre-defined options that the user can choose
|
|
// from.
|
|
Options []*PluginOption `json:"options,omitempty" yaml:"options,omitempty"`
|
|
}
|
|
|
|
type PluginSettingsSchema struct {
|
|
// Optional text to display above the settings. Supports Markdown formatting.
|
|
Header string `json:"header" yaml:"header"`
|
|
|
|
// Optional text to display below the settings. Supports Markdown formatting.
|
|
Footer string `json:"footer" yaml:"footer"`
|
|
|
|
// A list of setting definitions.
|
|
Settings []*PluginSetting `json:"settings" yaml:"settings"`
|
|
}
|
|
|
|
// The plugin manifest defines the metadata required to load and present your plugin. The manifest
|
|
// file should be named plugin.json or plugin.yaml and placed in the top of your
|
|
// plugin bundle.
|
|
//
|
|
// Example plugin.json:
|
|
//
|
|
//
|
|
// {
|
|
// "id": "com.mycompany.myplugin",
|
|
// "name": "My Plugin",
|
|
// "description": "This is my plugin",
|
|
// "homepage_url": "https://example.com",
|
|
// "support_url": "https://example.com/support",
|
|
// "release_notes_url": "https://example.com/releases/v0.0.1",
|
|
// "icon_path": "assets/logo.svg",
|
|
// "version": "0.1.0",
|
|
// "min_server_version": "5.6.0",
|
|
// "server": {
|
|
// "executables": {
|
|
// "linux-amd64": "server/dist/plugin-linux-amd64",
|
|
// "darwin-amd64": "server/dist/plugin-darwin-amd64",
|
|
// "windows-amd64": "server/dist/plugin-windows-amd64.exe"
|
|
// }
|
|
// },
|
|
// "webapp": {
|
|
// "bundle_path": "webapp/dist/main.js"
|
|
// },
|
|
// "settings_schema": {
|
|
// "header": "Some header text",
|
|
// "footer": "Some footer text",
|
|
// "settings": [{
|
|
// "key": "someKey",
|
|
// "display_name": "Enable Extra Feature",
|
|
// "type": "bool",
|
|
// "help_text": "When true, an extra feature will be enabled!",
|
|
// "default": "false"
|
|
// }]
|
|
// },
|
|
// "props": {
|
|
// "someKey": "someData"
|
|
// }
|
|
// }
|
|
type Manifest struct {
|
|
// The id is a globally unique identifier that represents your plugin. Ids must be at least
|
|
// 3 characters, at most 190 characters and must match ^[a-zA-Z0-9-_\.]+$.
|
|
// Reverse-DNS notation using a name you control is a good option, e.g. "com.mycompany.myplugin".
|
|
Id string `json:"id" yaml:"id"`
|
|
|
|
// The name to be displayed for the plugin.
|
|
Name string `json:"name" yaml:"name"`
|
|
|
|
// A description of what your plugin is and does.
|
|
Description string `json:"description,omitempty" yaml:"description,omitempty"`
|
|
|
|
// HomepageURL is an optional link to learn more about the plugin.
|
|
HomepageURL string `json:"homepage_url,omitempty" yaml:"homepage_url,omitempty"`
|
|
|
|
// SupportURL is an optional URL where plugin issues can be reported.
|
|
SupportURL string `json:"support_url,omitempty" yaml:"support_url,omitempty"`
|
|
|
|
// ReleaseNotesURL is an optional URL where a changelog for the release can be found.
|
|
ReleaseNotesURL string `json:"release_notes_url,omitempty" yaml:"release_notes_url,omitempty"`
|
|
|
|
// A relative file path in the bundle that points to the plugins svg icon for use with the Plugin Marketplace.
|
|
// This should be relative to the root of your bundle and the location of the manifest file. Bitmap image formats are not supported.
|
|
IconPath string `json:"icon_path,omitempty" yaml:"icon_path,omitempty"`
|
|
|
|
// A version number for your plugin. Semantic versioning is recommended: http://semver.org
|
|
Version string `json:"version" yaml:"version"`
|
|
|
|
// The minimum Mattermost server version required for your plugin.
|
|
//
|
|
// Minimum server version: 5.6
|
|
MinServerVersion string `json:"min_server_version,omitempty" yaml:"min_server_version,omitempty"`
|
|
|
|
// Server defines the server-side portion of your plugin.
|
|
Server *ManifestServer `json:"server,omitempty" yaml:"server,omitempty"`
|
|
|
|
// Backend is a deprecated flag for defining the server-side portion of your plugin. Going forward, use Server instead.
|
|
Backend *ManifestServer `json:"backend,omitempty" yaml:"backend,omitempty"`
|
|
|
|
// If your plugin extends the web app, you'll need to define webapp.
|
|
Webapp *ManifestWebapp `json:"webapp,omitempty" yaml:"webapp,omitempty"`
|
|
|
|
// To allow administrators to configure your plugin via the Mattermost system console, you can
|
|
// provide your settings schema.
|
|
SettingsSchema *PluginSettingsSchema `json:"settings_schema,omitempty" yaml:"settings_schema,omitempty"`
|
|
|
|
// Plugins can store any kind of data in Props to allow other plugins to use it.
|
|
Props map[string]interface{} `json:"props,omitempty" yaml:"props,omitempty"`
|
|
|
|
// RequiredConfig defines any required server configuration fields for the plugin to function properly.
|
|
//
|
|
// Use the plugin helpers CheckRequiredServerConfiguration method to enforce this.
|
|
RequiredConfig *Config `json:"required_configuration,omitempty" yaml:"required_configuration,omitempty"`
|
|
}
|
|
|
|
type ManifestServer struct {
|
|
// AllExecutables are the paths to your executable binaries, specifying multiple entry
|
|
// points for different platforms when bundled together in a single plugin.
|
|
AllExecutables map[string]string `json:"executables,omitempty" yaml:"executables,omitempty"`
|
|
|
|
// Executables is a legacy field populated with a subset of supported platform executables.
|
|
// When unmarshalling, Executables is authoritative for the platform executable paths it
|
|
// contains, overriding any values in AllExecutables. When marshalling, AllExecutables
|
|
// is authoritative.
|
|
//
|
|
// Code duplication is avoided when (un)marshalling by leveraging type aliases in the
|
|
// various (Un)Marshal(JSON|YAML) methods, since aliases don't inherit the aliased type's
|
|
// methods.
|
|
//
|
|
// In v6.0, we should remove this field and rename AllExecutables back to Executables.
|
|
Executables *ManifestExecutables `json:"-" yaml:"-"`
|
|
|
|
// Executable is the path to your executable binary. This should be relative to the root
|
|
// of your bundle and the location of the manifest file.
|
|
//
|
|
// On Windows, this file must have a ".exe" extension.
|
|
//
|
|
// If your plugin is compiled for multiple platforms, consider bundling them together
|
|
// and using the Executables field instead.
|
|
Executable string `json:"executable" yaml:"executable"`
|
|
}
|
|
|
|
func (ms *ManifestServer) MarshalJSON() ([]byte, error) {
|
|
type auxManifestServer ManifestServer
|
|
|
|
// Populate AllExecutables from Executables, if it exists.
|
|
if ms.Executables != nil {
|
|
if ms.AllExecutables == nil {
|
|
ms.AllExecutables = make(map[string]string)
|
|
}
|
|
|
|
ms.AllExecutables["linux-amd64"] = ms.Executables.LinuxAmd64
|
|
ms.AllExecutables["darwin-amd64"] = ms.Executables.DarwinAmd64
|
|
ms.AllExecutables["windows-amd64"] = ms.Executables.WindowsAmd64
|
|
}
|
|
|
|
return json.Marshal((*auxManifestServer)(ms))
|
|
}
|
|
|
|
func (ms *ManifestServer) UnmarshalJSON(data []byte) error {
|
|
type auxManifestServer ManifestServer
|
|
|
|
aux := (*auxManifestServer)(ms)
|
|
if err := json.Unmarshal(data, aux); err != nil {
|
|
return err
|
|
}
|
|
|
|
if len(aux.AllExecutables) > 0 {
|
|
ms.Executables = &ManifestExecutables{
|
|
LinuxAmd64: aux.AllExecutables["linux-amd64"],
|
|
DarwinAmd64: aux.AllExecutables["darwin-amd64"],
|
|
WindowsAmd64: aux.AllExecutables["windows-amd64"],
|
|
}
|
|
}
|
|
|
|
return nil
|
|
}
|
|
|
|
func (ms *ManifestServer) MarshalYAML() ([]byte, error) {
|
|
type auxManifestServer ManifestServer
|
|
|
|
// Populate AllExecutables from Executables, if it exists.
|
|
if ms.Executables != nil {
|
|
if ms.AllExecutables == nil {
|
|
ms.AllExecutables = make(map[string]string)
|
|
}
|
|
|
|
ms.AllExecutables["linux-amd64"] = ms.Executables.LinuxAmd64
|
|
ms.AllExecutables["darwin-amd64"] = ms.Executables.DarwinAmd64
|
|
ms.AllExecutables["windows-amd64"] = ms.Executables.WindowsAmd64
|
|
}
|
|
|
|
return yaml.Marshal((*auxManifestServer)(ms))
|
|
}
|
|
|
|
func (ms *ManifestServer) UnmarshalYAML(unmarshal func(interface{}) error) error {
|
|
type auxManifestServer ManifestServer
|
|
|
|
aux := (*auxManifestServer)(ms)
|
|
if err := unmarshal(&aux); err != nil {
|
|
return err
|
|
}
|
|
|
|
if len(aux.AllExecutables) > 0 {
|
|
ms.Executables = &ManifestExecutables{
|
|
LinuxAmd64: aux.AllExecutables["linux-amd64"],
|
|
DarwinAmd64: aux.AllExecutables["darwin-amd64"],
|
|
WindowsAmd64: aux.AllExecutables["windows-amd64"],
|
|
}
|
|
}
|
|
|
|
return nil
|
|
}
|
|
|
|
// ManifestExecutables is a legacy structure capturing a subet of the known platform executables.
|
|
type ManifestExecutables struct {
|
|
// LinuxAmd64 is the path to your executable binary for the corresponding platform
|
|
LinuxAmd64 string `json:"linux-amd64,omitempty" yaml:"linux-amd64,omitempty"`
|
|
// DarwinAmd64 is the path to your executable binary for the corresponding platform
|
|
DarwinAmd64 string `json:"darwin-amd64,omitempty" yaml:"darwin-amd64,omitempty"`
|
|
// WindowsAmd64 is the path to your executable binary for the corresponding platform
|
|
// This file must have a ".exe" extension
|
|
WindowsAmd64 string `json:"windows-amd64,omitempty" yaml:"windows-amd64,omitempty"`
|
|
}
|
|
|
|
type ManifestWebapp struct {
|
|
// The path to your webapp bundle. This should be relative to the root of your bundle and the
|
|
// location of the manifest file.
|
|
BundlePath string `json:"bundle_path" yaml:"bundle_path"`
|
|
|
|
// BundleHash is the 64-bit FNV-1a hash of the webapp bundle, computed when the plugin is loaded
|
|
BundleHash []byte `json:"-"`
|
|
}
|
|
|
|
func (m *Manifest) ToJson() string {
|
|
b, _ := json.Marshal(m)
|
|
return string(b)
|
|
}
|
|
|
|
func ManifestListToJson(m []*Manifest) string {
|
|
b, _ := json.Marshal(m)
|
|
return string(b)
|
|
}
|
|
|
|
func ManifestFromJson(data io.Reader) *Manifest {
|
|
var m *Manifest
|
|
json.NewDecoder(data).Decode(&m)
|
|
return m
|
|
}
|
|
|
|
func ManifestListFromJson(data io.Reader) []*Manifest {
|
|
var manifests []*Manifest
|
|
json.NewDecoder(data).Decode(&manifests)
|
|
return manifests
|
|
}
|
|
|
|
func (m *Manifest) HasClient() bool {
|
|
return m.Webapp != nil
|
|
}
|
|
|
|
func (m *Manifest) ClientManifest() *Manifest {
|
|
cm := new(Manifest)
|
|
*cm = *m
|
|
cm.Name = ""
|
|
cm.Description = ""
|
|
cm.Server = nil
|
|
if cm.Webapp != nil {
|
|
cm.Webapp = new(ManifestWebapp)
|
|
*cm.Webapp = *m.Webapp
|
|
cm.Webapp.BundlePath = "/static/" + m.Id + "/" + fmt.Sprintf("%s_%x_bundle.js", m.Id, m.Webapp.BundleHash)
|
|
}
|
|
return cm
|
|
}
|
|
|
|
// GetExecutableForRuntime returns the path to the executable for the given runtime architecture.
|
|
//
|
|
// If the manifest defines multiple executables, but none match, or if only a single executable
|
|
// is defined, the Executable field will be returned. This method does not guarantee that the
|
|
// resulting binary can actually execute on the given platform.
|
|
func (m *Manifest) GetExecutableForRuntime(goOs, goArch string) string {
|
|
server := m.Server
|
|
|
|
// Support the deprecated backend parameter.
|
|
if server == nil {
|
|
server = m.Backend
|
|
}
|
|
|
|
if server == nil {
|
|
return ""
|
|
}
|
|
|
|
var executable string
|
|
if len(server.AllExecutables) > 0 {
|
|
osArch := fmt.Sprintf("%s-%s", goOs, goArch)
|
|
executable = server.AllExecutables[osArch]
|
|
}
|
|
|
|
if executable == "" {
|
|
executable = server.Executable
|
|
}
|
|
|
|
return executable
|
|
}
|
|
|
|
func (m *Manifest) HasServer() bool {
|
|
return m.Server != nil || m.Backend != nil
|
|
}
|
|
|
|
func (m *Manifest) HasWebapp() bool {
|
|
return m.Webapp != nil
|
|
}
|
|
|
|
func (m *Manifest) MeetMinServerVersion(serverVersion string) (bool, error) {
|
|
minServerVersion, err := semver.Parse(m.MinServerVersion)
|
|
if err != nil {
|
|
return false, errors.New("failed to parse MinServerVersion")
|
|
}
|
|
sv := semver.MustParse(serverVersion)
|
|
if sv.LT(minServerVersion) {
|
|
return false, nil
|
|
}
|
|
return true, nil
|
|
}
|
|
|
|
func (m *Manifest) IsValid() error {
|
|
if !IsValidPluginId(m.Id) {
|
|
return errors.New("invalid plugin ID")
|
|
}
|
|
|
|
if strings.TrimSpace(m.Name) == "" {
|
|
return errors.New("a plugin name is needed")
|
|
}
|
|
|
|
if m.HomepageURL != "" && !IsValidHttpUrl(m.HomepageURL) {
|
|
return errors.New("invalid HomepageURL")
|
|
}
|
|
|
|
if m.SupportURL != "" && !IsValidHttpUrl(m.SupportURL) {
|
|
return errors.New("invalid SupportURL")
|
|
}
|
|
|
|
if m.ReleaseNotesURL != "" && !IsValidHttpUrl(m.ReleaseNotesURL) {
|
|
return errors.New("invalid ReleaseNotesURL")
|
|
}
|
|
|
|
if m.Version != "" {
|
|
_, err := semver.Parse(m.Version)
|
|
if err != nil {
|
|
return errors.Wrap(err, "failed to parse Version")
|
|
}
|
|
}
|
|
|
|
if m.MinServerVersion != "" {
|
|
_, err := semver.Parse(m.MinServerVersion)
|
|
if err != nil {
|
|
return errors.Wrap(err, "failed to parse MinServerVersion")
|
|
}
|
|
}
|
|
|
|
if m.SettingsSchema != nil {
|
|
err := m.SettingsSchema.isValid()
|
|
if err != nil {
|
|
return errors.Wrap(err, "invalid settings schema")
|
|
}
|
|
}
|
|
|
|
return nil
|
|
}
|
|
|
|
func (s *PluginSettingsSchema) isValid() error {
|
|
for _, setting := range s.Settings {
|
|
err := setting.isValid()
|
|
if err != nil {
|
|
return err
|
|
}
|
|
}
|
|
|
|
return nil
|
|
}
|
|
|
|
func (s *PluginSetting) isValid() error {
|
|
pluginSettingType, err := convertTypeToPluginSettingType(s.Type)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
if s.RegenerateHelpText != "" && pluginSettingType != Generated {
|
|
return errors.New("should not set RegenerateHelpText for setting type that is not generated")
|
|
}
|
|
|
|
if s.Placeholder != "" && !(pluginSettingType == Generated ||
|
|
pluginSettingType == Text ||
|
|
pluginSettingType == LongText ||
|
|
pluginSettingType == Number ||
|
|
pluginSettingType == Username) {
|
|
return errors.New("should not set Placeholder for setting type not in text, generated or username")
|
|
}
|
|
|
|
if s.Options != nil {
|
|
if pluginSettingType != Radio && pluginSettingType != Dropdown {
|
|
return errors.New("should not set Options for setting type not in radio or dropdown")
|
|
}
|
|
|
|
for _, option := range s.Options {
|
|
if option.DisplayName == "" || option.Value == "" {
|
|
return errors.New("should not have empty Displayname or Value for any option")
|
|
}
|
|
}
|
|
}
|
|
|
|
return nil
|
|
}
|
|
|
|
func convertTypeToPluginSettingType(t string) (PluginSettingType, error) {
|
|
var settingType PluginSettingType
|
|
switch t {
|
|
case "bool":
|
|
return Bool, nil
|
|
case "dropdown":
|
|
return Dropdown, nil
|
|
case "generated":
|
|
return Generated, nil
|
|
case "radio":
|
|
return Radio, nil
|
|
case "text":
|
|
return Text, nil
|
|
case "number":
|
|
return Number, nil
|
|
case "longtext":
|
|
return LongText, nil
|
|
case "username":
|
|
return Username, nil
|
|
case "custom":
|
|
return Custom, nil
|
|
default:
|
|
return settingType, errors.New("invalid setting type: " + t)
|
|
}
|
|
}
|
|
|
|
// FindManifest will find and parse the manifest in a given directory.
|
|
//
|
|
// In all cases other than a does-not-exist error, path is set to the path of the manifest file that was
|
|
// found.
|
|
//
|
|
// Manifests are JSON or YAML files named plugin.json, plugin.yaml, or plugin.yml.
|
|
func FindManifest(dir string) (manifest *Manifest, path string, err error) {
|
|
for _, name := range []string{"plugin.yml", "plugin.yaml"} {
|
|
path = filepath.Join(dir, name)
|
|
f, ferr := os.Open(path)
|
|
if ferr != nil {
|
|
if !os.IsNotExist(ferr) {
|
|
return nil, "", ferr
|
|
}
|
|
continue
|
|
}
|
|
b, ioerr := ioutil.ReadAll(f)
|
|
f.Close()
|
|
if ioerr != nil {
|
|
return nil, path, ioerr
|
|
}
|
|
var parsed Manifest
|
|
err = yaml.Unmarshal(b, &parsed)
|
|
if err != nil {
|
|
return nil, path, err
|
|
}
|
|
manifest = &parsed
|
|
manifest.Id = strings.ToLower(manifest.Id)
|
|
return manifest, path, nil
|
|
}
|
|
|
|
path = filepath.Join(dir, "plugin.json")
|
|
f, ferr := os.Open(path)
|
|
if ferr != nil {
|
|
if os.IsNotExist(ferr) {
|
|
path = ""
|
|
}
|
|
return nil, path, ferr
|
|
}
|
|
defer f.Close()
|
|
var parsed Manifest
|
|
err = json.NewDecoder(f).Decode(&parsed)
|
|
if err != nil {
|
|
return nil, path, err
|
|
}
|
|
manifest = &parsed
|
|
manifest.Id = strings.ToLower(manifest.Id)
|
|
return manifest, path, nil
|
|
}
|