2020-05-06 15:59:40 +02:00
|
|
|
// ⚡️ Fiber is an Express inspired web framework written in Go with ☕️
|
2020-05-07 20:15:27 +02:00
|
|
|
// 🤖 Github Repository: https://github.com/gofiber/fiber
|
2020-05-06 15:59:40 +02:00
|
|
|
// 📌 API Documentation: https://docs.gofiber.io
|
2020-02-21 18:06:08 +01:00
|
|
|
|
2019-12-30 07:29:42 -05:00
|
|
|
package fiber
|
|
|
|
|
|
|
|
import (
|
2020-02-05 17:37:04 +01:00
|
|
|
"bytes"
|
2020-02-12 04:02:58 +01:00
|
|
|
"fmt"
|
2020-04-28 19:45:23 +02:00
|
|
|
"hash/crc32"
|
2020-02-05 17:37:04 +01:00
|
|
|
"net"
|
2019-12-30 07:29:42 -05:00
|
|
|
"strings"
|
2020-02-05 17:37:04 +01:00
|
|
|
"time"
|
2020-05-16 05:13:01 +02:00
|
|
|
|
2020-05-23 09:30:21 +02:00
|
|
|
utils "github.com/gofiber/utils"
|
|
|
|
)
|
2020-05-07 22:49:26 +02:00
|
|
|
|
2020-06-20 17:26:48 +02:00
|
|
|
// Scan stack if other methods match
|
|
|
|
func setMethodNotAllowed(ctx *Ctx) {
|
2020-07-10 14:15:41 +02:00
|
|
|
var matched bool
|
2020-06-21 11:02:17 +02:00
|
|
|
for i := 0; i < len(intMethod); i++ {
|
2020-06-20 17:26:48 +02:00
|
|
|
// Skip original method
|
2020-07-10 13:57:53 +02:00
|
|
|
if ctx.methodINT == i {
|
2020-06-20 17:26:48 +02:00
|
|
|
continue
|
|
|
|
}
|
|
|
|
// Reset stack index
|
|
|
|
ctx.indexRoute = -1
|
|
|
|
// Get stack length
|
2020-06-21 11:02:17 +02:00
|
|
|
lenr := len(ctx.app.stack[i]) - 1
|
2020-06-21 10:47:03 +02:00
|
|
|
//Loop over the route stack starting from previous index
|
2020-06-20 17:26:48 +02:00
|
|
|
for ctx.indexRoute < lenr {
|
2020-06-21 10:47:03 +02:00
|
|
|
// Increment route index
|
|
|
|
ctx.indexRoute++
|
2020-06-20 17:26:48 +02:00
|
|
|
// Get *Route
|
2020-06-21 11:02:17 +02:00
|
|
|
route := ctx.app.stack[i][ctx.indexRoute]
|
2020-07-07 23:25:02 +08:00
|
|
|
// Skip use routes
|
|
|
|
if route.use {
|
|
|
|
continue
|
|
|
|
}
|
2020-06-20 17:26:48 +02:00
|
|
|
// Check if it matches the request path
|
2020-07-10 14:15:41 +02:00
|
|
|
match, _ := route.match(ctx.path, ctx.pathOriginal)
|
2020-06-20 17:26:48 +02:00
|
|
|
// No match, next route
|
|
|
|
if match {
|
2020-07-10 14:15:41 +02:00
|
|
|
// We matched
|
|
|
|
matched = true
|
2020-06-21 12:17:22 +02:00
|
|
|
// Add method to Allow header
|
2020-06-21 11:02:17 +02:00
|
|
|
ctx.Append(HeaderAllow, intMethod[i])
|
2020-06-21 11:03:17 +02:00
|
|
|
// Break stack loop
|
2020-06-20 17:26:48 +02:00
|
|
|
break
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2020-06-21 12:17:22 +02:00
|
|
|
// Update response status
|
2020-07-10 14:15:41 +02:00
|
|
|
if matched {
|
2020-06-21 12:17:22 +02:00
|
|
|
ctx.Status(StatusMethodNotAllowed)
|
|
|
|
}
|
2020-06-20 17:26:48 +02:00
|
|
|
}
|
|
|
|
|
2020-07-04 10:11:23 +02:00
|
|
|
// defaultString returns the value or a default value if it is set
|
|
|
|
func defaultString(value string, defaultValue []string) string {
|
|
|
|
if len(value) == 0 && len(defaultValue) > 0 {
|
|
|
|
return defaultValue[0]
|
|
|
|
}
|
|
|
|
return value
|
|
|
|
}
|
|
|
|
|
2020-05-07 17:57:21 +02:00
|
|
|
// Generate and set ETag header to response
|
2020-05-06 15:59:40 +02:00
|
|
|
func setETag(ctx *Ctx, weak bool) {
|
2020-06-01 13:45:04 +02:00
|
|
|
// Don't generate ETags for invalid responses
|
|
|
|
if ctx.Fasthttp.Response.StatusCode() != 200 {
|
|
|
|
return
|
|
|
|
}
|
2020-05-06 15:59:40 +02:00
|
|
|
body := ctx.Fasthttp.Response.Body()
|
2020-04-28 19:45:23 +02:00
|
|
|
// Skips ETag if no response body is present
|
|
|
|
if len(body) <= 0 {
|
|
|
|
return
|
|
|
|
}
|
|
|
|
// Get ETag header from request
|
2020-05-16 05:13:01 +02:00
|
|
|
clientEtag := ctx.Get(HeaderIfNoneMatch)
|
2020-04-28 19:45:23 +02:00
|
|
|
|
|
|
|
// Generate ETag for response
|
|
|
|
crc32q := crc32.MakeTable(0xD5828281)
|
|
|
|
etag := fmt.Sprintf("\"%d-%v\"", len(body), crc32.Checksum(body, crc32q))
|
|
|
|
|
|
|
|
// Enable weak tag
|
|
|
|
if weak {
|
2020-05-23 09:30:21 +02:00
|
|
|
etag = "W/" + etag
|
2020-04-28 19:45:23 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
// Check if client's ETag is weak
|
|
|
|
if strings.HasPrefix(clientEtag, "W/") {
|
|
|
|
// Check if server's ETag is weak
|
|
|
|
if clientEtag[2:] == etag || clientEtag[2:] == etag[2:] {
|
|
|
|
// W/1 == 1 || W/1 == W/1
|
|
|
|
ctx.SendStatus(304)
|
|
|
|
ctx.Fasthttp.ResetBody()
|
|
|
|
return
|
|
|
|
}
|
|
|
|
// W/1 != W/2 || W/1 != 2
|
2020-05-07 17:57:21 +02:00
|
|
|
ctx.Set(HeaderETag, etag)
|
2020-04-28 19:45:23 +02:00
|
|
|
return
|
|
|
|
}
|
|
|
|
if strings.Contains(clientEtag, etag) {
|
|
|
|
// 1 == 1
|
|
|
|
ctx.SendStatus(304)
|
|
|
|
ctx.Fasthttp.ResetBody()
|
|
|
|
return
|
|
|
|
}
|
|
|
|
// 1 != 2
|
2020-05-07 17:57:21 +02:00
|
|
|
ctx.Set(HeaderETag, etag)
|
2020-04-28 19:45:23 +02:00
|
|
|
}
|
|
|
|
|
2020-05-07 17:57:21 +02:00
|
|
|
func getGroupPath(prefix, path string) string {
|
2020-03-04 12:30:29 +01:00
|
|
|
if path == "/" {
|
2020-05-23 09:30:21 +02:00
|
|
|
return prefix
|
2020-05-16 05:13:01 +02:00
|
|
|
}
|
2020-05-23 09:30:21 +02:00
|
|
|
return utils.TrimRight(prefix, '/') + path
|
2020-04-12 14:58:05 +02:00
|
|
|
}
|
|
|
|
|
2020-05-12 19:24:04 +02:00
|
|
|
// return valid offer for header negotiation
|
|
|
|
func getOffer(header string, offers ...string) string {
|
|
|
|
if len(offers) == 0 {
|
|
|
|
return ""
|
2020-05-23 09:30:21 +02:00
|
|
|
} else if header == "" {
|
2020-05-12 19:24:04 +02:00
|
|
|
return offers[0]
|
|
|
|
}
|
|
|
|
|
2020-05-23 09:30:21 +02:00
|
|
|
spec, commaPos := "", 0
|
|
|
|
for len(header) > 0 && commaPos != -1 {
|
|
|
|
commaPos = strings.IndexByte(header, ',')
|
|
|
|
if commaPos != -1 {
|
|
|
|
spec = utils.Trim(header[:commaPos], ' ')
|
|
|
|
} else {
|
|
|
|
spec = header
|
|
|
|
}
|
|
|
|
if factorSign := strings.IndexByte(spec, ';'); factorSign != -1 {
|
|
|
|
spec = spec[:factorSign]
|
|
|
|
}
|
|
|
|
|
|
|
|
for _, offer := range offers {
|
|
|
|
// has star prefix
|
|
|
|
if len(spec) >= 1 && spec[len(spec)-1] == '*' {
|
|
|
|
return offer
|
|
|
|
} else if strings.HasPrefix(spec, offer) {
|
|
|
|
return offer
|
2020-05-12 19:24:04 +02:00
|
|
|
}
|
|
|
|
}
|
2020-05-23 09:30:21 +02:00
|
|
|
if commaPos != -1 {
|
|
|
|
header = header[commaPos+1:]
|
|
|
|
}
|
2020-05-12 19:24:04 +02:00
|
|
|
}
|
2020-05-23 09:30:21 +02:00
|
|
|
|
2020-05-12 19:24:04 +02:00
|
|
|
return ""
|
|
|
|
}
|
|
|
|
|
2020-04-29 01:44:22 +08:00
|
|
|
// Adapted from:
|
|
|
|
// https://github.com/jshttp/fresh/blob/10e0471669dbbfbfd8de65bc6efac2ddd0bfa057/index.js#L110
|
|
|
|
func parseTokenList(noneMatchBytes []byte) []string {
|
|
|
|
var (
|
|
|
|
start int
|
|
|
|
end int
|
|
|
|
list []string
|
|
|
|
)
|
2020-04-29 00:15:52 +02:00
|
|
|
for i := range noneMatchBytes {
|
|
|
|
switch noneMatchBytes[i] {
|
2020-04-29 01:44:22 +08:00
|
|
|
case 0x20:
|
|
|
|
if start == end {
|
|
|
|
start = i + 1
|
|
|
|
end = i + 1
|
|
|
|
}
|
|
|
|
case 0x2c:
|
|
|
|
list = append(list, getString(noneMatchBytes[start:end]))
|
|
|
|
start = i + 1
|
|
|
|
end = i + 1
|
|
|
|
default:
|
|
|
|
end = i + 1
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
list = append(list, getString(noneMatchBytes[start:end]))
|
|
|
|
return list
|
|
|
|
}
|
|
|
|
|
2020-05-11 04:30:31 +02:00
|
|
|
// https://golang.org/src/net/net.go#L113
|
|
|
|
// Helper methods for application#test
|
2020-07-05 11:09:01 +02:00
|
|
|
type testAddr string
|
|
|
|
|
|
|
|
func (a testAddr) Network() string {
|
|
|
|
return string(a)
|
|
|
|
}
|
|
|
|
func (a testAddr) String() string {
|
|
|
|
return string(a)
|
|
|
|
}
|
|
|
|
|
2020-05-11 04:30:31 +02:00
|
|
|
type testConn struct {
|
|
|
|
net.Conn
|
|
|
|
r bytes.Buffer
|
|
|
|
w bytes.Buffer
|
|
|
|
}
|
|
|
|
|
2020-07-05 11:09:01 +02:00
|
|
|
func (c *testConn) Read(b []byte) (int, error) { return c.r.Read(b) }
|
|
|
|
func (c *testConn) Write(b []byte) (int, error) { return c.w.Write(b) }
|
|
|
|
func (c *testConn) Close() error { return nil }
|
|
|
|
|
|
|
|
func (c *testConn) LocalAddr() net.Addr { return testAddr("local-addr") }
|
|
|
|
func (c *testConn) RemoteAddr() net.Addr { return testAddr("remote-addr") }
|
2020-05-11 04:30:31 +02:00
|
|
|
func (c *testConn) SetDeadline(t time.Time) error { return nil }
|
|
|
|
func (c *testConn) SetReadDeadline(t time.Time) error { return nil }
|
|
|
|
func (c *testConn) SetWriteDeadline(t time.Time) error { return nil }
|
|
|
|
|
|
|
|
// getString converts byte slice to a string without memory allocation.
|
2020-05-23 09:30:21 +02:00
|
|
|
var getString = utils.GetString
|
2020-05-11 04:30:31 +02:00
|
|
|
var getStringImmutable = func(b []byte) string {
|
|
|
|
return string(b)
|
|
|
|
}
|
|
|
|
|
|
|
|
// getBytes converts string to a byte slice without memory allocation.
|
2020-05-23 09:30:21 +02:00
|
|
|
var getBytes = utils.GetBytes
|
2020-05-11 04:30:31 +02:00
|
|
|
var getBytesImmutable = func(s string) (b []byte) {
|
|
|
|
return []byte(s)
|
|
|
|
}
|
|
|
|
|
2020-05-07 18:15:59 +02:00
|
|
|
// HTTP methods and their unique INTs
|
2020-06-29 22:51:41 +02:00
|
|
|
func methodInt(s string) int {
|
|
|
|
switch s {
|
|
|
|
case MethodGet:
|
|
|
|
return 0
|
|
|
|
case MethodHead:
|
|
|
|
return 1
|
|
|
|
case MethodPost:
|
|
|
|
return 2
|
|
|
|
case MethodPut:
|
|
|
|
return 3
|
|
|
|
case MethodDelete:
|
|
|
|
return 4
|
|
|
|
case MethodConnect:
|
|
|
|
return 5
|
|
|
|
case MethodOptions:
|
|
|
|
return 6
|
|
|
|
case MethodTrace:
|
|
|
|
return 7
|
|
|
|
case MethodPatch:
|
|
|
|
return 8
|
|
|
|
default:
|
|
|
|
return 0
|
|
|
|
}
|
2020-05-07 18:15:59 +02:00
|
|
|
}
|
|
|
|
|
2020-06-21 11:11:45 +02:00
|
|
|
// HTTP methods slice
|
2020-06-21 11:02:17 +02:00
|
|
|
var intMethod = []string{
|
|
|
|
MethodGet,
|
|
|
|
MethodHead,
|
|
|
|
MethodPost,
|
|
|
|
MethodPut,
|
|
|
|
MethodDelete,
|
|
|
|
MethodConnect,
|
|
|
|
MethodOptions,
|
|
|
|
MethodTrace,
|
|
|
|
MethodPatch,
|
|
|
|
}
|
|
|
|
|
2020-05-16 05:13:01 +02:00
|
|
|
// HTTP methods were copied from net/http.
|
|
|
|
const (
|
|
|
|
MethodGet = "GET" // RFC 7231, 4.3.1
|
|
|
|
MethodHead = "HEAD" // RFC 7231, 4.3.2
|
|
|
|
MethodPost = "POST" // RFC 7231, 4.3.3
|
|
|
|
MethodPut = "PUT" // RFC 7231, 4.3.4
|
|
|
|
MethodPatch = "PATCH" // RFC 5789
|
|
|
|
MethodDelete = "DELETE" // RFC 7231, 4.3.5
|
|
|
|
MethodConnect = "CONNECT" // RFC 7231, 4.3.6
|
|
|
|
MethodOptions = "OPTIONS" // RFC 7231, 4.3.7
|
|
|
|
MethodTrace = "TRACE" // RFC 7231, 4.3.8
|
|
|
|
)
|
2020-03-22 01:51:53 +01:00
|
|
|
|
2020-06-06 07:30:30 +02:00
|
|
|
// MIME types that are commonly used
|
2020-03-22 01:51:53 +01:00
|
|
|
const (
|
2020-05-07 17:57:21 +02:00
|
|
|
MIMETextXML = "text/xml"
|
|
|
|
MIMETextHTML = "text/html"
|
|
|
|
MIMETextPlain = "text/plain"
|
2020-06-06 07:30:30 +02:00
|
|
|
MIMEApplicationXML = "application/xml"
|
2020-03-22 01:51:53 +01:00
|
|
|
MIMEApplicationJSON = "application/json"
|
|
|
|
MIMEApplicationJavaScript = "application/javascript"
|
|
|
|
MIMEApplicationForm = "application/x-www-form-urlencoded"
|
2020-05-07 17:57:21 +02:00
|
|
|
MIMEOctetStream = "application/octet-stream"
|
2020-06-06 07:30:30 +02:00
|
|
|
MIMEMultipartForm = "multipart/form-data"
|
|
|
|
|
|
|
|
MIMETextXMLCharsetUTF8 = "text/xml; charset=utf-8"
|
|
|
|
MIMETextHTMLCharsetUTF8 = "text/html; charset=utf-8"
|
|
|
|
MIMETextPlainCharsetUTF8 = "text/plain; charset=utf-8"
|
|
|
|
MIMEApplicationXMLCharsetUTF8 = "application/xml; charset=utf-8"
|
|
|
|
MIMEApplicationJavaScriptCharsetUTF8 = "application/javascript; charset=utf-8"
|
2020-03-22 01:51:53 +01:00
|
|
|
)
|
|
|
|
|
2020-05-16 05:13:01 +02:00
|
|
|
// HTTP status codes were copied from net/http.
|
2020-03-22 01:51:53 +01:00
|
|
|
const (
|
2020-05-16 05:13:01 +02:00
|
|
|
StatusContinue = 100 // RFC 7231, 6.2.1
|
|
|
|
StatusSwitchingProtocols = 101 // RFC 7231, 6.2.2
|
|
|
|
StatusProcessing = 102 // RFC 2518, 10.1
|
|
|
|
StatusEarlyHints = 103 // RFC 8297
|
|
|
|
StatusOK = 200 // RFC 7231, 6.3.1
|
|
|
|
StatusCreated = 201 // RFC 7231, 6.3.2
|
|
|
|
StatusAccepted = 202 // RFC 7231, 6.3.3
|
|
|
|
StatusNonAuthoritativeInfo = 203 // RFC 7231, 6.3.4
|
|
|
|
StatusNoContent = 204 // RFC 7231, 6.3.5
|
|
|
|
StatusResetContent = 205 // RFC 7231, 6.3.6
|
|
|
|
StatusPartialContent = 206 // RFC 7233, 4.1
|
|
|
|
StatusMultiStatus = 207 // RFC 4918, 11.1
|
|
|
|
StatusAlreadyReported = 208 // RFC 5842, 7.1
|
|
|
|
StatusIMUsed = 226 // RFC 3229, 10.4.1
|
|
|
|
StatusMultipleChoices = 300 // RFC 7231, 6.4.1
|
|
|
|
StatusMovedPermanently = 301 // RFC 7231, 6.4.2
|
|
|
|
StatusFound = 302 // RFC 7231, 6.4.3
|
|
|
|
StatusSeeOther = 303 // RFC 7231, 6.4.4
|
|
|
|
StatusNotModified = 304 // RFC 7232, 4.1
|
|
|
|
StatusUseProxy = 305 // RFC 7231, 6.4.5
|
|
|
|
StatusTemporaryRedirect = 307 // RFC 7231, 6.4.7
|
|
|
|
StatusPermanentRedirect = 308 // RFC 7538, 3
|
|
|
|
StatusBadRequest = 400 // RFC 7231, 6.5.1
|
|
|
|
StatusUnauthorized = 401 // RFC 7235, 3.1
|
|
|
|
StatusPaymentRequired = 402 // RFC 7231, 6.5.2
|
|
|
|
StatusForbidden = 403 // RFC 7231, 6.5.3
|
|
|
|
StatusNotFound = 404 // RFC 7231, 6.5.4
|
|
|
|
StatusMethodNotAllowed = 405 // RFC 7231, 6.5.5
|
|
|
|
StatusNotAcceptable = 406 // RFC 7231, 6.5.6
|
|
|
|
StatusProxyAuthRequired = 407 // RFC 7235, 3.2
|
|
|
|
StatusRequestTimeout = 408 // RFC 7231, 6.5.7
|
|
|
|
StatusConflict = 409 // RFC 7231, 6.5.8
|
|
|
|
StatusGone = 410 // RFC 7231, 6.5.9
|
|
|
|
StatusLengthRequired = 411 // RFC 7231, 6.5.10
|
|
|
|
StatusPreconditionFailed = 412 // RFC 7232, 4.2
|
|
|
|
StatusRequestEntityTooLarge = 413 // RFC 7231, 6.5.11
|
|
|
|
StatusRequestURITooLong = 414 // RFC 7231, 6.5.12
|
|
|
|
StatusUnsupportedMediaType = 415 // RFC 7231, 6.5.13
|
|
|
|
StatusRequestedRangeNotSatisfiable = 416 // RFC 7233, 4.4
|
|
|
|
StatusExpectationFailed = 417 // RFC 7231, 6.5.14
|
|
|
|
StatusTeapot = 418 // RFC 7168, 2.3.3
|
|
|
|
StatusMisdirectedRequest = 421 // RFC 7540, 9.1.2
|
|
|
|
StatusUnprocessableEntity = 422 // RFC 4918, 11.2
|
|
|
|
StatusLocked = 423 // RFC 4918, 11.3
|
|
|
|
StatusFailedDependency = 424 // RFC 4918, 11.4
|
|
|
|
StatusTooEarly = 425 // RFC 8470, 5.2.
|
|
|
|
StatusUpgradeRequired = 426 // RFC 7231, 6.5.15
|
|
|
|
StatusPreconditionRequired = 428 // RFC 6585, 3
|
|
|
|
StatusTooManyRequests = 429 // RFC 6585, 4
|
|
|
|
StatusRequestHeaderFieldsTooLarge = 431 // RFC 6585, 5
|
|
|
|
StatusUnavailableForLegalReasons = 451 // RFC 7725, 3
|
|
|
|
StatusInternalServerError = 500 // RFC 7231, 6.6.1
|
|
|
|
StatusNotImplemented = 501 // RFC 7231, 6.6.2
|
|
|
|
StatusBadGateway = 502 // RFC 7231, 6.6.3
|
|
|
|
StatusServiceUnavailable = 503 // RFC 7231, 6.6.4
|
|
|
|
StatusGatewayTimeout = 504 // RFC 7231, 6.6.5
|
|
|
|
StatusHTTPVersionNotSupported = 505 // RFC 7231, 6.6.6
|
|
|
|
StatusVariantAlsoNegotiates = 506 // RFC 2295, 8.1
|
|
|
|
StatusInsufficientStorage = 507 // RFC 4918, 11.5
|
|
|
|
StatusLoopDetected = 508 // RFC 5842, 7.2
|
|
|
|
StatusNotExtended = 510 // RFC 2774, 7
|
|
|
|
StatusNetworkAuthenticationRequired = 511 // RFC 6585, 6
|
2020-03-22 01:51:53 +01:00
|
|
|
)
|
|
|
|
|
2020-06-06 18:53:51 +02:00
|
|
|
// Errors
|
|
|
|
var (
|
|
|
|
ErrContinue = NewError(StatusContinue) // RFC 7231, 6.2.1
|
|
|
|
ErrSwitchingProtocols = NewError(StatusSwitchingProtocols) // RFC 7231, 6.2.2
|
|
|
|
ErrProcessing = NewError(StatusProcessing) // RFC 2518, 10.1
|
|
|
|
ErrEarlyHints = NewError(StatusEarlyHints) // RFC 8297
|
|
|
|
ErrOK = NewError(StatusOK) // RFC 7231, 6.3.1
|
|
|
|
ErrCreated = NewError(StatusCreated) // RFC 7231, 6.3.2
|
|
|
|
ErrAccepted = NewError(StatusAccepted) // RFC 7231, 6.3.3
|
|
|
|
ErrNonAuthoritativeInfo = NewError(StatusNonAuthoritativeInfo) // RFC 7231, 6.3.4
|
|
|
|
ErrNoContent = NewError(StatusNoContent) // RFC 7231, 6.3.5
|
|
|
|
ErrResetContent = NewError(StatusResetContent) // RFC 7231, 6.3.6
|
|
|
|
ErrPartialContent = NewError(StatusPartialContent) // RFC 7233, 4.1
|
|
|
|
ErrMultiStatus = NewError(StatusMultiStatus) // RFC 4918, 11.1
|
|
|
|
ErrAlreadyReported = NewError(StatusAlreadyReported) // RFC 5842, 7.1
|
|
|
|
ErrIMUsed = NewError(StatusIMUsed) // RFC 3229, 10.4.1
|
|
|
|
ErrMultipleChoices = NewError(StatusMultipleChoices) // RFC 7231, 6.4.1
|
|
|
|
ErrMovedPermanently = NewError(StatusMovedPermanently) // RFC 7231, 6.4.2
|
|
|
|
ErrFound = NewError(StatusFound) // RFC 7231, 6.4.3
|
|
|
|
ErrSeeOther = NewError(StatusSeeOther) // RFC 7231, 6.4.4
|
|
|
|
ErrNotModified = NewError(StatusNotModified) // RFC 7232, 4.1
|
|
|
|
ErrUseProxy = NewError(StatusUseProxy) // RFC 7231, 6.4.5
|
|
|
|
ErrTemporaryRedirect = NewError(StatusTemporaryRedirect) // RFC 7231, 6.4.7
|
|
|
|
ErrPermanentRedirect = NewError(StatusPermanentRedirect) // RFC 7538, 3
|
|
|
|
ErrBadRequest = NewError(StatusBadRequest) // RFC 7231, 6.5.1
|
|
|
|
ErrUnauthorized = NewError(StatusUnauthorized) // RFC 7235, 3.1
|
|
|
|
ErrPaymentRequired = NewError(StatusPaymentRequired) // RFC 7231, 6.5.2
|
|
|
|
ErrForbidden = NewError(StatusForbidden) // RFC 7231, 6.5.3
|
|
|
|
ErrNotFound = NewError(StatusNotFound) // RFC 7231, 6.5.4
|
|
|
|
ErrMethodNotAllowed = NewError(StatusMethodNotAllowed) // RFC 7231, 6.5.5
|
|
|
|
ErrNotAcceptable = NewError(StatusNotAcceptable) // RFC 7231, 6.5.6
|
|
|
|
ErrProxyAuthRequired = NewError(StatusProxyAuthRequired) // RFC 7235, 3.2
|
|
|
|
ErrRequestTimeout = NewError(StatusRequestTimeout) // RFC 7231, 6.5.7
|
|
|
|
ErrConflict = NewError(StatusConflict) // RFC 7231, 6.5.8
|
|
|
|
ErrGone = NewError(StatusGone) // RFC 7231, 6.5.9
|
|
|
|
ErrLengthRequired = NewError(StatusLengthRequired) // RFC 7231, 6.5.10
|
|
|
|
ErrPreconditionFailed = NewError(StatusPreconditionFailed) // RFC 7232, 4.2
|
|
|
|
ErrRequestEntityTooLarge = NewError(StatusRequestEntityTooLarge) // RFC 7231, 6.5.11
|
|
|
|
ErrRequestURITooLong = NewError(StatusRequestURITooLong) // RFC 7231, 6.5.12
|
|
|
|
ErrUnsupportedMediaType = NewError(StatusUnsupportedMediaType) // RFC 7231, 6.5.13
|
|
|
|
ErrRequestedRangeNotSatisfiable = NewError(StatusRequestedRangeNotSatisfiable) // RFC 7233, 4.4
|
|
|
|
ErrExpectationFailed = NewError(StatusExpectationFailed) // RFC 7231, 6.5.14
|
|
|
|
ErrTeapot = NewError(StatusTeapot) // RFC 7168, 2.3.3
|
|
|
|
ErrMisdirectedRequest = NewError(StatusMisdirectedRequest) // RFC 7540, 9.1.2
|
|
|
|
ErrUnprocessableEntity = NewError(StatusUnprocessableEntity) // RFC 4918, 11.2
|
|
|
|
ErrLocked = NewError(StatusLocked) // RFC 4918, 11.3
|
|
|
|
ErrFailedDependency = NewError(StatusFailedDependency) // RFC 4918, 11.4
|
|
|
|
ErrTooEarly = NewError(StatusTooEarly) // RFC 8470, 5.2.
|
|
|
|
ErrUpgradeRequired = NewError(StatusUpgradeRequired) // RFC 7231, 6.5.15
|
|
|
|
ErrPreconditionRequired = NewError(StatusPreconditionRequired) // RFC 6585, 3
|
|
|
|
ErrTooManyRequests = NewError(StatusTooManyRequests) // RFC 6585, 4
|
|
|
|
ErrRequestHeaderFieldsTooLarge = NewError(StatusRequestHeaderFieldsTooLarge) // RFC 6585, 5
|
|
|
|
ErrUnavailableForLegalReasons = NewError(StatusUnavailableForLegalReasons) // RFC 7725, 3
|
|
|
|
ErrInternalServerError = NewError(StatusInternalServerError) // RFC 7231, 6.6.1
|
|
|
|
ErrNotImplemented = NewError(StatusNotImplemented) // RFC 7231, 6.6.2
|
|
|
|
ErrBadGateway = NewError(StatusBadGateway) // RFC 7231, 6.6.3
|
|
|
|
ErrServiceUnavailable = NewError(StatusServiceUnavailable) // RFC 7231, 6.6.4
|
|
|
|
ErrGatewayTimeout = NewError(StatusGatewayTimeout) // RFC 7231, 6.6.5
|
|
|
|
ErrHTTPVersionNotSupported = NewError(StatusHTTPVersionNotSupported) // RFC 7231, 6.6.6
|
|
|
|
ErrVariantAlsoNegotiates = NewError(StatusVariantAlsoNegotiates) // RFC 2295, 8.1
|
|
|
|
ErrInsufficientStorage = NewError(StatusInsufficientStorage) // RFC 4918, 11.5
|
|
|
|
ErrLoopDetected = NewError(StatusLoopDetected) // RFC 5842, 7.2
|
|
|
|
ErrNotExtended = NewError(StatusNotExtended) // RFC 2774, 7
|
|
|
|
ErrNetworkAuthenticationRequired = NewError(StatusNetworkAuthenticationRequired) // RFC 6585, 6
|
|
|
|
)
|
|
|
|
|
2020-03-22 01:51:53 +01:00
|
|
|
// HTTP Headers were copied from net/http.
|
|
|
|
const (
|
2020-05-07 17:57:21 +02:00
|
|
|
HeaderAuthorization = "Authorization"
|
|
|
|
HeaderProxyAuthenticate = "Proxy-Authenticate"
|
|
|
|
HeaderProxyAuthorization = "Proxy-Authorization"
|
|
|
|
HeaderWWWAuthenticate = "WWW-Authenticate"
|
|
|
|
HeaderAge = "Age"
|
|
|
|
HeaderCacheControl = "Cache-Control"
|
|
|
|
HeaderClearSiteData = "Clear-Site-Data"
|
|
|
|
HeaderExpires = "Expires"
|
|
|
|
HeaderPragma = "Pragma"
|
|
|
|
HeaderWarning = "Warning"
|
|
|
|
HeaderAcceptCH = "Accept-CH"
|
|
|
|
HeaderAcceptCHLifetime = "Accept-CH-Lifetime"
|
|
|
|
HeaderContentDPR = "Content-DPR"
|
|
|
|
HeaderDPR = "DPR"
|
|
|
|
HeaderEarlyData = "Early-Data"
|
|
|
|
HeaderSaveData = "Save-Data"
|
|
|
|
HeaderViewportWidth = "Viewport-Width"
|
|
|
|
HeaderWidth = "Width"
|
|
|
|
HeaderETag = "ETag"
|
|
|
|
HeaderIfMatch = "If-Match"
|
|
|
|
HeaderIfModifiedSince = "If-Modified-Since"
|
|
|
|
HeaderIfNoneMatch = "If-None-Match"
|
|
|
|
HeaderIfUnmodifiedSince = "If-Unmodified-Since"
|
|
|
|
HeaderLastModified = "Last-Modified"
|
|
|
|
HeaderVary = "Vary"
|
|
|
|
HeaderConnection = "Connection"
|
|
|
|
HeaderKeepAlive = "Keep-Alive"
|
|
|
|
HeaderAccept = "Accept"
|
|
|
|
HeaderAcceptCharset = "Accept-Charset"
|
|
|
|
HeaderAcceptEncoding = "Accept-Encoding"
|
|
|
|
HeaderAcceptLanguage = "Accept-Language"
|
|
|
|
HeaderCookie = "Cookie"
|
|
|
|
HeaderExpect = "Expect"
|
|
|
|
HeaderMaxForwards = "Max-Forwards"
|
|
|
|
HeaderSetCookie = "Set-Cookie"
|
|
|
|
HeaderAccessControlAllowCredentials = "Access-Control-Allow-Credentials"
|
|
|
|
HeaderAccessControlAllowHeaders = "Access-Control-Allow-Headers"
|
|
|
|
HeaderAccessControlAllowMethods = "Access-Control-Allow-Methods"
|
|
|
|
HeaderAccessControlAllowOrigin = "Access-Control-Allow-Origin"
|
|
|
|
HeaderAccessControlExposeHeaders = "Access-Control-Expose-Headers"
|
|
|
|
HeaderAccessControlMaxAge = "Access-Control-Max-Age"
|
|
|
|
HeaderAccessControlRequestHeaders = "Access-Control-Request-Headers"
|
|
|
|
HeaderAccessControlRequestMethod = "Access-Control-Request-Method"
|
|
|
|
HeaderOrigin = "Origin"
|
|
|
|
HeaderTimingAllowOrigin = "Timing-Allow-Origin"
|
|
|
|
HeaderXPermittedCrossDomainPolicies = "X-Permitted-Cross-Domain-Policies"
|
|
|
|
HeaderDNT = "DNT"
|
|
|
|
HeaderTk = "Tk"
|
|
|
|
HeaderContentDisposition = "Content-Disposition"
|
|
|
|
HeaderContentEncoding = "Content-Encoding"
|
|
|
|
HeaderContentLanguage = "Content-Language"
|
|
|
|
HeaderContentLength = "Content-Length"
|
|
|
|
HeaderContentLocation = "Content-Location"
|
|
|
|
HeaderContentType = "Content-Type"
|
|
|
|
HeaderForwarded = "Forwarded"
|
|
|
|
HeaderVia = "Via"
|
|
|
|
HeaderXForwardedFor = "X-Forwarded-For"
|
|
|
|
HeaderXForwardedHost = "X-Forwarded-Host"
|
|
|
|
HeaderXForwardedProto = "X-Forwarded-Proto"
|
2020-05-23 09:30:21 +02:00
|
|
|
HeaderXForwardedProtocol = "X-Forwarded-Protocol"
|
|
|
|
HeaderXForwardedSsl = "X-Forwarded-Ssl"
|
|
|
|
HeaderXUrlScheme = "X-Url-Scheme"
|
2020-05-07 17:57:21 +02:00
|
|
|
HeaderLocation = "Location"
|
|
|
|
HeaderFrom = "From"
|
|
|
|
HeaderHost = "Host"
|
|
|
|
HeaderReferer = "Referer"
|
|
|
|
HeaderReferrerPolicy = "Referrer-Policy"
|
|
|
|
HeaderUserAgent = "User-Agent"
|
|
|
|
HeaderAllow = "Allow"
|
|
|
|
HeaderServer = "Server"
|
|
|
|
HeaderAcceptRanges = "Accept-Ranges"
|
|
|
|
HeaderContentRange = "Content-Range"
|
|
|
|
HeaderIfRange = "If-Range"
|
|
|
|
HeaderRange = "Range"
|
2020-03-22 01:51:53 +01:00
|
|
|
HeaderContentSecurityPolicy = "Content-Security-Policy"
|
|
|
|
HeaderContentSecurityPolicyReportOnly = "Content-Security-Policy-Report-Only"
|
|
|
|
HeaderCrossOriginResourcePolicy = "Cross-Origin-Resource-Policy"
|
|
|
|
HeaderExpectCT = "Expect-CT"
|
|
|
|
HeaderFeaturePolicy = "Feature-Policy"
|
|
|
|
HeaderPublicKeyPins = "Public-Key-Pins"
|
|
|
|
HeaderPublicKeyPinsReportOnly = "Public-Key-Pins-Report-Only"
|
|
|
|
HeaderStrictTransportSecurity = "Strict-Transport-Security"
|
|
|
|
HeaderUpgradeInsecureRequests = "Upgrade-Insecure-Requests"
|
|
|
|
HeaderXContentTypeOptions = "X-Content-Type-Options"
|
|
|
|
HeaderXDownloadOptions = "X-Download-Options"
|
|
|
|
HeaderXFrameOptions = "X-Frame-Options"
|
|
|
|
HeaderXPoweredBy = "X-Powered-By"
|
|
|
|
HeaderXXSSProtection = "X-XSS-Protection"
|
2020-05-07 17:57:21 +02:00
|
|
|
HeaderLastEventID = "Last-Event-ID"
|
|
|
|
HeaderNEL = "NEL"
|
|
|
|
HeaderPingFrom = "Ping-From"
|
|
|
|
HeaderPingTo = "Ping-To"
|
|
|
|
HeaderReportTo = "Report-To"
|
|
|
|
HeaderTE = "TE"
|
|
|
|
HeaderTrailer = "Trailer"
|
|
|
|
HeaderTransferEncoding = "Transfer-Encoding"
|
|
|
|
HeaderSecWebSocketAccept = "Sec-WebSocket-Accept"
|
|
|
|
HeaderSecWebSocketExtensions = "Sec-WebSocket-Extensions"
|
|
|
|
HeaderSecWebSocketKey = "Sec-WebSocket-Key"
|
|
|
|
HeaderSecWebSocketProtocol = "Sec-WebSocket-Protocol"
|
|
|
|
HeaderSecWebSocketVersion = "Sec-WebSocket-Version"
|
|
|
|
HeaderAcceptPatch = "Accept-Patch"
|
|
|
|
HeaderAcceptPushPolicy = "Accept-Push-Policy"
|
|
|
|
HeaderAcceptSignature = "Accept-Signature"
|
|
|
|
HeaderAltSvc = "Alt-Svc"
|
|
|
|
HeaderDate = "Date"
|
|
|
|
HeaderIndex = "Index"
|
|
|
|
HeaderLargeAllocation = "Large-Allocation"
|
|
|
|
HeaderLink = "Link"
|
|
|
|
HeaderPushPolicy = "Push-Policy"
|
|
|
|
HeaderRetryAfter = "Retry-After"
|
|
|
|
HeaderServerTiming = "Server-Timing"
|
|
|
|
HeaderSignature = "Signature"
|
|
|
|
HeaderSignedHeaders = "Signed-Headers"
|
|
|
|
HeaderSourceMap = "SourceMap"
|
|
|
|
HeaderUpgrade = "Upgrade"
|
|
|
|
HeaderXDNSPrefetchControl = "X-DNS-Prefetch-Control"
|
|
|
|
HeaderXPingback = "X-Pingback"
|
|
|
|
HeaderXRequestID = "X-Request-ID"
|
|
|
|
HeaderXRequestedWith = "X-Requested-With"
|
|
|
|
HeaderXRobotsTag = "X-Robots-Tag"
|
|
|
|
HeaderXUACompatible = "X-UA-Compatible"
|
2020-03-22 01:51:53 +01:00
|
|
|
)
|