Go plugin 的性能和一些问题
最近我学习了下公司内的一个插件(lego)平台,插件平台深度使用了Go插件去帮助业务去实现一些动态可加载插件来解决业务中的一些问题,其核心就是利用了Go plugin 机制, 本文会介绍Go plugin的性能和一些坑!
项目地址: github.com/Anthony-Don…
Go 插件性能
Go调用Go插件
测试是 Go程序使用Go插件,执行func Add(x,y int) int
函数,测试代码在 benchmark_test.go
~ make golang_benchmark
cd plugin && bash -e build.sh
cd golang && CGO_ENABLED=1 go test -v -run=none -bench=Benchmark -count=2 -benchmem ./benchmark/...
goos: linux
goarch: amd64
pkg: github.com/anthony-dong/cgo_demo/golang/benchmark
cpu: Intel(R) Xeon(R) Platinum 8260 CPU @ 2.40GHz
BenchmarkGoPlugin
BenchmarkGoPlugin-8 569536491 2.010 ns/op 0 B/op 0 allocs/op
BenchmarkGoPlugin-8 579598886 2.037 ns/op 0 B/op 0 allocs/op
BenchmarkGoNative
BenchmarkGoNative-8 1000000000 0.3624 ns/op 0 B/op 0 allocs/op
BenchmarkGoNative-8 1000000000 0.3492 ns/op 0 B/op 0 allocs/op
PASS
ok github.com/anthony-dong/cgo_demo/golang/benchmark 3.550s
结论就是非常给力 !!!说实话这种函数调用的开销已经非常低了,Linux上也就是1ns左右(linux x86 && go1.18+)! 性能非常高! 损耗的原因还是动态库的损耗,都能接受!
C调用Go插件(Cgo)
c调用Go插件,是通过将Go插件编译成 c动态库 或者 c静态库,比较推荐静态库!可以参考 plugin4 、plugin5 , 测试代码在 add_benchmark.cpp
- 测试代码
#include <benchmark/benchmark.h>
#include <stdexcept>
#include "plugin.h"
static void BM_CGO(benchmark::State& state) {
// Perform setup here
for (auto _ : state) {
// This code gets timed
auto x = int(Add(GoInt(1), GoInt(2)));
if (x != 3) {
throw std::runtime_error("异常");
}
}
}
static void BM_Native(benchmark::State& state) {
// Perform setup here
for (auto _ : state) {
// This code gets timed
auto x = 1 + 2;
if (x != 3) {
throw std::runtime_error("异常");
}
}
}
BENCHMARK(BM_CGO);
BENCHMARK(BM_Native);
// Run the benchmark
BENCHMARK_MAIN();
- 测试结果:
Running output/add_benchmark
Run on (8 X 2394.37 MHz CPU s)
CPU Caches:
L1 Data 32 KiB (x4)
L1 Instruction 32 KiB (x4)
L2 Unified 1024 KiB (x4)
L3 Unified 36608 KiB (x2)
Load Average: 0.12, 0.17, 0.17
-----------------------------------------------------
Benchmark Time CPU Iterations
-----------------------------------------------------
BM_CGO 1480 ns 1480 ns 481544
BM_Native 0.000 ns 0.000 ns 1000000000000
- 性能损耗 (火焰图)
- 总结
cpp/c 调用 go插件(c lib) 会存在性能劣化的问题,大概单次调用损耗在 1480 ns (linux x86 && go1.18+) 左右! 所以不太推荐c调用go动态库!在字节内部很多业务线迁移至rust后,会存在存量Go插件的迁移,导致rust 调用 go插件性能劣化严重(这个代码比较简单,只涉及到单次调用的开开销,如果逻辑复杂点可能开销更大),本质上原因就是 Go/C 这个桥梁差异较大,我们通过火焰图可以看到大部分开销都在于 cgocallback 上!
Go调用C (Cgo)
这里也是性能劣化严重,推荐直接看我的文章:Cgo介绍和使用
Go插件的问题细说
plugin 示例
- go.mod
require github.com/tidwall/gjson v1.14.0 // 主程序版本是v1.14.0
- 插件代码实现相关函数实现
import "github.com/tidwall/gjson"
func GetJsonRow(input string, path string) string {
return gjson.Get(input, path).Raw
}
- 编译此plugin
bash build.sh
主程序 示例
- go.mod
require github.com/tidwall/gjson v1.11.0 // 主程序版本是v1.11.0
- 代码
package main
import (
"fmt"
"github.com/tidwall/gjson"
"plugin"
)
func loadGetJsonRowFunc(lib string) func(input string, path string) string {
p, err := plugin.Open(lib)
if err != nil {
panic(err)
}
foo, err := p.Lookup("GetJsonRow")
if err != nil {
panic(err)
}
return foo.(func(string, string) string)
}
func main() {
lib := "plugin1/output/plugin.so"
result := loadGetJsonRowFunc(lib)(`{"k1": true}`, "k1")
fmt.Println("result =", result)
}
- 运行报错
panic: plugin.Open("plugin1/output/plugin"): plugin was built with a different version of package github.com/tidwall/gjson
问题
panic: plugin.Open("plugin1/output/plugin"): plugin was built with a different version of package github.com/tidwall/gjson
如果插件中依赖的包的版本和主程序中依赖的版本不一致,会导致编译不通过!
如何解决
- github.com/tidwall/gjson 包存在不同的版本,那么办法简单就是把依赖的包名给全改了就行了! 那么就能避免Go检测的问题
github.com/tidwall/gjson -> t_xxxx/github.com/tidwall/gjson
import "github.com/tidwall/gjson"
func GetJsonPath(input string, path string) gjson.Result {
return gjson.Get(input, path)
}
- 此时就会出现一个问题
panic: interface conversion: plugin.Symbol is func(string, string) gjson.Result, not func(string, string) gjson.Result (types from different scopes)
gjson.Result 在插件中已经被我们替换了包名,实际上是 t_xxxx/github.com/tidwall/gjson.Result
,导致不一致,所以不能直接用结构体进行传输!!!
因此在做数据交互的时候尽可能的使用 builtin type. 如果不满足自行做序列话!
但是大部分插件都可以使用 接口
的方式避免此问题!
最佳实践
- 插件代码
// 全部都以接口的形式处理!!
// 这个接口也可以放到一个包里!!
type Request interface {
GetHeader(key string) string
GetBody() ([]byte, error)
}
type Response interface {
SetHeader(key, value string)
SetBody([]byte) error
}
type plugin struct {
}
// 插件初始化
func (*plugin) Init() error {
return nil
}
// 插件卸载
func (*plugin) Close() error {
return nil
}
// 插件逻辑
func (*plugin) Handle(ctx context.Context, req interface{}, resp interface{}) error {
request := req.(Request) // 接口
response := resp.(Response) // 接口
body, err := request.GetBody()
if err != nil {
return err
}
fmt.Printf("recevie http request body: %s\n", body)
if err := response.SetBody(body); err != nil {
return err
}
return nil
}
// NewPlugin 创建插件,每个插件都需要定义此函数,且函数签名一致
func NewPlugin() interface{} {
return &plugin{}
}
- 加载插件
package http_plugin
import (
"plugin"
)
func Load(lib string) Plugin {
p, err := plugin.Open(lib)
if err != nil {
panic(err)
}
foo, err := p.Lookup("NewPlugin")
if err != nil {
panic(err)
}
return foo.(func() interface{})().(Plugin)
}
- 使用
func main() {
lib := http_plugin.Load("plugin3/output/plugin.so") // 动态加载
if err := lib.Init(); err != nil {
panic(err)
}
defer lib.Close()
if err := http.ListenAndServe(":8080", http.HandlerFunc(func(writer http.ResponseWriter, request *http.Request) {
ctx := context.Background()
if err := lib.Handle(ctx, http_plugin.NewHTTPRequest(request), http_plugin.NewHTTPResponse(writer)); err != nil {
fmt.Printf("[ERROR] %s\n", err.Error())
}
})); err != nil {
panic(err)
}
}
总结
- Go插件提供了很高的性能,但是会存在一个问题,隔离性的问题,导致插件隔离不恰当会出现程序挂掉的问题,比如插件内部代码panic了,且未抓取程序直接挂了,
- 其次Go插件不支持卸载,所以插件内部尽可能的避免使用一些全局变量!
- 每个插件之间的包尽可能的采用 plugin2 脚本处理的方式将包名全部替换,避免依赖包之间的版本冲突!!
- 插件之间不应该直接依赖实体类型,例如
struct
等,推荐通过Go接口方式依赖,参考 plugin3 实现方式是比较好的做法,具体实现代码在 http_plugin - Go插件性能非常的高,Go调用Go插件,性能非常高,基本没有开销,非常适合Go业务服务去利用插件去解决一些动态化的东西或者可服用插件之类的需求!
- Go插件如果导出
C动态库/静态库
被cpp/rust等语言调用的化性能非常的差,劣化较为严重,甚用!
转载自:https://juejin.cn/post/7385163643297234979