package http
import "github.com/shuLhan/share/lib/http"
Package http implement custom HTTP server with memory file system and simplified routing handler.
Features
The following enhancements are added to Server and Client,
- Simplify registering routing with key binding in Server
- Add support for handling CORS in Server
- Serving files using memfs.MemFS in Server
- Simplify sending body with "application/x-www-form-urlencoded", "multipart/form-data", "application/json" with POST or PUT methods in Client.
- Add support for HTTP Range in Server and Client
- Add support for Server-Sent Events (SSE) in Server. For client see the sub package [sseclient].
Problems
There are two problems that this library try to handle. First, optimizing serving local file system; second, complexity of routing regarding to their method, request type, and response type.
Assuming that we want to serve file system and API using http.ServeMux, the simplest registered handler are,
mux.HandleFunc("/", handleFileSystem) mux.HandleFunc("/api", handleAPI)
The first problem is regarding to http.ServeFile. Everytime the request hit "handleFileSystem" the http.ServeFile try to locate the file regarding to request path in system, read the content of file, parse its content type, and finally write the content-type, content-length, and body as response. This is time consuming. Of course, on modern OS, they may caching readed file descriptor in memory to minimize disk lookup, so the next call to the same file path may not touch the hard storage back again.
The second problem is regarding to handling API. We must check the request method, checking content-type, parsing query parameter or POST form in every sub-handler of API. Assume that we have an API with method POST and query parameter, the method to handle it would be like these,
handleAPILogin(res, req) { // (1) Check if method is POST // (2) Parse query parameter // (3) Process request // (4) Write response }
The step number 1, 2, 4 needs to be written for every handler of our API.
Solutions
The solution to the first problem is by mapping all content of files to be served into memory. This cause more memory to be consumed on server side but we minimize path lookup, and cache-miss on OS level.
Serving file system is handled by package memfs, which can be set on ServerOptions. For example, to serve all content in directory "www", we can set the ServerOptions to,
opts := &http.ServerOptions{ Memfs: &memfs.MemFS{ Opts: &memfs.Options{ Root: `./www`, TryDirect: true, }, }, Address: ":8080", } httpServer, err := NewServer(opts)
There is a limit on size of file to be mapped on memory. See the package "lib/memfs" for more information.
The solution to the second problem is by mapping the registered request per method and by path. User just need to focus on step 3, handling on how to process request, all of process on step 1, 2, and 4 will be handled by our library.
import ( libhttp "github.com/shuLhan/share/lib/http" ) ... epAPILogin := &libhttp.Endpoint{ Method: libhttp.RequestMethodPost, Path: "/api/login", RequestType: libhttp.RequestTypeQuery, ResponseType: libhttp.ResponseTypeJSON, Call: handleLogin, } server.RegisterEndpoint(epAPILogin) ...
Upon receiving request to "POST /api/login", the library will call http.HttpRequest.ParseForm, read the content of body and pass them to "handleLogin",
func handleLogin(epr *EndpointRequest) (resBody []byte, err error) { // Process login input from epr.HttpRequest.Form, // epr.HttpRequest.PostForm, and/or epr.RequestBody. // Return response body and error. }
Routing
The Endpoint allow binding the unique key into path using colon ":" as the first character.
For example, after registering the following Endpoint,
epBinding := &libhttp.Endpoint{ Method: libhttp.RequestMethodGet, Path: "/category/:name", RequestType: libhttp.RequestTypeQuery, ResponseType: libhttp.ResponseTypeJSON, Call: handleCategory, } server.RegisterEndpoint(epBinding)
when the server receiving GET request using path "/category/book?limit=10", it will put the "book" and "10" into http.Request.Form with key is "name" and "limit"
fmt.Println("request.Form:", req.Form) // request.Form: map[name:[book] limit:[10]]
The key binding must be unique between path and query. If query has the same key then it will be overridden by value in path. For example, using the above endpoint, request with "/category/book?name=Hitchiker" will result in http.Request.Form:
map[name:[book]]
not
map[name:[book Hitchiker]]
Callback error handling
Each Endpoint can have their own error handler. If its nil, it will default to DefaultErrorHandler, which return the error as JSON with the following format,
{"code":<HTTP_STATUS_CODE>,"message":<err.Error()>}
Range request
The standard http package provide http.ServeContent function that support serving resources with Range request, except that it sometime it has an issue.
When server receive,
GET /big Range: bytes=0-
and the requested resources is quite larger, where writing all content of file result in i/o timeout, it is best practice if the server write only partial content and let the client continue with the subsequent Range request.
In the above case, the server should response with,
HTTP/1.1 206 Partial content Content-Range: bytes 0-<limit>/<size> Content-Length: <limit>
Where limit is maximum packet that is reasonable for most of the client. In this server we choose 8MB as limit, see DefRangeLimit.
Summary
The pseudocode below illustrate how Endpoint, Callback, and CallbackErrorHandler works when the Server receive HTTP request,
func (server *Server) (w http.ResponseWriter, req *http.Request) { for _, endpoint := range server.endpoints { if endpoint.Method.String() != req.Method { continue } epr := &EndpointRequest{ Endpoint: endpoint, HttpWriter: w, HttpRequest: req, } epr.RequestBody, _ = io.ReadAll(req.Body) // Check request type, and call ParseForm or // ParseMultipartForm if required. var resBody []byte resBody, epr.Error = endpoint.Call(epr) if err != nil { endpoint.ErrorHandler(epr) return } // Set content-type based on endpoint.ResponseType, // and write the response body, w.Write(resBody) return } // If request is HTTP GET, check if Path exist as static // contents in Memfs. }
Bugs and Limitations
The server does not handle CONNECT method
Missing test for request with content-type multipart-form
Server can not register path with ambigous route. For example, "/:x" and "/y" are ambiguous because one is dynamic path using key binding "x" and the last one is static path to "y".
Index ¶
- Constants
- Variables
- func DefaultErrorHandler(epr *EndpointRequest)
- func GenerateFormData(params map[string][]byte) (contentType, body string, err error)
- func HandleRange(res http.ResponseWriter, req *http.Request, bodyReader io.ReadSeeker, contentType string)
- func IPAddressOfRequest(headers http.Header, defAddr string) (addr string)
- func MarshalForm(in any) (out url.Values, err error)
- func ParseResponseHeader(raw []byte) (resp *http.Response, rest []byte, err error)
- func ParseXForwardedFor(val string) (clientAddr string, proxyAddrs []string)
- func UnmarshalForm(in url.Values, out interface{}) (err error)
- type CORSOptions
- type Callback
- type CallbackErrorHandler
- type Client
- func NewClient(opts *ClientOptions) (client *Client)
- func (client *Client) Delete(rpath string, hdr http.Header, params url.Values) ( res *http.Response, resBody []byte, err error, )
- func (client *Client) Do(req *http.Request) (res *http.Response, resBody []byte, err error)
- func (client *Client) Download(req DownloadRequest) (res *http.Response, err error)
- func (client *Client) GenerateHttpRequest( method RequestMethod, rpath string, rtype RequestType, hdr http.Header, params interface{}, ) (req *http.Request, err error)
- func (client *Client) Get(rpath string, hdr http.Header, params url.Values) ( res *http.Response, resBody []byte, err error, )
- func (client *Client) Head(rpath string, hdr http.Header, params url.Values) ( res *http.Response, resBody []byte, err error, )
- func (client *Client) Post(rpath string, hdr http.Header, params url.Values) ( res *http.Response, resBody []byte, err error, )
- func (client *Client) PostForm(rpath string, hdr http.Header, params url.Values) ( res *http.Response, resBody []byte, err error, )
- func (client *Client) PostFormData( rpath string, hdr http.Header, params map[string][]byte, ) ( res *http.Response, resBody []byte, err error, )
- func (client *Client) PostJSON(rpath string, hdr http.Header, params interface{}) ( res *http.Response, resBody []byte, err error, )
- func (client *Client) Put(rpath string, hdr http.Header, body []byte) ( *http.Response, []byte, error, )
- func (client *Client) PutForm(rpath string, hdr http.Header, params url.Values) ( res *http.Response, resBody []byte, err error, )
- func (client *Client) PutFormData(rpath string, hdr http.Header, params map[string][]byte) ( res *http.Response, resBody []byte, err error, )
- func (client *Client) PutJSON(rpath string, hdr http.Header, params interface{}) ( res *http.Response, resBody []byte, err error, )
- type ClientOptions
- type ClientRequest
- type DownloadRequest
- type Endpoint
- type EndpointRequest
- type EndpointResponse
- type Evaluator
- type FSHandler
- type Range
- func NewRange(unit string) (r *Range)
- func ParseMultipartRange(body io.Reader, boundary string) (r *Range, err error)
- func ParseRange(v string) (r Range)
- func (r *Range) Add(start, end *int64) bool
- func (r *Range) IsEmpty() bool
- func (r *Range) Positions() []*RangePosition
- func (r *Range) String() string
- type RangePosition
- func ParseContentRange(v string) (pos *RangePosition)
- func (pos RangePosition) Content() []byte
- func (pos RangePosition) ContentRange(unit string, size int64) (v string)
- func (pos RangePosition) String() string
- type RequestMethod
- type RequestType
- type ResponseType
- type SSECallback
- type SSEConn
- func (ep *SSEConn) WriteEvent(event, data string, id *string) (err error)
- func (ep *SSEConn) WriteRaw(msg []byte) (err error)
- func (ep *SSEConn) WriteRetry(retry time.Duration) (err error)
- type SSEEndpoint
- type Server
- func NewServer(opts *ServerOptions) (srv *Server, err error)
- func (srv *Server) HandleFS(res http.ResponseWriter, req *http.Request)
- func (srv *Server) RedirectTemp(res http.ResponseWriter, redirectURL string)
- func (srv *Server) RegisterEndpoint(ep *Endpoint) (err error)
- func (srv *Server) RegisterEvaluator(eval Evaluator)
- func (srv *Server) RegisterSSE(ep *SSEEndpoint) (err error)
- func (srv *Server) ServeHTTP(res http.ResponseWriter, req *http.Request)
- func (srv *Server) Start() (err error)
- func (srv *Server) Stop(wait time.Duration) (err error)
- type ServerOptions
Examples ¶
- Endpoint (ErrorHandler)
- EndpointResponse
- GenerateFormData
- IPAddressOfRequest
- MarshalForm
- ParseContentRange
- ParseMultipartRange
- ParseRange
- ParseXForwardedFor
- Range.Add
- Range.Positions
- Range.String
- Server (CustomHTTPStatusCode)
- UnmarshalForm
- UnmarshalForm (Error)
- UnmarshalForm (Slice)
- UnmarshalForm (Zero)
Constants ¶
const ( AcceptRangesBytes = `bytes` AcceptRangesNone = `none` )
List of header value for HTTP header Accept-Ranges.
const ( ContentEncodingBzip2 = `bzip2` ContentEncodingCompress = `compress` // Using LZW. ContentEncodingGzip = `gzip` ContentEncodingDeflate = `deflate` // Using zlib. )
List of known "Content-Encoding" header values.
const ( ContentTypeBinary = `application/octet-stream` ContentTypeEventStream = `text/event-stream` ContentTypeForm = `application/x-www-form-urlencoded` ContentTypeMultipartByteRanges = `multipart/byteranges` ContentTypeMultipartForm = `multipart/form-data` ContentTypeHTML = `text/html; charset=utf-8` ContentTypeJSON = `application/json` ContentTypePlain = `text/plain; charset=utf-8` ContentTypeXML = `text/xml; charset=utf-8` )
List of known "Content-Type" header values.
const ( HeaderACAllowCredentials = `Access-Control-Allow-Credentials` HeaderACAllowHeaders = `Access-Control-Allow-Headers` HeaderACAllowMethod = `Access-Control-Allow-Method` HeaderACAllowOrigin = `Access-Control-Allow-Origin` HeaderACExposeHeaders = `Access-Control-Expose-Headers` HeaderACMaxAge = `Access-Control-Max-Age` HeaderACRequestHeaders = `Access-Control-Request-Headers` HeaderACRequestMethod = `Access-Control-Request-Method` HeaderAccept = `Accept` HeaderAcceptEncoding = `Accept-Encoding` HeaderAcceptRanges = `Accept-Ranges` HeaderAllow = `Allow` HeaderAuthKeyBearer = `Bearer` HeaderAuthorization = `Authorization` HeaderCacheControl = `Cache-Control` HeaderContentEncoding = `Content-Encoding` HeaderContentLength = `Content-Length` HeaderContentRange = `Content-Range` HeaderContentType = `Content-Type` HeaderCookie = `Cookie` HeaderDate = `Date` HeaderETag = `Etag` HeaderHost = `Host` HeaderIfModifiedSince = `If-Modified-Since` HeaderIfNoneMatch = `If-None-Match` HeaderLastEventID = `Last-Event-ID` HeaderLocation = `Location` HeaderOrigin = `Origin` HeaderRange = `Range` HeaderSetCookie = `Set-Cookie` HeaderUserAgent = `User-Agent` HeaderXForwardedFor = `X-Forwarded-For` // https://en.wikipedia.org/wiki/X-Forwarded-For HeaderXRealIp = `X-Real-Ip` //revive:disable-line )
List of known header names.
const DefRangeLimit = 8388608
DefRangeLimit limit of content served by server when Range request without end, in example "0-".
Variables ¶
var ( // ErrClientDownloadNoOutput define an error when Client's // DownloadRequest does not define the Output. ErrClientDownloadNoOutput = errors.New(`invalid or empty client download output`) // ErrEndpointAmbiguous define an error when registering path that // already exist. For example, after registering "/:x", registering // "/:y" or "/z" on the same HTTP method will result in ambiguous. ErrEndpointAmbiguous = errors.New(`ambigous endpoint`) )
Functions ¶
func DefaultErrorHandler ¶
func DefaultErrorHandler(epr *EndpointRequest)
DefaultErrorHandler define the default function that will called to handle the error returned from Callback function, if the [Endpoint.ErrorHandler] is not defined.
First, it will check if error instance of *liberrors.E. If its true, it will use the Code value for HTTP status code, otherwise if its zero or invalid, it will set to http.StatusInternalServerError.
Second, it will set the HTTP header Content-Type to "application/json" and write the response body as JSON format,
{"code":<HTTP_STATUS_CODE>, "message":<err.Error()>}
func GenerateFormData ¶
GenerateFormData generate multipart/form-data body from params.
Code:play
Output:Example¶
package main
import (
"crypto/rand"
"fmt"
"log"
"strings"
libhttp "github.com/shuLhan/share/lib/http"
"github.com/shuLhan/share/lib/test/mock"
)
func main() {
// Mock the random reader for predictable output.
// NOTE: do not do this on real code.
rand.Reader = mock.NewRandReader([]byte(`randomseed`))
var data = map[string][]byte{
`name`: []byte(`test.txt`),
`size`: []byte(`42`),
}
var (
contentType string
body string
err error
)
contentType, body, err = libhttp.GenerateFormData(data)
if err != nil {
log.Fatal(err)
}
fmt.Println(`contentType:`, contentType)
fmt.Println(`body:`)
fmt.Println(strings.ReplaceAll(body, "\r\n", "\n"))
}
contentType: multipart/form-data; boundary=72616e646f6d7365656472616e646f6d7365656472616e646f6d73656564
body:
--72616e646f6d7365656472616e646f6d7365656472616e646f6d73656564
Content-Disposition: form-data; name="name"
test.txt
--72616e646f6d7365656472616e646f6d7365656472616e646f6d73656564
Content-Disposition: form-data; name="size"
42
--72616e646f6d7365656472616e646f6d7365656472616e646f6d73656564--
func HandleRange ¶
func HandleRange(res http.ResponseWriter, req *http.Request, bodyReader io.ReadSeeker, contentType string)
HandleRange handle HTTP Range request using "bytes" unit.
The body parameter contains the content of resource being requested that implement Reader and Seeker.
If the http.Request.Method is not GET, or no "Range" in http.Request.Header, it will return all the body RFC7233 S-3.1.
The contentType is optional, if its empty, it will detected by http.ResponseWriter during Write.
It will return HTTP Code,
- 406 http.StatusNotAcceptable, if the Range unit is not "bytes".
- 416 http.StatusRequestedRangeNotSatisfiable, if the request Range start position is greater than resource size.
func IPAddressOfRequest ¶
IPAddressOfRequest get the client IP address from HTTP request header
"X-Real-IP" or "X-Forwarded-For", which ever non-empty first.
If no headers present, use the default address.
Code:
Output:Example¶
{
defAddress := "192.168.100.1"
headers := http.Header{
"X-Real-Ip": []string{"127.0.0.1"},
}
fmt.Println("Request with X-Real-IP:", IPAddressOfRequest(headers, defAddress))
headers = http.Header{
"X-Forwarded-For": []string{"127.0.0.2, 192.168.100.1"},
}
fmt.Println("Request with X-Forwarded-For:", IPAddressOfRequest(headers, defAddress))
headers = http.Header{}
fmt.Println("Request without X-* headers:", IPAddressOfRequest(headers, defAddress))
// Output:
// Request with X-Real-IP: 127.0.0.1
// Request with X-Forwarded-For: 127.0.0.2
// Request without X-* headers: 192.168.100.1
}
Request with X-Real-IP: 127.0.0.1
Request with X-Forwarded-For: 127.0.0.2
Request without X-* headers: 192.168.100.1
func MarshalForm ¶
MarshalForm marshal struct fields tagged with `form:` into url.Values.
The rules for marshaling follow the same rules as in UnmarshalForm.
It will return an error if the input is not pointer to or a struct.
Code:
Output:Example¶
{
type T struct {
Rat *big.Rat `form:"big.Rat"`
String string `form:"string"`
Bytes []byte `form:"bytes"`
Int int `form:""` // With empty tag.
F64 float64 `form:"f64"`
F32 float32 `form:"f32"`
NotSet int16 `form:"notset"`
Uint8 uint8 `form:" uint8 "`
Bool bool // Without tag.
}
var (
in = T{
Rat: big.NewRat(`1.2345`),
String: `a_string`,
Bytes: []byte(`bytes`),
Int: 1,
F64: 6.4,
F32: 3.2,
Uint8: 2,
Bool: true,
}
out url.Values
err error
)
out, err = MarshalForm(in)
if err != nil {
fmt.Println(err)
}
fmt.Println(out.Encode())
//Output:
//Bool=true&Int=1&big.Rat=1.2345&bytes=bytes&f32=3.2&f64=6.4¬set=0&string=a_string&uint8=2
}
Bool=true&Int=1&big.Rat=1.2345&bytes=bytes&f32=3.2&f64=6.4¬set=0&string=a_string&uint8=2
func ParseResponseHeader ¶
ParseResponseHeader parse HTTP response header and return it as standard HTTP Response with unreaded packet.
func ParseXForwardedFor ¶
ParseXForwardedFor parse the HTTP header "X-Forwarded-For" value from the
following format "client, proxy1, proxy2" into client address and list of
proxy addressess.
Code:
Output:Example¶
{
values := []string{
"",
"203.0.113.195",
"203.0.113.195, 70.41.3.18, 150.172.238.178",
"2001:db8:85a3:8d3:1319:8a2e:370:7348",
}
for _, val := range values {
clientAddr, proxyAddrs := ParseXForwardedFor(val)
fmt.Println(clientAddr, proxyAddrs)
}
// Output:
// []
// 203.0.113.195 []
// 203.0.113.195 [70.41.3.18 150.172.238.178]
// 2001:db8:85a3:8d3:1319:8a2e:370:7348 []
}
[]
203.0.113.195 []
203.0.113.195 [70.41.3.18 150.172.238.178]
2001:db8:85a3:8d3:1319:8a2e:370:7348 []
func UnmarshalForm ¶
UnmarshalForm read struct fields tagged with `form:` from out as key and set its using the value from url.Values based on that key. If the field does not have `form:` tag but it is exported, then it will use the field name, in case insensitive manner.
Only the following types are supported: bool, int/intX, uint/uintX, floatX, string, []byte, or type that implement encoding.BinaryUnmarshaler (UnmarshalBinary), json.Unmarshaler (UnmarshalJSON), or encoding.TextUnmarshaler (UnmarshalText).
A bool type can be set to true using the following string value: "true", "yes", or "1".
If the input contains multiple values but the field type is not slice, the field will be set using the first value.
It will return an error if the out variable is not set-able (the type is
not a pointer to a struct).
It will not return an error if one of the input value is not match with
field type.
Code:
Output: Code:
Output: Code:
Output: Code:
Output:Example¶
{
type T struct {
Rat *big.Rat `form:"big.Rat"`
String string `form:"string"`
Bytes []byte `form:"bytes"`
Int int `form:""` // With empty tag.
F64 float64 `form:"f64"`
F32 float32 `form:"f32"`
NotSet int16 `form:"notset"`
Uint8 uint8 `form:" uint8 "`
Bool bool // Without tag.
}
var (
in = url.Values{}
out T
ptrOut *T
err error
)
in.Set("big.Rat", "1.2345")
in.Set("string", "a_string")
in.Set("bytes", "bytes")
in.Set("int", "1")
in.Set("f64", "6.4")
in.Set("f32", "3.2")
in.Set("uint8", "2")
in.Set("bool", "true")
err = UnmarshalForm(in, &out)
if err != nil {
fmt.Println(err)
} else {
fmt.Printf("%+v\n", out)
}
// Set the struct without initialized.
err = UnmarshalForm(in, &ptrOut)
if err != nil {
fmt.Println(err)
} else {
fmt.Printf("%+v\n", ptrOut)
}
//Output:
//{Rat:1.2345 String:a_string Bytes:[98 121 116 101 115] Int:1 F64:6.4 F32:3.2 NotSet:0 Uint8:2 Bool:true}
//&{Rat:1.2345 String:a_string Bytes:[98 121 116 101 115] Int:1 F64:6.4 F32:3.2 NotSet:0 Uint8:2 Bool:true}
}
{Rat:1.2345 String:a_string Bytes:[98 121 116 101 115] Int:1 F64:6.4 F32:3.2 NotSet:0 Uint8:2 Bool:true}
&{Rat:1.2345 String:a_string Bytes:[98 121 116 101 115] Int:1 F64:6.4 F32:3.2 NotSet:0 Uint8:2 Bool:true}
Example (Error)¶
{
type T struct {
Int int
}
var (
in = url.Values{}
out T
ptrOut *T
err error
)
// Passing out as unsetable by function.
err = UnmarshalForm(in, out)
if err != nil {
fmt.Println(err)
} else {
fmt.Println(out)
}
// Passing out as un-initialized pointer.
err = UnmarshalForm(in, ptrOut)
if err != nil {
fmt.Println(err)
} else {
fmt.Println(out)
}
// Set the field with invalid type.
in.Set("int", "a")
err = UnmarshalForm(in, &out)
if err != nil {
fmt.Println(err)
} else {
fmt.Println(out)
}
//Output:
//UnmarshalForm: expecting *T got http.T
//UnmarshalForm: *http.T is not initialized
//{0}
}
UnmarshalForm: expecting *T got http.T
UnmarshalForm: *http.T is not initialized
{0}
Example (Slice)¶
{
type SliceT struct {
NotSlice string `form:"multi_value"`
SliceString []string `form:"slice_string"`
SliceInt []int `form:"slice_int"`
}
var (
in = url.Values{}
sliceOut SliceT
ptrSliceOut *SliceT
err error
)
in.Add("multi_value", "first")
in.Add("multi_value", "second")
in.Add("slice_string", "multi")
in.Add("slice_string", "value")
in.Add("slice_int", "123")
in.Add("slice_int", "456")
err = UnmarshalForm(in, &sliceOut)
if err != nil {
fmt.Println(err)
} else {
fmt.Printf("%+v\n", sliceOut)
}
err = UnmarshalForm(in, &ptrSliceOut)
if err != nil {
fmt.Println(err)
} else {
fmt.Printf("%+v\n", ptrSliceOut)
}
//Output:
//{NotSlice:first SliceString:[multi value] SliceInt:[123 456]}
//&{NotSlice:first SliceString:[multi value] SliceInt:[123 456]}
}
{NotSlice:first SliceString:[multi value] SliceInt:[123 456]}
&{NotSlice:first SliceString:[multi value] SliceInt:[123 456]}
Example (Zero)¶
{
type T struct {
Rat *big.Rat `form:"big.Rat"`
String string `form:"string"`
Bytes []byte `form:"bytes"`
Int int `form:""` // With empty tag.
F64 float64 `form:"f64"`
F32 float32 `form:"f32"`
NotSet int16 `form:"notset"`
Uint8 uint8 `form:" uint8 "`
Bool bool // Without tag.
}
var (
in = url.Values{}
out T
err error
)
in.Set("big.Rat", "1.2345")
in.Set("string", "a_string")
in.Set("bytes", "bytes")
in.Set("int", "1")
in.Set("f64", "6.4")
in.Set("f32", "3.2")
in.Set("uint8", "2")
in.Set("bool", "true")
err = UnmarshalForm(in, &out)
if err != nil {
fmt.Println(err)
} else {
fmt.Printf("%+v\n", out)
}
in.Set("bool", "")
in.Set("int", "")
in.Set("uint8", "")
in.Set("f32", "")
in.Set("f64", "")
in.Set("string", "")
in.Set("bytes", "")
in.Set("big.Rat", "")
err = UnmarshalForm(in, &out)
if err != nil {
fmt.Println(err)
} else {
fmt.Printf("%+v\n", out)
}
//Output:
//{Rat:1.2345 String:a_string Bytes:[98 121 116 101 115] Int:1 F64:6.4 F32:3.2 NotSet:0 Uint8:2 Bool:true}
//{Rat:0 String: Bytes:[] Int:0 F64:0 F32:0 NotSet:0 Uint8:0 Bool:false}
}
{Rat:1.2345 String:a_string Bytes:[98 121 116 101 115] Int:1 F64:6.4 F32:3.2 NotSet:0 Uint8:2 Bool:true}
{Rat:0 String: Bytes:[] Int:0 F64:0 F32:0 NotSet:0 Uint8:0 Bool:false}
Types ¶
type CORSOptions ¶
type CORSOptions struct { // AllowOrigins contains global list of cross-site Origin that are // allowed during preflight requests by the OPTIONS method. // The list is case-sensitive. // To allow all Origin, one must add "*" string to the list. AllowOrigins []string // AllowHeaders contains global list of allowed headers during // preflight requests by the OPTIONS method. // The list is case-insensitive. // To allow all headers, one must add "*" string to the list. AllowHeaders []string // ExposeHeaders contains list of allowed headers. // This list will be send when browser request OPTIONS without // request-method. ExposeHeaders []string // MaxAge gives the value in seconds for how long the response to // the preflight request can be cached for without sending another // preflight request. MaxAge int // AllowCredentials indicates whether or not the actual request // can be made using credentials. AllowCredentials bool // contains filtered or unexported fields }
CORSOptions define optional options for server to allow other servers to access its resources.
type Callback ¶
type Callback func(req *EndpointRequest) (resBody []byte, err error)
Callback define a type of function for handling registered handler.
The function will have the query URL, request multipart form data, and request body ready to be used in [EndpointRequest.HttpRequest] and [EndpointRequest.RequestBody] fields.
The [EndpointRequest.HttpWriter] can be used to write custom header or to write cookies but should not be used to write response body.
The error return type should be an instance of *liberrors.E, with liberrors.E.Code define the HTTP status code. If error is not nil and not *liberrors.E, server will response with http.StatusInternalServerError status code.
type CallbackErrorHandler ¶
type CallbackErrorHandler func(epr *EndpointRequest)
CallbackErrorHandler define the function that can be used to handle an error returned from [Endpoint.Call]. By default, if [Endpoint.Call] is nil, it will use DefaultErrorHandler.
type Client ¶
Client is a wrapper for standard http.Client with simplified usabilities, including setting default headers, uncompressing response body.
func NewClient ¶
func NewClient(opts *ClientOptions) (client *Client)
NewClient create and initialize new Client.
The client will have net.Dialer.KeepAlive set to 30 seconds, with one http.Transport.MaxIdleConns, and 90 seconds http.Transport.IdleConnTimeout.
func (*Client) Delete ¶
func (client *Client) Delete(rpath string, hdr http.Header, params url.Values) ( res *http.Response, resBody []byte, err error, )
Delete send the DELETE request to server using rpath as target endpoint and params as query parameters. On success, it will return the uncompressed response body.
func (*Client) Do ¶
Do overwrite the standard http.Client.Do to allow debugging request and response, and to read and return the response body immediately.
func (*Client) Download ¶
func (client *Client) Download(req DownloadRequest) (res *http.Response, err error)
Download a resource from remote server and write it into [DownloadRequest.Output].
If the [DownloadRequest.Output] is nil, it will return an error ErrClientDownloadNoOutput. If server return HTTP code beside 200, it will return non-nil http.Response with an error.
func (*Client) GenerateHttpRequest ¶
func (client *Client) GenerateHttpRequest( method RequestMethod, rpath string, rtype RequestType, hdr http.Header, params interface{}, ) (req *http.Request, err error)
GenerateHttpRequest generate http.Request from method, rpath, rtype, hdr, and params.
For HTTP method GET, CONNECT, DELETE, HEAD, OPTIONS, or TRACE; the params value should be nil or url.Values. If its url.Values, then the params will be encoded as query parameters.
For HTTP method is PATCH, POST, or PUT; the params will converted based on rtype rules below,
- If rtype is RequestTypeQuery and params is url.Values it will be added as query parameters in the rpath.
- If rtype is RequestTypeForm and params is url.Values it will be added as URL encoded in the body.
- If rtype is RequestTypeMultipartForm and params type is map[string][]byte, then it will be converted as multipart form in the body.
- If rtype is RequestTypeJSON and params is not nil, the params will be encoded as JSON in body.
func (*Client) Get ¶
func (client *Client) Get(rpath string, hdr http.Header, params url.Values) ( res *http.Response, resBody []byte, err error, )
Get send the GET request to server using rpath as target endpoint and params as query parameters. On success, it will return the uncompressed response body.
func (*Client) Head ¶
func (client *Client) Head(rpath string, hdr http.Header, params url.Values) ( res *http.Response, resBody []byte, err error, )
Head send the HEAD request to rpath endpoint, with optional hdr and params in query parameters. The returned resBody shoule be always nil.
func (*Client) Post ¶
func (client *Client) Post(rpath string, hdr http.Header, params url.Values) ( res *http.Response, resBody []byte, err error, )
Post send the POST request to rpath without setting "Content-Type". If the params is not nil, it will send as query parameters in the rpath.
func (*Client) PostForm ¶
func (client *Client) PostForm(rpath string, hdr http.Header, params url.Values) ( res *http.Response, resBody []byte, err error, )
PostForm send the POST request to rpath using "application/x-www-form-urlencoded".
func (*Client) PostFormData ¶
func (client *Client) PostFormData( rpath string, hdr http.Header, params map[string][]byte, ) ( res *http.Response, resBody []byte, err error, )
PostFormData send the POST request to rpath with all parameters is send using "multipart/form-data".
func (*Client) PostJSON ¶
func (client *Client) PostJSON(rpath string, hdr http.Header, params interface{}) ( res *http.Response, resBody []byte, err error, )
PostJSON send the POST request with content type set to "application/json" and params encoded automatically to JSON. The params must be a type than can be marshalled with json.Marshal or type that implement json.Marshaler.
func (*Client) Put ¶
func (client *Client) Put(rpath string, hdr http.Header, body []byte) ( *http.Response, []byte, error, )
Put send the HTTP PUT request to rpath with optional, raw body. The Content-Type can be set in the hdr.
func (*Client) PutForm ¶
func (client *Client) PutForm(rpath string, hdr http.Header, params url.Values) ( res *http.Response, resBody []byte, err error, )
PutForm send the PUT request with params set in body using content type "application/x-www-form-urlencoded".
func (*Client) PutFormData ¶
func (client *Client) PutFormData(rpath string, hdr http.Header, params map[string][]byte) ( res *http.Response, resBody []byte, err error, )
PutFormData send the PUT request with params set in body using content type "multipart/form-data".
func (*Client) PutJSON ¶
func (client *Client) PutJSON(rpath string, hdr http.Header, params interface{}) ( res *http.Response, resBody []byte, err error, )
PutJSON send the PUT request with content type set to "application/json" and params encoded automatically to JSON.
type ClientOptions ¶
type ClientOptions struct { // Headers define default headers that will be send in any request to // server. // This field is optional. Headers http.Header // ServerUrl define the server address without path, for example // "https://example.com" or "http://10.148.0.12:8080". // This value should not changed during call of client's method. // This field is required. ServerUrl string //revive:disable-line // Timeout affect the http Transport Timeout and TLSHandshakeTimeout. // This field is optional, if not set it will set to 10 seconds. Timeout time.Duration // AllowInsecure if its true, it will allow to connect to server with // unknown certificate authority. // This field is optional. AllowInsecure bool }
ClientOptions options for HTTP client.
type ClientRequest ¶
type ClientRequest struct { // Headers additional header to be send on request. // This field is optional. Headers http.Header // // Params define parameter to be send on request. // This field is optional. // // For Method GET, CONNECT, DELETE, HEAD, OPTIONS, or TRACE; the // params value should be nil or url.Values. // If its url.Values, then the params will be encoded as query // parameters. // // For Method PATCH, POST, or PUT; the Params will converted based on // Type rules below, // // * If Type is RequestTypeQuery and Params is url.Values it will be // added as query parameters in the Path. // // * If Type is RequestTypeForm and Params is url.Values it will be // added as URL encoded in the body. // // * If Type is RequestTypeMultipartForm and Params is // map[string][]byte, then it will be converted as multipart form in // the body. // // * If Type is RequestTypeJSON and Params is not nil, the params will // be encoded as JSON in body using json.Encode(). // Params interface{} // The Path to resource on the server. // This field is required, if its empty default to "/". Path string // The HTTP method of request. // This field is optional, if its empty default to RequestMethodGet // (GET). Method RequestMethod // The Type of request. // This field is optional, it's affect how the Params field encoded in // the path or body. Type RequestType }
ClientRequest define the parameters for each Client methods.
type DownloadRequest ¶
type DownloadRequest struct { // Output define where the downloaded resource from server will be // writen. // This field is required. Output io.Writer ClientRequest }
DownloadRequest define the parameter for Client.Download method.
type Endpoint ¶
type Endpoint struct { // ErrorHandler define the function that will handle the error // returned from Call. ErrorHandler CallbackErrorHandler // Eval define evaluator for route that will be called after global // evaluators and before callback. Eval Evaluator // Call is the main process of route. Call Callback // Path contains route to be served, default to "/" if its empty. Path string // RequestType contains type of request, default to RequestTypeNone. RequestType RequestType // ResponseType contains type of request, default to ResponseTypeNone. ResponseType ResponseType // Method contains HTTP method, default to GET. Method RequestMethod }
Endpoint represent route that will be handled by server.
Each route have their own evaluator that will be evaluated after global
evaluators from server.
Code:
Output:Example (ErrorHandler)¶
{
serverOpts := &ServerOptions{
Address: "127.0.0.1:8123",
}
server, _ := NewServer(serverOpts)
endpointError := &Endpoint{
Method: RequestMethodGet,
Path: "/",
RequestType: RequestTypeQuery,
ResponseType: ResponseTypePlain,
Call: func(epr *EndpointRequest) ([]byte, error) {
return nil, fmt.Errorf(epr.HttpRequest.Form.Get("error"))
},
ErrorHandler: func(epr *EndpointRequest) {
epr.HttpWriter.Header().Set(HeaderContentType, ContentTypePlain)
codeMsg := strings.Split(epr.Error.Error(), ":")
if len(codeMsg) != 2 {
epr.HttpWriter.WriteHeader(http.StatusInternalServerError)
_, _ = epr.HttpWriter.Write([]byte(epr.Error.Error()))
} else {
code, _ := strconv.Atoi(codeMsg[0])
epr.HttpWriter.WriteHeader(code)
_, _ = epr.HttpWriter.Write([]byte(codeMsg[1]))
}
},
}
_ = server.RegisterEndpoint(endpointError)
go func() {
_ = server.Start()
}()
defer func() {
_ = server.Stop(1 * time.Second)
}()
time.Sleep(1 * time.Second)
clientOpts := &ClientOptions{
ServerUrl: "http://" + serverOpts.Address,
}
client := NewClient(clientOpts)
params := url.Values{}
params.Set("error", "400:error with status code")
httpres, resbody, err := client.Get(`/`, nil, params)
if err != nil {
log.Fatal(err)
}
fmt.Printf("%d: %s\n", httpres.StatusCode, resbody)
params.Set("error", "error without status code")
httpres, resbody, err = client.Get(`/`, nil, params)
if err != nil {
log.Fatal(err)
}
fmt.Printf("%d: %s\n", httpres.StatusCode, resbody)
// Output:
// 400: error with status code
// 500: error without status code
}
400: error with status code
500: error without status code
func (*Endpoint) HTTPMethod ¶
HTTPMethod return the string representation of HTTP method as predefined in "net/http".
type EndpointRequest ¶
type EndpointRequest struct { HttpWriter http.ResponseWriter //revive:disable-line Error error Endpoint *Endpoint HttpRequest *http.Request //revive:disable-line RequestBody []byte }
EndpointRequest wrap the called Endpoint and common two parameters in HTTP handler: the http.ResponseWriter and http.Request.
The RequestBody field contains the full http.Request.Body that has been read.
The Error field is used by CallbackErrorHandler.
type EndpointResponse ¶
type EndpointResponse struct { Data interface{} `json:"data,omitempty"` liberrors.E // The Limit field contains the maximum number of records per page. Limit int64 `json:"limit,omitempty"` // The Offset field contains the start index of paging. // If Page values is from request then the offset can be set to // Page times Limit. Offset int64 `json:"offset,omitempty"` // The Page field contains the requested or current page of response. Page int64 `json:"page,omitempty"` // Count field contains the total number of records in Data. Count int64 `json:"count,omitempty"` // Total field contains the total number of all records. Total int64 `json:"total,omitempty"` }
EndpointResponse is one of the common HTTP response container that can be used by Server implementor. Its embed the liberrors.E type to work seamlessly with [Endpoint.Call] handler for checking the returned error.
If the response is paging, contains more than one item in Data, one can set the current status of paging in field Limit, Offset, Page, and Count.
See the example below on how to use it with [Endpoint.Call] handler.
Code:
Output:Example¶
{
type myData struct {
ID string
}
server, err := NewServer(&ServerOptions{
Address: "127.0.0.1:7016",
})
if err != nil {
log.Fatal(err)
}
// Lest say we have an endpoint that echoing back the request
// parameter "id" back to client inside the EndpointResponse.Data using
// myData as JSON format.
// If the parameter "id" is missing or empty it will return an HTTP
// status code with message as defined in EndpointResponse.
err = server.RegisterEndpoint(&Endpoint{
Method: RequestMethodGet,
RequestType: RequestTypeQuery,
ResponseType: ResponseTypeJSON,
Call: func(epr *EndpointRequest) ([]byte, error) {
res := &EndpointResponse{}
id := epr.HttpRequest.Form.Get("id")
if len(id) == 0 {
res.E.Code = http.StatusBadRequest
res.E.Message = "empty parameter id"
return nil, res
}
if id == "0" {
// If the EndpointResponse.Code is 0, it will
// default to http.StatusInternalServerError
res.E.Message = "id value 0 cause internal server error"
return nil, res
}
res.E.Code = http.StatusOK
res.Data = &myData{
ID: id,
}
return json.Marshal(res)
},
})
if err != nil {
log.Fatal(err)
}
go func() {
var errStart = server.Start()
if errStart != nil {
log.Fatal(errStart)
}
}()
time.Sleep(1 * time.Second)
clientOpts := &ClientOptions{
ServerUrl: "http://127.0.0.1:7016",
}
cl := NewClient(clientOpts)
params := url.Values{}
// Test call endpoint without "id" parameter.
_, resBody, err := cl.Get("/", nil, params)
if err != nil {
log.Fatal(err)
}
fmt.Printf("GET / => %s\n", resBody)
// Test call endpoint with "id" parameter set to "0", it should return
// HTTP status 500 with custom message.
params.Set("id", "0")
_, resBody, err = cl.Get("/", nil, params)
if err != nil {
log.Fatal(err)
}
fmt.Printf("GET /?id=0 => %s\n", resBody)
// Test with "id" parameter is set.
params.Set("id", "1000")
_, resBody, err = cl.Get("/", nil, params)
if err != nil {
log.Fatal(err)
}
fmt.Printf("GET /?id=1000 => %s\n", resBody)
// Output:
// GET / => {"message":"empty parameter id","code":400}
// GET /?id=0 => {"message":"id value 0 cause internal server error","code":500}
// GET /?id=1000 => {"data":{"ID":"1000"},"code":200}
}
GET / => {"message":"empty parameter id","code":400}
GET /?id=0 => {"message":"id value 0 cause internal server error","code":500}
GET /?id=1000 => {"data":{"ID":"1000"},"code":200}
func (*EndpointResponse) Error ¶
func (epr *EndpointResponse) Error() string
func (*EndpointResponse) Unwrap ¶
func (epr *EndpointResponse) Unwrap() (err error)
Unwrap return the error as instance of *liberrors.E.
type Evaluator ¶
Evaluator evaluate the request. If request is invalid, the error will tell the response code and the error message to be written back to client.
type FSHandler ¶
FSHandler define the function to inspect each GET request to Server memfs.MemFS instance. The node parameter contains the requested file inside the memfs.
If the handler return true, server will continue processing the node (writing the memfs.Node content type, body, and so on).
If the handler return false, server stop processing the node and return immediately, which means the function should have already handle writing the header, status code, and/or body.
type Range ¶
type Range struct {
// contains filtered or unexported fields
}
Range define the unit and list of start-end positions for resource.
func NewRange ¶
NewRange create new Range with specified unit. The default unit is "bytes" if its empty.
func ParseMultipartRange ¶
ParseMultipartRange parse "multipart/byteranges" response body.
Each Content-Range position and body part in the multipart will be stored
under RangePosition.
Code:play
Output:Example¶
package main
import (
"bytes"
"fmt"
"log"
"strings"
libhttp "github.com/shuLhan/share/lib/http"
)
func main() {
var (
boundary = `zxcv`
)
var body = `--zxcv
Content-Range: bytes 0-6/50
Part 1
--zxcv
Missing Content-Range header, skipped.
--zxcv
Content-Range: bytes 7-13
Invalid Content-Range, missing size, skipped.
--zxcv
Content-Range: bytes 14-19/50
Part 2
--zxcv--
`
body = strings.ReplaceAll(body, "\n", "\r\n")
var (
reader = bytes.NewReader([]byte(body))
r *libhttp.Range
err error
)
r, err = libhttp.ParseMultipartRange(reader, boundary)
if err != nil {
log.Fatal(err)
}
var pos *libhttp.RangePosition
for _, pos = range r.Positions() {
fmt.Printf("%s: %s\n", pos.String(), pos.Content())
}
}
0-6: Part 1
14-19: Part 2
func ParseRange ¶
ParseRange parses raw range value in the following format,
range = unit "=" position *("," position) unit = 1*VCHAR position = "-" last / start "-" / start "-" end last = 1*DIGIT start = 1*DIGIT end = 1*DIGIT
An invalid position will be skipped.
Code:play
Output:Example¶
package main
import (
"fmt"
libhttp "github.com/shuLhan/share/lib/http"
)
func main() {
var r libhttp.Range
// Empty range due to missing "=".
r = libhttp.ParseRange(`bytes`)
fmt.Println(r.String())
r = libhttp.ParseRange(`bytes=10-`)
fmt.Println(r.String())
// The "20-30" is overlap with "10-".
r = libhttp.ParseRange(`bytes=10-,20-30`)
fmt.Println(r.String())
// The "10-" is ignored since its overlap with the first range
// "20-30".
r = libhttp.ParseRange(`bytes=20 - 30 , 10 -`)
fmt.Println(r.String())
r = libhttp.ParseRange(`bytes=10-20`)
fmt.Println(r.String())
r = libhttp.ParseRange(`bytes=-20`)
fmt.Println(r.String())
r = libhttp.ParseRange(`bytes=0-9,10-19,-20`)
fmt.Println(r.String())
r = libhttp.ParseRange(`bytes=0-`)
fmt.Println(r.String())
// The only valid position here is 0-9, 10-19, and -20.
// The x, -x, x-9, 0-x, 0-9-, and -0-9 is not valid position.
// The -10 is overlap with -20.
r = libhttp.ParseRange(`bytes=,x,-x,x-9,0-x,0-9-,-0-9,0-9,10-19,-20,-10,`)
fmt.Println(r.String())
}
bytes=10-
bytes=10-
bytes=20-30
bytes=10-20
bytes=-20
bytes=0-9,10-19,-20
bytes=0-
bytes=0-9,10-19,-20
func (*Range) Add ¶
Add start and end as requested position to Range. The start and end position is inclusive, closed interval [start, end], with end position must equal or greater than start position, unless its zero. For example,
- [0,+x] is valid, from offset 0 until x+1.
- [0,0] is valid and equal to first byte (but unusual).
- [+x,+y] is valid iff x <= y.
- [+x,-y] is invalid.
- [-x,+y] is invalid.
The start or end can be nil, but not both. For example,
- [nil,+x] is valid, equal to "-x" or the last x bytes.
- [nil,0] is invalid.
- [nil,-x] is invalid.
- [x,nil] is valid, equal to "x-" or from offset x until end of file.
- [-x,nil] is invalid.
The new position will be added and return true iff it does not overlap
with existing list.
Code:play
Output:Example¶
package main
import (
"fmt"
libhttp "github.com/shuLhan/share/lib/http"
)
func ptrInt64(v int64) *int64 { return &v }
func main() {
var listpos = []struct {
start *int64
end *int64
}{
{ptrInt64(0), ptrInt64(9)}, // OK.
{ptrInt64(0), ptrInt64(5)}, // Overlap with [0,9].
{ptrInt64(9), ptrInt64(19)}, // Overlap with [0,9].
{ptrInt64(10), ptrInt64(19)}, // OK.
{ptrInt64(19), ptrInt64(20)}, // Overlap with [10,19].
{ptrInt64(20), ptrInt64(19)}, // End less than start.
{nil, ptrInt64(10)}, // OK.
{nil, ptrInt64(20)}, // Overlap with [nil,10].
{ptrInt64(20), nil}, // Overlap with [nil,10].
{ptrInt64(30), ptrInt64(40)}, // Overlap with [20,nil].
{ptrInt64(30), nil}, // Overlap with [20,nil].
}
var r = libhttp.NewRange(``)
for _, pos := range listpos {
fmt.Println(r.Add(pos.start, pos.end), r.String())
}
}
true bytes=0-9
false bytes=0-9
false bytes=0-9
true bytes=0-9,10-19
false bytes=0-9,10-19
false bytes=0-9,10-19
true bytes=0-9,10-19,-10
false bytes=0-9,10-19,-10
false bytes=0-9,10-19,-10
true bytes=0-9,10-19,-10,30-40
false bytes=0-9,10-19,-10,30-40
func (*Range) IsEmpty ¶
IsEmpty return true if Range has no registered positions.
func (*Range) Positions ¶
func (r *Range) Positions() []*RangePosition
Positions return the list of range position.
Code:play
Output:Example¶
package main
import (
"fmt"
libhttp "github.com/shuLhan/share/lib/http"
)
func ptrInt64(v int64) *int64 { return &v }
func main() {
var r = libhttp.NewRange(``)
fmt.Println(r.Positions()) // Empty positions.
r.Add(ptrInt64(10), ptrInt64(20))
fmt.Println(r.Positions())
}
[]
[10-20]
func (*Range) String ¶
String return the Range as value for HTTP header.
It will return an empty string if no position registered.
Code:play
Output:Example¶
package main
import (
"fmt"
libhttp "github.com/shuLhan/share/lib/http"
)
func ptrInt64(v int64) *int64 { return &v }
func main() {
var r = libhttp.NewRange(`MyUnit`)
fmt.Println(r.String()) // Empty range will return empty string.
r.Add(ptrInt64(0), ptrInt64(9))
fmt.Println(r.String())
}
myunit=0-9
type RangePosition ¶
type RangePosition struct {
// contains filtered or unexported fields
}
RangePosition contains the parsed value of Content-Range header.
func ParseContentRange ¶
func ParseContentRange(v string) (pos *RangePosition)
ParseContentRange parse the HTTP header "Content-Range" value, as response from server, with the following format,
Content-Range = unit SP valid-range / invalid-range SP = " " valid-range = position "/" size invalid-range = "*" "/" size position = start "-" end size = 1*DIGIT / "*" start = 1*DIGIT end = 1*DIGIT
It will return nil if the v is invalid.
Code:play
Output:Example¶
package main
import (
"fmt"
libhttp "github.com/shuLhan/share/lib/http"
)
func main() {
fmt.Println(libhttp.ParseContentRange(`bytes 10-/20`)) // Invalid, missing end.
fmt.Println(libhttp.ParseContentRange(`bytes 10-19/20`)) // OK
fmt.Println(libhttp.ParseContentRange(`bytes -10/20`)) // Invalid, missing start.
fmt.Println(libhttp.ParseContentRange(`10-20/20`)) // Invalid, missing unit.
fmt.Println(libhttp.ParseContentRange(`bytes 10-`)) // Invalid, missing "/size".
fmt.Println(libhttp.ParseContentRange(`bytes -10/x`)) // Invalid, invalid "size".
fmt.Println(libhttp.ParseContentRange(`bytes`)) // Invalid, missing position.
}
<nil>
10-19
<nil>
<nil>
<nil>
<nil>
<nil>
func (RangePosition) Content ¶
func (pos RangePosition) Content() []byte
Content return the range content body in multipart.
func (RangePosition) ContentRange ¶
func (pos RangePosition) ContentRange(unit string, size int64) (v string)
ContentRange return the string that can be used for HTTP Content-Range header value.
func (RangePosition) String ¶
func (pos RangePosition) String() string
type RequestMethod ¶
type RequestMethod int
RequestMethod define type of HTTP method.
const ( RequestMethodGet RequestMethod = iota RequestMethodConnect RequestMethodDelete RequestMethodHead RequestMethodOptions RequestMethodPatch RequestMethodPost RequestMethodPut RequestMethodTrace )
List of known HTTP methods.
func (RequestMethod) String ¶
func (rm RequestMethod) String() string
String return the string representation of request method.
type RequestType ¶
type RequestType int
RequestType define type of HTTP request.
const ( RequestTypeNone RequestType = iota RequestTypeQuery RequestTypeForm RequestTypeMultipartForm RequestTypeJSON RequestTypeXML )
List of valid request type.
func (RequestType) String ¶
func (rt RequestType) String() string
String return the string representation of request type as in "Content-Type" header. For RequestTypeNone or RequestTypeQuery it will return an empty string "".
type ResponseType ¶
type ResponseType int
ResponseType define type for HTTP response.
const ( ResponseTypeNone ResponseType = iota ResponseTypeBinary ResponseTypeHTML ResponseTypeJSON ResponseTypePlain ResponseTypeXML )
List of valid response type.
func (ResponseType) String ¶
func (restype ResponseType) String() string
String return the string representation of ResponseType as in "Content-Type" header. For ResponseTypeNone it will return an empty string "".
type SSECallback ¶
type SSECallback func(sse *SSEConn)
SSECallback define the handler for Server-Sent Events (SSE).
SSECallback type pass SSEConn that contains original HTTP request. This allow the server to check for header "Last-Event-ID" and/or for authentication. Remember that "the original http.Request.Body must not be used" according to http.Hijacker documentation.
type SSEConn ¶
type SSEConn struct { HttpRequest *http.Request //revive:disable-line // contains filtered or unexported fields }
SSEConn define the connection when the SSE request accepted by server.
func (*SSEConn) WriteEvent ¶
WriteEvent write message with optional event type and id to client.
The "event" parameter is optional. If its empty, no "event:" line will be send to client.
The "data" parameter must not be empty, otherwise no message will be send. If "data" value contains new line character ('\n'), the message will be split into multiple "data:".
The id parameter is optional. If its nil, it will be ignored. if its non-nil and empty, it will be send as empty ID.
It will return an error if its failed to write to peer connection.
func (*SSEConn) WriteRaw ¶
WriteRaw write raw event message directly, without any parsing.
func (*SSEConn) WriteRetry ¶
WriteRetry inform user how long they should wait, after disconnect, before re-connecting back to server.
The duration must be in millisecond.
type SSEEndpoint ¶
type SSEEndpoint struct { // Call handler that will called when request to Path accepted. Call SSECallback // Path where server accept the request for SSE. Path string // KeepAliveInterval define the interval where server will send an // empty message to active connection periodically. // This field is optional, default and minimum value is 5 seconds. KeepAliveInterval time.Duration }
SSEEndpoint endpoint to create Server-Sent Events (SSE) on server.
For creating the SSE client see subpackage [sseclient].
type Server ¶
type Server struct { *http.Server // Options for server, set by calling NewServer. // This field is exported only for reference, for example logging in // the Options when server started. // Modifying the value of Options after server has been started may // cause undefined effects. Options *ServerOptions // contains filtered or unexported fields }
Server define HTTP server.
Code:
Output:Example (CustomHTTPStatusCode)¶
{
type CustomResponse struct {
Status int `json:"status"`
}
var (
exp = CustomResponse{
Status: http.StatusBadRequest,
}
opts = &ServerOptions{
Address: "127.0.0.1:8123",
}
srv *Server
err error
)
srv, err = NewServer(opts)
if err != nil {
log.Fatal(err)
}
go func() {
err = srv.Start()
if err != nil {
log.Println(err)
}
}()
defer func() {
_ = srv.Stop(5 * time.Second)
}()
epCustom := &Endpoint{
Path: "/error/custom",
RequestType: RequestTypeJSON,
ResponseType: ResponseTypeJSON,
Call: func(epr *EndpointRequest) (
resbody []byte, err error,
) {
epr.HttpWriter.WriteHeader(exp.Status)
return json.Marshal(exp)
},
}
err = srv.registerPost(epCustom)
if err != nil {
log.Fatal(err)
}
// Wait for the server fully started.
time.Sleep(1 * time.Second)
clientOpts := &ClientOptions{
ServerUrl: "http://127.0.0.1:8123",
}
client := NewClient(clientOpts)
httpRes, resBody, err := client.PostJSON(epCustom.Path, nil, nil)
if err != nil {
log.Fatal(err)
}
fmt.Printf("%d\n", httpRes.StatusCode)
fmt.Printf("%s\n", resBody)
// Output:
// 400
// {"status":400}
}
400
{"status":400}
func NewServer ¶
func NewServer(opts *ServerOptions) (srv *Server, err error)
NewServer create and initialize new HTTP server that serve root directory with custom connection.
func (*Server) HandleFS ¶
func (srv *Server) HandleFS(res http.ResponseWriter, req *http.Request)
HandleFS handle the request as resource in the memory file system. This method only works if the [ServerOptions.Memfs] is not nil.
If the request Path exists and [ServerOptions.HandleFS] is set and returning false, it will return immediately.
If the request Path exists in file system, it will return 200 OK with the header Content-Type set accordingly to the detected file type and the response body set to the content of file. If the request Method is HEAD, only the header will be sent back to client.
If the request Path is not exist it will return 404 Not Found.
func (*Server) RedirectTemp ¶
func (srv *Server) RedirectTemp(res http.ResponseWriter, redirectURL string)
RedirectTemp make the request to temporary redirect (307) to new URL.
func (*Server) RegisterEndpoint ¶
RegisterEndpoint register the Endpoint based on Method. If [Endpoint.Method] field is not set, it will default to GET. The [Endpoint.Call] field MUST be set, or it will return an error.
Endpoint with Method HEAD or OPTIONS does not have any effect because it already handled automatically by server.
Endpoint with Method CONNECT or TRACE will return an error because its not supported, yet.
func (*Server) RegisterEvaluator ¶
RegisterEvaluator register HTTP middleware that will be called before [Endpoint.Eval] and [Endpoint.Call] is called.
func (*Server) RegisterSSE ¶
func (srv *Server) RegisterSSE(ep *SSEEndpoint) (err error)
RegisterSSE register Server-Sent Events endpoint. It will return an error if the [SSEEndpoint.Call] field is not set or ErrEndpointAmbiguous if the same path is already registered.
func (*Server) ServeHTTP ¶
func (srv *Server) ServeHTTP(res http.ResponseWriter, req *http.Request)
ServeHTTP handle mapping of client request to registered endpoints.
func (*Server) Start ¶
Start the HTTP server.
func (*Server) Stop ¶
Stop the server using Shutdown method. The wait is set default and minimum to five seconds.
type ServerOptions ¶
type ServerOptions struct { // Memfs contains the content of file systems to be served in memory. // The MemFS instance to be served should be already embedded in Go // file, generated using memfs.MemFS.GoEmbed(). // Otherwise, it will try to read from file system directly. // // See https://pkg.go.dev/github.com/shuLhan/share/lib/memfs#hdr-Go_embed Memfs *memfs.MemFS // HandleFS inspect each GET request to Memfs. // Some usage of this handler is to check for authorization on // specific path, handling redirect, and so on. // If nil it means all request are allowed. // See FSHandler for more information. HandleFS FSHandler // Address define listen address, using ip:port format. // This field is optional, default to ":80". Address string // Conn contains custom HTTP server connection. // This fields is optional. Conn *http.Server // ErrorWriter define the writer where output from panic in handler // will be written. Basically this will create new log.Logger and set // the default Server.ErrorLog. // This field is optional, but if its set it will be used only if Conn // is not set by caller. ErrorWriter io.Writer // The options for Cross-Origin Resource Sharing. CORS CORSOptions // If true, server generate index.html automatically if its not // exist in the directory. // The index.html contains the list of files inside the requested // path. EnableIndexHtml bool //revive:disable-line }
ServerOptions define an options to initialize HTTP server.
Source Files ¶
callback.go callback_error_handler.go client.go client_options.go client_request.go cors_options.go download_request.go endpoint.go endpoint_request.go endpoint_response.go evaluator.go form.go fs_handler.go http.go range.go range_position.go requestmethod.go requesttype.go response.go responsetype.go route.go server.go serveroptions.go sse_conn.go sse_endpoint.go
Directories ¶
Path | Synopsis |
---|---|
lib/http/sseclient | Package sseclient implement HTTP client for Server-Sent Events (SSE). |
- Version
- v0.53.1 (latest)
- Published
- Mar 2, 2024
- Platform
- linux/amd64
- Imports
- 34 packages
- Last checked
- 5 days ago –
Tools for package owners.