Maison >développement back-end >Golang >Versionnement basé sur l'en-tête sur Golang
Le contrôle de version basé sur les en-têtes sur Golang est un moyen efficace de gérer les versions de code pendant le développement. En ajoutant des informations de version au début des fichiers de code, les développeurs peuvent facilement suivre les modifications du code et l'historique des versions. Cette méthode de contrôle de version est non seulement facile à utiliser, mais également adaptée aux projets de toutes tailles. L'éditeur PHP Xigua présentera en détail l'utilisation et les précautions du contrôle de version basé sur les en-têtes sur Golang dans cet article pour aider les développeurs à mieux gérer et maintenir le code. Que vous soyez débutant ou développeur expérimenté, cet article vous fournira des références et des conseils précieux. Explorons ensemble cette technologie de contrôle de version intéressante et pratique !
Je souhaite utiliser gin pour implémenter un contrôle de version basé sur l'en-tête lors de mes déplacements. Je pense utiliser une fonction middleware pour faire cela sur le routeur.
Le client appellera la même URL d'API et la version sera dans un en-tête http personnalisé comme ceci :
Appelez la version 1 Récupérer /users/12345678 Version acceptée : v1
Appelez la version 2 : Récupérer /users/12345678 Version acceptée : v2
Ainsi, le routeur peut reconnaître l'en-tête et appeler la version spécifique. Des choses comme ça :
router := gin.Default() v1 := router.Group("/v1") v1.Use(VersionMiddleware()) v1.GET("/user/:id", func(c *gin.Context) { c.String(http.StatusOK, "This is the v1 API") }) v2 := router.Group("/v2") v2.Use(VersionMiddleware()) v2.GET("/user/:id", func(c *gin.Context) { c.String(http.StatusOK, "This is the v2 API") }) func VersionMiddleware() gin.HandlerFunc { return func(c *gin.Context) { version := c.Request.Header.Get(configuration.GetConfigValue("VersionHeader")) // Construct the new URL path based on the version number path := fmt.Sprintf("/%s%s", version, c.Request.URL.Path) // Modify the request URL path to point to the new version-specific endpoint c.Request.URL.Path = path c.Next() } }
Veuillez vérifier l'extrait de code ci-dessous. J'utilise reverseproxy pour rediriger vers une version donnée. Vous devez vérifier soigneusement une version donnée. Sinon, cela entraînera des appels récursifs.
Remarque : j'ai utilisé deux versions de /user
/user
get(/v1/user
和 /v2/user
get
package main import ( "net/http" "net/http/httputil" "regexp" "github.com/gin-gonic/gin" ) func main() { router := gin.default() router.use(versionmiddleware()) v1 := router.group("/v1") v1.get("/user", func(c *gin.context) { c.string(http.statusok, "this is the v1 api") }) v2 := router.group("/v2") v2.get("/user", func(c *gin.context) { c.string(http.statusok, "this is the v2 api") }) router.run(":8082") } func versionmiddleware() gin.handlerfunc { return func(c *gin.context) { // you need to check c.request.url.path whether // already have a version or not, if it has a valid // version, return. regex, _ := regexp.compile("/v[0-9]+") ver := regex.matchstring(c.request.url.path) if ver { return } version := c.request.header.get("accept-version") // you need to validate given version by the user here. // if version is not a valid version, return error // mentioning that given version is invalid. director := func(req *http.request) { r := c.request req.url.scheme = "http" req.url.host = r.host req.url.path = "/"+ version + r.url.path } proxy := &httputil.reverseproxy{director: director} proxy.servehttp(c.writer, c.request) } }ou
package main import ( "net/http" "github.com/gin-gonic/gin" "github.com/udayangaac/stackoverflow/golang/75860989/ginwrapper" ) func main() { engine := gin.default() router := ginwrapper.newrouter(engine) defaultrouter := router.default() defaultrouter.get("/profile",func(ctx *gin.context) { }) v1 := router.withversion("/v1") v1.get("/user",func(ctx *gin.context) { ctx.string(http.statusok, "this is the profile v1 api") }) v2 := router.withversion("/v2") v2.get("/user",func(ctx *gin.context) { ctx.string(http.statusok, "this is the profile v2 api") }) engine.run(":8082") }
package ginwrapper import ( "fmt" "net/http" "github.com/gin-gonic/gin" ) type router struct { router *gin.engine versiongroups map[string]*gin.routergroup } type versionedrouter struct { version string router } func newrouter(router *gin.engine) *router { return &router{ router: router, versiongroups: make(map[string]*gin.routergroup), } } func (a *router) default() versionedrouter { return versionedrouter{router: *a } } func (a *router) withversion(version string) versionedrouter { if _,ok := a.versiongroups[version]; ok { panic("cannot initialize same version multiple times") } a.versiongroups[version] = a.router.group(version) return versionedrouter{router: *a,version:version } } func (vr versionedrouter) get(relativepath string, handlers ...gin.handlerfunc) { vr.handle(http.methodget,relativepath,handlers...) } // note: you need to follow the same for other http methods. // as an example, we can write a method for post http method as below, // // func (vr versionedrouter) post(relativepath string, handlers ...gin.handlerfunc) { // vr.handle(http.methodpost,relativepath,handlers...) // } func (vr versionedrouter)handle(method,relativepath string, handlers ...gin.handlerfunc) { if !vr.isrouteexist(method,relativepath) { vr.router.handle(method,relativepath,func(ctx *gin.context) { version := ctx.request.header.get("accept-version") if len(version) == 0 { ctx.string(http.statusbadrequest,"accept-version header is empty") } ctx.request.url.path = fmt.sprintf("/%s%s", version, ctx.request.url.path) vr.router.handlecontext(ctx) }) } versionedrelativepath := vr.version + relativepath if !vr.isrouteexist(method,versionedrelativepath) { vr.router.handle(method,versionedrelativepath,handlers... ) } } func (a versionedrouter) isrouteexist(method,relativepath string) bool { for _,route := range a.router.routes() { if route.method == method && relativepath == route.path { return true } } return false }
/v1/user
/v2/user
curl --location 'localhost:8082/user' \ --header 'accept-version: v1'
curl --location 'localhost:8082/user' \ --header 'Accept-version: v2'
Ce qui précède est le contenu détaillé de. pour plus d'informations, suivez d'autres articles connexes sur le site Web de PHP en chinois!