Documentation ¶
Overview ¶
Package garcon is a server for API and static website including middlewares to manage rate-limit, Cookies, JWT, CORS, OPA, web traffic, Prometheus export and PProf.
Index ¶
- Constants
- Variables
- func AppendPrefixes(origins []string, prefixes ...string) []string
- func AppendURLs(urls []*url.URL, prefixes ...*url.URL) []*url.URL
- func ConvertSize(sizeInBytes int) string
- func ConvertSize64(sizeInBytes int64) string
- func DecodeJSONRequest(w http.ResponseWriter, r *http.Request, msg any, maxBytes ...int) error
- func DecodeJSONResponse(r *http.Response, msg any, maxBytes ...int) error
- func Deduplicate[T comparable](duplicates []T) []T
- func DefaultContactSettings() map[string][2]int
- func DefaultFileSettings() map[string][2]int
- func DevOrigins() []*url.URL
- func EnvInt(envvar string, fallback ...int) int
- func EnvStr(envvar string, fallback ...string) string
- func ExtractWords(csv string, dictionary []string) []string
- func FingerprintMD(r *http.Request) string
- func InvalidPath(w http.ResponseWriter, r *http.Request)
- func ListenAndServe(server *http.Server) error
- func LoadPolicy(filenames []string) (*ast.Compiler, error)
- func LogVersion()
- func MiddlewareCORS(origins, methods, headers []string, debug bool) func(next http.Handler) http.Handler
- func MiddlewareLogDuration(next http.Handler) http.Handler
- func MiddlewareLogDurationSafe(next http.Handler) http.Handler
- func MiddlewareLogFingerprint(next http.Handler) http.Handler
- func MiddlewareLogFingerprintSafe(next http.Handler) http.Handler
- func MiddlewareLogRequest(next http.Handler) http.Handler
- func MiddlewareLogRequestSafe(next http.Handler) http.Handler
- func MiddlewareRejectUnprintableURI(next http.Handler) http.Handler
- func MiddlewareSecureHTTPHeader(secure bool) func(next http.Handler) http.Handler
- func MiddlewareServerHeader(version string) func(next http.Handler) http.Handler
- func NewHash() (hash.Hash, error)
- func NotImplemented(w http.ResponseWriter, r *http.Request)
- func Obfuscate(str string) (string, error)
- func OriginsFromURLs(urls []*url.URL) []string
- func OverwriteBufferContent(b []byte)
- func ParseURLs(origins []string) []*url.URL
- func PrintVersion(program string)
- func Printable(array ...string) int
- func PrintableRune(r rune) bool
- func ProbeCPU() interface{ ... }
- func RandomBytes(n int) []byte
- func SafeHeader(r *http.Request, header string) string
- func Sanitize(str string) string
- func ServeVersion() func(w http.ResponseWriter, r *http.Request)
- func Server(h http.Handler, port int, connState ...func(net.Conn, http.ConnState)) http.Server
- func SetCustomVersionFlag(fs *flag.FlagSet, flagName, program string)
- func SetVersionFlag()
- func SplitClean(values string) []string
- func StartPProfServer(port int)
- func TraversalPath(w http.ResponseWriter, r *http.Request) bool
- func UnmarshalJSONRequest[T json.Unmarshaler](w http.ResponseWriter, r *http.Request, msg T, maxBytes ...int) error
- func UnmarshalJSONResponse[T json.Unmarshaler](r *http.Response, msg T, maxBytes ...int) error
- func Value(r *http.Request, key, header string) (string, error)
- func Values(r *http.Request, key string) ([]string, error)
- func Version(serverName string) string
- func WriteErr(w http.ResponseWriter, r *http.Request, statusCode int, kv ...any)
- func WriteErrSafe(w http.ResponseWriter, r *http.Request, statusCode int, kv ...any)
- func WriteOK(w http.ResponseWriter, kv ...any)
- type AdaptiveRate
- type Chain
- type Garcon
- func (g *Garcon) IncorruptibleChecker(secretKeyHex string, maxAge int, setIP bool) *incorruptible.Incorruptible
- func (g *Garcon) JWTChecker(secretKeyHex string, planPerm ...any) *JWTChecker
- func (g *Garcon) MiddlewareCORS() Middleware
- func (g *Garcon) MiddlewareCORSWithMethodsHeaders(methods, headers []string) Middleware
- func (g *Garcon) MiddlewareLogDuration(safe ...bool) Middleware
- func (g *Garcon) MiddlewareLogRequest(settings ...string) Middleware
- func (g *Garcon) MiddlewareOPA(opaFilenames ...string) Middleware
- func (g *Garcon) MiddlewareRateLimiter(settings ...int) Middleware
- func (Garcon) MiddlewareRejectUnprintableURI() Middleware
- func (g *Garcon) MiddlewareServerHeader(serverName ...string) Middleware
- func (g *Garcon) NewContactForm(redirectURL string) WebForm
- func (g *Garcon) NewStaticWebServer(dir string) StaticWebServer
- func (g *Garcon) StartMetricsServer(expPort int) (Chain, func(net.Conn, http.ConnState))
- type JWTChecker
- func (ck *JWTChecker) Chk(next http.Handler) http.Handler
- func (ck *JWTChecker) Cookie(i int) *http.Cookie
- func (ck *JWTChecker) NewCookie(name, plan, user string, secure bool, dns, dir string) http.Cookie
- func (ck *JWTChecker) PermFromBearerOrCookie(r *http.Request) (Perm, []any)
- func (ck *JWTChecker) PermFromCookie(r *http.Request) (Perm, []any)
- func (ck *JWTChecker) PermFromJWT(JWT string) (Perm, []any)
- func (ck *JWTChecker) Set(next http.Handler) http.Handler
- func (ck *JWTChecker) Vet(next http.Handler) http.Handler
- type Middleware
- type Muter
- type Option
- type Perm
- type Policy
- type RTChain
- type RTMiddleware
- type ReqLimiter
- type RoundTripperFunc
- type ServerName
- type StaticWebServer
- func (ws *StaticWebServer) ServeAssets() func(w http.ResponseWriter, r *http.Request)
- func (ws *StaticWebServer) ServeDir(contentType string) func(w http.ResponseWriter, r *http.Request)
- func (ws *StaticWebServer) ServeFile(urlPath, contentType string) func(w http.ResponseWriter, r *http.Request)
- func (ws *StaticWebServer) ServeImages() func(w http.ResponseWriter, r *http.Request)
- type TokenChecker
- type WebForm
- type Writer
- func (gw Writer) InvalidPath(w http.ResponseWriter, r *http.Request)
- func (gw Writer) NotImplemented(w http.ResponseWriter, r *http.Request)
- func (gw Writer) WriteErr(w http.ResponseWriter, r *http.Request, statusCode int, kv ...any)
- func (gw Writer) WriteErrSafe(w http.ResponseWriter, r *http.Request, statusCode int, kv ...any)
- func (gw Writer) WriteOK(w http.ResponseWriter, kv ...any)
Constants ¶
const ( // DefaultPlan is the plan name in absence of "permissions" parametric parameters. DefaultPlan = "VIP" // DefaultPerm is the perm value in absence of "permissions" parametric parameters. DefaultPerm = 1 )
const FingerprintExplanation = `` /* 683-byte string literal not displayed */
FingerprintExplanation provides a description of the logged HTTP headers.
Variables ¶
var ( ErrExpiredToken = errors.New("expired or invalid refresh token") ErrJWTSignature = errors.New("JWT signature mismatch") ErrNoAuthorization = errors.New("provide your JWT within the 'Authorization Bearer' HTTP header") ErrNoBase64JWT = errors.New("the token claims (second part of the JWT) is not base64-valid") ErrNoBearer = errors.New("malformed HTTP Authorization, must be Bearer") ErrNoValidJWT = errors.New("cannot find a valid JWT in either the cookie or the first 'Authorization' HTTP header") )
var ErrEmptyOPAFilename = errors.New("OPA: missing filename")
var ErrNonPrintable = errors.New("non-printable")
var V string
V is set at build time using the `-ldflags` build flag:
v="$(git describe --tags --always --broken)" go build -ldflags="-X 'github.com/teal-finance/garcon.V=$v'" ./cmd/main/package
The following commands provide a semver-like version format such as "v1.2.0-my-branch+3" where "+3" is the number of commits since "v1.2.0". If no tag in the Git repo, $t is the long SHA1 of the last commit.
t="$(git describe --tags --abbrev=0 --always)" b="$(git branch --show-current)" [ _$b = _main ] && b="" || b="-$b" n="$(git rev-list --count "$t"..)" [ "$n" -eq 0 ] && n="" || n="+$n" go build -ldflags="-X 'github.com/teal-finance/garcon.V=$t$b$n'" ./cmd/main/package
Functions ¶
func AppendPrefixes ¶ added in v0.5.1
func ConvertSize ¶ added in v0.17.0
ConvertSize converts a size in bytes into the most appropriate unit among KiB, MiB, GiB, TiB, PiB and EiB. 1 KiB is 1024 bytes as defined by the ISO/IEC 80000-13:2008 standard. See: https://wikiless.org/wiki/ISO%2FIEC_80000#Units_of_the_ISO_and_IEC_80000_series
func ConvertSize64 ¶ added in v0.17.0
ConvertSize64 is similar ConvertSize but takes in input an int64.
func DecodeJSONRequest ¶ added in v0.23.5
DecodeJSONRequest decodes the JSON from the request body.
func DecodeJSONResponse ¶ added in v0.23.5
DecodeJSONResponse decodes the JSON from the request body.
func Deduplicate ¶ added in v0.24.0
func Deduplicate[T comparable](duplicates []T) []T
Deduplicate makes a slice of elements unique: it returns a slice with only the unique elements in it.
func DefaultContactSettings ¶ added in v0.16.0
DefaultContactSettings is compliant with standard names for web form input fields: https://html.spec.whatwg.org/multipage/form-control-infrastructure.html#inappropriate-for-the-control
func DefaultFileSettings ¶ added in v0.16.0
DefaultFileSettings sets FileLimits with only "file".
func DevOrigins ¶ added in v0.4.3
DevOrigins provides the development origins: - yarn run vite --port 3000 - yarn run vite preview --port 5000 - localhost:8085 on multi devices: web auto-reload using https://github.com/synw/fwr - flutter run --web-port=8080 - 192.168.1.x + any port on tablet: mobile app using fast builtin auto-reload.
func EnvInt ¶ added in v0.24.0
EnvInt does the same as EnvStr but expects the value is an integer. EnvInt panics if the envvar value cannot be parsed as an integer.
func EnvStr ¶ added in v0.24.0
EnvStr searches the environment variable (envvar) and returns its value if found, otherwise returns the optional fallback value. In absence of fallback, "" is returned.
func ExtractWords ¶ added in v0.24.0
ExtractWords converts comma-separated values into a slice of unique words found in the dictionary.
The search is case-insensitive and is based on common prefix: the input value "foo" selects the first word in the dictionary that starts with "foo" (as "food" for example).
Moreover the special value "ALL" means all the dictionary words.
No guarantees are made about ordering. However the returned words are not duplicated. Note this operation alters the content of the dictionary: the found words are replaced by the last dictionary words. Clone the input dictionary if it needs to be preserved:
d2 := append([]string{}, dictionary...) words := garcon.ExtractWords(csv, d2)
func FingerprintMD ¶ added in v0.17.0
FingerprintMD provide the browser fingerprint in markdown format. Attention: read the .
func InvalidPath ¶ added in v0.17.0
func InvalidPath(w http.ResponseWriter, r *http.Request)
func ListenAndServe ¶ added in v0.23.0
ListenAndServe runs the HTTP server(s) in foreground. Optionally it also starts a metrics server in background (if export port > 0). The metrics server is for use with Prometheus or another compatible monitoring tool.
func LoadPolicy ¶ added in v0.17.0
LoadPolicy checks the Rego filenames and loads them to build the OPA compiler.
func LogVersion ¶ added in v0.20.0
func LogVersion()
LogVersion logs the version and (Git) commit information.
func MiddlewareCORS ¶ added in v0.23.0
func MiddlewareCORS(origins, methods, headers []string, debug bool) func(next http.Handler) http.Handler
MiddlewareCORS uses restrictive CORS values.
func MiddlewareLogDuration ¶ added in v0.23.0
MiddlewareLogDuration logs the requested URL along with the time to handle it.
func MiddlewareLogDurationSafe ¶ added in v0.23.0
MiddlewareLogDurationSafe is similar to MiddlewareLogDurations but also sanitizes the URL.
func MiddlewareLogFingerprint ¶ added in v0.23.0
MiddlewareLogFingerprint is the middleware to log incoming HTTP request and browser fingerprint.
func MiddlewareLogFingerprintSafe ¶ added in v0.23.0
MiddlewareLogFingerprintSafe is similar to MiddlewareLogFingerprints but sanitize the URL.
func MiddlewareLogRequest ¶ added in v0.23.0
MiddlewareLogRequest is the middleware to log the requester IP and the requested URL.
func MiddlewareLogRequestSafe ¶ added in v0.23.0
MiddlewareLogRequestSafe is similar to LogRequest but sanitize the URL.
func MiddlewareRejectUnprintableURI ¶ added in v0.23.0
MiddlewareRejectUnprintableURI is a middleware rejecting HTTP requests having a Carriage Return "\r" or a Line Feed "\n" within the URI to prevent log injection.
func MiddlewareSecureHTTPHeader ¶ added in v0.23.0
MiddlewareSecureHTTPHeader is a middleware adding recommended HTTP response headers to secure the web application.
func MiddlewareServerHeader ¶ added in v0.23.0
MiddlewareServerHeader is the middleware setting the Server HTTP header in the response.
func NewHash ¶ added in v0.17.0
NewHash is based on HighwayHash, a hashing algorithm enabling high speed (especially on AMD64). See the study on HighwayHash and some other hash functions: https://github.com/fwessels/HashCompare
func NotImplemented ¶ added in v0.17.0
func NotImplemented(w http.ResponseWriter, r *http.Request)
func Obfuscate ¶ added in v0.17.0
Obfuscate hashes the input string to prevent logging sensitive information.
func OriginsFromURLs ¶ added in v0.9.0
func OverwriteBufferContent ¶ added in v0.22.0
func OverwriteBufferContent(b []byte)
OverwriteBufferContent is to erase a secret when it is no longer required.
func PrintVersion ¶ added in v0.17.0
func PrintVersion(program string)
PrintVersion prints the version and exits. The version may also contain the (Git) commit information.
func Printable ¶ added in v0.17.0
Printable returns -1 when all the strings are safely printable else returns the position of the rejected character.
The non printable characters are:
- Carriage Return "\r"
- Line Feed "\n"
- other ASCII control codes (except space)
- invalid UTF-8 codes
Printable can be used to preventing log injection.
When multiple strings are passed, the returned position is sum with the string index multiplied by 1000.
func PrintableRune ¶ added in v0.17.0
PrintableRune returns false if rune is a Carriage Return "\r", a Line Feed "\n", another ASCII control code (except space), or an invalid UTF-8 code. PrintableRune can be used to prevent log injection.
func ProbeCPU ¶ added in v0.17.0
func ProbeCPU() interface{ Stop() }
ProbeCPU is used like the following:
defer pprof.ProbeCPU.Stop()
When the caller reaches its function end, the defer executes Stop() that writes the file "cpu.pprof". To visualize "cpu.pprof" use the pprof tool:
cd ~/go go get -u github.com/google/pprof cd - pprof -http=: cpu.pprof
or using one single command line:
go run github.com/google/pprof@latest -http=: cpu.pprof
func RandomBytes ¶ added in v0.21.0
func SafeHeader ¶ added in v0.17.0
SafeHeader stringifies a safe list of HTTP header values.
func Sanitize ¶ added in v0.17.0
Sanitize replaces control codes by the tofu symbol and invalid UTF-8 codes by the replacement character. Sanitize can be used to prevent log injection.
Inspired from: - https://wikiless.org/wiki/Replacement_character#Replacement_character - https://graphicdesign.stackexchange.com/q/108297
func ServeVersion ¶ added in v0.22.0
func ServeVersion() func(w http.ResponseWriter, r *http.Request)
ServeVersion send HTML or JSON depending on Accept header.
func Server ¶ added in v0.23.0
Server returns a default http.Server ready to handle API endpoints, static web pages...
func SetCustomVersionFlag ¶ added in v0.20.0
SetCustomVersionFlag register PrintVersion() for the version flag.
Example with default values:
import "github.com/teal-finance/garcon" func main() { garcon.SetCustomVersionFlag(nil, "", "") flag.Parse() }
Example with custom values values:
import "github.com/teal-finance/garcon" func main() { garcon.SetCustomVersionFlag(nil, "v", "MyApp") flag.Parse() }
func SetVersionFlag ¶ added in v0.17.0
func SetVersionFlag()
SetVersionFlag defines -version flag to print the version stored in V. See SetCustomVersionFlag for a more flexibility.
func SplitClean ¶
SplitClean splits the values and trim them.
func StartPProfServer ¶ added in v0.17.0
func StartPProfServer(port int)
StartPProfServer starts a PProf server in background. Endpoints usage example:
curl http://localhost:6063/debug/pprof/allocs > allocs.pprof pprof -http=: allocs.pprof wget http://localhost:31415/debug/pprof/goroutine pprof -http=: goroutine wget http://localhost:31415/debug/pprof/heap pprof -http=: heap wget http://localhost:31415/debug/pprof/trace pprof -http=: trace
func TraversalPath ¶ added in v0.17.0
func TraversalPath(w http.ResponseWriter, r *http.Request) bool
TraversalPath returns true when path contains ".." to prevent path traversal attack.
func UnmarshalJSONRequest ¶ added in v0.23.5
func UnmarshalJSONRequest[T json.Unmarshaler](w http.ResponseWriter, r *http.Request, msg T, maxBytes ...int) error
UnmarshalJSONRequest unmarshals the JSON from the request body.
func UnmarshalJSONResponse ¶ added in v0.23.5
UnmarshalJSONResponse unmarshals the JSON from the request body.
func Value ¶ added in v0.12.3
Value returns the /endpoint/{key} (URL path) else the "key" form (HTTP body) else the "key" query string (URL) else the HTTP header. Value requires chi.URLParam().
func Version ¶ added in v0.17.0
Version format is "Program-1.2.3". If the program argument is empty, the format is "v1.2.3". If V is empty, Version uses the main module version.
func WriteErrSafe ¶ added in v0.21.0
func WriteOK ¶ added in v0.21.0
func WriteOK(w http.ResponseWriter, kv ...any)
Types ¶
type AdaptiveRate ¶ added in v0.24.0
AdaptiveRate continuously adjusts the timing between requests to prevent the API responds "429 Too Many Requests". AdaptiveRate increases/decreases the rate depending on absence/presence of the 429 status code.
func NewAdaptiveRate ¶ added in v0.24.0
func NewAdaptiveRate(name string, d time.Duration) AdaptiveRate
func (*AdaptiveRate) Get ¶ added in v0.24.0
func (ar *AdaptiveRate) Get(symbol, url string, msg any, maxBytes ...int) error
func (*AdaptiveRate) LogStats ¶ added in v0.24.0
func (ar *AdaptiveRate) LogStats()
type Chain ¶ added in v0.17.0
type Chain []Middleware
Chain acts as a list of http.Handler middleware. Chain is effectively immutable: once created, it will always hold the same set of middleware in the same order.
func NewChain ¶ added in v0.17.0
func NewChain(chain ...Middleware) Chain
NewChain creates a new chain, memorizing the given list of middleware. NewChain serves no other function, middleware is only constructed upon a call to chain.Then().
func StartMetricsServer ¶ added in v0.17.0
StartMetricsServer creates and starts the Prometheus export server.
func (Chain) Append ¶ added in v0.17.0
func (c Chain) Append(chain ...Middleware) Chain
Append extends a chain, adding the provided middleware as the last ones in the request flow.
chain := garcon.NewChain(m1, m2) chain = chain.Append(m3, m4) // requests in chain go m1 -> m2 -> m3 -> m4
func (Chain) Then ¶ added in v0.17.0
Then chains the middlewares and returns the final http.Handler.
garcon.NewChain(m1, m2, m3).Then(h)
is equivalent to:
m1(m2(m3(h)))
When the request comes in, it will be passed to m1, then m2, then m3 and finally, the given handler (assuming every middleware calls the following one).
A chain can be safely reused by calling Then() several times.
chain := garcon.NewChain(rateLimitHandler, csrfHandler) indexPipe = chain.Then(indexHandler) authPipe = chain.Then(authHandler)
Note: every call to Then() calls all middleware pieces. Thus several instances of the same middleware will be created when a chain is reused in this previous example. For proper middleware, this should cause no problem.
Then() treats nil as http.DefaultServeMux.
func (Chain) ThenFunc ¶ added in v0.17.0
func (c Chain) ThenFunc(fn http.HandlerFunc) http.Handler
ThenFunc works identically to Then, but takes a HandlerFunc instead of a Handler.
The following two statements are equivalent:
c.Then(http.HandlerFunc(fn)) c.ThenFunc(fn)
ThenFunc provides all the guarantees of Then.
type Garcon ¶
type Garcon struct { ServerName ServerName Writer Writer // contains filtered or unexported fields }
func (*Garcon) IncorruptibleChecker ¶ added in v0.23.0
func (g *Garcon) IncorruptibleChecker(secretKeyHex string, maxAge int, setIP bool) *incorruptible.Incorruptible
IncorruptibleChecker uses cookies based the fast and tiny Incorruptible token. IncorruptibleChecker requires g.WithURLs() to set the Cookie secure, domain and path.
func (*Garcon) JWTChecker ¶ added in v0.6.0
func (g *Garcon) JWTChecker(secretKeyHex string, planPerm ...any) *JWTChecker
JWTChecker requires WithURLs() to set the Cookie name, secure, domain and path.
func (*Garcon) MiddlewareCORS ¶ added in v0.23.0
func (g *Garcon) MiddlewareCORS() Middleware
MiddlewareCORS is a middleware to handle Cross-Origin Resource Sharing (CORS).
func (*Garcon) MiddlewareCORSWithMethodsHeaders ¶ added in v0.23.3
func (g *Garcon) MiddlewareCORSWithMethodsHeaders(methods, headers []string) Middleware
MiddlewareCORSWithMethodsHeaders is a middleware to handle Cross-Origin Resource Sharing (CORS).
func (*Garcon) MiddlewareLogDuration ¶ added in v0.23.0
func (g *Garcon) MiddlewareLogDuration(safe ...bool) Middleware
MiddlewareLogDuration logs the requested URL along with its handling time. When the optional parameter safe is true, this middleware sanitizes the URL before printing it.
func (*Garcon) MiddlewareLogRequest ¶ added in v0.23.0
func (g *Garcon) MiddlewareLogRequest(settings ...string) Middleware
MiddlewareLogRequest logs the incoming request URL. If one of its optional parameter is "fingerprint", this middleware also logs the browser fingerprint. If the other optional parameter is "safe", this middleware sanitizes the URL before printing it.
func (*Garcon) MiddlewareOPA ¶ added in v0.23.0
func (g *Garcon) MiddlewareOPA(opaFilenames ...string) Middleware
MiddlewareOPA creates the middleware for Authentication rules (Open Policy Agent).
func (*Garcon) MiddlewareRateLimiter ¶ added in v0.23.0
func (g *Garcon) MiddlewareRateLimiter(settings ...int) Middleware
func (Garcon) MiddlewareRejectUnprintableURI ¶ added in v0.23.0
func (Garcon) MiddlewareRejectUnprintableURI() Middleware
MiddlewareRejectUnprintableURI is a middleware rejecting HTTP requests having a Carriage Return "\r" or a Line Feed "\n" within the URI to prevent log injection.
func (*Garcon) MiddlewareServerHeader ¶ added in v0.23.0
func (g *Garcon) MiddlewareServerHeader(serverName ...string) Middleware
func (*Garcon) NewContactForm ¶ added in v0.22.0
func (*Garcon) NewStaticWebServer ¶ added in v0.23.0
func (g *Garcon) NewStaticWebServer(dir string) StaticWebServer
NewStaticWebServer creates a StaticWebServer.
type JWTChecker ¶ added in v0.20.0
type JWTChecker struct {
// contains filtered or unexported fields
}
func NewJWTChecker ¶ added in v0.20.0
func (*JWTChecker) Chk ¶ added in v0.20.0
func (ck *JWTChecker) Chk(next http.Handler) http.Handler
Chk is a middleware to accept only HTTP requests having a valid cookie. Then, Chk puts the permission (of the JWT) in the request context.
func (*JWTChecker) Cookie ¶ added in v0.20.0
func (ck *JWTChecker) Cookie(i int) *http.Cookie
Cookie returns a default cookie to facilitate testing.
func (*JWTChecker) PermFromBearerOrCookie ¶ added in v0.20.0
func (ck *JWTChecker) PermFromBearerOrCookie(r *http.Request) (Perm, []any)
func (*JWTChecker) PermFromCookie ¶ added in v0.20.0
func (ck *JWTChecker) PermFromCookie(r *http.Request) (Perm, []any)
func (*JWTChecker) PermFromJWT ¶ added in v0.20.0
func (ck *JWTChecker) PermFromJWT(JWT string) (Perm, []any)
func (*JWTChecker) Set ¶ added in v0.20.0
func (ck *JWTChecker) Set(next http.Handler) http.Handler
Set is a middleware putting a HttpOnly cookie in the HTTP response header when no valid cookie is present. The new cookie conveys the JWT of the first plan. Set also puts the permission from the JWT in the request context.
func (*JWTChecker) Vet ¶ added in v0.20.0
func (ck *JWTChecker) Vet(next http.Handler) http.Handler
Vet is a middleware to accept only the HTTP request having a valid JWT. The JWT can be either in the cookie or in the first "Authorization" header. Then, Vet puts the permission (of the JWT) in the request context.
type Middleware ¶ added in v0.17.0
Middleware is a constructor function returning a http.Handler.
type Muter ¶ added in v0.21.0
type Muter struct { // Threshold is the level enabling the muted state. Threshold int // NoAlertDuration allows to consider the verbosity is back to normal. NoAlertDuration time.Duration // RemindMuteState allows to remind the state is still muted. // Set value 100 to send this reminder every 100 increments. // Set to zero to disable this feature, RemindMuteState int // contains filtered or unexported fields }
Muter can be used to limit the logger/alerting verbosity. Muter detects when its internal counter is over the Threshold and waits the counter goes back to zero or after NoAlertDuration to return to normal situation. Muter uses the Hysteresis principle: https://wikiless.org/wiki/Hysteresis Similar wording: quieter, stopper, limiter, reducer, inhibitor, mouth-closer.
type Option ¶ added in v0.6.0
type Option func(*Garcon)
func WithDocURL ¶ added in v0.6.0
func WithServerName ¶ added in v0.23.0
type Perm ¶ added in v0.17.0
type Perm struct {
Value int
}
func PermFromCtx ¶ added in v0.20.0
PermFromCtx gets the permission information from the request context.
type Policy ¶ added in v0.17.0
type Policy struct {
// contains filtered or unexported fields
}
Policy manages the Open Policy Agent (OPA). see https://www.openpolicyagent.org/docs/edge/integration/#integrating-with-the-go-api
type RTChain ¶ added in v0.21.0
type RTChain []RTMiddleware
RTChain acts as a list of http.RoundTripper middlewares. RTChain is effectively immutable: once created, it will always hold the same set of middleware in the same order.
func NewRTChain ¶ added in v0.21.0
func NewRTChain(chain ...RTMiddleware) RTChain
NewRTChain creates a new chain of HTTP round trippers, memorizing the given list of RoundTrip middlewares. NewRTChain serves no other function, middlewares are only called upon a call to Then().
func (RTChain) Append ¶ added in v0.21.0
func (c RTChain) Append(chain ...RTMiddleware) RTChain
Append extends a chain, adding the specified middlewares as the last ones in the request flow.
Append returns a new chain, leaving the original one untouched.
Example #1
stdChain := garcon.NewRTChain(m1, m2) extChain := stdChain.Append(m3, m4) // requests in stdChain go m1 -> m2 // requests in extChain go m1 -> m2 -> m3 -> m4
Example #2
stdChain := garcon.NewRTChain(m1, m2) ext1Chain := garcon.NewRTChain(m3, m4) ext2Chain := stdChain.Append(ext1Chain...) // requests in stdChain go m1 -> m2 // requests in ext1Chain go m3 -> m4 // requests in ext2Chain go m1 -> m2 -> m3 -> m4
Example #3
aHtmlAfterNosurf := garcon.NewRTChain(m2) aHtml := garcon.NewRTChain(m1, func(rt http.RoundTripper) http.RoundTripper { csrf := nosurf.New(rt) csrf.SetFailureHandler(aHtmlAfterNosurf.ThenFunc(csrfFail)) return csrf }).Append(aHtmlAfterNosurf) // requests to aHtml hitting nosurfs success handler go: m1 -> nosurf -> m2 -> rt // requests to aHtml hitting nosurfs failure handler go: m1 -> nosurf -> m2 -> csrfFail
func (RTChain) Then ¶ added in v0.21.0
func (c RTChain) Then(rt http.RoundTripper) http.RoundTripper
Then chains the middleware and returns the final http.RoundTripper.
garcon.NewRTChain(m1, m2, m3).Then(rt)
is equivalent to:
m1(m2(m3(rt)))
When the request goes out, it will be passed to m1, then m2, then m3 and finally, the given round tripper (assuming every middleware calls the following one).
A chain can be safely reused by calling Then() several times.
stdStack := garcon.NewRTChain(rateLimitHandler, csrfHandler) indexPipe = stdStack.Then(indexHandler) authPipe = stdStack.Then(authHandler)
Note that middlewares are called on every call to Then() and thus several instances of the same middleware will be created when a chain is reused in this way. For proper middleware, this should cause no problems.
Then() treats nil as http.DefaultTransport.
func (RTChain) ThenFunc ¶ added in v0.21.0
func (c RTChain) ThenFunc(fn RoundTripperFunc) http.RoundTripper
ThenFunc works identically to Then, but takes a RoundTripperFunc instead of a RoundTripper.
The following two statements are equivalent:
c.Then(http.RoundTripperFunc(fn)) c.ThenFunc(fn)
ThenFunc provides all the guarantees of Then.
type RTMiddleware ¶ added in v0.21.0
type RTMiddleware func(http.RoundTripper) http.RoundTripper
RTMiddleware for a piece of RoundTrip middleware. Some middleware uses this out of the box, so in most cases you can just use somepackage.New().
type ReqLimiter ¶ added in v0.17.0
type ReqLimiter struct {
// contains filtered or unexported fields
}
func NewRateLimiter ¶ added in v0.23.0
func NewRateLimiter(gw Writer, maxReqBurst, maxReqPerMinute int, devMode bool) ReqLimiter
func (*ReqLimiter) MiddlewareRateLimiter ¶ added in v0.23.0
func (rl *ReqLimiter) MiddlewareRateLimiter(next http.Handler) http.Handler
type RoundTripperFunc ¶ added in v0.21.0
RoundTripperFunc is to RoundTripper what HandlerFunc is to Handler. It is a higher-order function that enables chaining of RoundTrippers with the middleware pattern.
type ServerName ¶ added in v0.23.0
type ServerName string
ServerName is used in multiple parts in Garcon: - the HTTP Server header, - the Prometheus namespace.
func ExtractName ¶ added in v0.23.0
func ExtractName(str string) ServerName
ExtractName extracts the wider "[a-zA-Z0-9_]+" string from the end of str. If str is a path or an URL, keep the last basename. Example: keep "myapp" from "https://example.com/path/myapp/" ExtractName also removes all punctuation characters except "_".
func (ServerName) MiddlewareExportTrafficMetrics ¶ added in v0.23.0
func (ns ServerName) MiddlewareExportTrafficMetrics(next http.Handler) http.Handler
MiddlewareExportTrafficMetrics measures the time to handle a request.
func (*ServerName) SetPromNamingRule ¶ added in v0.23.0
func (ns *ServerName) SetPromNamingRule()
SetPromNamingRule verifies Prom naming rules for namespace and fixes it if necessary. valid namespace = [a-zA-Z][a-zA-Z0-9_]* https://prometheus.io/docs/concepts/data_model/#metric-names-and-labels
func (ServerName) String ¶ added in v0.23.0
func (ns ServerName) String() string
type StaticWebServer ¶ added in v0.16.0
StaticWebServer is a webserver serving static files among HTML, CSS, JS and popular image formats.
func NewStaticWebServer ¶ added in v0.16.0
func NewStaticWebServer(gw Writer, dir string) StaticWebServer
NewStaticWebServer creates a StaticWebServer.
func (*StaticWebServer) ServeAssets ¶ added in v0.16.0
func (ws *StaticWebServer) ServeAssets() func(w http.ResponseWriter, r *http.Request)
ServeAssets detects the Content-Type depending on the asset extension.
func (*StaticWebServer) ServeDir ¶ added in v0.16.0
func (ws *StaticWebServer) ServeDir(contentType string) func(w http.ResponseWriter, r *http.Request)
ServeDir handles the static files using the same Content-Type.
func (*StaticWebServer) ServeFile ¶ added in v0.16.0
func (ws *StaticWebServer) ServeFile(urlPath, contentType string) func(w http.ResponseWriter, r *http.Request)
ServeFile handles one specific file (and its specific Content-Type).
func (*StaticWebServer) ServeImages ¶ added in v0.16.0
func (ws *StaticWebServer) ServeImages() func(w http.ResponseWriter, r *http.Request)
ServeImages detects the Content-Type depending on the image extension.
type TokenChecker ¶ added in v0.13.0
type TokenChecker interface { // Set is a middleware setting a cookie in the response when the request has no valid token. // Set searches the token in a cookie and in the first "Authorization" header. // Finally, Set stores the decoded token fields within the request context. Set(next http.Handler) http.Handler // Chk is a middleware accepting requests only if it has a valid cookie: // other requests are rejected with http.StatusUnauthorized. // Chk does not verify the "Authorization" header. // See also the Vet() function if the token should also be verified in the "Authorization" header. // Finally, Chk stores the decoded token fields within the request context. // In dev. mode, Chk accepts any request but does not store invalid tokens. Chk(next http.Handler) http.Handler // Vet is a middleware accepting accepting requests having a valid token // either in the cookie or in the first "Authorization" header: // other requests are rejected with http.StatusUnauthorized. // Vet also stores the decoded token in the request context. // In dev. mode, Vet accepts any request but does not store invalid tokens. Vet(next http.Handler) http.Handler // Cookie returns a default cookie to facilitate testing. Cookie(i int) *http.Cookie }
TokenChecker is the common interface to Incorruptible and JWTChecker.
type WebForm ¶ added in v0.16.0
type WebForm struct { Writer Writer Notifier notifier.Notifier Redirect string // TextLimits are used as security limits // to avoid being flooded by large web forms // and unexpected field names. // The map key is the input field name. // The map value is a pair of integers: // the max length and the max line breaks. // Use 0 to disable any limit. TextLimits map[string][2]int // FileLimits is similar to TextLimits // but for uploaded files. // The map value is a pair of integers: // the max size in runes of one file // and the max occurrences having same field name. // Use 0 to disable any limit. FileLimits map[string][2]int // MaxTotalLength includes the // form fields and browser fingerprints. MaxTotalLength int // contains filtered or unexported fields }
func NewContactForm ¶ added in v0.16.0
NewContactForm initializes a new WebForm with the default contact-form settings.
type Writer ¶ added in v0.21.0
type Writer string
Writer enables writing useful JSON error message in the HTTP response body.
func (Writer) InvalidPath ¶ added in v0.21.0
func (gw Writer) InvalidPath(w http.ResponseWriter, r *http.Request)
func (Writer) NotImplemented ¶ added in v0.21.0
func (gw Writer) NotImplemented(w http.ResponseWriter, r *http.Request)
func (Writer) WriteErr ¶ added in v0.21.0
WriteErr is a fast pretty-JSON marshaler dedicated to the HTTP error response. WriteErr extends the JSON content when more than two key-values (kv) are provided.
func (Writer) WriteErrSafe ¶ added in v0.21.0
WriteErrSafe is a safe alternative to Write, may be slower despite the easyjson generated code. Disadvantage: WriteErrSafe concatenates all key-values (kv) in "message" field.