| 49 | } |
| 50 | |
| 51 | func LoadGoPlugins(basicRes context.BasicRes) errors.Error { |
| 52 | pluginsDir := basicRes.GetConfig("PLUGIN_DIR") |
| 53 | var wg sync.WaitGroup |
| 54 | walkErr := filepath.WalkDir(pluginsDir, func(path string, d fs.DirEntry, err error) error { |
| 55 | if err != nil { |
| 56 | return err |
| 57 | } |
| 58 | fileName := d.Name() |
| 59 | if strings.HasSuffix(fileName, ".so") && fileName != ".so" { |
| 60 | pluginName := fileName[0 : len(d.Name())-3] |
| 61 | plug, loadErr := goplugin.Open(path) |
| 62 | if loadErr != nil { |
| 63 | return loadErr |
| 64 | } |
| 65 | symPluginEntry, pluginEntryError := plug.Lookup("PluginEntry") |
| 66 | if pluginEntryError != nil { |
| 67 | return pluginEntryError |
| 68 | } |
| 69 | pluginMeta, ok := symPluginEntry.(plugin.PluginMeta) |
| 70 | if !ok { |
| 71 | return errors.Default.New(fmt.Sprintf("%s PluginEntry must implement PluginMeta interface", pluginName)) |
| 72 | } |
| 73 | wg.Add(1) |
| 74 | go func(pluginName string, pluginMeta plugin.PluginMeta) { |
| 75 | defer func() { |
| 76 | wg.Done() |
| 77 | }() |
| 78 | err = plugin.RegisterPlugin(pluginName, pluginMeta) |
| 79 | if err != nil { |
| 80 | panic(err) |
| 81 | } |
| 82 | basicRes.GetLogger().Info(`plugin loaded %s`, pluginName) |
| 83 | }(pluginName, pluginMeta) |
| 84 | } |
| 85 | return nil |
| 86 | }) |
| 87 | wg.Wait() |
| 88 | return errors.Convert(walkErr) |
| 89 | } |
| 90 | |
| 91 | func LoadRemotePlugins(basicRes context.BasicRes) errors.Error { |
| 92 | remotePluginDir := basicRes.GetConfig("REMOTE_PLUGIN_DIR") |