目的和需求:部分go的核心文件不开源,例如验证,主程序核心逻辑等等
第一个想法,把子程序代码打包成静态文件,然后主程序执行
子程序
package mainimport ("fmt""github.com/gogf/gf/v2/os/gfile"
)func Valid() {contents := gfile.GetContents("cert.key")if contents != "123456" {fmt.Println("不通过")//主程序执行静态编译的文件,停止不到主程序//panic("not pass")}}func main() {Valid()}
执行打包命令:gf build -o valid
valid文件拷贝到主程序根目录
主程序
package hello
import ("context""fmt""gfdemo/api/hello/v1""gfdemo/internal/model""gfdemo/internal/service""github.com/gogf/gf/v2/frame/g""github.com/gogf/gf/v2/util/gconv""os""os/exec"
)
func init() {//方法一:直接执行二进制静态文件cmd := exec.Command("./valid")// 将标准输出连接到当前程序的标准输出cmd.Stdout = os.Stdout// 运行命令err := cmd.Run()if err != nil {fmt.Println("Error:", err.Error())}
}
结论:这个方案,达不到预期,因为子程序验证不通过,没办法停止主程序
第二个想法,把子程序打包成动态库,主程序调用
子程序
package mainfunc Valid(name string) string {if name != "123456" {panic("不通过")}return "Hello world"
}
执行打包命令:go build -buildmode=plugin -o core.so main.go
core.so文件拷贝到主程序路径 :core/core.so
主程序
package hello
import ("context""fmt""gfdemo/api/hello/v1""gfdemo/internal/model""gfdemo/internal/service""github.com/gogf/gf/v2/frame/g""github.com/gogf/gf/v2/util/gconv""os""os/exec"
)
func init() {contents := gfile.GetContents("cert.key")plug, err := plugin.Open("core/core.so")if err != nil {fmt.Println("Error loading plugin:", err)return}validSymbol, err := plug.Lookup("Valid")if err != nil {fmt.Println("Error Lookup plugin:", err)return}validFunc, ok := validSymbol.(func(string) string)if !ok {fmt.Println("Error asserting function type")return}result := validFunc(contents)fmt.Println(result)
}
结论:满足需求,子程序可以停止主程序
这里只是简单演示,如果要真正在项目上面实现这个需求,肯定不能单纯靠验证根目录文件限制,我觉得可以把一些核心逻辑代码放到子程序,主程序再调用执行,同时子程序也要验证根目录证书,这样主程序必定要用到子程序的逻辑才能完整运行,达到预期目的
Plugin文档:https://pkg.go.dev/plugin
demo代码上传在 github:https://github.com/gzdzh/gfdemo