2
0
Files
logikonline 12f4ea03a8
Some checks failed
Build and Release / Create Release (push) Successful in 0s
Trigger Vault Plugin Rebuild / Trigger Vault Rebuild (push) Successful in 0s
Build and Release / Integration Tests (PostgreSQL) (push) Successful in 2m48s
Build and Release / Lint (push) Failing after 5m2s
Build and Release / Build Binaries (amd64, windows, windows-latest) (push) Has been skipped
Build and Release / Build Binaries (amd64, darwin, linux-latest) (push) Has been skipped
Build and Release / Build Binaries (amd64, linux, linux-latest) (push) Has been skipped
Build and Release / Build Binaries (arm64, darwin, linux-latest) (push) Has been skipped
Build and Release / Build Binaries (arm64, linux, linux-latest) (push) Has been skipped
Build and Release / Unit Tests (push) Successful in 5m37s
refactor: add /v3 suffix to module path for proper Go semver
Go's semantic import versioning requires v2+ modules to include the
major version in the module path. This enables using proper version
tags (v3.x.x) instead of pseudo-versions.

Updated module path: code.gitcaddy.com/server/v3
2026-01-17 17:53:59 -05:00

90 lines
2.1 KiB
Go

// Copyright 2026 MarketAlly. All rights reserved.
// SPDX-License-Identifier: MIT
package middleware
import (
"context"
"net/http"
"unicode"
"code.gitcaddy.com/server/v3/modules/setting"
"github.com/google/uuid"
)
type requestIDKeyType struct{}
var requestIDKey = requestIDKeyType{}
// RequestIDHeader is the header name for request ID
const RequestIDHeader = "X-Request-ID"
// maxRequestIDByteLength is the maximum length for a request ID
const maxRequestIDByteLength = 40
// GetRequestID returns the request ID from context
func GetRequestID(ctx context.Context) string {
if id, ok := ctx.Value(requestIDKey).(string); ok {
return id
}
return ""
}
// isSafeRequestID checks if the request ID contains only printable characters
func isSafeRequestID(id string) bool {
for _, r := range id {
if !unicode.IsPrint(r) {
return false
}
}
return true
}
// parseOrGenerateRequestID extracts request ID from headers or generates a new one
func parseOrGenerateRequestID(req *http.Request) string {
// Try to get from configured headers
for _, key := range setting.Log.RequestIDHeaders {
if id := req.Header.Get(key); id != "" {
if isSafeRequestID(id) {
if len(id) > maxRequestIDByteLength {
return id[:maxRequestIDByteLength]
}
return id
}
}
}
// Also check X-Request-ID header explicitly
if id := req.Header.Get(RequestIDHeader); id != "" {
if isSafeRequestID(id) {
if len(id) > maxRequestIDByteLength {
return id[:maxRequestIDByteLength]
}
return id
}
}
// Generate a new request ID (short form of UUID)
id := uuid.New()
return id.String()[:8]
}
// RequestID returns a middleware that sets X-Request-ID header
func RequestID() func(http.Handler) http.Handler {
return func(next http.Handler) http.Handler {
return http.HandlerFunc(func(w http.ResponseWriter, req *http.Request) {
requestID := parseOrGenerateRequestID(req)
// Store in context
ctx := context.WithValue(req.Context(), requestIDKey, requestID)
req = req.WithContext(ctx)
// Set response header
w.Header().Set(RequestIDHeader, requestID)
next.ServeHTTP(w, req)
})
}
}