dsgdhtr_43654 2017-11-10 08:21 采纳率: 100%
浏览 44

如何基于请求路径创建自定义超时处理程序?

I wrote a timeout middleware. In this timeout middleware i created a goroutine for processing next handlers but after two seconds when timeout happens the response returns with 408 timeout status code but the goroutine runs in background and executes next handlers that are in handlers chain. So how to write a correct timeout middleware?

Timeout Middleware:

type TimeoutHandler struct {
    Next http.Handler
}

func (handler *TimeoutHandler) ServeHTTP(responseWriter http.ResponseWriter, request *http.Request) {
     /*if nil 'Next' handler.*/
     if handler.Next == nil {
         /*delegate to default serve mux.*/
         handler.Next = http.DefaultServeMux
     }

     /*context with timeout.*/
     requestContext := request.Context()
     timeoutContext, cancelFunction := context.WithTimeout(requestContext, 2*time.Second)
     defer cancelFunction()

     /*request with new context.*/
     request = request.WithContext(timeoutContext)
     chanDone := make(chan bool)

     go func() {
          /*delegate request to `Next` handler*/
          handler.Next.ServeHTTP(responseWriter, request)
          chanDone <- true
     }()

     select {
     case <-timeoutContext.Done():
          /*status 408 request timeout.*/
          responseWriter.WriteHeader(http.StatusRequestTimeout)
     case <-chanDone:
          return
    }
}

These are the problems that i have:

  1. After timeout, next handler runs no matter what.
  2. If i use more handlers after timeout middleware, there is warning that says "http: multiple response.WriteHeader calls."
  • 写回答

1条回答 默认 最新

  • douyunjiaok300404 2017-11-11 14:27
    关注

    I found better solution in a forum and i hope this solution helps other go newcomers.

    /**
    *
    */
    type TimeoutHandler struct {
        Next http.Handler
    }
    
    /**
     *
     */
    func (h *TimeoutHandler) ServeHTTP(w http.ResponseWriter, r *http.Request) {
        /*nil 'Next' handler.*/
        if h.Next == nil {
            /*delegate to default serve mux.*/
            h.Next = http.DefaultServeMux
        }
    
        /*delegate request to new timeout handler.*/
        timeoutHandler := http.TimeoutHandler(h.Next, 4 * time.Second, `Request Timeout.`)
        timeoutHandler.ServeHTTP(w, r)
    }
    
    评论

报告相同问题?

悬赏问题

  • ¥20 我想使用一些网络协议或者部分协议也行,主要想实现类似于traceroute的一定步长内的路由拓扑功能
  • ¥30 深度学习,前后端连接
  • ¥15 孟德尔随机化结果不一致
  • ¥15 apm2.8飞控罗盘bad health,加速度计校准失败
  • ¥15 求解O-S方程的特征值问题给出边界层布拉休斯平行流的中性曲线
  • ¥15 谁有desed数据集呀
  • ¥20 手写数字识别运行c仿真时,程序报错错误代码sim211-100
  • ¥15 关于#hadoop#的问题
  • ¥15 (标签-Python|关键词-socket)
  • ¥15 keil里为什么main.c定义的函数在it.c调用不了