2020-05-28 15:20:58 +02:00
|
|
|
package requestid
|
|
|
|
|
|
|
|
import (
|
2021-01-09 04:41:20 +01:00
|
|
|
"crypto/sha256"
|
2020-07-15 10:18:40 +02:00
|
|
|
"encoding/hex"
|
|
|
|
"net/http/httputil"
|
|
|
|
|
2020-05-28 15:20:58 +02:00
|
|
|
"github.com/kataras/iris/v12/context"
|
|
|
|
|
|
|
|
"github.com/google/uuid"
|
|
|
|
)
|
|
|
|
|
|
|
|
func init() {
|
|
|
|
context.SetHandlerName("iris/middleware/requestid.*", "iris.request.id")
|
|
|
|
}
|
|
|
|
|
2020-07-15 10:18:40 +02:00
|
|
|
const xRequestIDHeaderKey = "X-Request-Id"
|
2020-05-28 15:20:58 +02:00
|
|
|
|
|
|
|
// Generator defines the function which should extract or generate
|
|
|
|
// a Request ID. See `DefaultGenerator` and `New` package-level functions.
|
2020-07-10 22:21:09 +02:00
|
|
|
type Generator func(ctx *context.Context) string
|
2020-05-28 15:20:58 +02:00
|
|
|
|
|
|
|
// DefaultGenerator is the default `Generator` that is used
|
|
|
|
// when nil is passed on `New` package-level function.
|
|
|
|
// It extracts the ID from the "X-Request-ID" request header value
|
|
|
|
// or, if missing, it generates a new UUID(v4) and sets the header and context value.
|
|
|
|
//
|
|
|
|
// See `Get` package-level function too.
|
2020-07-10 22:21:09 +02:00
|
|
|
var DefaultGenerator Generator = func(ctx *context.Context) string {
|
2020-07-15 10:18:40 +02:00
|
|
|
id := ctx.ResponseWriter().Header().Get(xRequestIDHeaderKey)
|
|
|
|
if id != "" {
|
|
|
|
return id
|
|
|
|
}
|
2020-05-28 15:20:58 +02:00
|
|
|
|
2020-07-15 10:18:40 +02:00
|
|
|
id = ctx.GetHeader(xRequestIDHeaderKey)
|
2020-05-28 15:20:58 +02:00
|
|
|
if id == "" {
|
|
|
|
uid, err := uuid.NewRandom()
|
|
|
|
if err != nil {
|
|
|
|
ctx.StopWithStatus(500)
|
|
|
|
return ""
|
|
|
|
}
|
|
|
|
|
|
|
|
id = uid.String()
|
|
|
|
}
|
|
|
|
|
2020-07-15 10:18:40 +02:00
|
|
|
ctx.Header(xRequestIDHeaderKey, id)
|
2020-05-28 15:20:58 +02:00
|
|
|
return id
|
|
|
|
}
|
|
|
|
|
2020-07-15 10:18:40 +02:00
|
|
|
// HashGenerator uses the request's hash to generate a fixed-length Request ID.
|
|
|
|
// Note that one or many requests may contain the same ID, so it's not unique.
|
|
|
|
func HashGenerator(includeBody bool) Generator {
|
|
|
|
return func(ctx *context.Context) string {
|
|
|
|
ctx.Header(xRequestIDHeaderKey, Hash(ctx, includeBody))
|
|
|
|
return DefaultGenerator(ctx)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-05-28 15:20:58 +02:00
|
|
|
// New returns a new request id middleware.
|
2020-06-07 14:26:06 +02:00
|
|
|
// It optionally accepts an ID Generator.
|
2020-05-28 15:20:58 +02:00
|
|
|
// The Generator can stop the handlers chain with an error or
|
|
|
|
// return a valid ID (string).
|
|
|
|
// If it's nil then the `DefaultGenerator` will be used instead.
|
2020-06-07 14:26:06 +02:00
|
|
|
func New(generator ...Generator) context.Handler {
|
|
|
|
gen := DefaultGenerator
|
|
|
|
if len(generator) > 0 {
|
|
|
|
gen = generator[0]
|
2020-05-28 15:20:58 +02:00
|
|
|
}
|
|
|
|
|
2020-07-10 22:21:09 +02:00
|
|
|
return func(ctx *context.Context) {
|
2020-05-28 15:20:58 +02:00
|
|
|
if Get(ctx) != "" {
|
|
|
|
ctx.Next()
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
id := gen(ctx)
|
|
|
|
if ctx.IsStopped() {
|
|
|
|
// ctx.Next checks that
|
|
|
|
// but we don't want to call SetID if generator failed.
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
ctx.SetID(id)
|
|
|
|
ctx.Next()
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Get returns the Request ID or empty string.
|
|
|
|
//
|
|
|
|
// A shortcut of `context.GetID().(string)`.
|
2020-07-10 22:21:09 +02:00
|
|
|
func Get(ctx *context.Context) string {
|
2020-05-28 15:20:58 +02:00
|
|
|
v := ctx.GetID()
|
|
|
|
if v != nil {
|
|
|
|
if id, ok := v.(string); ok {
|
|
|
|
return id
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return ""
|
|
|
|
}
|
2020-07-15 10:18:40 +02:00
|
|
|
|
|
|
|
// Hash returns the sha1 hash of the request.
|
|
|
|
// It does not capture error, instead it returns an empty string.
|
|
|
|
func Hash(ctx *context.Context, includeBody bool) string {
|
2021-01-09 04:41:20 +01:00
|
|
|
h := sha256.New() // sha1 fits here as well.
|
2020-07-15 10:18:40 +02:00
|
|
|
b, err := httputil.DumpRequest(ctx.Request(), includeBody)
|
|
|
|
if err != nil {
|
|
|
|
return ""
|
|
|
|
}
|
|
|
|
h.Write(b)
|
|
|
|
return hex.EncodeToString(h.Sum(nil))
|
|
|
|
}
|