Golang HTTP服务器等待数据发送到客户端

时间:2017-06-21 13:00:08

标签: go

我正在创建类似于Twitter firehose / streaming API的流式传输API。

据我所知,这是基于保持打开的HTTP连接,当后端获取数据时,它会写入chucked HTTP连接。似乎我编写的任何代码都会在连接时立即关闭HTTP连接。

有没有办法让这个开放?

func startHTTP(pathPrefix string) {
    log.Println("Starting HTTPS Server")
    http.HandleFunc("/", func(w http.ResponseWriter, r *http.Request) {
        // Wait here until a write happens to w
        // Or we timeout, we can reset this timeout after each write
    })

    log.Print("HTTPS listening on :5556")
    log.Fatal(http.ListenAndServeTLS(":5556", pathPrefix+".crt", pathPrefix+".key", nil))
}

1 个答案:

答案 0 :(得分:4)

当您想要立即向客户端发送HTTP响应但在某些事件之后,它被称为long polling

以下是在客户端断开连接时请求取消的长轮询的简单示例:

package main

import (
    "context"
    "fmt"
    "net/http"
    "time"
)

func longOperation(ctx context.Context, ch chan<- string) {
    // Simulate long operation.
    // Change it to more than 10 seconds to get server timeout.
    select {
    case <-time.After(time.Second * 3):
        ch <- "Successful result."
    case <-ctx.Done():
        close(ch)
    }
}

func handler(w http.ResponseWriter, _ *http.Request) {
    notifier, ok := w.(http.CloseNotifier)
    if !ok {
        panic("Expected http.ResponseWriter to be an http.CloseNotifier")
    }

    ctx, cancel := context.WithCancel(context.Background())
    ch := make(chan string)
    go longOperation(ctx, ch)

    select {
    case result := <-ch:
        fmt.Fprint(w, result)
        cancel()
        return
    case <-time.After(time.Second * 10):
        fmt.Fprint(w, "Server is busy.")
    case <-notifier.CloseNotify():
        fmt.Println("Client has disconnected.")
    }
    cancel()
    <-ch
}

func main() {
    http.HandleFunc("/", handler)
    http.ListenAndServe("localhost:8080", nil)
}

网址:

  1. Golang : anonymous struct and empty struct
  2. Send a chunked HTTP response from a Go server
  3. Go Concurrency Patterns: Context
  4. 要旨:

    1. Golang long polling example
    2. Golang long polling example with request cancellation