Make ResponseWriter an interface
Make ResponseWriter an interface with an unexported method. Implementors must embed a ResponseWriter from elsewhere. This gives us the flexibility of an interface while allowing us to add new methods in the future.
This commit is contained in:
parent
526d232ab0
commit
3660698a4b
12
fs.go
12
fs.go
@ -33,7 +33,7 @@ type fileServer struct {
|
|||||||
fs.FS
|
fs.FS
|
||||||
}
|
}
|
||||||
|
|
||||||
func (fs fileServer) ServeGemini(ctx context.Context, w *ResponseWriter, r *Request) {
|
func (fs fileServer) ServeGemini(ctx context.Context, w ResponseWriter, r *Request) {
|
||||||
serveFile(w, r, fs, path.Clean(r.URL.Path), true)
|
serveFile(w, r, fs, path.Clean(r.URL.Path), true)
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -43,11 +43,11 @@ func (fs fileServer) ServeGemini(ctx context.Context, w *ResponseWriter, r *Requ
|
|||||||
//
|
//
|
||||||
// ServeContent tries to deduce the type from name's file extension.
|
// ServeContent tries to deduce the type from name's file extension.
|
||||||
// The name is otherwise unused; it is never sent in the response.
|
// The name is otherwise unused; it is never sent in the response.
|
||||||
func ServeContent(w *ResponseWriter, r *Request, name string, content io.Reader) {
|
func ServeContent(w ResponseWriter, r *Request, name string, content io.Reader) {
|
||||||
serveContent(w, name, content)
|
serveContent(w, name, content)
|
||||||
}
|
}
|
||||||
|
|
||||||
func serveContent(w *ResponseWriter, name string, content io.Reader) {
|
func serveContent(w ResponseWriter, name string, content io.Reader) {
|
||||||
// Detect mimetype from file extension
|
// Detect mimetype from file extension
|
||||||
ext := path.Ext(name)
|
ext := path.Ext(name)
|
||||||
mimetype := mime.TypeByExtension(ext)
|
mimetype := mime.TypeByExtension(ext)
|
||||||
@ -75,7 +75,7 @@ func serveContent(w *ResponseWriter, name string, content io.Reader) {
|
|||||||
// Outside of those two special cases, ServeFile does not use r.URL.Path for
|
// Outside of those two special cases, ServeFile does not use r.URL.Path for
|
||||||
// selecting the file or directory to serve; only the file or directory
|
// selecting the file or directory to serve; only the file or directory
|
||||||
// provided in the name argument is used.
|
// provided in the name argument is used.
|
||||||
func ServeFile(w *ResponseWriter, r *Request, fsys fs.FS, name string) {
|
func ServeFile(w ResponseWriter, r *Request, fsys fs.FS, name string) {
|
||||||
if containsDotDot(r.URL.Path) {
|
if containsDotDot(r.URL.Path) {
|
||||||
// Too many programs use r.URL.Path to construct the argument to
|
// Too many programs use r.URL.Path to construct the argument to
|
||||||
// serveFile. Reject the request under the assumption that happened
|
// serveFile. Reject the request under the assumption that happened
|
||||||
@ -102,7 +102,7 @@ func containsDotDot(v string) bool {
|
|||||||
|
|
||||||
func isSlashRune(r rune) bool { return r == '/' || r == '\\' }
|
func isSlashRune(r rune) bool { return r == '/' || r == '\\' }
|
||||||
|
|
||||||
func serveFile(w *ResponseWriter, r *Request, fsys fs.FS, name string, redirect bool) {
|
func serveFile(w ResponseWriter, r *Request, fsys fs.FS, name string, redirect bool) {
|
||||||
const indexPage = "/index.gmi"
|
const indexPage = "/index.gmi"
|
||||||
|
|
||||||
// Redirect .../index.gmi to .../
|
// Redirect .../index.gmi to .../
|
||||||
@ -177,7 +177,7 @@ func serveFile(w *ResponseWriter, r *Request, fsys fs.FS, name string, redirect
|
|||||||
serveContent(w, name, f)
|
serveContent(w, name, f)
|
||||||
}
|
}
|
||||||
|
|
||||||
func dirList(w *ResponseWriter, f fs.File) {
|
func dirList(w ResponseWriter, f fs.File) {
|
||||||
var entries []fs.DirEntry
|
var entries []fs.DirEntry
|
||||||
var err error
|
var err error
|
||||||
d, ok := f.(fs.ReadDirFile)
|
d, ok := f.(fs.ReadDirFile)
|
||||||
|
12
handler.go
12
handler.go
@ -21,23 +21,23 @@ import (
|
|||||||
//
|
//
|
||||||
// Handlers should not modify the provided Request.
|
// Handlers should not modify the provided Request.
|
||||||
type Handler interface {
|
type Handler interface {
|
||||||
ServeGemini(context.Context, *ResponseWriter, *Request)
|
ServeGemini(context.Context, ResponseWriter, *Request)
|
||||||
}
|
}
|
||||||
|
|
||||||
// The HandlerFunc type is an adapter to allow the use of ordinary functions
|
// The HandlerFunc type is an adapter to allow the use of ordinary functions
|
||||||
// as Gemini handlers. If f is a function with the appropriate signature,
|
// as Gemini handlers. If f is a function with the appropriate signature,
|
||||||
// HandlerFunc(f) is a Handler that calls f.
|
// HandlerFunc(f) is a Handler that calls f.
|
||||||
type HandlerFunc func(context.Context, *ResponseWriter, *Request)
|
type HandlerFunc func(context.Context, ResponseWriter, *Request)
|
||||||
|
|
||||||
// ServeGemini calls f(ctx, w, r).
|
// ServeGemini calls f(ctx, w, r).
|
||||||
func (f HandlerFunc) ServeGemini(ctx context.Context, w *ResponseWriter, r *Request) {
|
func (f HandlerFunc) ServeGemini(ctx context.Context, w ResponseWriter, r *Request) {
|
||||||
f(ctx, w, r)
|
f(ctx, w, r)
|
||||||
}
|
}
|
||||||
|
|
||||||
// StatusHandler returns a request handler that responds to each request
|
// StatusHandler returns a request handler that responds to each request
|
||||||
// with the provided status code and meta.
|
// with the provided status code and meta.
|
||||||
func StatusHandler(status Status, meta string) Handler {
|
func StatusHandler(status Status, meta string) Handler {
|
||||||
return HandlerFunc(func(ctx context.Context, w *ResponseWriter, r *Request) {
|
return HandlerFunc(func(ctx context.Context, w ResponseWriter, r *Request) {
|
||||||
w.WriteHeader(status, meta)
|
w.WriteHeader(status, meta)
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
@ -58,7 +58,7 @@ func StripPrefix(prefix string, h Handler) Handler {
|
|||||||
if prefix == "" {
|
if prefix == "" {
|
||||||
return h
|
return h
|
||||||
}
|
}
|
||||||
return HandlerFunc(func(ctx context.Context, w *ResponseWriter, r *Request) {
|
return HandlerFunc(func(ctx context.Context, w ResponseWriter, r *Request) {
|
||||||
p := strings.TrimPrefix(r.URL.Path, prefix)
|
p := strings.TrimPrefix(r.URL.Path, prefix)
|
||||||
rp := strings.TrimPrefix(r.URL.RawPath, prefix)
|
rp := strings.TrimPrefix(r.URL.RawPath, prefix)
|
||||||
if len(p) < len(r.URL.Path) && (r.URL.RawPath == "" || len(rp) < len(r.URL.RawPath)) {
|
if len(p) < len(r.URL.Path) && (r.URL.RawPath == "" || len(rp) < len(r.URL.RawPath)) {
|
||||||
@ -92,7 +92,7 @@ type timeoutHandler struct {
|
|||||||
dt time.Duration
|
dt time.Duration
|
||||||
}
|
}
|
||||||
|
|
||||||
func (t *timeoutHandler) ServeGemini(ctx context.Context, w *ResponseWriter, r *Request) {
|
func (t *timeoutHandler) ServeGemini(ctx context.Context, w ResponseWriter, r *Request) {
|
||||||
ctx, cancel := context.WithTimeout(ctx, t.dt)
|
ctx, cancel := context.WithTimeout(ctx, t.dt)
|
||||||
defer cancel()
|
defer cancel()
|
||||||
|
|
||||||
|
2
mux.go
2
mux.go
@ -212,7 +212,7 @@ func (mux *ServeMux) Handler(r *Request) Handler {
|
|||||||
|
|
||||||
// ServeGemini dispatches the request to the handler whose
|
// ServeGemini dispatches the request to the handler whose
|
||||||
// pattern most closely matches the request URL.
|
// pattern most closely matches the request URL.
|
||||||
func (mux *ServeMux) ServeGemini(ctx context.Context, w *ResponseWriter, r *Request) {
|
func (mux *ServeMux) ServeGemini(ctx context.Context, w ResponseWriter, r *Request) {
|
||||||
h := mux.Handler(r)
|
h := mux.Handler(r)
|
||||||
h.ServeGemini(ctx, w, r)
|
h.ServeGemini(ctx, w, r)
|
||||||
}
|
}
|
||||||
|
@ -8,7 +8,7 @@ import (
|
|||||||
|
|
||||||
type nopHandler struct{}
|
type nopHandler struct{}
|
||||||
|
|
||||||
func (*nopHandler) ServeGemini(context.Context, *ResponseWriter, *Request) {}
|
func (*nopHandler) ServeGemini(context.Context, ResponseWriter, *Request) {}
|
||||||
|
|
||||||
func TestServeMuxMatch(t *testing.T) {
|
func TestServeMuxMatch(t *testing.T) {
|
||||||
type Match struct {
|
type Match struct {
|
||||||
|
129
response.go
129
response.go
@ -120,7 +120,70 @@ func (r *Response) TLS() *tls.ConnectionState {
|
|||||||
//
|
//
|
||||||
// A ResponseWriter may not be used after the Handler.ServeGemini method
|
// A ResponseWriter may not be used after the Handler.ServeGemini method
|
||||||
// has returned.
|
// has returned.
|
||||||
type ResponseWriter struct {
|
type ResponseWriter interface {
|
||||||
|
// SetMediaType sets the media type that will be sent by Write for a
|
||||||
|
// successful response. If no media type is set, a default of
|
||||||
|
// "text/gemini; charset=utf-8" will be used.
|
||||||
|
//
|
||||||
|
// Setting the media type after a call to Write or WriteHeader has
|
||||||
|
// no effect.
|
||||||
|
SetMediaType(mediatype string)
|
||||||
|
|
||||||
|
// Write writes the data to the connection as part of a Gemini response.
|
||||||
|
//
|
||||||
|
// If WriteHeader has not yet been called, Write calls WriteHeader with
|
||||||
|
// StatusSuccess and the media type set in SetMediaType before writing the data.
|
||||||
|
// If no media type was set, Write uses a default media type of
|
||||||
|
// "text/gemini; charset=utf-8".
|
||||||
|
Write([]byte) (int, error)
|
||||||
|
|
||||||
|
// WriteHeader sends a Gemini response header with the provided
|
||||||
|
// status code and meta.
|
||||||
|
//
|
||||||
|
// If WriteHeader is not called explicitly, the first call to Write
|
||||||
|
// will trigger an implicit call to WriteHeader with a successful
|
||||||
|
// status code and the media type set in SetMediaType.
|
||||||
|
//
|
||||||
|
// The provided code must be a valid Gemini status code.
|
||||||
|
// The provided meta must not be longer than 1024 bytes.
|
||||||
|
// Only one header may be written.
|
||||||
|
WriteHeader(status Status, meta string)
|
||||||
|
|
||||||
|
// Flush sends any buffered data to the client.
|
||||||
|
Flush() error
|
||||||
|
|
||||||
|
// Close closes the connection.
|
||||||
|
// Any blocked Write operations will be unblocked and return errors.
|
||||||
|
Close() error
|
||||||
|
|
||||||
|
// Conn returns the underlying network connection.
|
||||||
|
// To take over the connection, use Hijack.
|
||||||
|
Conn() net.Conn
|
||||||
|
|
||||||
|
// TLS returns information about the underlying TLS connection.
|
||||||
|
TLS() *tls.ConnectionState
|
||||||
|
|
||||||
|
// Hijack lets the caller take over the connection.
|
||||||
|
// After a call to Hijack the Gemini server library
|
||||||
|
// will not do anything else with the connection.
|
||||||
|
// It becomes the caller's responsibility to manage
|
||||||
|
// and close the connection.
|
||||||
|
//
|
||||||
|
// The returned net.Conn may have read or write deadlines
|
||||||
|
// already set, depending on the configuration of the
|
||||||
|
// Server. It is the caller's responsibility to set
|
||||||
|
// or clear those deadlines as needed.
|
||||||
|
Hijack() net.Conn
|
||||||
|
|
||||||
|
reset(io.WriteCloser)
|
||||||
|
|
||||||
|
// unexported method so we can extend this interface over time
|
||||||
|
// without breaking existing code. Implementers must embed a concrete
|
||||||
|
// type from elsewhere.
|
||||||
|
unexported()
|
||||||
|
}
|
||||||
|
|
||||||
|
type responseWriter struct {
|
||||||
bw *bufio.Writer
|
bw *bufio.Writer
|
||||||
cl io.Closer
|
cl io.Closer
|
||||||
mediatype string
|
mediatype string
|
||||||
@ -130,38 +193,26 @@ type ResponseWriter struct {
|
|||||||
conn net.Conn
|
conn net.Conn
|
||||||
}
|
}
|
||||||
|
|
||||||
func newResponseWriter(w io.WriteCloser) *ResponseWriter {
|
func newResponseWriter(w io.WriteCloser) *responseWriter {
|
||||||
return &ResponseWriter{
|
return &responseWriter{
|
||||||
bw: bufio.NewWriter(w),
|
bw: bufio.NewWriter(w),
|
||||||
cl: w,
|
cl: w,
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
func (w *ResponseWriter) reset(wc io.WriteCloser) {
|
func (w *responseWriter) reset(wc io.WriteCloser) {
|
||||||
w.bw.Reset(wc)
|
w.bw.Reset(wc)
|
||||||
*w = ResponseWriter{
|
*w = responseWriter{
|
||||||
bw: w.bw,
|
bw: w.bw,
|
||||||
cl: wc,
|
cl: wc,
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// SetMediaType sets the media type that will be sent by Write for a
|
func (w *responseWriter) SetMediaType(mediatype string) {
|
||||||
// successful response. If no media type is set, a default of
|
|
||||||
// "text/gemini; charset=utf-8" will be used.
|
|
||||||
//
|
|
||||||
// Setting the media type after a call to Write or WriteHeader has
|
|
||||||
// no effect.
|
|
||||||
func (w *ResponseWriter) SetMediaType(mediatype string) {
|
|
||||||
w.mediatype = mediatype
|
w.mediatype = mediatype
|
||||||
}
|
}
|
||||||
|
|
||||||
// Write writes the data to the connection as part of a Gemini response.
|
func (w *responseWriter) Write(b []byte) (int, error) {
|
||||||
//
|
|
||||||
// If WriteHeader has not yet been called, Write calls WriteHeader with
|
|
||||||
// StatusSuccess and the media type set in SetMediaType before writing the data.
|
|
||||||
// If no media type was set, Write uses a default media type of
|
|
||||||
// "text/gemini; charset=utf-8".
|
|
||||||
func (w *ResponseWriter) Write(b []byte) (int, error) {
|
|
||||||
if w.hijacked {
|
if w.hijacked {
|
||||||
return 0, ErrHijacked
|
return 0, ErrHijacked
|
||||||
}
|
}
|
||||||
@ -179,17 +230,7 @@ func (w *ResponseWriter) Write(b []byte) (int, error) {
|
|||||||
return w.bw.Write(b)
|
return w.bw.Write(b)
|
||||||
}
|
}
|
||||||
|
|
||||||
// WriteHeader sends a Gemini response header with the provided
|
func (w *responseWriter) WriteHeader(status Status, meta string) {
|
||||||
// status code and meta.
|
|
||||||
//
|
|
||||||
// If WriteHeader is not called explicitly, the first call to Write
|
|
||||||
// will trigger an implicit call to WriteHeader with a successful
|
|
||||||
// status code and the media type set in SetMediaType.
|
|
||||||
//
|
|
||||||
// The provided code must be a valid Gemini status code.
|
|
||||||
// The provided meta must not be longer than 1024 bytes.
|
|
||||||
// Only one header may be written.
|
|
||||||
func (w *ResponseWriter) WriteHeader(status Status, meta string) {
|
|
||||||
if w.hijacked {
|
if w.hijacked {
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
@ -208,8 +249,7 @@ func (w *ResponseWriter) WriteHeader(status Status, meta string) {
|
|||||||
w.wroteHeader = true
|
w.wroteHeader = true
|
||||||
}
|
}
|
||||||
|
|
||||||
// Flush sends any buffered data to the client.
|
func (w *responseWriter) Flush() error {
|
||||||
func (w *ResponseWriter) Flush() error {
|
|
||||||
if w.hijacked {
|
if w.hijacked {
|
||||||
return ErrHijacked
|
return ErrHijacked
|
||||||
}
|
}
|
||||||
@ -220,23 +260,18 @@ func (w *ResponseWriter) Flush() error {
|
|||||||
return w.bw.Flush()
|
return w.bw.Flush()
|
||||||
}
|
}
|
||||||
|
|
||||||
// Close closes the connection.
|
func (w *responseWriter) Close() error {
|
||||||
// Any blocked Write operations will be unblocked and return errors.
|
|
||||||
func (w *ResponseWriter) Close() error {
|
|
||||||
if w.hijacked {
|
if w.hijacked {
|
||||||
return ErrHijacked
|
return ErrHijacked
|
||||||
}
|
}
|
||||||
return w.cl.Close()
|
return w.cl.Close()
|
||||||
}
|
}
|
||||||
|
|
||||||
// Conn returns the underlying network connection.
|
func (w *responseWriter) Conn() net.Conn {
|
||||||
// To take over the connection, use Hijack.
|
|
||||||
func (w *ResponseWriter) Conn() net.Conn {
|
|
||||||
return w.conn
|
return w.conn
|
||||||
}
|
}
|
||||||
|
|
||||||
// TLS returns information about the underlying TLS connection.
|
func (w *responseWriter) TLS() *tls.ConnectionState {
|
||||||
func (w *ResponseWriter) TLS() *tls.ConnectionState {
|
|
||||||
if tlsConn, ok := w.conn.(*tls.Conn); ok {
|
if tlsConn, ok := w.conn.(*tls.Conn); ok {
|
||||||
state := tlsConn.ConnectionState()
|
state := tlsConn.ConnectionState()
|
||||||
return &state
|
return &state
|
||||||
@ -244,17 +279,9 @@ func (w *ResponseWriter) TLS() *tls.ConnectionState {
|
|||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
// Hijack lets the caller take over the connection.
|
func (w *responseWriter) Hijack() net.Conn {
|
||||||
// After a call to Hijack the Gemini server library
|
|
||||||
// will not do anything else with the connection.
|
|
||||||
// It becomes the caller's responsibility to manage
|
|
||||||
// and close the connection.
|
|
||||||
//
|
|
||||||
// The returned net.Conn may have read or write deadlines
|
|
||||||
// already set, depending on the configuration of the
|
|
||||||
// Server. It is the caller's responsibility to set
|
|
||||||
// or clear those deadlines as needed.
|
|
||||||
func (w *ResponseWriter) Hijack() net.Conn {
|
|
||||||
w.hijacked = true
|
w.hijacked = true
|
||||||
return w.conn
|
return w.conn
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func (w *responseWriter) unexported() {}
|
||||||
|
Loading…
Reference in New Issue
Block a user