Documentation ¶
Overview ¶
Package bhttp provides utilities for creating http handlers with buffered responses.
Index ¶
- Variables
- func ChainStd(h http.Handler, m ...StdMiddleware) http.Handler
- func Serve[V any](hdlr Handler[V], os ...Option) http.Handler
- func ServeFunc[V any](hdlr HandlerFunc[V], os ...Option) http.Handler
- type Context
- type Handler
- type HandlerFunc
- type Logger
- type Middleware
- type Option
- type ResponseBuffer
- func (w *ResponseBuffer) FlushError() error
- func (w *ResponseBuffer) Free()
- func (w *ResponseBuffer) Header() http.Header
- func (w *ResponseBuffer) ImplicitFlush() error
- func (w *ResponseBuffer) Reset()
- func (w *ResponseBuffer) Unwrap() http.ResponseWriter
- func (w *ResponseBuffer) Write(buf []byte) (int, error)
- func (w *ResponseBuffer) WriteHeader(statusCode int)
- type ResponseWriter
- type Reverser
- type ServeMux
- func (m *ServeMux[V]) BHandle(pattern string, handler Handler[V], name ...string)
- func (m *ServeMux[V]) BHandleFunc(pattern string, handler HandlerFunc[V], name ...string)
- func (m *ServeMux[V]) BUse(mw ...Middleware[V])
- func (m *ServeMux[V]) Handle(pattern string, handler http.Handler, name ...string)
- func (m *ServeMux[V]) HandleFunc(pattern string, handler http.HandlerFunc, name ...string)
- func (m *ServeMux[V]) Reverse(name string, vals ...string) (string, error)
- func (m ServeMux[V]) ServeHTTP(w http.ResponseWriter, r *http.Request)
- func (m *ServeMux[V]) Use(mw ...StdMiddleware)
- type StdMiddleware
Constants ¶
This section is empty.
Variables ¶
var ErrBufferFull = errors.New("buffer is full")
ErrBufferFull is returned when the write call will cause the buffer to be filled beyond its limit.
Functions ¶
func ChainStd ¶
func ChainStd(h http.Handler, m ...StdMiddleware) http.Handler
ChainStd turns a slice of standard middleware into wrapped calls. The left-most middleware will become the outer middleware. 'h' will be come the inner handler.
func Serve ¶
Serve takes a handler with a customizable context that is able to return an error. To support this the response is buffered until the handler is done. If an error occurs the buffer is discarded and a full replacement response can be formulated. The underlying buffer is re-used between requests for improved performance.
Types ¶
type Context ¶
type Context[V any] struct { // V may hold extra typed values. This can be set by middleware and provide type safety for downstream // middleware and the handler itself. V V // contains filtered or unexported fields }
Context implements context.Context but takes a type parameter so applications can declare typed data accessors that middleware can set.
func NewContext ¶
NewContext inits a context instance with an embedded context 'c'.
func (Context[V]) Done ¶
func (c Context[V]) Done() <-chan struct{}
Done makes that the context implement context.Context.
func (*Context[V]) WithValue ¶
WithValue is a utility method that is used by middleware to set (untyped) contextual data. It will set it on the request's context as well as the embedded context. So both should be passed further down the middleware chain. Even though this package's Context encourages a type-safe approach for contextual data it can still be useful to use untyped context data to transfer over api boundaries that just accept a bare context.Context.
type Handler ¶
type Handler[V any] interface { ServeBHTTP(ctx *Context[V], w ResponseWriter, r *http.Request) error }
Handler mirrors http.Handler but it supports typed context values and a buffered response allow returning error.
func Chain ¶
func Chain[V any](h Handler[V], m ...Middleware[V]) Handler[V]
Chain takes the inner handler h and wraps it with middleware. The order is that of the Gorilla and Chi router. That is: the middleware provided first is called first and is the "outer" most wrapping, the middleware provided last will be the "inner most" wrapping (closest to the handler).
type HandlerFunc ¶
HandlerFunc allow casting a function to imple Handler.
func (HandlerFunc[V]) ServeBHTTP ¶
func (f HandlerFunc[V]) ServeBHTTP(ctx *Context[V], w ResponseWriter, r *http.Request) error
ServeBHTTP implements the Handler interface.
type Middleware ¶
Middleware functions wrap each other to create unilateral functionality.
type Option ¶
type Option func(*opts)
Option allows for customization of the (buffered) handling logic.
func WithBufferLimit ¶
WithBufferLimit allows limiting the buffered writer (if it buffered). This can protect against buffered response writers taking up too much memory per response.
func WithErrorLog ¶
WithErrorLog will add a standard library logger to allow observing some failure situations. If not specified it will log to log.Default().
type ResponseBuffer ¶
type ResponseBuffer struct {
// contains filtered or unexported fields
}
ResponseBuffer is a http.ResponseWriter implementation that buffers writes up to configurable amount of bytes. This allows the implementation of handlers that can error halfway and return a completely different response instead of what was written before the error occurred.
func NewBufferResponse ¶
func NewBufferResponse(resp http.ResponseWriter, limit int) *ResponseBuffer
NewBufferResponse inits a buffered response writer. It has a configurable limit after which the writing will return an error. This is to protect unchecked handlers from claiming too much memory. Limit can be set to -1 to disable this check.
func (*ResponseBuffer) FlushError ¶
func (w *ResponseBuffer) FlushError() error
FlushError any buffered bytes to the underlying response writer and resets the buffer. After flush has been called the response data should be considered sent and in-transport to the client.
func (*ResponseBuffer) Free ¶
func (w *ResponseBuffer) Free()
Free resets all members of the ResponseBuffer and puts it back in the sync pool to allow it to be re-used for a possible next initilization. It should be called after the handling has completed and the buffer should not be used after.
func (*ResponseBuffer) Header ¶
func (w *ResponseBuffer) Header() http.Header
Header allows users to modify the headers (and trailers) sent to the client. The headers are not actually flushed to the underlying writer until a write or flush is being triggered.
func (*ResponseBuffer) ImplicitFlush ¶
func (w *ResponseBuffer) ImplicitFlush() error
ImplicitFlush flushes data to the underlying writer without calling .Flush on it by proxy. This is provided separately from FlushError to allow for emulating the original ResponseWriter behaviour more correctly.
func (*ResponseBuffer) Reset ¶
func (w *ResponseBuffer) Reset()
Reset provides the differentiating feature from a regular ResponseWriter: it allows changing the response completely even if some data has been written already. This behaviour cannot be guaranteed if flush has been called explicitly so in that case it will panic.
func (*ResponseBuffer) Unwrap ¶
func (w *ResponseBuffer) Unwrap() http.ResponseWriter
Unwrap returns the underlying response writer. This is expected by the http.ResponseController to allow it to call appropriate optional interface implementations.
func (*ResponseBuffer) Write ¶
func (w *ResponseBuffer) Write(buf []byte) (int, error)
Write appends the contents of p to the buffered response, growing the internal buffer as needed. If the write will cause the buffer be larger then the configure limit it will return ErrBufferFull.
func (*ResponseBuffer) WriteHeader ¶
func (w *ResponseBuffer) WriteHeader(statusCode int)
WriteHeader will cause headers to be flushed to the underlying writer while calling WriteHeader on the underlying writer with the given status code.
type ResponseWriter ¶
type ResponseWriter interface { http.ResponseWriter Reset() }
ResponseWriter implements the http.ResponseWriter but the underlying bytes are buffered. This allows middleware to reset the writer and formulate a completely new response.
type Reverser ¶
type Reverser struct {
// contains filtered or unexported fields
}
Reverser keeps track of named patterns and allows building URLS.
func (Reverser) NamedPattern ¶
NamedPattern will parse 's' as a path pattern while returning it as well.
type ServeMux ¶
type ServeMux[V any] struct { // contains filtered or unexported fields }
ServeMux is an extension to the standard http.ServeMux. It supports handling requests with a buffered response for error returns, typed context values and named routes.
func (*ServeMux[V]) BHandle ¶
BHandle will invoke 'handler' with a buffered response for the named route and pattern.
func (*ServeMux[V]) BHandleFunc ¶
func (m *ServeMux[V]) BHandleFunc(pattern string, handler HandlerFunc[V], name ...string)
BHandleFunc will invoke a handler func with a buffered response.
func (*ServeMux[V]) BUse ¶
func (m *ServeMux[V]) BUse(mw ...Middleware[V])
BUse will add a middleware ONLY for any buffered http handling, that is handlers setup using BHandle or BHandleFunc.
func (*ServeMux[V]) Handle ¶
Handle will invoke 'handler' with an unbuffered response for the named route and pattern.
func (*ServeMux[V]) HandleFunc ¶
func (m *ServeMux[V]) HandleFunc(pattern string, handler http.HandlerFunc, name ...string)
HandleFunc will invoke 'handler' with a unbuffered response for the named route and pattern.
func (ServeMux[V]) ServeHTTP ¶
func (m ServeMux[V]) ServeHTTP(w http.ResponseWriter, r *http.Request)
ServeHTTP maxes the mux implement http.Handler.
func (*ServeMux[V]) Use ¶
func (m *ServeMux[V]) Use(mw ...StdMiddleware)
Use will add a standard http middleware triggered for both buffered and unbuffered request handling.