1
0
mirror of https://github.com/gofiber/fiber.git synced 2025-02-12 03:01:21 +00:00
fiber/app.go

541 lines
17 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
import (
"bufio"
2020-03-04 12:30:29 +01:00
"crypto/tls"
2020-02-21 18:07:43 +01:00
"fmt"
2020-02-27 04:10:26 -05:00
"log"
2020-02-21 18:07:43 +01:00
"net"
"net/http"
"net/http/httputil"
"os"
"os/exec"
2020-02-27 04:10:26 -05:00
"reflect"
2020-02-21 18:07:43 +01:00
"runtime"
"strconv"
"strings"
"time"
fasthttp "github.com/valyala/fasthttp"
)
2020-03-24 05:31:51 +01:00
// Version of current package
const Version = "1.9.5"
2020-02-21 18:07:43 +01:00
2020-03-24 05:31:51 +01:00
// Map is a shortcut for map[string]interface{}
type Map map[string]interface{}
2020-03-31 10:03:39 +02:00
// App denotes the Fiber application.
type App struct {
2020-03-24 05:31:51 +01:00
server *fasthttp.Server // FastHTTP server
routes [][]*Route // Route stack
2020-03-24 05:31:51 +01:00
Settings *Settings // Fiber settings
}
// Settings holds is a struct holding the server settings
type Settings struct {
// This will spawn multiple Go processes listening on the same port
Prefork bool // default: false
// Enable strict routing. When enabled, the router treats "/foo" and "/foo/" as different.
StrictRouting bool // default: false
// Enable case sensitivity. When enabled, "/Foo" and "/foo" are different routes.
CaseSensitive bool // default: false
// Enables the "Server: value" HTTP header.
ServerHeader string // default: ""
// Enables handler values to be immutable even if you return from handler
Immutable bool // default: false
// Enable or disable ETag header generation, since both weak and strong etags are generated
// using the same hashing method (CRC-32). Weak ETags are the default when enabled.
// Optional. Default value false
ETag bool
2020-03-24 05:31:51 +01:00
// Max body size that the server accepts
BodyLimit int // default: 4 * 1024 * 1024
2020-04-19 16:10:19 +02:00
// Maximum number of concurrent connections.
Concurrency int // default: 256 * 1024
// Disable keep-alive connections, the server will close incoming connections after sending the first response to client
DisableKeepalive bool // default: false
// When set to true causes the default date header to be excluded from the response.
DisableDefaultDate bool // default: false
// When set to true, causes the default Content-Type header to be excluded from the Response.
DisableDefaultContentType bool // default: false
// When set to true, it will not print out the fiber ASCII and "listening" on message
DisableStartupMessage bool
2020-03-24 05:31:51 +01:00
// Folder containing template files
TemplateFolder string // default: ""
// Template engine: html, amber, handlebars , mustache or pug
TemplateEngine func(raw string, bind interface{}) (string, error) // default: nil
// Extension for the template files
TemplateExtension string // default: ""
// The amount of time allowed to read the full request including body.
ReadTimeout time.Duration // default: unlimited
// The maximum duration before timing out writes of the response.
WriteTimeout time.Duration // default: unlimited
// The maximum amount of time to wait for the next request when keep-alive is enabled.
IdleTimeout time.Duration // default: unlimited
}
2020-02-21 18:07:43 +01:00
// Group struct
type Group struct {
prefix string
2020-03-31 10:03:39 +02:00
app *App
}
// Static struct
type Static struct {
// Transparently compresses responses if set to true
// This works differently than the github.com/gofiber/compression middleware
// The server tries minimizing CPU usage by caching compressed files.
// It adds ".fiber.gz" suffix to the original file name.
// Optional. Default value false
Compress bool
// Enables byte range requests if set to true.
// Optional. Default value false
ByteRange bool
// Enable directory browsing.
// Optional. Default value false.
Browse bool
// Index file for serving a directory.
// Optional. Default value "index.html".
Index string
}
2020-03-24 05:31:51 +01:00
// New creates a new Fiber named instance.
2020-03-16 17:18:25 +01:00
// You can pass optional settings when creating a new instance.
2020-03-31 10:03:39 +02:00
func New(settings ...*Settings) *App {
2020-04-13 09:01:27 +02:00
schemaDecoderForm.SetAliasTag("form")
schemaDecoderForm.IgnoreUnknownKeys(true)
2020-04-13 09:01:27 +02:00
schemaDecoderQuery.SetAliasTag("query")
schemaDecoderQuery.IgnoreUnknownKeys(true)
2020-03-24 05:31:51 +01:00
// Create app
2020-03-31 10:03:39 +02:00
app := new(App)
// Create route stack
app.routes = make([][]*Route, len(methodINT), len(methodINT))
2020-03-24 05:31:51 +01:00
// Create settings
app.Settings = new(Settings)
// Set default settings
2020-04-12 14:58:05 +02:00
app.Settings.Prefork = isPrefork()
2020-03-24 05:31:51 +01:00
app.Settings.BodyLimit = 4 * 1024 * 1024
2020-03-27 06:56:58 +01:00
// If settings exist, set defaults
2020-02-21 18:07:43 +01:00
if len(settings) > 0 {
2020-03-14 12:30:21 +01:00
app.Settings = settings[0] // Set custom settings
if !app.Settings.Prefork { // Default to -prefork flag if false
2020-04-12 14:58:05 +02:00
app.Settings.Prefork = isPrefork()
2020-02-26 19:31:43 -05:00
}
2020-04-19 16:10:19 +02:00
if app.Settings.BodyLimit <= 0 { // Default MaxRequestBodySize
2020-03-14 12:30:21 +01:00
app.Settings.BodyLimit = 4 * 1024 * 1024
2020-02-21 18:07:43 +01:00
}
2020-04-19 16:10:19 +02:00
if app.Settings.Concurrency <= 0 {
app.Settings.Concurrency = 256 * 1024
}
2020-03-14 12:30:21 +01:00
if app.Settings.Immutable { // Replace unsafe conversion funcs
2020-03-24 05:31:51 +01:00
getString = getStringImmutable
getBytes = getBytesImmutable
2020-02-21 18:07:43 +01:00
}
2020-03-14 12:30:21 +01:00
}
2020-03-04 12:30:29 +01:00
return app
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.
2020-03-31 10:03:39 +02:00
func (app *App) Group(prefix string, handlers ...func(*Ctx)) *Group {
2020-02-27 04:10:26 -05:00
if len(handlers) > 0 {
2020-03-03 12:21:34 -05:00
app.registerMethod("USE", prefix, handlers...)
2020-02-27 04:10:26 -05:00
}
return &Group{
prefix: prefix,
app: app,
}
2020-02-21 22:36:26 -05:00
}
2020-03-24 05:31:51 +01:00
// Static registers a new route with path prefix to serve static files from the provided root directory.
2020-03-31 10:03:39 +02:00
func (app *App) Static(prefix, root string, config ...Static) *App {
2020-03-15 14:00:03 +01:00
app.registerStatic(prefix, root, config...)
2020-02-21 18:07:43 +01:00
return app
}
2020-03-24 05:31:51 +01:00
// Use registers a middleware route.
// Middleware matches requests beginning with the provided prefix.
// Providing a prefix is optional, it defaults to "/"
2020-03-31 10:03:39 +02:00
func (app *App) Use(args ...interface{}) *App {
2020-02-27 04:10:26 -05:00
var path = ""
var handlers []func(*Ctx)
for i := 0; i < len(args); i++ {
switch arg := args[i].(type) {
case string:
path = arg
case func(*Ctx):
handlers = append(handlers, arg)
default:
log.Fatalf("Invalid handler: %v", reflect.TypeOf(arg))
}
}
2020-03-03 12:21:34 -05:00
app.registerMethod("USE", path, handlers...)
2020-02-21 18:07:43 +01:00
return app
}
2020-02-26 19:31:43 -05:00
// Connect : https://fiber.wiki/application#http-methods
2020-03-31 10:03:39 +02:00
func (app *App) Connect(path string, handlers ...func(*Ctx)) *App {
2020-03-20 16:49:46 +01:00
app.registerMethod(MethodConnect, path, handlers...)
2020-02-21 18:07:43 +01:00
return app
}
2020-02-26 19:31:43 -05:00
// Put : https://fiber.wiki/application#http-methods
2020-03-31 10:03:39 +02:00
func (app *App) Put(path string, handlers ...func(*Ctx)) *App {
2020-03-20 16:49:46 +01:00
app.registerMethod(MethodPut, path, handlers...)
2020-02-21 18:07:43 +01:00
return app
}
2020-02-26 19:31:43 -05:00
// Post : https://fiber.wiki/application#http-methods
2020-03-31 10:03:39 +02:00
func (app *App) Post(path string, handlers ...func(*Ctx)) *App {
2020-03-20 16:49:46 +01:00
app.registerMethod(MethodPost, path, handlers...)
2020-02-21 18:07:43 +01:00
return app
}
2020-02-26 19:31:43 -05:00
// Delete : https://fiber.wiki/application#http-methods
2020-03-31 10:03:39 +02:00
func (app *App) Delete(path string, handlers ...func(*Ctx)) *App {
2020-03-20 16:49:46 +01:00
app.registerMethod(MethodDelete, path, handlers...)
2020-02-21 18:07:43 +01:00
return app
}
2020-02-26 19:31:43 -05:00
// Head : https://fiber.wiki/application#http-methods
2020-03-31 10:03:39 +02:00
func (app *App) Head(path string, handlers ...func(*Ctx)) *App {
2020-03-20 16:49:46 +01:00
app.registerMethod(MethodHead, path, handlers...)
2020-02-21 18:07:43 +01:00
return app
}
2020-02-26 19:31:43 -05:00
// Patch : https://fiber.wiki/application#http-methods
2020-03-31 10:03:39 +02:00
func (app *App) Patch(path string, handlers ...func(*Ctx)) *App {
2020-03-20 16:49:46 +01:00
app.registerMethod(MethodPatch, path, handlers...)
2020-02-21 18:07:43 +01:00
return app
}
2020-02-26 19:31:43 -05:00
// Options : https://fiber.wiki/application#http-methods
2020-03-31 10:03:39 +02:00
func (app *App) Options(path string, handlers ...func(*Ctx)) *App {
2020-03-20 16:49:46 +01:00
app.registerMethod(MethodOptions, path, handlers...)
2020-02-21 18:07:43 +01:00
return app
}
2020-02-26 19:31:43 -05:00
// Trace : https://fiber.wiki/application#http-methods
2020-03-31 10:03:39 +02:00
func (app *App) Trace(path string, handlers ...func(*Ctx)) *App {
2020-03-20 16:49:46 +01:00
app.registerMethod(MethodTrace, path, handlers...)
2020-02-21 18:07:43 +01:00
return app
}
2020-02-26 19:31:43 -05:00
// Get : https://fiber.wiki/application#http-methods
2020-03-31 10:03:39 +02:00
func (app *App) Get(path string, handlers ...func(*Ctx)) *App {
2020-03-20 16:49:46 +01:00
app.registerMethod(MethodGet, path, handlers...)
2020-02-21 18:07:43 +01:00
return app
}
2020-03-16 17:18:25 +01:00
// All matches all HTTP methods and complete paths
2020-03-31 10:03:39 +02:00
func (app *App) All(path string, handlers ...func(*Ctx)) *App {
2020-03-03 12:21:34 -05:00
app.registerMethod("ALL", path, handlers...)
2020-02-21 18:07:43 +01:00
return app
}
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.
func (grp *Group) Group(prefix string, handlers ...func(*Ctx)) *Group {
prefix = getGroupPath(grp.prefix, prefix)
if len(handlers) > 0 {
grp.app.registerMethod("USE", prefix, handlers...)
}
return &Group{
prefix: prefix,
app: grp.app,
}
}
// Static : https://fiber.wiki/application#static
func (grp *Group) Static(prefix, root string, config ...Static) *Group {
prefix = getGroupPath(grp.prefix, prefix)
grp.app.registerStatic(prefix, root, config...)
return grp
}
2020-03-24 05:31:51 +01:00
// Use registers a middleware route.
// Middleware matches requests beginning with the provided prefix.
// Providing a prefix is optional, it defaults to "/"
func (grp *Group) Use(args ...interface{}) *Group {
var path = ""
var handlers []func(*Ctx)
for i := 0; i < len(args); i++ {
switch arg := args[i].(type) {
case string:
path = arg
case func(*Ctx):
handlers = append(handlers, arg)
default:
log.Fatalf("Invalid Use() arguments, must be (prefix, handler) or (handler)")
}
}
grp.app.registerMethod("USE", getGroupPath(grp.prefix, path), handlers...)
return grp
}
// Connect : https://fiber.wiki/application#http-methods
func (grp *Group) Connect(path string, handlers ...func(*Ctx)) *Group {
grp.app.registerMethod(MethodConnect, getGroupPath(grp.prefix, path), handlers...)
return grp
}
// Put : https://fiber.wiki/application#http-methods
func (grp *Group) Put(path string, handlers ...func(*Ctx)) *Group {
grp.app.registerMethod(MethodPut, getGroupPath(grp.prefix, path), handlers...)
return grp
}
// Post : https://fiber.wiki/application#http-methods
func (grp *Group) Post(path string, handlers ...func(*Ctx)) *Group {
grp.app.registerMethod(MethodPost, getGroupPath(grp.prefix, path), handlers...)
return grp
}
// Delete : https://fiber.wiki/application#http-methods
func (grp *Group) Delete(path string, handlers ...func(*Ctx)) *Group {
grp.app.registerMethod(MethodDelete, getGroupPath(grp.prefix, path), handlers...)
return grp
}
// Head : https://fiber.wiki/application#http-methods
func (grp *Group) Head(path string, handlers ...func(*Ctx)) *Group {
grp.app.registerMethod(MethodHead, getGroupPath(grp.prefix, path), handlers...)
return grp
}
// Patch : https://fiber.wiki/application#http-methods
func (grp *Group) Patch(path string, handlers ...func(*Ctx)) *Group {
grp.app.registerMethod(MethodPatch, getGroupPath(grp.prefix, path), handlers...)
return grp
}
// Options : https://fiber.wiki/application#http-methods
func (grp *Group) Options(path string, handlers ...func(*Ctx)) *Group {
grp.app.registerMethod(MethodOptions, getGroupPath(grp.prefix, path), handlers...)
return grp
}
// Trace : https://fiber.wiki/application#http-methods
func (grp *Group) Trace(path string, handlers ...func(*Ctx)) *Group {
grp.app.registerMethod(MethodTrace, getGroupPath(grp.prefix, path), handlers...)
return grp
}
// Get : https://fiber.wiki/application#http-methods
func (grp *Group) Get(path string, handlers ...func(*Ctx)) *Group {
grp.app.registerMethod(MethodGet, getGroupPath(grp.prefix, path), handlers...)
return grp
}
// All matches all HTTP methods and complete paths
func (grp *Group) All(path string, handlers ...func(*Ctx)) *Group {
grp.app.registerMethod("ALL", getGroupPath(grp.prefix, path), handlers...)
return grp
}
2020-04-12 14:58:05 +02:00
// Serve can be used to pass a custom listener
// This method does not support the Prefork feature
2020-04-23 00:33:36 +02:00
// Preforkin is not available using app.Serve(ln net.Listener)
2020-04-12 14:58:05 +02:00
// You can pass an optional *tls.Config to enable TLS.
func (app *App) Serve(ln net.Listener, tlsconfig ...*tls.Config) error {
// Create fasthttp server
app.server = app.newServer()
// TLS config
if len(tlsconfig) > 0 {
ln = tls.NewListener(ln, tlsconfig[0])
}
// Print listening message
if !app.Settings.DisableStartupMessage {
fmt.Printf(" _______ __\n ____ / ____(_) /_ ___ _____\n_____ / /_ / / __ \\/ _ \\/ ___/\n __ / __/ / / /_/ / __/ /\n /_/ /_/_.___/\\___/_/ v%s\n", Version)
fmt.Printf("Started listening on %s\n", ln.Addr().String())
}
2020-04-12 14:58:05 +02:00
return app.server.Serve(ln)
}
2020-03-24 05:31:51 +01:00
// Listen serves HTTP requests from the given addr or port.
// You can pass an optional *tls.Config to enable TLS.
2020-03-31 10:03:39 +02:00
func (app *App) Listen(address interface{}, tlsconfig ...*tls.Config) error {
2020-02-21 18:07:43 +01:00
addr, ok := address.(string)
if !ok {
port, ok := address.(int)
if !ok {
return fmt.Errorf("Listen: Host must be an INT port or STRING address")
}
addr = strconv.Itoa(port)
}
if !strings.Contains(addr, ":") {
addr = ":" + addr
}
// Create fasthttp server
app.server = app.newServer()
2020-04-12 14:58:05 +02:00
2020-02-21 18:07:43 +01:00
var ln net.Listener
var err error
// Prefork enabled
2020-04-12 14:58:05 +02:00
if app.Settings.Prefork && runtime.NumCPU() > 1 && runtime.GOOS != "windows" {
2020-02-21 18:07:43 +01:00
if ln, err = app.prefork(addr); err != nil {
return err
}
} else {
if ln, err = net.Listen("tcp4", addr); err != nil {
return err
}
}
2020-03-04 12:30:29 +01:00
// TLS config
if len(tlsconfig) > 0 {
ln = tls.NewListener(ln, tlsconfig[0])
2020-02-21 18:07:43 +01:00
}
2020-04-12 14:58:05 +02:00
// Print listening message
if !app.Settings.DisableStartupMessage && !isChild() {
2020-04-25 14:33:40 +02:00
fmt.Printf(" _______ __\n ____ / ____(_) /_ ___ _____\n_____ / /_ / / __ \\/ _ \\/ ___/\n __ / __/ / / /_/ / __/ /\n /_/ /_/_.___/\\___/_/ v%s\n", Version)
fmt.Printf("Started listening on %s\n", ln.Addr().String())
2020-04-12 14:58:05 +02:00
}
2020-02-21 18:07:43 +01:00
return app.server.Serve(ln)
}
2020-03-24 05:31:51 +01: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 and then shut down.
//
// When Shutdown is called, Serve, ListenAndServe, and ListenAndServeTLS immediately return nil.
// 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 {
2020-02-21 18:07:43 +01:00
if app.server == nil {
return fmt.Errorf("Server is not running")
}
return app.server.Shutdown()
}
2020-04-12 14:58:05 +02:00
// Test is used for internal debugging by passing a *http.Request
// Timeout is optional and defaults to 1s, -1 will disable it completely.
2020-03-31 10:03:39 +02:00
func (app *App) Test(request *http.Request, msTimeout ...int) (*http.Response, error) {
timeout := 1000 // 1 second default
2020-03-22 17:35:12 +01:00
if len(msTimeout) > 0 {
timeout = msTimeout[0]
}
2020-03-20 16:43:28 +01:00
// Dump raw http request
dump, err := httputil.DumpRequest(request, true)
2020-02-21 18:07:43 +01:00
if err != nil {
return nil, err
}
2020-03-20 16:43:28 +01:00
// Setup server
2020-02-21 18:07:43 +01:00
app.server = app.newServer()
2020-03-20 16:43:28 +01:00
// Create conn
conn := new(testConn)
// Write raw http request
if _, err = conn.r.Write(dump); err != nil {
2020-02-21 18:07:43 +01:00
return nil, err
}
// Serve conn to server
channel := make(chan error)
go func() {
channel <- app.server.ServeConn(conn)
}()
2020-03-23 21:52:37 +01:00
// Wait for callback
if timeout >= 0 {
// With timeout
select {
case err = <-channel:
case <-time.After(time.Duration(timeout) * time.Millisecond):
return nil, fmt.Errorf("Timeout error %vms", timeout)
2020-02-21 18:07:43 +01:00
}
} else {
// Without timeout
select {
case err = <-channel:
}
}
// Check for errors
if err != nil {
return nil, err
2020-03-24 03:43:13 +01:00
}
2020-03-20 16:43:28 +01:00
// Read response
buffer := bufio.NewReader(&conn.w)
// Convert raw http response to *http.Response
2020-02-26 19:31:43 -05:00
resp, err := http.ReadResponse(buffer, request)
2020-02-21 18:07:43 +01:00
if err != nil {
return nil, err
}
// Return *http.Response
return resp, nil
}
2020-03-27 06:56:58 +01:00
// Sharding: https://www.nginx.com/blog/socket-sharding-nginx-release-1-9-1/
2020-03-31 10:03:39 +02:00
func (app *App) prefork(address string) (ln net.Listener, err error) {
2020-02-21 18:07:43 +01:00
// Master proc
2020-04-12 14:58:05 +02:00
if !isChild() {
2020-03-01 07:31:14 +01:00
addr, err := net.ResolveTCPAddr("tcp", address)
2020-02-21 18:07:43 +01:00
if err != nil {
return ln, err
}
2020-03-01 07:31:14 +01:00
tcplistener, err := net.ListenTCP("tcp", addr)
2020-02-21 18:07:43 +01:00
if err != nil {
return ln, err
}
fl, err := tcplistener.File()
if err != nil {
return ln, err
}
2020-02-26 19:31:43 -05:00
files := []*os.File{fl}
2020-02-21 18:07:43 +01:00
childs := make([]*exec.Cmd, runtime.NumCPU()/2)
// #nosec G204
for i := range childs {
2020-02-26 19:31:43 -05:00
childs[i] = exec.Command(os.Args[0], append(os.Args[1:], "-prefork", "-child")...)
2020-02-21 18:07:43 +01:00
childs[i].Stdout = os.Stdout
childs[i].Stderr = os.Stderr
2020-02-26 19:31:43 -05:00
childs[i].ExtraFiles = files
2020-02-21 18:07:43 +01:00
if err := childs[i].Start(); err != nil {
return ln, err
}
}
for k := range childs {
if err := childs[k].Wait(); err != nil {
2020-02-21 18:07:43 +01:00
return ln, err
}
}
os.Exit(0)
} else {
2020-03-27 06:56:58 +01:00
// 1 core per child
2020-02-26 19:31:43 -05:00
runtime.GOMAXPROCS(1)
2020-02-21 18:07:43 +01:00
ln, err = net.FileListener(os.NewFile(3, ""))
}
return ln, err
}
type disableLogger struct{}
func (dl *disableLogger) Printf(format string, args ...interface{}) {
// fmt.Println(fmt.Sprintf(format, args...))
}
2020-03-31 10:03:39 +02:00
func (app *App) newServer() *fasthttp.Server {
2020-02-21 18:07:43 +01:00
return &fasthttp.Server{
2020-03-04 12:30:29 +01:00
Handler: app.handler,
Name: app.Settings.ServerHeader,
2020-04-19 16:10:19 +02:00
Concurrency: app.Settings.Concurrency,
NoDefaultDate: app.Settings.DisableDefaultDate,
NoDefaultContentType: app.Settings.DisableDefaultContentType,
DisableKeepalive: app.Settings.DisableKeepalive,
2020-03-24 03:36:52 +01:00
MaxRequestBodySize: app.Settings.BodyLimit,
2020-03-24 03:42:28 +01:00
NoDefaultServerHeader: app.Settings.ServerHeader == "",
2020-03-23 05:25:46 +01:00
ReadTimeout: app.Settings.ReadTimeout,
WriteTimeout: app.Settings.WriteTimeout,
IdleTimeout: app.Settings.IdleTimeout,
Logger: &disableLogger{},
2020-03-14 12:30:21 +01:00
LogAllErrors: false,
2020-02-21 18:07:43 +01:00
ErrorHandler: func(ctx *fasthttp.RequestCtx, err error) {
2020-03-01 06:56:41 +01:00
if err.Error() == "body size exceeds the given limit" {
2020-04-29 10:52:51 +08:00
ctx.Response.SetStatusCode(StatusRequestEntityTooLarge)
2020-03-01 06:56:41 +01:00
ctx.Response.SetBodyString("Request Entity Too Large")
} else {
2020-04-29 10:52:51 +08:00
ctx.Response.SetStatusCode(StatusBadRequest)
2020-03-01 06:56:41 +01:00
ctx.Response.SetBodyString("Bad Request")
}
2020-02-21 18:07:43 +01:00
},
}
}