Move TimeoutHandler to handler.go
This commit is contained in:
parent
310bd16344
commit
f08efa330f
44
handler.go
44
handler.go
@ -1,9 +1,11 @@
|
|||||||
package gemini
|
package gemini
|
||||||
|
|
||||||
import (
|
import (
|
||||||
|
"bytes"
|
||||||
"context"
|
"context"
|
||||||
"net/url"
|
"net/url"
|
||||||
"strings"
|
"strings"
|
||||||
|
"time"
|
||||||
)
|
)
|
||||||
|
|
||||||
// A Handler responds to a Gemini request.
|
// A Handler responds to a Gemini request.
|
||||||
@ -72,3 +74,45 @@ func StripPrefix(prefix string, h Handler) Handler {
|
|||||||
}
|
}
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// TimeoutHandler returns a Handler that runs h with the given time limit.
|
||||||
|
//
|
||||||
|
// The new Handler calls h.ServeGemini to handle each request, but
|
||||||
|
// if a call runs for longer than its time limit, the handler responds with a
|
||||||
|
// 40 Temporary Failure error. After such a timeout, writes by h to its
|
||||||
|
// ResponseWriter will return ErrHandlerTimeout.
|
||||||
|
func TimeoutHandler(h Handler, dt time.Duration) Handler {
|
||||||
|
return &timeoutHandler{
|
||||||
|
h: h,
|
||||||
|
dt: dt,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
type timeoutHandler struct {
|
||||||
|
h Handler
|
||||||
|
dt time.Duration
|
||||||
|
}
|
||||||
|
|
||||||
|
func (t *timeoutHandler) ServeGemini(ctx context.Context, w *ResponseWriter, r *Request) {
|
||||||
|
ctx, cancel := context.WithTimeout(ctx, t.dt)
|
||||||
|
defer cancel()
|
||||||
|
|
||||||
|
conn := w.Hijack()
|
||||||
|
|
||||||
|
var b bytes.Buffer
|
||||||
|
w.reset(nopCloser{&b})
|
||||||
|
|
||||||
|
done := make(chan struct{})
|
||||||
|
go func() {
|
||||||
|
t.h.ServeGemini(ctx, w, r)
|
||||||
|
close(done)
|
||||||
|
}()
|
||||||
|
|
||||||
|
select {
|
||||||
|
case <-done:
|
||||||
|
conn.Write(b.Bytes())
|
||||||
|
case <-ctx.Done():
|
||||||
|
w.reset(conn)
|
||||||
|
w.WriteHeader(StatusTemporaryFailure, "Timeout")
|
||||||
|
}
|
||||||
|
}
|
||||||
|
49
timeout.go
49
timeout.go
@ -1,49 +0,0 @@
|
|||||||
package gemini
|
|
||||||
|
|
||||||
import (
|
|
||||||
"bytes"
|
|
||||||
"context"
|
|
||||||
"time"
|
|
||||||
)
|
|
||||||
|
|
||||||
// TimeoutHandler returns a Handler that runs h with the given time limit.
|
|
||||||
//
|
|
||||||
// The new Handler calls h.ServeGemini to handle each request, but
|
|
||||||
// if a call runs for longer than its time limit, the handler responds with a
|
|
||||||
// 40 Temporary Failure error. After such a timeout, writes by h to its
|
|
||||||
// ResponseWriter will return ErrHandlerTimeout.
|
|
||||||
func TimeoutHandler(h Handler, dt time.Duration) Handler {
|
|
||||||
return &timeoutHandler{
|
|
||||||
h: h,
|
|
||||||
dt: dt,
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
type timeoutHandler struct {
|
|
||||||
h Handler
|
|
||||||
dt time.Duration
|
|
||||||
}
|
|
||||||
|
|
||||||
func (t *timeoutHandler) ServeGemini(ctx context.Context, w *ResponseWriter, r *Request) {
|
|
||||||
ctx, cancel := context.WithTimeout(ctx, t.dt)
|
|
||||||
defer cancel()
|
|
||||||
|
|
||||||
conn := w.Hijack()
|
|
||||||
|
|
||||||
var b bytes.Buffer
|
|
||||||
w.reset(nopCloser{&b})
|
|
||||||
|
|
||||||
done := make(chan struct{})
|
|
||||||
go func() {
|
|
||||||
t.h.ServeGemini(ctx, w, r)
|
|
||||||
close(done)
|
|
||||||
}()
|
|
||||||
|
|
||||||
select {
|
|
||||||
case <-done:
|
|
||||||
conn.Write(b.Bytes())
|
|
||||||
case <-ctx.Done():
|
|
||||||
w.reset(conn)
|
|
||||||
w.WriteHeader(StatusTemporaryFailure, "Timeout")
|
|
||||||
}
|
|
||||||
}
|
|
Loading…
Reference in New Issue
Block a user