您好,登錄后才能下訂單哦!
這篇文章主要介紹如何解決Golang庫插件注冊加載機制的問題,文中介紹的非常詳細,具有一定的參考價值,感興趣的小伙伴們一定要看完!
最近看到一個內部項目的插件加載機制,非常贊。當然這里說的插件并不是指的golang原生的可以在buildmode中加載指定so文件的那種加載機制。而是軟件設計上的「插件」。如果你的軟件是一個框架,或者一個平臺性產品,想要提升擴展性,即可以讓第三方進行第三方庫開發,最終能像搭積木一樣將這些庫組裝起來。那么就可能需要這種庫加載機制。
我們的目標是什么?對第三方庫進行某種庫規范,只要按照這種庫規范進行開發,這個庫就可以被加載到框架中。
我們先定義一個插件的數據結構,這里肯定是需要使用接口來規范,這個可以根據你的項目自由發揮,比如我希望插件有一個Setup方法來在啟動的時候加載即可。然后我就定義如下的Plugin結構。
type Plugin interface{ Name() string Setup(config map[string]string) error }
而在框架啟動的時候,我啟動了一個如下的全局變量:
var plugins map[string]Plugin
有人可能會問,這里有了加載函數setup,但是為什么沒有注冊邏輯呢?
答案是注冊的邏輯放在庫的init函數中。
即框架還提供了一個注冊函數。
// package plugin Register(plugin Plugin)
這個register就是實現了將第三方plugin放到plugins全局變量中。
所以第三方的plugin庫大致實現如下:
package MyPlugin type MyPlugin struct{ } func (m *MyPlugin) Setup(config map[string]string) error { // TODO func (m *MyPlugin) Name() string { return "myPlugin" func init() { plugin.Register(&MyPlugin)
這樣注冊的邏輯就變成了,如果你要加載一個插件,那么你在main.go中直接以 _ import的形式引入即可。
package main _ import "github.com/foo/myplugin" func main() { }
整體的感覺,這樣子插件的注冊就被“隱藏”到import中了。
注冊的邏輯其實看起來也平平無奇,但是加載的邏輯就考驗細節了。
首先插件的加載其實有兩點需要考慮:
配置
依賴
配置指的是插件一定是有某種配置的,這些配置以配置文件yaml中plugins.myplugin的路徑存在。
plugins: myplugin: foo: bar
其實我對這種實現持保留意見。配置文件以一個文件中配置項的形式存在,好像不如以配置文件的形式存在,即以config/plugins/myplugin.yaml 的文件。
這樣不會出現一個大配置文件的問題。畢竟每個配置文件本身就是一門DSL語言。如果你將配置文件的邏輯變復雜,一定會有很多附帶的bug是由于配置文件錯誤導致的。
第二個說的是依賴。插件A依賴與插件B,那么這里就有加載函數Setup的先后順序了。這種先后順序如果純依賴用戶的“經驗”,將某個插件的Setup調用放在某個插件的Setup調用之前,是非常痛苦的。(雖然一定是有辦法可以做到)。更好的辦法是依賴于框架自身的加載機制來進行加載。
首先我們在plugin包中定義一個接口:
type Depend interface{ DependOn() []string }
如果我的插件依賴一個名字為 “fooPlugin” 的插件,那么我的插件 MyPlugin就會實現這個接口。
package MyPlugin type MyPlugin struct{ } func (m *MyPlugin) Setup(config map[string]string) error { // TODO func (m *MyPlugin) Name() string { return "myPlugin" func init() { plugin.Register(&MyPlugin) func (m *MyPlugin) DependOn() []string { return []string{"fooPlugin"}
在最終加載所有插件的時候,我們并不是簡單地將所有插件調用Setup,而是使用一個channel,將所有插件放在channel中,然后一個個調用Setup,遇到有Depend其他插件的,且依賴插件還未被加載,則將當前插件放在隊列最后(重新塞入channel)。
var setupStatus map[string]bool // 獲取所有注冊插件 func loadPlugins() (plugin chan Plugin, setupStatus map[string]bool) { // 這里定義一個長度為10的隊列 var sortPlugin = make(chan Plugin, 10) var setupStatus = make[string]bool // 所有的插件 for name, plugin := range plugins { sortPlugin <- plugin setupStatus[name] = false } return sortPlugin, setupStatus } // 加載所有插件 func SetupPlugins(pluginChan chan Plugin, setupStatus map[string]bool) error { num := len(pluginChan) for num > 0 { plugin <- pluginChan canSetup := true if deps, ok := p.(Depend); ok { depends := deps.DependOn() for _, dependName := range depends{ if _, setuped := setupStatus[dependName]; !setup { // 有未加載的插件 canSetup = false break } } } // 如果這個插件能被setup if canSetup { plugin.Setup(xxx) setupStatus[p.Name()] = true } else { // 如果插件不能被setup, 這個plugin就塞入到最后一個隊列 pluginChan <- plugin return nil }
上面這段代碼最精妙的就是使用了一個有buffer的channel作為一個隊列,消費隊列一方SetupPlugins,除了消費隊列,也有可能生產數據到隊列,這樣就保證了隊列中所有plugin都是被按照標記的依賴被順序加載的。
以上是“如何解決Golang庫插件注冊加載機制的問題”這篇文章的所有內容,感謝各位的閱讀!希望分享的內容對大家有幫助,更多相關知識,歡迎關注億速云行業資訊頻道!
免責聲明:本站發布的內容(圖片、視頻和文字)以原創、轉載和分享為主,文章觀點不代表本網站立場,如果涉及侵權請聯系站長郵箱:is@yisu.com進行舉報,并提供相關證據,一經查實,將立刻刪除涉嫌侵權內容。