/
plugins.go
119 lines (92 loc) · 2.75 KB
/
plugins.go
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
/*
Copyright 2018 The Kubernetes Authors.
Licensed under the Apache License, Version 2.0 (the "License");
you may not use this file except in compliance with the License.
You may obtain a copy of the License at
http://www.apache.org/licenses/LICENSE-2.0
Unless required by applicable law or agreed to in writing, software
distributed under the License is distributed on an "AS IS" BASIS,
WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
See the License for the specific language governing permissions and
limitations under the License.
*/
package framework
import (
"fmt"
"path/filepath"
"plugin"
"strings"
"sync"
"k8s.io/klog"
)
var pluginMutex sync.Mutex
// PluginBuilder plugin management
type PluginBuilder = func(Arguments) Plugin
// Plugin management
var pluginBuilders = map[string]PluginBuilder{}
// RegisterPluginBuilder register the plugin
func RegisterPluginBuilder(name string, pc PluginBuilder) {
pluginMutex.Lock()
defer pluginMutex.Unlock()
pluginBuilders[name] = pc
}
// CleanupPluginBuilders cleans up all the plugin
func CleanupPluginBuilders() {
pluginMutex.Lock()
defer pluginMutex.Unlock()
pluginBuilders = map[string]PluginBuilder{}
}
// GetPluginBuilder get the pluginbuilder by name
func GetPluginBuilder(name string) (PluginBuilder, bool) {
pluginMutex.Lock()
defer pluginMutex.Unlock()
pb, found := pluginBuilders[name]
return pb, found
}
// LoadCustomPlugins loads custom implement plugins
func LoadCustomPlugins(pluginsDir string) error {
pluginPaths, _ := filepath.Glob(fmt.Sprintf("%s/*.so", pluginsDir))
for _, pluginPath := range pluginPaths {
pluginBuilder, err := loadPluginBuilder(pluginPath)
if err != nil {
return err
}
pluginName := getPluginName(pluginPath)
RegisterPluginBuilder(pluginName, pluginBuilder)
klog.V(4).Infof("Custom plugin %s loaded", pluginName)
}
return nil
}
func getPluginName(pluginPath string) string {
return strings.TrimSuffix(filepath.Base(pluginPath), filepath.Ext(pluginPath))
}
func loadPluginBuilder(pluginPath string) (PluginBuilder, error) {
plug, err := plugin.Open(pluginPath)
if err != nil {
return nil, err
}
symBuilder, err := plug.Lookup("New")
if err != nil {
return nil, err
}
builder, ok := symBuilder.(PluginBuilder)
if !ok {
return nil, fmt.Errorf("unexpected plugin: %s, failed to convert PluginBuilder `New`", pluginPath)
}
return builder, nil
}
// Action management
var actionMap = map[string]Action{}
// RegisterAction register action
func RegisterAction(act Action) {
pluginMutex.Lock()
defer pluginMutex.Unlock()
actionMap[act.Name()] = act
}
// GetAction get the action by name
func GetAction(name string) (Action, bool) {
pluginMutex.Lock()
defer pluginMutex.Unlock()
act, found := actionMap[name]
return act, found
}