2018-11-10 22:29:24 +01:00
|
|
|
package versioning
|
|
|
|
|
|
|
|
import (
|
2018-11-18 01:41:24 +01:00
|
|
|
"net/http"
|
|
|
|
|
2019-10-25 00:27:02 +02:00
|
|
|
"github.com/kataras/iris/v12/context"
|
|
|
|
"github.com/kataras/iris/v12/core/router"
|
2018-11-10 22:29:24 +01:00
|
|
|
)
|
|
|
|
|
|
|
|
type (
|
|
|
|
vroute struct {
|
|
|
|
method string
|
|
|
|
path string
|
|
|
|
versions Map
|
|
|
|
}
|
|
|
|
|
2018-11-18 01:41:24 +01:00
|
|
|
// Group is a group of version-based routes.
|
|
|
|
// One version per one or more routes.
|
2018-11-10 22:29:24 +01:00
|
|
|
Group struct {
|
2018-11-11 16:27:31 +01:00
|
|
|
version string
|
|
|
|
extraMethods []string
|
|
|
|
routes []vroute
|
2018-11-10 22:29:24 +01:00
|
|
|
|
|
|
|
deprecation DeprecationOptions
|
|
|
|
}
|
|
|
|
)
|
|
|
|
|
2018-11-18 01:41:24 +01:00
|
|
|
// NewGroup returns a ptr to Group based on the given "version".
|
|
|
|
//
|
|
|
|
// See `Handle` and `RegisterGroups` for more.
|
2018-11-10 22:29:24 +01:00
|
|
|
func NewGroup(version string) *Group {
|
|
|
|
return &Group{
|
|
|
|
version: version,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Deprecated marks this group and all its versioned routes
|
|
|
|
// as deprecated versions of that endpoint.
|
2018-11-18 01:41:24 +01:00
|
|
|
// It can be called in the end just before `RegisterGroups`
|
2018-11-10 22:29:24 +01:00
|
|
|
// or first by `NewGroup(...).Deprecated(...)`. It returns itself.
|
|
|
|
func (g *Group) Deprecated(options DeprecationOptions) *Group {
|
|
|
|
// if `Deprecated` is called in the end.
|
|
|
|
for _, r := range g.routes {
|
|
|
|
r.versions[g.version] = Deprecated(r.versions[g.version], options)
|
|
|
|
}
|
|
|
|
|
|
|
|
// store the options if called before registering any versioned routes.
|
|
|
|
g.deprecation = options
|
|
|
|
|
|
|
|
return g
|
|
|
|
}
|
|
|
|
|
2018-11-18 01:41:24 +01:00
|
|
|
// AllowMethods can be called before `Handle/Get/Post...`
|
|
|
|
// to tell the underline router that all routes should be registered
|
|
|
|
// to these "methods" as well.
|
2018-11-11 16:27:31 +01:00
|
|
|
func (g *Group) AllowMethods(methods ...string) *Group {
|
|
|
|
g.extraMethods = append(g.extraMethods, methods...)
|
|
|
|
return g
|
|
|
|
}
|
|
|
|
|
|
|
|
func (g *Group) addVRoute(method, path string, handler context.Handler) {
|
2018-11-18 01:41:24 +01:00
|
|
|
for _, r := range g.routes { // check if route already exists.
|
2018-11-11 16:27:31 +01:00
|
|
|
if r.method == method && r.path == path {
|
|
|
|
return
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
g.routes = append(g.routes, vroute{
|
|
|
|
method: method,
|
|
|
|
path: path,
|
|
|
|
versions: Map{g.version: handler},
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
2018-11-10 22:29:24 +01:00
|
|
|
// Handle registers a versioned route to the group.
|
2018-11-18 01:41:24 +01:00
|
|
|
// A call of `RegisterGroups` is necessary in order to register the actual routes
|
|
|
|
// when the group is complete.
|
2018-11-10 22:29:24 +01:00
|
|
|
//
|
2018-11-18 01:41:24 +01:00
|
|
|
// `RegisterGroups` for more.
|
2018-11-11 16:27:31 +01:00
|
|
|
func (g *Group) Handle(method string, path string, handler context.Handler) {
|
2018-11-10 22:29:24 +01:00
|
|
|
if g.deprecation.ShouldHandle() { // if `Deprecated` called first.
|
|
|
|
handler = Deprecated(handler, g.deprecation)
|
|
|
|
}
|
|
|
|
|
2018-11-11 16:27:31 +01:00
|
|
|
methods := append(g.extraMethods, method)
|
|
|
|
|
|
|
|
for _, method := range methods {
|
|
|
|
g.addVRoute(method, path, handler)
|
|
|
|
}
|
2018-11-10 22:29:24 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
// None registers an "offline" versioned route
|
|
|
|
// see `context#ExecRoute(routeName)` and routing examples.
|
|
|
|
func (g *Group) None(path string, handler context.Handler) {
|
|
|
|
g.Handle(router.MethodNone, path, handler)
|
|
|
|
}
|
|
|
|
|
|
|
|
// Get registers a versioned route for the Get http method.
|
|
|
|
func (g *Group) Get(path string, handler context.Handler) {
|
2018-11-18 01:41:24 +01:00
|
|
|
g.Handle(http.MethodGet, path, handler)
|
2018-11-10 22:29:24 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
// Post registers a versioned route for the Post http method.
|
|
|
|
func (g *Group) Post(path string, handler context.Handler) {
|
2018-11-18 01:41:24 +01:00
|
|
|
g.Handle(http.MethodPost, path, handler)
|
2018-11-10 22:29:24 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
// Put registers a versioned route for the Put http method
|
|
|
|
func (g *Group) Put(path string, handler context.Handler) {
|
2018-11-18 01:41:24 +01:00
|
|
|
g.Handle(http.MethodPut, path, handler)
|
2018-11-10 22:29:24 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
// Delete registers a versioned route for the Delete http method.
|
|
|
|
func (g *Group) Delete(path string, handler context.Handler) {
|
2018-11-18 01:41:24 +01:00
|
|
|
g.Handle(http.MethodDelete, path, handler)
|
2018-11-10 22:29:24 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
// Connect registers a versioned route for the Connect http method.
|
|
|
|
func (g *Group) Connect(path string, handler context.Handler) {
|
2018-11-18 01:41:24 +01:00
|
|
|
g.Handle(http.MethodConnect, path, handler)
|
2018-11-10 22:29:24 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
// Head registers a versioned route for the Head http method.
|
|
|
|
func (g *Group) Head(path string, handler context.Handler) {
|
2018-11-18 01:41:24 +01:00
|
|
|
g.Handle(http.MethodHead, path, handler)
|
2018-11-10 22:29:24 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
// Options registers a versioned route for the Options http method.
|
|
|
|
func (g *Group) Options(path string, handler context.Handler) {
|
2018-11-18 01:41:24 +01:00
|
|
|
g.Handle(http.MethodOptions, path, handler)
|
2018-11-10 22:29:24 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
// Patch registers a versioned route for the Patch http method.
|
|
|
|
func (g *Group) Patch(path string, handler context.Handler) {
|
2018-11-18 01:41:24 +01:00
|
|
|
g.Handle(http.MethodPatch, path, handler)
|
2018-11-10 22:29:24 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
// Trace registers a versioned route for the Trace http method.
|
|
|
|
func (g *Group) Trace(path string, handler context.Handler) {
|
2018-11-18 01:41:24 +01:00
|
|
|
g.Handle(http.MethodTrace, path, handler)
|
2018-11-10 22:29:24 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
// Any registers a versioned route for ALL of the http methods
|
|
|
|
// (Get,Post,Put,Head,Patch,Options,Connect,Delete).
|
|
|
|
func (g *Group) Any(registeredPath string, handler context.Handler) {
|
|
|
|
g.Get(registeredPath, handler)
|
|
|
|
g.Post(registeredPath, handler)
|
|
|
|
g.Put(registeredPath, handler)
|
|
|
|
g.Delete(registeredPath, handler)
|
|
|
|
g.Connect(registeredPath, handler)
|
|
|
|
g.Head(registeredPath, handler)
|
|
|
|
g.Options(registeredPath, handler)
|
|
|
|
g.Patch(registeredPath, handler)
|
|
|
|
g.Trace(registeredPath, handler)
|
|
|
|
}
|
|
|
|
|
2018-11-18 01:41:24 +01:00
|
|
|
// RegisterGroups registers one or more groups to an `iris.Party` or to the root router.
|
|
|
|
// See `NewGroup` and `NotFoundHandler` too.
|
|
|
|
func RegisterGroups(r router.Party, notFoundHandler context.Handler, groups ...*Group) (actualRoutes []*router.Route) {
|
2018-11-10 22:29:24 +01:00
|
|
|
var total []vroute
|
|
|
|
for _, g := range groups {
|
|
|
|
inner:
|
|
|
|
for _, r := range g.routes {
|
|
|
|
for i, tr := range total {
|
|
|
|
if tr.method == r.method && tr.path == r.path {
|
2018-11-18 01:41:24 +01:00
|
|
|
total[i].versions[g.version] = r.versions[g.version]
|
2018-11-10 22:29:24 +01:00
|
|
|
continue inner
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-11-18 01:41:24 +01:00
|
|
|
total = append(total, r)
|
|
|
|
}
|
2018-11-10 22:29:24 +01:00
|
|
|
}
|
|
|
|
|
2018-11-18 01:41:24 +01:00
|
|
|
for _, vr := range total {
|
|
|
|
if notFoundHandler != nil {
|
|
|
|
vr.versions[NotFound] = notFoundHandler
|
2018-11-10 22:29:24 +01:00
|
|
|
}
|
|
|
|
|
2018-11-18 01:41:24 +01:00
|
|
|
route := r.Handle(vr.method, vr.path, NewMatcher(vr.versions))
|
|
|
|
actualRoutes = append(actualRoutes, route)
|
2018-11-10 22:29:24 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
return
|
|
|
|
}
|