Maison >développement back-end >Golang >Versionnement basé sur l'en-tête sur Golang

Versionnement basé sur l'en-tête sur Golang

WBOY
WBOYavant
2024-02-10 19:03:08495parcourir

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 !

Contenu de la question

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()
    }
}

Solution de contournement

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/userget

(

et

).

Exemple de code

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
  • Vous pouvez le faire en utilisant l'emballage de gin ci-dessous.
    Exemple
  • 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")
    }
    

Emballage

    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
    }
    
    
  • Demande d'échantillon/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!

Déclaration:
Cet article est reproduit dans:. en cas de violation, veuillez contacter admin@php.cn Supprimer