2022-05-03 11:17:04 +00:00
|
|
|
package plugins
|
|
|
|
|
|
|
|
import (
|
|
|
|
"fmt"
|
|
|
|
|
|
|
|
"github.com/rs/zerolog"
|
|
|
|
|
2022-07-14 00:58:22 +02:00
|
|
|
"github.com/demodesk/neko/pkg/types"
|
2022-05-03 11:17:04 +00:00
|
|
|
)
|
|
|
|
|
|
|
|
type dependency struct {
|
|
|
|
plugin types.Plugin
|
|
|
|
dependsOn []*dependency
|
|
|
|
invoked bool
|
|
|
|
logger zerolog.Logger
|
|
|
|
}
|
|
|
|
|
2023-03-31 12:02:33 +02:00
|
|
|
func (a *dependency) findPlugin(name string) (*dependency, bool) {
|
|
|
|
if a == nil {
|
|
|
|
return nil, false
|
|
|
|
}
|
|
|
|
|
|
|
|
if a.plugin.Name() == name {
|
|
|
|
return a, true
|
|
|
|
}
|
|
|
|
|
|
|
|
for _, dep := range a.dependsOn {
|
|
|
|
plug, ok := dep.findPlugin(name)
|
|
|
|
if ok {
|
|
|
|
return plug, true
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return nil, false
|
|
|
|
}
|
|
|
|
|
|
|
|
func (a *dependency) startPlugin(pm types.PluginManagers) error {
|
|
|
|
if a.invoked {
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
a.invoked = true
|
|
|
|
|
|
|
|
for _, do := range a.dependsOn {
|
|
|
|
if err := do.startPlugin(pm); err != nil {
|
|
|
|
return fmt.Errorf("plugin's '%s' dependency: %w", a.plugin.Name(), err)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
err := a.plugin.Start(pm)
|
|
|
|
if err != nil {
|
|
|
|
return fmt.Errorf("plugin '%s' failed to start: %w", a.plugin.Name(), err)
|
|
|
|
}
|
|
|
|
|
|
|
|
a.logger.Info().Str("plugin", a.plugin.Name()).Msg("plugin started")
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2022-05-03 11:17:04 +00:00
|
|
|
type dependiencies struct {
|
|
|
|
deps map[string]*dependency
|
|
|
|
logger zerolog.Logger
|
|
|
|
}
|
|
|
|
|
|
|
|
func (d *dependiencies) addPlugin(plugin types.Plugin) error {
|
2023-03-31 12:02:33 +02:00
|
|
|
pluginName := plugin.Name()
|
|
|
|
|
|
|
|
plug, ok := d.deps[pluginName]
|
2022-05-03 11:17:04 +00:00
|
|
|
if !ok {
|
|
|
|
plug = &dependency{}
|
|
|
|
} else if plug.plugin != nil {
|
2023-03-31 12:02:33 +02:00
|
|
|
return fmt.Errorf("plugin '%s' already added", pluginName)
|
2022-05-03 11:17:04 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
plug.plugin = plugin
|
|
|
|
plug.logger = d.logger
|
2023-03-31 12:02:33 +02:00
|
|
|
d.deps[pluginName] = plug
|
2022-05-03 11:17:04 +00:00
|
|
|
|
|
|
|
dplug, ok := plugin.(types.DependablePlugin)
|
|
|
|
if !ok {
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2023-03-31 12:02:33 +02:00
|
|
|
for _, depName := range dplug.DependsOn() {
|
|
|
|
dependsOn, ok := d.deps[depName]
|
2022-05-03 11:17:04 +00:00
|
|
|
if !ok {
|
|
|
|
dependsOn = &dependency{}
|
|
|
|
} else if dependsOn.plugin != nil {
|
|
|
|
// if there is a cyclical dependency, break it and return error
|
2023-03-31 12:02:33 +02:00
|
|
|
if tdep, ok := dependsOn.findPlugin(pluginName); ok {
|
2022-05-03 11:17:04 +00:00
|
|
|
dependsOn.dependsOn = nil
|
2023-03-31 12:02:33 +02:00
|
|
|
delete(d.deps, pluginName)
|
|
|
|
return fmt.Errorf("cyclical dependency detected: '%s' <-> '%s'", pluginName, tdep.plugin.Name())
|
2022-05-03 11:17:04 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
plug.dependsOn = append(plug.dependsOn, dependsOn)
|
2023-03-31 12:02:33 +02:00
|
|
|
d.deps[depName] = dependsOn
|
2022-05-03 11:17:04 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func (d *dependiencies) findPlugin(name string) (*dependency, bool) {
|
2023-03-31 12:02:33 +02:00
|
|
|
for _, dep := range d.deps {
|
|
|
|
plug, ok := dep.findPlugin(name)
|
|
|
|
if ok {
|
|
|
|
return plug, true
|
2022-05-03 11:17:04 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
return nil, false
|
|
|
|
}
|
|
|
|
|
2023-03-31 12:02:33 +02:00
|
|
|
func (d *dependiencies) start(pm types.PluginManagers) error {
|
|
|
|
for _, dep := range d.deps {
|
|
|
|
if err := dep.startPlugin(pm); err != nil {
|
|
|
|
return err
|
2022-05-03 11:17:04 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func (d *dependiencies) forEach(f func(*dependency) error) error {
|
2023-03-31 12:02:33 +02:00
|
|
|
for _, dep := range d.deps {
|
|
|
|
if err := f(dep); err != nil {
|
2022-05-03 11:17:04 +00:00
|
|
|
return err
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func (d *dependiencies) len() int {
|
|
|
|
return len(d.deps)
|
|
|
|
}
|