假设我正在编写一个http处理程序,在返回响应之前执行其他操作,我是否必须设置一个侦听器来检查http请求上下文是否已被取消?以便它可以立即返回,或者是否有其他方法在请求上下文取消时退出处理程序?
func handleSomething(w http.ResponseWriter, r *http.Request) {
done := make(chan error)
go func() {
if err := doSomething(r.Context()); err != nil {
done <- err
return
}
done <- nil
}()
select {
case <-r.Context().Done():
http.Error(w, r.Context().Err().Error(), http.StatusInternalServerError)
return
case err := <-done:
if err != nil {
http.Error(w, err.Error(), http.StatusInternalServerError)
return
}
w.WriteHeader(http.StatusOK)
w.Write([]byte("ok"))
}
}
func doSomething(ctx context.Context) error {
// simulate doing something for 1 second.
time.Sleep(time.Second)
return nil
}
我试着对它进行测试,但是在上下文被取消后,doSomething
函数没有停止,仍然在后台运行。
func TestHandler(t *testing.T) {
mux := http.NewServeMux()
mux.HandleFunc("/something", handleSomething)
srv := http.Server{
Addr: ":8989",
Handler: mux,
}
var wg sync.WaitGroup
wg.Add(1)
go func() {
defer wg.Done()
if err := srv.ListenAndServe(); err != nil {
log.Println(err)
}
}()
time.Sleep(time.Second)
req, err := http.NewRequest(http.MethodGet, "http://localhost:8989/something", nil)
if err != nil {
t.Fatal(err)
}
cl := http.Client{
Timeout: 3 * time.Second,
}
res, err := cl.Do(req)
if err != nil {
t.Logf("error: %s", err.Error())
} else {
t.Logf("request is done with status code %d", res.StatusCode)
}
go func() {
<-time.After(10 * time.Second)
shutdown, cancel := context.WithTimeout(context.Background(), 10*time.Second)
defer cancel()
srv.Shutdown(shutdown)
}()
wg.Wait()
}
func handleSomething(w http.ResponseWriter, r *http.Request) {
done := make(chan error)
go func() {
if err := doSomething(r.Context()); err != nil {
log.Println(err)
done <- err
}
done <- nil
}()
select {
case <-r.Context().Done():
log.Println("context is done!")
return
case err := <-done:
if err != nil {
http.Error(w, err.Error(), http.StatusInternalServerError)
return
}
w.WriteHeader(http.StatusOK)
w.Write([]byte("ok"))
}
}
func doSomething(ctx context.Context) error {
return runInContext(ctx, func() {
log.Println("doing something")
defer log.Println("done doing something")
time.Sleep(10 * time.Second)
})
}
func runInContext(ctx context.Context, fn func()) error {
ch := make(chan struct{})
go func() {
defer close(ch)
fn()
}()
select {
case <-ctx.Done():
return ctx.Err()
case <-ch:
return nil
}
}
1条答案
按热度按时间b1payxdu1#
我刚刚对提供的解决方案进行了一点重构,现在它应该可以工作了。
doSomething
函数它等待取消输入,或者在延迟
3
秒后返回调用方。它接受要侦听的上下文。handleSomething
函数这里的逻辑与您的非常相似。在select中,我们检查接收到的错误是否为
nil
,并基于此向调用者返回正确的HTTP状态代码。如果我们接收到取消输入,则取消所有上下文链。TestHandler
函数这里,我们启动一个HTTP服务器,并通过
http.Client
向它发出一个HTTP请求。您可以看到有两个语句用于设置上下文超时。如果您使用带有注解// request canceled
的语句,则所有操作都将被取消,否则,如果您使用另一个语句,则请求将被处理。我希望这澄清了你的问题!