文章插图
在前两篇文章(Golang 模块获取包modfetch研读,Golang模块代理goproxy.io源码研读),我们学习了Golang Module Proxy的工作原理以及实现原理 。
本文尝试独立实现一个Golang Module Proxy服务 。
实现逻辑主要涉及这几块内容:
a)main.go 负责服务启动,服务优雅终止;
b)generate.sh 负责将$GOROOT中的internal包拷贝至当前项目并替换引用路径;
【实现一个 Golang Module Proxy】c)proxy.go 核心逻辑部分,负责工作目录设定,路径检查,Module请求处理 。
下面详细看一下这几部分的代码 。
1 main.go头部的//go:generate注释指定脚本generate.sh,当执行go generate时,其会调用generate.sh将modfetch包及其依赖包从$GOROOT中的internal文件夹拷贝至当前项目,然后即可以在当前项目直接使用了 。
初始化一个http.Server,其Handler使用proxy.go的proxy.Proxy函数 。
启动一个goroutine监听中断信号,以便优雅的终止服务(如何优雅的终止一个服务?) 。
//go:generate sh generate.shpackage mainimport ( ... "github.com/olzhy/goproxy/pkg/proxy")var port = flag.String("serverPort", ":8080", "server port")func main() { // server srv := http.Server{ Addr: *port, Handler: proxy.Proxy(), } // server startup / gracefully shutdown ... srv.ListenAndServe() ...}2 generate.shgenerate.sh负责将$GOROOT中的internal包拷贝至当前项目并将引用路径替换为新的引用路径 。
#!/bin/bashmkdir internal# copy dependenciescp -r $GOROOT/src/cmd/go/internal/modfetch ./internal/...cp -r $GOROOT/src/cmd/internal/sys ./internal/...# replace import pathsfind . -type f -name "*.go" -exec sed -i '' 's#cmd/go/internal/#github.com/olzhy/goproxy/internal/#g' {} ; ...3 proxy.gopkg/proxy/proxy.go提供proxy.Proxy函数 。
proxy.go首先会设置工作目录,启动后对于一个GET请求,首先会校验请求路径,对不满足规则的请求直接返回404,然后仅对这几类符合Module请求格式的请求作处理:
a)后缀为“/@v/list”
如GET github.com/olzhy/quote/@v/list
从请求路径截取mod名称,调用modfetch.Lookup函数返回所有可用版本 。
b)后缀为“/@latest”
如GET github.com/olzhy/quote/@latest
从请求路径截取mod名称,调用modfetch.Lookup函数获取最近一次提交信息 。
c)后缀为“.info”
如GET github.com/olzhy/quote/@v/v1.0.0.info
从请求路径截取mod及version信息,调用modfetch.Stat函数获取info 。
d)后缀为“.mod”
如GET github.com/olzhy/quote/@v/v1.0.0.mod
从请求路径截取mod及version信息,调用modfetch.GoMod函数获取mod内容 。
e)后缀为“.zip”
如GET github.com/olzhy/quote/@v/v1.0.0.zip
从请求路径截取mod及version信息,调用modfetch.DownloadZip函数获取zip文件路径名称并提供下载 。
package proxyimport ( ... "github.com/olzhy/goproxy/internal/modfetch" ...)const ( ListSuffix = "/@v/list" LatestSuffix = "/@latest" InfoSuffix = ".info" ModSuffix = ".mod" ZipSuffix = ".zip" VInfix = "/@v/")func init() { modfetch.PkgMod = ... codehost.WorkRoot = ...}func Proxy() http.HandlerFunc { return http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) { path := strings.Trim(r.RequestURI, "/") // req path validation if err := pathValidation(path); nil != err { w.WriteHeader(http.StatusBadRequest) fmt.Fprintln(w, err) return } switch { // suffix is /@v/list case strings.HasSuffix(path, ListSuffix): ... // call modfetch.Lookup(mod) lookupVersions(mod) ... return // suffix is /@latest case strings.HasSuffix(path, LatestSuffix):... // modfetch.Lookup(mod) lookupLatestRev(mod) ... return // suffix is .info case strings.HasSuffix(path, InfoSuffix): ... // call modfetch.Stat(mod, rev) loadRev(mod, ver) ...return // suffix is .mod case strings.HasSuffix(path, ModSuffix): ... // call modfetch.GoMod(mod, rev)loadModContent(mod, ver) ...return // suffix is .zip case strings.HasSuffix(path, ZipSuffix): ... // call modfetch.DownloadZip(module.Version{Path: mod, Version: rev}) loadZip(mod, ver) ...return default:w.WriteHeader(http.StatusBadRequest)fmt.Fprintln(w, "please give me a correct module query") } })}完整实现代码已提交至GitHub(github.com/olzhy/goproxy),欢迎大家关注 。
推荐阅读
- 怎么样给皮肤深层补水
- 吃不完的葱如何保存,放一个月也不会坏,有需要的收藏
- 又甜又脆的羊角蜜怎么挑?很简单,看这5点,一挑一个准
- 如何运营好一个淘宝店铺 淘宝店铺运营规划方案
- 7个华为超实用小技巧,别说你一个都不知道
- Java 分布式系统如何实现session共享?
- 如何用 Fedora 学习 Golang
- 麦当劳原味甜筒多少钱一个 麦当劳香芋脆皮甜筒多少钱一个
- 梦见自己生了两个男孩是什么征兆 梦见自己生了两个男孩一个女孩
- 皮皮虾剥皮有妙招,一根筷子就搞定,3秒钟剥一个,再不怕扎手了