1
0
mirror of https://github.com/gofiber/fiber.git synced 2025-02-11 23:01:22 +00:00
fiber/app.go

1124 lines
35 KiB
Go
Raw Normal View History

// ⚡️ Fiber is an Express inspired web framework written in Go with ☕️
// 🤖 Github Repository: https://github.com/gofiber/fiber
// 📌 API Documentation: https://docs.gofiber.io
2020-02-21 18:07:43 +01:00
// Package fiber is an Express inspired web framework built on top of Fasthttp,
2020-07-06 17:12:35 +02:00
// the fastest HTTP engine for Go. Designed to ease things up for fast
// development with zero memory allocation and performance in mind.
2020-02-21 18:07:43 +01:00
package fiber
import (
"bufio"
"context"
"encoding/json"
"encoding/xml"
"errors"
2020-02-21 18:07:43 +01:00
"fmt"
"net"
"net/http"
"net/http/httputil"
2020-02-27 04:10:26 -05:00
"reflect"
2020-02-21 18:07:43 +01:00
"strconv"
"strings"
"sync"
2020-02-21 18:07:43 +01:00
"time"
2023-11-07 20:22:31 +03:00
"github.com/gofiber/fiber/v3/log"
"github.com/gofiber/utils/v2"
2020-09-13 11:20:11 +02:00
"github.com/valyala/fasthttp"
2020-02-21 18:07:43 +01:00
)
// Version of current fiber package
2024-04-07 20:27:48 +02:00
const Version = "3.0.0-beta.2"
2020-02-21 18:07:43 +01:00
2020-10-31 07:51:44 +01:00
// Handler defines a function to serve HTTP requests.
type Handler = func(Ctx) error
2020-10-31 07:51:44 +01:00
2022-05-31 17:41:38 +03:00
// Map is a shortcut for map[string]any, useful for JSON returns
type Map map[string]any
2020-03-24 05:31:51 +01:00
// Storage interface for communicating with different database/key-value
// providers
2020-10-26 00:10:35 +00:00
type Storage interface {
// Get gets the value for the given key.
2022-02-22 10:10:55 +03:00
// `nil, nil` is returned when the key does not exist
2020-10-31 06:36:02 +01:00
Get(key string) ([]byte, error)
2022-02-22 10:10:55 +03:00
// Set stores the given value for the given key along
// with an expiration value, 0 means no expiration.
// Empty key or value will be ignored without an error.
2022-02-22 10:10:55 +03:00
Set(key string, val []byte, exp time.Duration) error
2020-10-31 06:36:02 +01:00
// Delete deletes the value for the given key.
// It returns no error if the storage does not contain the key,
2020-10-31 06:36:02 +01:00
Delete(key string) error
// Reset resets the storage and delete all keys.
Reset() error
// Close closes the storage and will stop any running garbage
// collectors and open connections.
Close() error
2020-10-26 00:10:35 +00:00
}
2020-09-13 11:20:11 +02:00
// ErrorHandler defines a function that will process all errors
// returned from any handlers in the stack
//
// cfg := fiber.Config{}
// cfg.ErrorHandler = func(c Ctx, err error) error {
// code := StatusInternalServerError
// var e *fiber.Error
// if errors.As(err, &e) {
// code = e.Code
// }
// c.Set(HeaderContentType, MIMETextPlainCharsetUTF8)
// return c.Status(code).SendString(err.Error())
// }
// app := fiber.New(cfg)
type ErrorHandler = func(Ctx, error) error
// Error represents an error that occurred while handling a request.
type Error struct {
Code int `json:"code"`
Message string `json:"message"`
}
2020-06-07 20:57:55 +02:00
2020-03-31 10:03:39 +02:00
// App denotes the Fiber application.
type App struct {
mutex sync.Mutex
2020-06-20 17:26:48 +02:00
// Route stack divided by HTTP methods
stack [][]*Route
// Route stack divided by HTTP methods and route prefixes
treeStack []map[string][]*Route
// contains the information if the route stack has been changed to build the optimized tree
routesRefreshed bool
// Amount of registered routes
routesCount uint32
// Amount of registered handlers
handlersCount uint32
// Ctx pool
pool sync.Pool
// Fasthttp server
2020-05-16 05:11:25 +02:00
server *fasthttp.Server
2020-09-13 11:20:11 +02:00
// App config
config Config
// Converts string to a byte slice
getBytes func(s string) (b []byte)
// Converts byte slice to a string
getString func(b []byte) string
// Hooks
2022-08-16 09:13:38 +03:00
hooks *Hooks
// Latest route & group
latestRoute *Route
// newCtxFunc
newCtxFunc func(app *App) CustomCtx
:sparkles: v3 (feature): initial support for binding (#1981) * :sparkles: v3 (feature): initial support for binding * ✨ v3 (feature): initial support for binding #1981 use pointer/references instead of copies * :sparkles: v3 (feature): initial support for binding embed bind into the ctx * :sparkles: v3 (feature): initial support for binding - add URI binder. * :sparkles: v3 (feature): initial support for binding - add response header binder. * :sparkles: v3 (feature): initial support for binding - add response header binder. * :sparkles: v3 (feature): initial support for binding - add cookie binder. * :sparkles: v3 (feature): initial support for binding - custom binder support for body binding. - test case for custom binder. * :sparkles: v3 (feature): initial support for binding - add map[string][]string & map[string]string support for binders. * :sparkles: v3 (feature): initial support for binding - fix Test_Bind_Header_Map * :sparkles: v3 (feature): initial support for binding - Functional Should/Must * :sparkles: v3 (feature): initial support for binding - custom struct validator support. * :sparkles: v3 (feature): initial support for binding - README for binding. - Docs for binding methods. * :sparkles: v3 (feature): initial support for binding - Bind() -> BindVars(), Binding() -> Bind() * :sparkles: v3 (feature): initial support for binding - fix doc problems * :sparkles: v3 (feature): initial support for binding - fix doc problems Co-authored-by: wernerr <rene@gofiber.io>
2022-08-08 10:16:08 +03:00
// custom binders
customBinders []CustomBinder
// TLS handler
tlsHandler *TLSHandler
// Mount fields
mountFields *mountFields
// Indicates if the value was explicitly configured
configured Config
// customConstraints is a list of external constraints
customConstraints []CustomConstraint
2020-09-13 11:20:11 +02:00
}
// Config is a struct holding the server settings.
type Config struct {
2020-06-03 00:01:55 +02:00
// Enables the "Server: value" HTTP header.
2020-09-17 01:39:42 +02:00
//
2020-06-03 00:01:55 +02:00
// Default: ""
2020-07-02 20:26:38 +02:00
ServerHeader string `json:"server_header"`
// When set to true, the router treats "/foo" and "/foo/" as different.
// By default this is disabled and both "/foo" and "/foo/" will execute the same handler.
2020-09-17 01:39:42 +02:00
//
// Default: false
2020-07-02 20:26:38 +02:00
StrictRouting bool `json:"strict_routing"`
// When set to true, enables case sensitive routing.
// E.g. "/FoO" and "/foo" are treated as different routes.
// By default this is disabled and both "/FoO" and "/foo" will execute the same handler.
2020-09-17 01:39:42 +02:00
//
// Default: false
2020-07-02 20:26:38 +02:00
CaseSensitive bool `json:"case_sensitive"`
// When set to true, this relinquishes the 0-allocation promise in certain
2020-07-12 22:48:12 +09:00
// cases in order to access the handler values (e.g. request bodies) in an
// immutable fashion so that these values are available even if you return
// from handler.
2020-09-17 01:39:42 +02:00
//
2020-06-03 00:01:55 +02:00
// Default: false
2020-07-02 20:26:38 +02:00
Immutable bool `json:"immutable"`
// When set to true, converts all encoded characters in the route back
// before setting the path for the context, so that the routing,
// the returning of the current url from the context `ctx.Path()`
// and the parameters `ctx.Params(%key%)` with decoded characters will work
2020-09-17 01:39:42 +02:00
//
// Default: false
2020-07-02 20:26:38 +02:00
UnescapePath bool `json:"unescape_path"`
// Max body size that the server accepts.
// -1 will decline any body size
//
2020-06-03 00:01:55 +02:00
// Default: 4 * 1024 * 1024
2020-07-02 20:26:38 +02:00
BodyLimit int `json:"body_limit"`
2020-04-19 16:10:19 +02:00
// Maximum number of concurrent connections.
2020-09-17 01:39:42 +02:00
//
2020-06-03 00:01:55 +02:00
// Default: 256 * 1024
2020-07-02 20:26:38 +02:00
Concurrency int `json:"concurrency"`
2020-06-12 12:29:57 +02:00
// Views is the interface that wraps the Render function.
2020-09-17 01:39:42 +02:00
//
2020-06-12 12:29:57 +02:00
// Default: nil
2020-07-02 20:26:38 +02:00
Views Views `json:"-"`
2020-06-12 12:29:57 +02:00
// Views Layout is the global layout for all template render until override on Render function.
//
// Default: ""
ViewsLayout string `json:"views_layout"`
// PassLocalsToViews Enables passing of the locals set on a fiber.Ctx to the template engine
//
// Default: false
PassLocalsToViews bool `json:"pass_locals_to_views"`
2020-03-24 05:31:51 +01:00
// The amount of time allowed to read the full request including body.
2020-06-21 18:00:57 +02:00
// It is reset after the request handler has returned.
// The connection's read deadline is reset when the connection opens.
2020-09-17 01:39:42 +02:00
//
2020-06-03 00:01:55 +02:00
// Default: unlimited
2020-07-02 20:26:38 +02:00
ReadTimeout time.Duration `json:"read_timeout"`
2020-03-24 05:31:51 +01:00
// The maximum duration before timing out writes of the response.
2020-06-21 18:00:57 +02:00
// It is reset after the request handler has returned.
2020-09-17 01:39:42 +02:00
//
2020-06-03 00:01:55 +02:00
// Default: unlimited
2020-07-02 20:26:38 +02:00
WriteTimeout time.Duration `json:"write_timeout"`
2020-03-24 05:31:51 +01:00
// The maximum amount of time to wait for the next request when keep-alive is enabled.
2020-06-21 18:00:57 +02:00
// If IdleTimeout is zero, the value of ReadTimeout is used.
2020-09-17 01:39:42 +02:00
//
2020-06-03 00:01:55 +02:00
// Default: unlimited
2020-07-02 20:26:38 +02:00
IdleTimeout time.Duration `json:"idle_timeout"`
2020-02-21 18:07:43 +01:00
2020-06-15 13:36:50 +02:00
// Per-connection buffer size for requests' reading.
// This also limits the maximum header size.
// Increase this buffer if your clients send multi-KB RequestURIs
// and/or multi-KB headers (for example, BIG cookies).
2020-09-17 01:39:42 +02:00
//
// Default: 4096
2020-07-02 20:26:38 +02:00
ReadBufferSize int `json:"read_buffer_size"`
2020-06-15 13:36:50 +02:00
// Per-connection buffer size for responses' writing.
2020-09-17 01:39:42 +02:00
//
// Default: 4096
2020-07-02 20:26:38 +02:00
WriteBufferSize int `json:"write_buffer_size"`
2020-06-15 13:36:50 +02:00
2020-06-07 20:35:41 +02:00
// CompressedFileSuffix adds suffix to the original file name and
// tries saving the resulting compressed file under the new file name.
2020-09-17 01:39:42 +02:00
//
2020-06-07 20:35:41 +02:00
// Default: ".fiber.gz"
2020-07-02 20:26:38 +02:00
CompressedFileSuffix string `json:"compressed_file_suffix"`
2020-06-07 20:35:41 +02:00
2020-09-13 11:20:11 +02:00
// ProxyHeader will enable c.IP() to return the value of the given header key
// By default c.IP() will return the Remote IP from the TCP connection
// This property can be useful if you are behind a load balancer: X-Forwarded-*
// NOTE: headers are easily spoofed and the detected IP addresses are unreliable.
2020-09-17 01:39:42 +02:00
//
2020-09-13 11:20:11 +02:00
// Default: ""
ProxyHeader string `json:"proxy_header"`
// GETOnly rejects all non-GET requests if set to true.
// This option is useful as anti-DoS protection for servers
// accepting only GET requests. The request size is limited
// by ReadBufferSize if GETOnly is set.
2020-09-17 01:39:42 +02:00
//
// Default: false
2020-09-13 11:20:11 +02:00
GETOnly bool `json:"get_only"`
// ErrorHandler is executed when an error is returned from fiber.Handler.
2020-09-17 01:39:42 +02:00
//
// Default: DefaultErrorHandler
2020-09-13 11:20:11 +02:00
ErrorHandler ErrorHandler `json:"-"`
// When set to true, disables keep-alive connections.
// The server will close incoming connections after sending the first response to client.
2020-09-17 01:39:42 +02:00
//
2020-09-13 11:20:11 +02:00
// Default: false
2020-09-14 04:54:26 +02:00
DisableKeepalive bool `json:"disable_keepalive"`
2020-09-13 11:20:11 +02:00
// When set to true, causes the default date header to be excluded from the response.
2020-09-17 01:39:42 +02:00
//
2020-09-13 11:20:11 +02:00
// Default: false
DisableDefaultDate bool `json:"disable_default_date"`
// When set to true, causes the default Content-Type header to be excluded from the response.
2020-09-17 01:39:42 +02:00
//
2020-09-13 11:20:11 +02:00
// Default: false
DisableDefaultContentType bool `json:"disable_default_content_type"`
// When set to true, disables header normalization.
// By default all header names are normalized: conteNT-tYPE -> Content-Type.
2020-09-17 01:39:42 +02:00
//
2020-09-13 11:20:11 +02:00
// Default: false
DisableHeaderNormalizing bool `json:"disable_header_normalizing"`
// This function allows to setup app name for the app
//
// Default: nil
AppName string `json:"app_name"`
// StreamRequestBody enables request body streaming,
// and calls the handler sooner when given body is
// larger than the current limit.
StreamRequestBody bool
// Will not pre parse Multipart Form data if set to true.
//
// This option is useful for servers that desire to treat
// multipart form data as a binary blob, or choose when to parse the data.
//
// Server pre parses multipart form data by default.
DisablePreParseMultipartForm bool
// Aggressively reduces memory usage at the cost of higher CPU usage
// if set to true.
//
// Try enabling this option only if the server consumes too much memory
// serving mostly idle keep-alive connections. This may reduce memory
// usage by more than 50%.
//
// Default: false
ReduceMemoryUsage bool `json:"reduce_memory_usage"`
2020-11-11 14:18:19 +01:00
// FEATURE: v2.3.x
// The router executes the same handler by default if StrictRouting or CaseSensitive is disabled.
// Enabling RedirectFixedPath will change this behavior into a client redirect to the original route path.
// Using the status code 301 for GET requests and 308 for all other request methods.
2020-09-17 01:39:42 +02:00
//
// Default: false
// RedirectFixedPath bool
// When set by an external client of Fiber it will use the provided implementation of a
// JSONMarshal
//
// Allowing for flexibility in using another json library for encoding
// Default: json.Marshal
JSONEncoder utils.JSONMarshal `json:"-"`
2021-02-07 13:55:13 +08:00
// When set by an external client of Fiber it will use the provided implementation of a
// JSONUnmarshal
//
2022-02-01 22:42:23 +03:00
// Allowing for flexibility in using another json library for decoding
// Default: json.Unmarshal
JSONDecoder utils.JSONUnmarshal `json:"-"`
// XMLEncoder set by an external client of Fiber it will use the provided implementation of a
// XMLMarshal
//
// Allowing for flexibility in using another XML library for encoding
// Default: xml.Marshal
XMLEncoder utils.XMLMarshal `json:"-"`
// If you find yourself behind some sort of proxy, like a load balancer,
// then certain header information may be sent to you using special X-Forwarded-* headers or the Forwarded header.
// For example, the Host HTTP header is usually used to return the requested host.
// But when youre behind a proxy, the actual host may be stored in an X-Forwarded-Host header.
//
// If you are behind a proxy, you should enable TrustedProxyCheck to prevent header spoofing.
// If you enable EnableTrustedProxyCheck and leave TrustedProxies empty Fiber will skip
// all headers that could be spoofed.
// If request ip in TrustedProxies whitelist then:
// 1. c.Scheme() get value from X-Forwarded-Proto, X-Forwarded-Protocol, X-Forwarded-Ssl or X-Url-Scheme header
// 2. c.IP() get value from ProxyHeader header.
// 3. c.Host() and c.Hostname() get value from X-Forwarded-Host header
// But if request ip NOT in Trusted Proxies whitelist then:
// 1. c.Scheme() WON't get value from X-Forwarded-Proto, X-Forwarded-Protocol, X-Forwarded-Ssl or X-Url-Scheme header,
// will return https in case when tls connection is handled by the app, of http otherwise
// 2. c.IP() WON'T get value from ProxyHeader header, will return RemoteIP() from fasthttp context
// 3. c.Host() and c.Hostname() WON'T get value from X-Forwarded-Host header, fasthttp.Request.URI().Host()
// will be used to get the hostname.
//
// Default: false
EnableTrustedProxyCheck bool `json:"enable_trusted_proxy_check"`
// Read EnableTrustedProxyCheck doc.
//
// Default: []string
TrustedProxies []string `json:"trusted_proxies"`
trustedProxiesMap map[string]struct{}
trustedProxyRanges []*net.IPNet
// If set to true, c.IP() and c.IPs() will validate IP addresses before returning them.
// Also, c.IP() will return only the first valid IP rather than just the raw header
// WARNING: this has a performance cost associated with it.
//
// Default: false
EnableIPValidation bool `json:"enable_ip_validation"`
// You can define custom color scheme. They'll be used for startup message, route list and some middlewares.
//
// Optional. Default: DefaultColors
ColorScheme Colors `json:"color_scheme"`
:sparkles: v3 (feature): initial support for binding (#1981) * :sparkles: v3 (feature): initial support for binding * ✨ v3 (feature): initial support for binding #1981 use pointer/references instead of copies * :sparkles: v3 (feature): initial support for binding embed bind into the ctx * :sparkles: v3 (feature): initial support for binding - add URI binder. * :sparkles: v3 (feature): initial support for binding - add response header binder. * :sparkles: v3 (feature): initial support for binding - add response header binder. * :sparkles: v3 (feature): initial support for binding - add cookie binder. * :sparkles: v3 (feature): initial support for binding - custom binder support for body binding. - test case for custom binder. * :sparkles: v3 (feature): initial support for binding - add map[string][]string & map[string]string support for binders. * :sparkles: v3 (feature): initial support for binding - fix Test_Bind_Header_Map * :sparkles: v3 (feature): initial support for binding - Functional Should/Must * :sparkles: v3 (feature): initial support for binding - custom struct validator support. * :sparkles: v3 (feature): initial support for binding - README for binding. - Docs for binding methods. * :sparkles: v3 (feature): initial support for binding - Bind() -> BindVars(), Binding() -> Bind() * :sparkles: v3 (feature): initial support for binding - fix doc problems * :sparkles: v3 (feature): initial support for binding - fix doc problems Co-authored-by: wernerr <rene@gofiber.io>
2022-08-08 10:16:08 +03:00
// If you want to validate header/form/query... automatically when to bind, you can define struct validator.
// Fiber doesn't have default validator, so it'll skip validator step if you don't use any validator.
//
// Default: nil
StructValidator StructValidator
// RequestMethods provides customizibility for HTTP methods. You can add/remove methods as you wish.
//
2022-11-21 08:54:34 +01:00
// Optional. Default: DefaultMethods
RequestMethods []string
// EnableSplittingOnParsers splits the query/body/header parameters by comma when it's true.
// For example, you can use it to parse multiple values from a query parameter like this:
// /api?foo=bar,baz == foo[]=bar&foo[]=baz
//
// Optional. Default: false
EnableSplittingOnParsers bool `json:"enable_splitting_on_parsers"`
}
// Static defines configuration options when defining static assets.
type Static struct {
// When set to true, the server tries minimizing CPU usage by caching compressed files.
// This works differently than the github.com/gofiber/compression middleware.
// Optional. Default value false
2020-09-13 11:20:11 +02:00
Compress bool `json:"compress"`
// When set to true, enables byte range requests.
// Optional. Default value false
2020-09-13 11:20:11 +02:00
ByteRange bool `json:"byte_range"`
// When set to true, enables directory browsing.
// Optional. Default value false.
2020-09-13 11:20:11 +02:00
Browse bool `json:"browse"`
// When set to true, enables direct download.
// Optional. Default value false.
Download bool `json:"download"`
// The name of the index file for serving a directory.
// Optional. Default value "index.html".
2020-09-13 11:20:11 +02:00
Index string `json:"index"`
2020-11-11 14:18:19 +01:00
// Expiration duration for inactive file handlers.
// Use a negative time.Duration to disable it.
//
// Optional. Default value 10 * time.Second.
CacheDuration time.Duration `json:"cache_duration"`
// The value for the Cache-Control HTTP-header
// that is set on the file response. MaxAge is defined in seconds.
//
// Optional. Default value 0.
MaxAge int `json:"max_age"`
// ModifyResponse defines a function that allows you to alter the response.
//
// Optional. Default: nil
ModifyResponse Handler
// Next defines a function to skip this middleware when returned true.
//
// Optional. Default: nil
Next func(c Ctx) bool
}
// RouteMessage is some message need to be print when server starts
type RouteMessage struct {
name string
method string
path string
handlers string
}
2020-09-13 11:39:55 +02:00
// Default Config values
const (
2020-09-13 11:20:11 +02:00
DefaultBodyLimit = 4 * 1024 * 1024
DefaultConcurrency = 256 * 1024
DefaultReadBufferSize = 4096
DefaultWriteBufferSize = 4096
DefaultCompressedFileSuffix = ".fiber.gz"
)
// HTTP methods enabled by default
var DefaultMethods = []string{
MethodGet,
MethodHead,
MethodPost,
MethodPut,
MethodDelete,
MethodConnect,
MethodOptions,
MethodTrace,
MethodPatch,
}
// DefaultErrorHandler that process return errors from handlers
func DefaultErrorHandler(c Ctx, err error) error {
code := StatusInternalServerError
var e *Error
if errors.As(err, &e) {
code = e.Code
}
2020-09-13 11:20:11 +02:00
c.Set(HeaderContentType, MIMETextPlainCharsetUTF8)
return c.Status(code).SendString(err.Error())
}
2020-03-24 05:31:51 +01:00
// New creates a new Fiber named instance.
//
// app := fiber.New()
//
2020-09-14 04:54:26 +02:00
// You can pass optional configuration options by passing a Config struct:
//
// app := fiber.New(fiber.Config{
// Prefork: true,
// ServerHeader: "Fiber",
// })
2020-09-13 11:20:11 +02:00
func New(config ...Config) *App {
// Create a new app
app := &App{
2020-09-13 11:20:11 +02:00
// Create config
:sparkles: v3 (feature): initial support for binding (#1981) * :sparkles: v3 (feature): initial support for binding * ✨ v3 (feature): initial support for binding #1981 use pointer/references instead of copies * :sparkles: v3 (feature): initial support for binding embed bind into the ctx * :sparkles: v3 (feature): initial support for binding - add URI binder. * :sparkles: v3 (feature): initial support for binding - add response header binder. * :sparkles: v3 (feature): initial support for binding - add response header binder. * :sparkles: v3 (feature): initial support for binding - add cookie binder. * :sparkles: v3 (feature): initial support for binding - custom binder support for body binding. - test case for custom binder. * :sparkles: v3 (feature): initial support for binding - add map[string][]string & map[string]string support for binders. * :sparkles: v3 (feature): initial support for binding - fix Test_Bind_Header_Map * :sparkles: v3 (feature): initial support for binding - Functional Should/Must * :sparkles: v3 (feature): initial support for binding - custom struct validator support. * :sparkles: v3 (feature): initial support for binding - README for binding. - Docs for binding methods. * :sparkles: v3 (feature): initial support for binding - Bind() -> BindVars(), Binding() -> Bind() * :sparkles: v3 (feature): initial support for binding - fix doc problems * :sparkles: v3 (feature): initial support for binding - fix doc problems Co-authored-by: wernerr <rene@gofiber.io>
2022-08-08 10:16:08 +03:00
config: Config{},
getBytes: utils.UnsafeBytes,
getString: utils.UnsafeString,
latestRoute: &Route{},
customBinders: []CustomBinder{},
}
// Create Ctx pool
app.pool = sync.Pool{
New: func() any {
V2 to v3 merge (#2864) * Update pull_request_template.md * Update v3-changes.md * Update CONTRIBUTING.md (#2752) Grammar correction. * chore(encryptcookie)!: update default config (#2753) * chore(encryptcookie)!: update default config docs(encryptcookie): enhance documentation and examples BREAKING CHANGE: removed the hardcoded "csrf_" from the Except. * docs(encryptcookie): reads or modifies cookies * chore(encryptcookie): csrf config example * docs(encryptcookie): md table spacing * build(deps): bump actions/setup-go from 4 to 5 (#2754) Bumps [actions/setup-go](https://github.com/actions/setup-go) from 4 to 5. - [Release notes](https://github.com/actions/setup-go/releases) - [Commits](https://github.com/actions/setup-go/compare/v4...v5) --- updated-dependencies: - dependency-name: actions/setup-go dependency-type: direct:production update-type: version-update:semver-major ... Signed-off-by: dependabot[bot] <support@github.com> Co-authored-by: dependabot[bot] <49699333+dependabot[bot]@users.noreply.github.com> * 🩹 middleware/logger/: log client IP address by default (#2755) * middleware/logger: Log client IP address by default. * Update doc. * fix: don't constrain middlewares' context-keys to strings :bug: (#2751) * Revert "Revert ":bug: requestid.Config.ContextKey is interface{} (#2369)" (#2742)" This reverts commit 28be17f929cfa7d3c27dd292fc3956f2f9882e22. * fix: request ContextKey default value condition Should check for `nil` since it is `any`. * fix: don't constrain middlewares' context-keys to strings `context` recommends using "unexported type" as context keys to avoid collisions https://pkg.go.dev/github.com/gofiber/fiber/v2#Ctx.Locals. The official go blog also recommends this https://go.dev/blog/context. `fiber.Ctx.Locals(key any, value any)` correctly allows consumers to use unexported types or e.g. strings. But some fiber middlewares constrain their context-keys to `string` in their "default config structs", making it impossible to use unexported types. This PR removes the `string` _constraint_ from all middlewares, allowing to now use unexported types as per the official guidelines. However the default value is still a string, so it's not a breaking change, and anyone still using strings as context keys is not affected. * 📚 Update app.md for indentation (#2761) Update app.md for indentation * build(deps): bump github.com/google/uuid from 1.4.0 to 1.5.0 (#2762) Bumps [github.com/google/uuid](https://github.com/google/uuid) from 1.4.0 to 1.5.0. - [Release notes](https://github.com/google/uuid/releases) - [Changelog](https://github.com/google/uuid/blob/master/CHANGELOG.md) - [Commits](https://github.com/google/uuid/compare/v1.4.0...v1.5.0) --- updated-dependencies: - dependency-name: github.com/google/uuid dependency-type: direct:production update-type: version-update:semver-minor ... Signed-off-by: dependabot[bot] <support@github.com> Co-authored-by: dependabot[bot] <49699333+dependabot[bot]@users.noreply.github.com> * build(deps): bump github/codeql-action from 2 to 3 (#2763) Bumps [github/codeql-action](https://github.com/github/codeql-action) from 2 to 3. - [Release notes](https://github.com/github/codeql-action/releases) - [Changelog](https://github.com/github/codeql-action/blob/main/CHANGELOG.md) - [Commits](https://github.com/github/codeql-action/compare/v2...v3) --- updated-dependencies: - dependency-name: github/codeql-action dependency-type: direct:production update-type: version-update:semver-major ... Signed-off-by: dependabot[bot] <support@github.com> Co-authored-by: dependabot[bot] <49699333+dependabot[bot]@users.noreply.github.com> * Changing default log output (#2730) changing default log output Closes #2729 * Update hooks.md fix wrong hooks signature * 🩹 Fix: CORS middleware should use the defined AllowedOriginsFunc config when AllowedOrigins is empty (#2771) * 🐛 [Bug]: Adaptator + otelfiber issue #2641 (#2772) * 🩹🚨 - fix for redirect with query params (#2748) * redirect with query params did not work, fix it and add test for it * redirect middleware - fix test typo * ♻️ logger/middleware colorize logger error message #2593 (#2773) * :sparkles: feat: add liveness and readiness checks (#2509) * :sparkles: feat: add liveness and readiness checkers * :memo: docs: add docs for liveness and readiness * :sparkles: feat: add options method for probe checkers * :white_check_mark: tests: add tests for liveness and readiness * :recycle: refactor: change default endpoint values * :recycle: refactor: change default value for liveness endpoint * :memo: docs: add return status for liveness and readiness probes * :recycle: refactor: change probechecker to middleware * :memo: docs: move docs to middleware session * :recycle: refactor: apply gofumpt formatting * :recycle: refactor: remove unused parameter * split config and apply a review * apply reviews and add testcases * add benchmark * cleanup * rename middleware * fix linter * Update docs and config values * Revert change to IsReady * Updates based on code review * Update docs to match other middlewares --------- Co-authored-by: Muhammed Efe Cetin <efectn@protonmail.com> Co-authored-by: Juan Calderon-Perez <835733+gaby@users.noreply.github.com> Co-authored-by: Juan Calderon-Perez <jgcalderonperez@protonmail.com> * prepare release v2.52.0 - add more Parser tests * fix healthcheck.md * configure workflows for V2 branch * configure workflows for V2 branch * Fix default value to false in docs of QueryBool (#2811) fix default value to false in docs of QueryBool * update queryParser config * Update ctx.md * Update routing.md * merge v2 in v3 * merge v2 in v3 * lint fixes * :books: Doc: Fix code snippet indentation in /docs/api/middleware/keyauth.md Removes an an extra level of indentation in line 51 of `keyauth.md` [here](https://github.com/gofiber/fiber/blob/v2/docs/api/middleware/keyauth.md?plain=1#L51) * fix: healthcheck middleware not working with route group (#2863) * fix: healthcheck middleware not working with route group * perf: change verification method to improve perf * Update healthcheck_test.go * test: add not matching route test for strict routing * add more test cases * correct tests * correct test helpers * correct tests * correct tests --------- Co-authored-by: Juan Calderon-Perez <835733+gaby@users.noreply.github.com> Co-authored-by: René Werner <rene@gofiber.io> * merge v2 in v3 * Merge pull request from GHSA-fmg4-x8pw-hjhg * Enforce Wildcard Origins with AllowCredentials check * Expand unit-tests, fix issues with subdomains logic, update docs * Update cors.md * Added test using localhost, ipv4, and ipv6 address * improve documentation markdown --------- Co-authored-by: René Werner <rene@gofiber.io> * Update app.go prepare release v2.52.1 * fix cors domain normalize * fix sync-docs workflow * test: fix failing tests * fix sync-docs workflow * test: cors middleware use testify require * chore: fix lint warnings * chore: revert test isolation. * fixed the fasthttp ctx race condition problem * Update middleware/cors/utils.go Co-authored-by: Renan Bastos <renanbastos.tec@gmail.com> * fix sync_docs.sh * fix review comments/hints * fix review comments/hints * stabilize Test_Proxy_Timeout_Slow_Server test * stabilize Test_Proxy_.* tests * ignore bodyclose linter for tests use http.NoBody instead of nil * revert(tests): undo http.NoBody usage * fix(ctx pool): postpone the reset for some values shortly before the release in the pool * refactor(tests): use testify panic method instead of custom solution --------- Signed-off-by: dependabot[bot] <support@github.com> Co-authored-by: tokelo-12 <113810058+tokelo-12@users.noreply.github.com> Co-authored-by: Jason McNeil <sixcolors@mac.com> Co-authored-by: dependabot[bot] <49699333+dependabot[bot]@users.noreply.github.com> Co-authored-by: iRedMail <2048991+iredmail@users.noreply.github.com> Co-authored-by: Benjamin Grosse <ste3ls@gmail.com> Co-authored-by: Mehmet Firat KOMURCU <mehmetfiratkomurcu@hotmail.com> Co-authored-by: Bruno <bdm2943@icloud.com> Co-authored-by: Muhammad Kholid B <muhammadkholidb@gmail.com> Co-authored-by: gilwo <gilwo@users.noreply.github.com> Co-authored-by: Lucas Lemos <lucashenriqueblemos@gmail.com> Co-authored-by: Muhammed Efe Cetin <efectn@protonmail.com> Co-authored-by: Juan Calderon-Perez <835733+gaby@users.noreply.github.com> Co-authored-by: Juan Calderon-Perez <jgcalderonperez@protonmail.com> Co-authored-by: Jongmin Kim <kjongmin26@gmail.com> Co-authored-by: Giovanni Rivera <rivera.giovanni271@gmail.com> Co-authored-by: Renan Bastos <renanbastos.tec@gmail.com>
2024-02-29 08:29:59 +01:00
return app.newCtx()
},
}
// Define hooks
app.hooks = newHooks(app)
// Define mountFields
app.mountFields = newMountFields(app)
2020-09-13 11:20:11 +02:00
// Override config if provided
if len(config) > 0 {
app.config = config[0]
2020-03-14 12:30:21 +01:00
}
// Initialize configured before defaults are set
app.configured = app.config
2020-09-13 11:20:11 +02:00
// Override default values
if app.config.BodyLimit == 0 {
2020-09-13 11:20:11 +02:00
app.config.BodyLimit = DefaultBodyLimit
2020-06-07 20:57:55 +02:00
}
2020-09-13 11:20:11 +02:00
if app.config.Concurrency <= 0 {
app.config.Concurrency = DefaultConcurrency
2020-06-15 13:36:50 +02:00
}
2020-09-13 11:20:11 +02:00
if app.config.ReadBufferSize <= 0 {
app.config.ReadBufferSize = DefaultReadBufferSize
2020-06-15 13:36:50 +02:00
}
2020-09-13 11:20:11 +02:00
if app.config.WriteBufferSize <= 0 {
app.config.WriteBufferSize = DefaultWriteBufferSize
2020-06-07 20:57:55 +02:00
}
2020-09-13 11:20:11 +02:00
if app.config.CompressedFileSuffix == "" {
app.config.CompressedFileSuffix = DefaultCompressedFileSuffix
2020-06-07 20:57:55 +02:00
}
2020-09-13 11:20:11 +02:00
if app.config.Immutable {
app.getBytes, app.getString = getBytesImmutable, getStringImmutable
2020-06-07 20:57:55 +02:00
}
2020-09-13 11:20:11 +02:00
if app.config.ErrorHandler == nil {
app.config.ErrorHandler = DefaultErrorHandler
}
if app.config.JSONEncoder == nil {
app.config.JSONEncoder = json.Marshal
}
if app.config.JSONDecoder == nil {
app.config.JSONDecoder = json.Unmarshal
}
if app.config.XMLEncoder == nil {
app.config.XMLEncoder = xml.Marshal
}
if len(app.config.RequestMethods) == 0 {
app.config.RequestMethods = DefaultMethods
}
app.config.trustedProxiesMap = make(map[string]struct{}, len(app.config.TrustedProxies))
for _, ipAddress := range app.config.TrustedProxies {
app.handleTrustedProxy(ipAddress)
}
// Create router stack
app.stack = make([][]*Route, len(app.config.RequestMethods))
app.treeStack = make([]map[string][]*Route, len(app.config.RequestMethods))
// Override colors
app.config.ColorScheme = defaultColors(app.config.ColorScheme)
2020-09-13 11:20:11 +02:00
// Init app
app.init()
2020-06-12 12:29:57 +02:00
// Return app
return app
2020-02-21 18:07:43 +01:00
}
// Adds an ip address to trustedProxyRanges or trustedProxiesMap based on whether it is an IP range or not
func (app *App) handleTrustedProxy(ipAddress string) {
if strings.Contains(ipAddress, "/") {
_, ipNet, err := net.ParseCIDR(ipAddress)
if err != nil {
log.Warnf("IP range %q could not be parsed: %v", ipAddress, err)
} else {
app.config.trustedProxyRanges = append(app.config.trustedProxyRanges, ipNet)
}
} else {
app.config.trustedProxiesMap[ipAddress] = struct{}{}
}
}
// NewCtxFunc allows to customize ctx methods as we want.
// Note: It doesn't allow adding new methods, only customizing exist methods.
func (app *App) NewCtxFunc(function func(app *App) CustomCtx) {
app.newCtxFunc = function
}
// RegisterCustomConstraint allows to register custom constraint.
func (app *App) RegisterCustomConstraint(constraint CustomConstraint) {
app.customConstraints = append(app.customConstraints, constraint)
}
:sparkles: v3 (feature): initial support for binding (#1981) * :sparkles: v3 (feature): initial support for binding * ✨ v3 (feature): initial support for binding #1981 use pointer/references instead of copies * :sparkles: v3 (feature): initial support for binding embed bind into the ctx * :sparkles: v3 (feature): initial support for binding - add URI binder. * :sparkles: v3 (feature): initial support for binding - add response header binder. * :sparkles: v3 (feature): initial support for binding - add response header binder. * :sparkles: v3 (feature): initial support for binding - add cookie binder. * :sparkles: v3 (feature): initial support for binding - custom binder support for body binding. - test case for custom binder. * :sparkles: v3 (feature): initial support for binding - add map[string][]string & map[string]string support for binders. * :sparkles: v3 (feature): initial support for binding - fix Test_Bind_Header_Map * :sparkles: v3 (feature): initial support for binding - Functional Should/Must * :sparkles: v3 (feature): initial support for binding - custom struct validator support. * :sparkles: v3 (feature): initial support for binding - README for binding. - Docs for binding methods. * :sparkles: v3 (feature): initial support for binding - Bind() -> BindVars(), Binding() -> Bind() * :sparkles: v3 (feature): initial support for binding - fix doc problems * :sparkles: v3 (feature): initial support for binding - fix doc problems Co-authored-by: wernerr <rene@gofiber.io>
2022-08-08 10:16:08 +03:00
// You can register custom binders to use as Bind().Custom("name").
// They should be compatible with CustomBinder interface.
func (app *App) RegisterCustomBinder(binder CustomBinder) {
app.customBinders = append(app.customBinders, binder)
}
// You can use SetTLSHandler to use ClientHelloInfo when using TLS with Listener.
func (app *App) SetTLSHandler(tlsHandler *TLSHandler) {
// Attach the tlsHandler to the config
app.mutex.Lock()
app.tlsHandler = tlsHandler
app.mutex.Unlock()
}
// Name Assign name to specific route.
func (app *App) Name(name string) Router {
app.mutex.Lock()
defer app.mutex.Unlock()
for _, routes := range app.stack {
for _, route := range routes {
isMethodValid := route.Method == app.latestRoute.Method || app.latestRoute.use ||
(app.latestRoute.Method == MethodGet && route.Method == MethodHead)
if route.Path == app.latestRoute.Path && isMethodValid {
route.Name = name
if route.group != nil {
route.Name = route.group.name + route.Name
}
}
}
}
if err := app.hooks.executeOnNameHooks(*app.latestRoute); err != nil {
panic(err)
}
return app
}
// GetRoute Get route by name
func (app *App) GetRoute(name string) Route {
for _, routes := range app.stack {
for _, route := range routes {
if route.Name == name {
return *route
}
}
}
return Route{}
}
// GetRoutes Get all routes. When filterUseOption equal to true, it will filter the routes registered by the middleware.
func (app *App) GetRoutes(filterUseOption ...bool) []Route {
var rs []Route
var filterUse bool
if len(filterUseOption) != 0 {
filterUse = filterUseOption[0]
}
for _, routes := range app.stack {
for _, route := range routes {
if filterUse && route.use {
continue
}
rs = append(rs, *route)
}
}
return rs
}
2020-09-14 04:54:26 +02:00
// Use registers a middleware route that will match requests
// with the provided prefix (which is optional and defaults to "/").
// Also, you can pass another app instance as a sub-router along a routing path.
// It's very useful to split up a large API as many independent routers and
// compose them as a single service using Use. The fiber's error handler and
// any of the fiber's sub apps are added to the application's error handlers
// to be invoked on errors that happen within the prefix route.
2020-09-13 11:20:11 +02:00
//
// app.Use(func(c fiber.Ctx) error {
// return c.Next()
// })
// app.Use("/api", func(c fiber.Ctx) error {
// return c.Next()
// })
// app.Use("/api", handler, func(c fiber.Ctx) error {
// return c.Next()
// })
// subApp := fiber.New()
// app.Use("/mounted-path", subApp)
2020-05-16 05:11:25 +02:00
//
2020-09-13 11:20:11 +02:00
// This method will match all HTTP verbs: GET, POST, PUT, HEAD etc...
2022-05-31 17:41:38 +03:00
func (app *App) Use(args ...any) Router {
2020-05-16 05:11:25 +02:00
var prefix string
var subApp *App
var prefixes []string
var handlers []Handler
2020-05-16 05:11:25 +02:00
2020-02-27 04:10:26 -05:00
for i := 0; i < len(args); i++ {
switch arg := args[i].(type) {
case string:
2020-05-16 05:11:25 +02:00
prefix = arg
case *App:
subApp = arg
case []string:
prefixes = arg
case Handler:
2020-02-27 04:10:26 -05:00
handlers = append(handlers, arg)
default:
2020-07-15 17:43:30 +02:00
panic(fmt.Sprintf("use: invalid handler %v\n", reflect.TypeOf(arg)))
2020-02-27 04:10:26 -05:00
}
}
if len(prefixes) == 0 {
prefixes = append(prefixes, prefix)
}
for _, prefix := range prefixes {
if subApp != nil {
app.mount(prefix, subApp)
return app
}
app.register([]string{methodUse}, prefix, nil, nil, handlers...)
}
return app
2020-02-21 18:07:43 +01:00
}
2020-07-06 17:57:00 +02:00
// Get registers a route for GET methods that requests a representation
// of the specified resource. Requests using GET should only retrieve data.
func (app *App) Get(path string, handler Handler, middleware ...Handler) Router {
return app.Add([]string{MethodGet}, path, handler, middleware...)
2020-02-21 18:07:43 +01:00
}
2020-07-06 17:57:00 +02:00
// Head registers a route for HEAD methods that asks for a response identical
// to that of a GET request, but without the response body.
func (app *App) Head(path string, handler Handler, middleware ...Handler) Router {
return app.Add([]string{MethodHead}, path, handler, middleware...)
2020-02-21 18:07:43 +01:00
}
2020-07-06 17:57:00 +02:00
// Post registers a route for POST methods that is used to submit an entity to the
// specified resource, often causing a change in state or side effects on the server.
func (app *App) Post(path string, handler Handler, middleware ...Handler) Router {
return app.Add([]string{MethodPost}, path, handler, middleware...)
2020-02-21 18:07:43 +01:00
}
2020-07-06 17:57:00 +02:00
// Put registers a route for PUT methods that replaces all current representations
// of the target resource with the request payload.
func (app *App) Put(path string, handler Handler, middleware ...Handler) Router {
return app.Add([]string{MethodPut}, path, handler, middleware...)
2020-02-21 18:07:43 +01:00
}
2020-07-06 17:57:00 +02:00
// Delete registers a route for DELETE methods that deletes the specified resource.
func (app *App) Delete(path string, handler Handler, middleware ...Handler) Router {
return app.Add([]string{MethodDelete}, path, handler, middleware...)
2020-02-21 18:07:43 +01:00
}
2020-07-06 17:57:00 +02:00
// Connect registers a route for CONNECT methods that establishes a tunnel to the
// server identified by the target resource.
func (app *App) Connect(path string, handler Handler, middleware ...Handler) Router {
return app.Add([]string{MethodConnect}, path, handler, middleware...)
2020-02-21 18:07:43 +01:00
}
2020-07-06 17:57:00 +02:00
// Options registers a route for OPTIONS methods that is used to describe the
// communication options for the target resource.
func (app *App) Options(path string, handler Handler, middleware ...Handler) Router {
return app.Add([]string{MethodOptions}, path, handler, middleware...)
2020-02-21 18:07:43 +01:00
}
2020-07-06 17:57:00 +02:00
// Trace registers a route for TRACE methods that performs a message loop-back
// test along the path to the target resource.
func (app *App) Trace(path string, handler Handler, middleware ...Handler) Router {
return app.Add([]string{MethodTrace}, path, handler, middleware...)
2020-02-21 18:07:43 +01:00
}
2020-07-06 17:57:00 +02:00
// Patch registers a route for PATCH methods that is used to apply partial
// modifications to a resource.
func (app *App) Patch(path string, handler Handler, middleware ...Handler) Router {
return app.Add([]string{MethodPatch}, path, handler, middleware...)
2020-02-21 18:07:43 +01:00
}
// Add allows you to specify multiple HTTP methods to register a route.
func (app *App) Add(methods []string, path string, handler Handler, middleware ...Handler) Router {
app.register(methods, path, nil, handler, middleware...)
return app
2020-05-16 05:11:25 +02:00
}
2020-09-13 11:20:11 +02:00
// Static will create a file server serving static files
func (app *App) Static(prefix, root string, config ...Static) Router {
app.registerStatic(prefix, root, config...)
return app
2020-05-16 05:11:25 +02:00
}
2020-09-13 11:20:11 +02:00
// All will register the handler on all HTTP methods
func (app *App) All(path string, handler Handler, middleware ...Handler) Router {
return app.Add(app.config.RequestMethods, path, handler, middleware...)
2020-02-21 18:07:43 +01:00
}
2020-03-24 05:31:51 +01:00
// Group is used for Routes with common prefix to define a new sub-router with optional middleware.
//
// api := app.Group("/api")
// api.Get("/users", handler)
func (app *App) Group(prefix string, handlers ...Handler) Router {
grp := &Group{Prefix: prefix, app: app}
if len(handlers) > 0 {
app.register([]string{methodUse}, prefix, grp, nil, handlers...)
2020-05-16 05:11:25 +02:00
}
if err := app.hooks.executeOnGroupHooks(*grp); err != nil {
panic(err)
}
return grp
}
// Route is used to define routes with a common prefix inside the common function.
// Uses Group method to define new sub-router.
func (app *App) Route(path string) Register {
// Create new route
route := &Registering{app: app, path: path}
return route
}
2020-09-13 11:20:11 +02:00
// Error makes it compatible with the `error` interface.
2020-06-22 15:12:50 +02:00
func (e *Error) Error() string {
return e.Message
2020-06-22 15:12:50 +02:00
}
// NewError creates a new Error instance with an optional message
func NewError(code int, message ...string) *Error {
err := &Error{
Code: code,
Message: utils.StatusMessage(code),
2020-09-13 11:20:11 +02:00
}
if len(message) > 0 {
err.Message = message[0]
2020-06-22 15:12:50 +02:00
}
return err
}
2020-09-13 11:20:11 +02:00
// Config returns the app config as value ( read-only ).
func (app *App) Config() Config {
return app.config
}
// Handler returns the server handler.
func (app *App) Handler() fasthttp.RequestHandler { //revive:disable-line:confusing-naming // Having both a Handler() (uppercase) and a handler() (lowercase) is fine. TODO: Use nolint:revive directive instead. See https://github.com/golangci/golangci-lint/issues/3476
// prepare the server for the start
app.startupProcess()
return app.requestHandler
}
2020-09-13 11:20:11 +02:00
// Stack returns the raw router stack.
func (app *App) Stack() [][]*Route {
return app.stack
}
// HandlersCount returns the amount of registered handlers.
func (app *App) HandlersCount() uint32 {
return app.handlersCount
}
2020-09-13 11:20:11 +02:00
// Shutdown gracefully shuts down the server without interrupting any active connections.
// Shutdown works by first closing all open listeners and then waiting indefinitely for all connections to return to idle before shutting down.
2020-03-24 05:31:51 +01:00
//
// Make sure the program doesn't exit and waits instead for Shutdown to return.
//
// Shutdown does not close keepalive connections so its recommended to set ReadTimeout to something else than 0.
2020-03-31 10:03:39 +02:00
func (app *App) Shutdown() error {
return app.ShutdownWithContext(context.Background())
}
// ShutdownWithTimeout gracefully shuts down the server without interrupting any active connections. However, if the timeout is exceeded,
// ShutdownWithTimeout will forcefully close any active connections.
// ShutdownWithTimeout works by first closing all open listeners and then waiting for all connections to return to idle before shutting down.
//
// Make sure the program doesn't exit and waits instead for ShutdownWithTimeout to return.
//
// ShutdownWithTimeout does not close keepalive connections so its recommended to set ReadTimeout to something else than 0.
func (app *App) ShutdownWithTimeout(timeout time.Duration) error {
ctx, cancelFunc := context.WithTimeout(context.Background(), timeout)
defer cancelFunc()
return app.ShutdownWithContext(ctx)
}
// ShutdownWithContext shuts down the server including by force if the context's deadline is exceeded.
//
// Make sure the program doesn't exit and waits instead for ShutdownWithTimeout to return.
//
// ShutdownWithContext does not close keepalive connections so its recommended to set ReadTimeout to something else than 0.
func (app *App) ShutdownWithContext(ctx context.Context) error {
if app.hooks != nil {
// TODO: check should be defered?
app.hooks.executeOnShutdownHooks()
}
app.mutex.Lock()
defer app.mutex.Unlock()
2020-02-21 18:07:43 +01:00
if app.server == nil {
return ErrNotRunning
2020-02-21 18:07:43 +01:00
}
return app.server.ShutdownWithContext(ctx)
2020-02-21 18:07:43 +01:00
}
2020-09-27 23:23:28 +02:00
// Server returns the underlying fasthttp server
func (app *App) Server() *fasthttp.Server {
return app.server
}
// Hooks returns the hook struct to register hooks.
2022-08-16 09:13:38 +03:00
func (app *App) Hooks() *Hooks {
return app.hooks
}
// Test is used for internal debugging by passing a *http.Request.
// Timeout is optional and defaults to 1s, -1 will disable it completely.
func (app *App) Test(req *http.Request, timeout ...time.Duration) (*http.Response, error) {
2020-09-13 11:20:11 +02:00
// Set timeout
to := 1 * time.Second
if len(timeout) > 0 {
to = timeout[0]
2020-03-22 17:35:12 +01:00
}
2020-09-13 11:20:11 +02:00
// Add Content-Length if not provided with body
2020-09-13 11:20:11 +02:00
if req.Body != http.NoBody && req.Header.Get(HeaderContentLength) == "" {
req.Header.Add(HeaderContentLength, strconv.FormatInt(req.ContentLength, 10))
}
2020-09-13 11:20:11 +02:00
2020-03-20 16:43:28 +01:00
// Dump raw http request
2020-09-13 11:20:11 +02:00
dump, err := httputil.DumpRequest(req, true)
2020-02-21 18:07:43 +01:00
if err != nil {
return nil, fmt.Errorf("failed to dump request: %w", err)
2020-02-21 18:07:43 +01:00
}
2020-09-13 11:20:11 +02:00
Use param support + optimizations (#361) * Benchmark workflow * Update router.go * Clean root * Add mutex * Benchmark workflow * Benchmark workflow * Add mutex * Enable benchmark tests * Enable race testing Co-Authored-By: ReneWerner87 <renewerner87@users.noreply.github.com> * Benchmark Workflow * Benchmark workflow * Add mutex * Enable benchmark tests * Enable race testing Co-Authored-By: ReneWerner87 <renewerner87@users.noreply.github.com> * Update security workflow * Benchmark workflow * Add mutex * Enable benchmark tests * Enable race testing Co-Authored-By: ReneWerner87 <renewerner87@users.noreply.github.com> * Make Ctx pool accessible - Add ctx benchmarks * v1.9.6 * v1.9.6 Co-Authored-By: ReneWerner87 <renewerner87@googlemail.com> * Improve context functions * Add utils benchmarks * Update benchmarks & tests * Add utils tests * New tests * update test * Move fastpath tests * offer negotiation * Cleanup * Update Vary Co-Authored-By: RW <renewerner87@googlemail.com> * Optimize Append Co-Authored-By: RW <renewerner87@googlemail.com> * Optimize more methods Co-Authored-By: RW <renewerner87@googlemail.com> Co-Authored-By: Vic Shóstak <vikkyshostak@gmail.com> * Add param support to Use Co-Authored-By: RW <renewerner87@googlemail.com> Co-Authored-By: Vic Shóstak <vikkyshostak@gmail.com> * Add use_params tests Co-Authored-By: RW <renewerner87@googlemail.com> Co-Authored-By: Vic Shóstak <vikkyshostak@gmail.com> * v1.9.7 Co-Authored-By: RW <renewerner87@googlemail.com> Co-Authored-By: Vic Shóstak <vikkyshostak@gmail.com> * Tests Co-Authored-By: RW <renewerner87@googlemail.com> Co-Authored-By: Vic Shóstak <vikkyshostak@gmail.com> * v1.9.7 Co-Authored-By: RW <renewerner87@googlemail.com> Co-Authored-By: Vic Shóstak <vikkyshostak@gmail.com> Co-Authored-By: József Sallai <jozsef@sallai.me> Co-Authored-By: Thomas van Vugt <thomasvvugt@users.noreply.github.com> * Update app_test.go Co-Authored-By: RW <renewerner87@googlemail.com> Co-Authored-By: Vic Shóstak <vikkyshostak@gmail.com> Co-Authored-By: József Sallai <jozsef@sallai.me> Co-Authored-By: Thomas van Vugt <thomasvvugt@users.noreply.github.com> Co-Authored-By: Nifty255 <nifty255@users.noreply.github.com> * Rename argument Co-Authored-By: RW <renewerner87@googlemail.com> * Add nosec for WriteByte Co-Authored-By: RW <renewerner87@googlemail.com> * Add media article * Update media articles * Fix typo Co-Authored-By: Thomas van Vugt <thomasvvugt@users.noreply.github.com> * Fix typo Co-authored-by: ReneWerner87 <renewerner87@users.noreply.github.com> Co-authored-by: ReneWerner87 <renewerner87@googlemail.com> Co-authored-by: Vic Shóstak <vikkyshostak@gmail.com> Co-authored-by: József Sallai <jozsef@sallai.me> Co-authored-by: Thomas van Vugt <thomasvvugt@users.noreply.github.com> Co-authored-by: Nifty255 <nifty255@users.noreply.github.com>
2020-05-12 19:24:04 +02:00
// Create test connection
conn := new(testConn)
2020-09-13 11:20:11 +02:00
2020-03-20 16:43:28 +01:00
// Write raw http request
if _, err := conn.r.Write(dump); err != nil {
return nil, fmt.Errorf("failed to write: %w", err)
2020-02-21 18:07:43 +01:00
}
// prepare the server for the start
app.startupProcess()
2020-09-13 11:20:11 +02:00
2020-02-21 18:07:43 +01:00
// Serve conn to server
channel := make(chan error)
go func() {
var returned bool
defer func() {
if !returned {
channel <- ErrHandlerExited
}
}()
Use param support + optimizations (#361) * Benchmark workflow * Update router.go * Clean root * Add mutex * Benchmark workflow * Benchmark workflow * Add mutex * Enable benchmark tests * Enable race testing Co-Authored-By: ReneWerner87 <renewerner87@users.noreply.github.com> * Benchmark Workflow * Benchmark workflow * Add mutex * Enable benchmark tests * Enable race testing Co-Authored-By: ReneWerner87 <renewerner87@users.noreply.github.com> * Update security workflow * Benchmark workflow * Add mutex * Enable benchmark tests * Enable race testing Co-Authored-By: ReneWerner87 <renewerner87@users.noreply.github.com> * Make Ctx pool accessible - Add ctx benchmarks * v1.9.6 * v1.9.6 Co-Authored-By: ReneWerner87 <renewerner87@googlemail.com> * Improve context functions * Add utils benchmarks * Update benchmarks & tests * Add utils tests * New tests * update test * Move fastpath tests * offer negotiation * Cleanup * Update Vary Co-Authored-By: RW <renewerner87@googlemail.com> * Optimize Append Co-Authored-By: RW <renewerner87@googlemail.com> * Optimize more methods Co-Authored-By: RW <renewerner87@googlemail.com> Co-Authored-By: Vic Shóstak <vikkyshostak@gmail.com> * Add param support to Use Co-Authored-By: RW <renewerner87@googlemail.com> Co-Authored-By: Vic Shóstak <vikkyshostak@gmail.com> * Add use_params tests Co-Authored-By: RW <renewerner87@googlemail.com> Co-Authored-By: Vic Shóstak <vikkyshostak@gmail.com> * v1.9.7 Co-Authored-By: RW <renewerner87@googlemail.com> Co-Authored-By: Vic Shóstak <vikkyshostak@gmail.com> * Tests Co-Authored-By: RW <renewerner87@googlemail.com> Co-Authored-By: Vic Shóstak <vikkyshostak@gmail.com> * v1.9.7 Co-Authored-By: RW <renewerner87@googlemail.com> Co-Authored-By: Vic Shóstak <vikkyshostak@gmail.com> Co-Authored-By: József Sallai <jozsef@sallai.me> Co-Authored-By: Thomas van Vugt <thomasvvugt@users.noreply.github.com> * Update app_test.go Co-Authored-By: RW <renewerner87@googlemail.com> Co-Authored-By: Vic Shóstak <vikkyshostak@gmail.com> Co-Authored-By: József Sallai <jozsef@sallai.me> Co-Authored-By: Thomas van Vugt <thomasvvugt@users.noreply.github.com> Co-Authored-By: Nifty255 <nifty255@users.noreply.github.com> * Rename argument Co-Authored-By: RW <renewerner87@googlemail.com> * Add nosec for WriteByte Co-Authored-By: RW <renewerner87@googlemail.com> * Add media article * Update media articles * Fix typo Co-Authored-By: Thomas van Vugt <thomasvvugt@users.noreply.github.com> * Fix typo Co-authored-by: ReneWerner87 <renewerner87@users.noreply.github.com> Co-authored-by: ReneWerner87 <renewerner87@googlemail.com> Co-authored-by: Vic Shóstak <vikkyshostak@gmail.com> Co-authored-by: József Sallai <jozsef@sallai.me> Co-authored-by: Thomas van Vugt <thomasvvugt@users.noreply.github.com> Co-authored-by: Nifty255 <nifty255@users.noreply.github.com>
2020-05-12 19:24:04 +02:00
channel <- app.server.ServeConn(conn)
returned = true
2020-02-21 18:07:43 +01:00
}()
2020-09-13 11:20:11 +02:00
2020-03-23 21:52:37 +01:00
// Wait for callback
if to >= 0 {
// With timeout
select {
case err = <-channel:
case <-time.After(to):
return nil, fmt.Errorf("test: timeout error after %s", to)
2020-02-21 18:07:43 +01:00
}
} else {
// Without timeout
2020-05-12 23:24:04 +02:00
err = <-channel
}
2020-09-13 11:20:11 +02:00
// Check for errors
if err != nil && !errors.Is(err, fasthttp.ErrGetOnly) {
return nil, err
2020-03-24 03:43:13 +01:00
}
2020-09-13 11:20:11 +02:00
2020-03-20 16:43:28 +01:00
// Read response
Use param support + optimizations (#361) * Benchmark workflow * Update router.go * Clean root * Add mutex * Benchmark workflow * Benchmark workflow * Add mutex * Enable benchmark tests * Enable race testing Co-Authored-By: ReneWerner87 <renewerner87@users.noreply.github.com> * Benchmark Workflow * Benchmark workflow * Add mutex * Enable benchmark tests * Enable race testing Co-Authored-By: ReneWerner87 <renewerner87@users.noreply.github.com> * Update security workflow * Benchmark workflow * Add mutex * Enable benchmark tests * Enable race testing Co-Authored-By: ReneWerner87 <renewerner87@users.noreply.github.com> * Make Ctx pool accessible - Add ctx benchmarks * v1.9.6 * v1.9.6 Co-Authored-By: ReneWerner87 <renewerner87@googlemail.com> * Improve context functions * Add utils benchmarks * Update benchmarks & tests * Add utils tests * New tests * update test * Move fastpath tests * offer negotiation * Cleanup * Update Vary Co-Authored-By: RW <renewerner87@googlemail.com> * Optimize Append Co-Authored-By: RW <renewerner87@googlemail.com> * Optimize more methods Co-Authored-By: RW <renewerner87@googlemail.com> Co-Authored-By: Vic Shóstak <vikkyshostak@gmail.com> * Add param support to Use Co-Authored-By: RW <renewerner87@googlemail.com> Co-Authored-By: Vic Shóstak <vikkyshostak@gmail.com> * Add use_params tests Co-Authored-By: RW <renewerner87@googlemail.com> Co-Authored-By: Vic Shóstak <vikkyshostak@gmail.com> * v1.9.7 Co-Authored-By: RW <renewerner87@googlemail.com> Co-Authored-By: Vic Shóstak <vikkyshostak@gmail.com> * Tests Co-Authored-By: RW <renewerner87@googlemail.com> Co-Authored-By: Vic Shóstak <vikkyshostak@gmail.com> * v1.9.7 Co-Authored-By: RW <renewerner87@googlemail.com> Co-Authored-By: Vic Shóstak <vikkyshostak@gmail.com> Co-Authored-By: József Sallai <jozsef@sallai.me> Co-Authored-By: Thomas van Vugt <thomasvvugt@users.noreply.github.com> * Update app_test.go Co-Authored-By: RW <renewerner87@googlemail.com> Co-Authored-By: Vic Shóstak <vikkyshostak@gmail.com> Co-Authored-By: József Sallai <jozsef@sallai.me> Co-Authored-By: Thomas van Vugt <thomasvvugt@users.noreply.github.com> Co-Authored-By: Nifty255 <nifty255@users.noreply.github.com> * Rename argument Co-Authored-By: RW <renewerner87@googlemail.com> * Add nosec for WriteByte Co-Authored-By: RW <renewerner87@googlemail.com> * Add media article * Update media articles * Fix typo Co-Authored-By: Thomas van Vugt <thomasvvugt@users.noreply.github.com> * Fix typo Co-authored-by: ReneWerner87 <renewerner87@users.noreply.github.com> Co-authored-by: ReneWerner87 <renewerner87@googlemail.com> Co-authored-by: Vic Shóstak <vikkyshostak@gmail.com> Co-authored-by: József Sallai <jozsef@sallai.me> Co-authored-by: Thomas van Vugt <thomasvvugt@users.noreply.github.com> Co-authored-by: Nifty255 <nifty255@users.noreply.github.com>
2020-05-12 19:24:04 +02:00
buffer := bufio.NewReader(&conn.w)
2020-09-13 11:20:11 +02:00
2020-03-20 16:43:28 +01:00
// Convert raw http response to *http.Response
res, err := http.ReadResponse(buffer, req)
if err != nil {
return nil, fmt.Errorf("failed to read response: %w", err)
}
return res, nil
2020-02-21 18:07:43 +01:00
}
type disableLogger struct{}
func (*disableLogger) Printf(string, ...any) {
}
func (app *App) init() *App {
2020-09-13 11:20:11 +02:00
// lock application
Use param support + optimizations (#361) * Benchmark workflow * Update router.go * Clean root * Add mutex * Benchmark workflow * Benchmark workflow * Add mutex * Enable benchmark tests * Enable race testing Co-Authored-By: ReneWerner87 <renewerner87@users.noreply.github.com> * Benchmark Workflow * Benchmark workflow * Add mutex * Enable benchmark tests * Enable race testing Co-Authored-By: ReneWerner87 <renewerner87@users.noreply.github.com> * Update security workflow * Benchmark workflow * Add mutex * Enable benchmark tests * Enable race testing Co-Authored-By: ReneWerner87 <renewerner87@users.noreply.github.com> * Make Ctx pool accessible - Add ctx benchmarks * v1.9.6 * v1.9.6 Co-Authored-By: ReneWerner87 <renewerner87@googlemail.com> * Improve context functions * Add utils benchmarks * Update benchmarks & tests * Add utils tests * New tests * update test * Move fastpath tests * offer negotiation * Cleanup * Update Vary Co-Authored-By: RW <renewerner87@googlemail.com> * Optimize Append Co-Authored-By: RW <renewerner87@googlemail.com> * Optimize more methods Co-Authored-By: RW <renewerner87@googlemail.com> Co-Authored-By: Vic Shóstak <vikkyshostak@gmail.com> * Add param support to Use Co-Authored-By: RW <renewerner87@googlemail.com> Co-Authored-By: Vic Shóstak <vikkyshostak@gmail.com> * Add use_params tests Co-Authored-By: RW <renewerner87@googlemail.com> Co-Authored-By: Vic Shóstak <vikkyshostak@gmail.com> * v1.9.7 Co-Authored-By: RW <renewerner87@googlemail.com> Co-Authored-By: Vic Shóstak <vikkyshostak@gmail.com> * Tests Co-Authored-By: RW <renewerner87@googlemail.com> Co-Authored-By: Vic Shóstak <vikkyshostak@gmail.com> * v1.9.7 Co-Authored-By: RW <renewerner87@googlemail.com> Co-Authored-By: Vic Shóstak <vikkyshostak@gmail.com> Co-Authored-By: József Sallai <jozsef@sallai.me> Co-Authored-By: Thomas van Vugt <thomasvvugt@users.noreply.github.com> * Update app_test.go Co-Authored-By: RW <renewerner87@googlemail.com> Co-Authored-By: Vic Shóstak <vikkyshostak@gmail.com> Co-Authored-By: József Sallai <jozsef@sallai.me> Co-Authored-By: Thomas van Vugt <thomasvvugt@users.noreply.github.com> Co-Authored-By: Nifty255 <nifty255@users.noreply.github.com> * Rename argument Co-Authored-By: RW <renewerner87@googlemail.com> * Add nosec for WriteByte Co-Authored-By: RW <renewerner87@googlemail.com> * Add media article * Update media articles * Fix typo Co-Authored-By: Thomas van Vugt <thomasvvugt@users.noreply.github.com> * Fix typo Co-authored-by: ReneWerner87 <renewerner87@users.noreply.github.com> Co-authored-by: ReneWerner87 <renewerner87@googlemail.com> Co-authored-by: Vic Shóstak <vikkyshostak@gmail.com> Co-authored-by: József Sallai <jozsef@sallai.me> Co-authored-by: Thomas van Vugt <thomasvvugt@users.noreply.github.com> Co-authored-by: Nifty255 <nifty255@users.noreply.github.com>
2020-05-12 19:24:04 +02:00
app.mutex.Lock()
2020-08-11 00:08:04 +02:00
// Only load templates if a view engine is specified
2020-09-13 11:20:11 +02:00
if app.config.Views != nil {
if err := app.config.Views.Load(); err != nil {
log.Warnf("failed to load views: %v", err)
2020-06-12 12:29:57 +02:00
}
}
2020-09-13 11:20:11 +02:00
// create fasthttp server
app.server = &fasthttp.Server{
Logger: &disableLogger{},
LogAllErrors: false,
ErrorHandler: app.serverErrorHandler,
2020-05-16 05:11:25 +02:00
}
2020-09-13 11:20:11 +02:00
// fasthttp server settings
app.server.Handler = app.requestHandler
2020-09-13 11:20:11 +02:00
app.server.Name = app.config.ServerHeader
app.server.Concurrency = app.config.Concurrency
app.server.NoDefaultDate = app.config.DisableDefaultDate
app.server.NoDefaultContentType = app.config.DisableDefaultContentType
app.server.DisableHeaderNamesNormalizing = app.config.DisableHeaderNormalizing
app.server.DisableKeepalive = app.config.DisableKeepalive
app.server.MaxRequestBodySize = app.config.BodyLimit
app.server.NoDefaultServerHeader = app.config.ServerHeader == ""
app.server.ReadTimeout = app.config.ReadTimeout
app.server.WriteTimeout = app.config.WriteTimeout
app.server.IdleTimeout = app.config.IdleTimeout
app.server.ReadBufferSize = app.config.ReadBufferSize
app.server.WriteBufferSize = app.config.WriteBufferSize
app.server.GetOnly = app.config.GETOnly
app.server.ReduceMemoryUsage = app.config.ReduceMemoryUsage
app.server.StreamRequestBody = app.config.StreamRequestBody
app.server.DisablePreParseMultipartForm = app.config.DisablePreParseMultipartForm
2020-09-13 11:20:11 +02:00
// unlock application
app.mutex.Unlock()
return app
2020-02-21 18:07:43 +01:00
}
2020-06-07 21:10:38 +02:00
// ErrorHandler is the application's method in charge of finding the
// appropriate handler for the given request. It searches any mounted
// sub fibers by their prefixes and if it finds a match, it uses that
// error handler. Otherwise it uses the configured error handler for
// the app, which if not set is the DefaultErrorHandler.
func (app *App) ErrorHandler(ctx Ctx, err error) error {
var (
mountedErrHandler ErrorHandler
mountedPrefixParts int
)
for prefix, subApp := range app.mountFields.appList {
if prefix != "" && strings.HasPrefix(ctx.Path(), prefix) {
parts := len(strings.Split(prefix, "/"))
if mountedPrefixParts <= parts {
if subApp.configured.ErrorHandler != nil {
mountedErrHandler = subApp.config.ErrorHandler
}
mountedPrefixParts = parts
}
}
}
if mountedErrHandler != nil {
return mountedErrHandler(ctx, err)
}
return app.config.ErrorHandler(ctx, err)
}
// serverErrorHandler is a wrapper around the application's error handler method
// user for the fasthttp server configuration. It maps a set of fasthttp errors to fiber
// errors before calling the application's error handler method.
func (app *App) serverErrorHandler(fctx *fasthttp.RequestCtx, err error) {
// Acquire Ctx with fasthttp request from pool
V2 to v3 merge (#2864) * Update pull_request_template.md * Update v3-changes.md * Update CONTRIBUTING.md (#2752) Grammar correction. * chore(encryptcookie)!: update default config (#2753) * chore(encryptcookie)!: update default config docs(encryptcookie): enhance documentation and examples BREAKING CHANGE: removed the hardcoded "csrf_" from the Except. * docs(encryptcookie): reads or modifies cookies * chore(encryptcookie): csrf config example * docs(encryptcookie): md table spacing * build(deps): bump actions/setup-go from 4 to 5 (#2754) Bumps [actions/setup-go](https://github.com/actions/setup-go) from 4 to 5. - [Release notes](https://github.com/actions/setup-go/releases) - [Commits](https://github.com/actions/setup-go/compare/v4...v5) --- updated-dependencies: - dependency-name: actions/setup-go dependency-type: direct:production update-type: version-update:semver-major ... Signed-off-by: dependabot[bot] <support@github.com> Co-authored-by: dependabot[bot] <49699333+dependabot[bot]@users.noreply.github.com> * 🩹 middleware/logger/: log client IP address by default (#2755) * middleware/logger: Log client IP address by default. * Update doc. * fix: don't constrain middlewares' context-keys to strings :bug: (#2751) * Revert "Revert ":bug: requestid.Config.ContextKey is interface{} (#2369)" (#2742)" This reverts commit 28be17f929cfa7d3c27dd292fc3956f2f9882e22. * fix: request ContextKey default value condition Should check for `nil` since it is `any`. * fix: don't constrain middlewares' context-keys to strings `context` recommends using "unexported type" as context keys to avoid collisions https://pkg.go.dev/github.com/gofiber/fiber/v2#Ctx.Locals. The official go blog also recommends this https://go.dev/blog/context. `fiber.Ctx.Locals(key any, value any)` correctly allows consumers to use unexported types or e.g. strings. But some fiber middlewares constrain their context-keys to `string` in their "default config structs", making it impossible to use unexported types. This PR removes the `string` _constraint_ from all middlewares, allowing to now use unexported types as per the official guidelines. However the default value is still a string, so it's not a breaking change, and anyone still using strings as context keys is not affected. * 📚 Update app.md for indentation (#2761) Update app.md for indentation * build(deps): bump github.com/google/uuid from 1.4.0 to 1.5.0 (#2762) Bumps [github.com/google/uuid](https://github.com/google/uuid) from 1.4.0 to 1.5.0. - [Release notes](https://github.com/google/uuid/releases) - [Changelog](https://github.com/google/uuid/blob/master/CHANGELOG.md) - [Commits](https://github.com/google/uuid/compare/v1.4.0...v1.5.0) --- updated-dependencies: - dependency-name: github.com/google/uuid dependency-type: direct:production update-type: version-update:semver-minor ... Signed-off-by: dependabot[bot] <support@github.com> Co-authored-by: dependabot[bot] <49699333+dependabot[bot]@users.noreply.github.com> * build(deps): bump github/codeql-action from 2 to 3 (#2763) Bumps [github/codeql-action](https://github.com/github/codeql-action) from 2 to 3. - [Release notes](https://github.com/github/codeql-action/releases) - [Changelog](https://github.com/github/codeql-action/blob/main/CHANGELOG.md) - [Commits](https://github.com/github/codeql-action/compare/v2...v3) --- updated-dependencies: - dependency-name: github/codeql-action dependency-type: direct:production update-type: version-update:semver-major ... Signed-off-by: dependabot[bot] <support@github.com> Co-authored-by: dependabot[bot] <49699333+dependabot[bot]@users.noreply.github.com> * Changing default log output (#2730) changing default log output Closes #2729 * Update hooks.md fix wrong hooks signature * 🩹 Fix: CORS middleware should use the defined AllowedOriginsFunc config when AllowedOrigins is empty (#2771) * 🐛 [Bug]: Adaptator + otelfiber issue #2641 (#2772) * 🩹🚨 - fix for redirect with query params (#2748) * redirect with query params did not work, fix it and add test for it * redirect middleware - fix test typo * ♻️ logger/middleware colorize logger error message #2593 (#2773) * :sparkles: feat: add liveness and readiness checks (#2509) * :sparkles: feat: add liveness and readiness checkers * :memo: docs: add docs for liveness and readiness * :sparkles: feat: add options method for probe checkers * :white_check_mark: tests: add tests for liveness and readiness * :recycle: refactor: change default endpoint values * :recycle: refactor: change default value for liveness endpoint * :memo: docs: add return status for liveness and readiness probes * :recycle: refactor: change probechecker to middleware * :memo: docs: move docs to middleware session * :recycle: refactor: apply gofumpt formatting * :recycle: refactor: remove unused parameter * split config and apply a review * apply reviews and add testcases * add benchmark * cleanup * rename middleware * fix linter * Update docs and config values * Revert change to IsReady * Updates based on code review * Update docs to match other middlewares --------- Co-authored-by: Muhammed Efe Cetin <efectn@protonmail.com> Co-authored-by: Juan Calderon-Perez <835733+gaby@users.noreply.github.com> Co-authored-by: Juan Calderon-Perez <jgcalderonperez@protonmail.com> * prepare release v2.52.0 - add more Parser tests * fix healthcheck.md * configure workflows for V2 branch * configure workflows for V2 branch * Fix default value to false in docs of QueryBool (#2811) fix default value to false in docs of QueryBool * update queryParser config * Update ctx.md * Update routing.md * merge v2 in v3 * merge v2 in v3 * lint fixes * :books: Doc: Fix code snippet indentation in /docs/api/middleware/keyauth.md Removes an an extra level of indentation in line 51 of `keyauth.md` [here](https://github.com/gofiber/fiber/blob/v2/docs/api/middleware/keyauth.md?plain=1#L51) * fix: healthcheck middleware not working with route group (#2863) * fix: healthcheck middleware not working with route group * perf: change verification method to improve perf * Update healthcheck_test.go * test: add not matching route test for strict routing * add more test cases * correct tests * correct test helpers * correct tests * correct tests --------- Co-authored-by: Juan Calderon-Perez <835733+gaby@users.noreply.github.com> Co-authored-by: René Werner <rene@gofiber.io> * merge v2 in v3 * Merge pull request from GHSA-fmg4-x8pw-hjhg * Enforce Wildcard Origins with AllowCredentials check * Expand unit-tests, fix issues with subdomains logic, update docs * Update cors.md * Added test using localhost, ipv4, and ipv6 address * improve documentation markdown --------- Co-authored-by: René Werner <rene@gofiber.io> * Update app.go prepare release v2.52.1 * fix cors domain normalize * fix sync-docs workflow * test: fix failing tests * fix sync-docs workflow * test: cors middleware use testify require * chore: fix lint warnings * chore: revert test isolation. * fixed the fasthttp ctx race condition problem * Update middleware/cors/utils.go Co-authored-by: Renan Bastos <renanbastos.tec@gmail.com> * fix sync_docs.sh * fix review comments/hints * fix review comments/hints * stabilize Test_Proxy_Timeout_Slow_Server test * stabilize Test_Proxy_.* tests * ignore bodyclose linter for tests use http.NoBody instead of nil * revert(tests): undo http.NoBody usage * fix(ctx pool): postpone the reset for some values shortly before the release in the pool * refactor(tests): use testify panic method instead of custom solution --------- Signed-off-by: dependabot[bot] <support@github.com> Co-authored-by: tokelo-12 <113810058+tokelo-12@users.noreply.github.com> Co-authored-by: Jason McNeil <sixcolors@mac.com> Co-authored-by: dependabot[bot] <49699333+dependabot[bot]@users.noreply.github.com> Co-authored-by: iRedMail <2048991+iredmail@users.noreply.github.com> Co-authored-by: Benjamin Grosse <ste3ls@gmail.com> Co-authored-by: Mehmet Firat KOMURCU <mehmetfiratkomurcu@hotmail.com> Co-authored-by: Bruno <bdm2943@icloud.com> Co-authored-by: Muhammad Kholid B <muhammadkholidb@gmail.com> Co-authored-by: gilwo <gilwo@users.noreply.github.com> Co-authored-by: Lucas Lemos <lucashenriqueblemos@gmail.com> Co-authored-by: Muhammed Efe Cetin <efectn@protonmail.com> Co-authored-by: Juan Calderon-Perez <835733+gaby@users.noreply.github.com> Co-authored-by: Juan Calderon-Perez <jgcalderonperez@protonmail.com> Co-authored-by: Jongmin Kim <kjongmin26@gmail.com> Co-authored-by: Giovanni Rivera <rivera.giovanni271@gmail.com> Co-authored-by: Renan Bastos <renanbastos.tec@gmail.com>
2024-02-29 08:29:59 +01:00
c := app.AcquireCtx(fctx)
defer app.ReleaseCtx(c)
var (
errNetOP *net.OpError
netErr net.Error
)
switch {
case errors.As(err, new(*fasthttp.ErrSmallBuffer)):
err = ErrRequestHeaderFieldsTooLarge
case errors.As(err, &errNetOP) && errNetOP.Timeout():
err = ErrRequestTimeout
case errors.As(err, &netErr):
err = ErrBadGateway
case errors.Is(err, fasthttp.ErrBodyTooLarge):
err = ErrRequestEntityTooLarge
case errors.Is(err, fasthttp.ErrGetOnly):
err = ErrMethodNotAllowed
case strings.Contains(err.Error(), "timeout"):
err = ErrRequestTimeout
default:
err = NewError(StatusBadRequest, err.Error())
}
if catch := app.ErrorHandler(c, err); catch != nil {
log.Errorf("serverErrorHandler: failed to call ErrorHandler: %v", catch)
_ = c.SendStatus(StatusInternalServerError) //nolint:errcheck // It is fine to ignore the error here
return
}
}
// startupProcess Is the method which executes all the necessary processes just before the start of the server.
func (app *App) startupProcess() *App {
app.mutex.Lock()
defer app.mutex.Unlock()
app.mountStartupProcess()
// build route tree stack
app.buildTree()
return app
}
// Run onListen hooks. If they return an error, panic.
func (app *App) runOnListenHooks(listenData ListenData) {
if err := app.hooks.executeOnListenHooks(listenData); err != nil {
panic(err)
}
}