Как получить HTTP-ответ для потоковой передачи - PullRequest
0 голосов
/ 29 августа 2018

При отправке HTTP-запроса с Go и получении ответа я хочу получить ответ во время потоковой передачи, учитывая случай, когда ResponseBody огромен (1 ГБ или более).

resp, err: = http.Client.Do(req)

В этом случае, если тело огромно, я не могу прочитать Заголовок и не знаю состояние ответа. Есть ли решение?

1 Ответ

0 голосов
/ 29 августа 2018

(Изменить: если вы не можете получить заголовок «Content-length» из ответа, возможно, веб-служба, к которой вы обращаетесь, не возвращает этот заголовок. В таком случае, нет никакого способа узнать длину тела ответа, не прочитав его полностью. Это можно смоделировать в следующем примере, удалив строку, которая устанавливает заголовок Content-length в ответе.)

Стандартный пакет Go net/http отлично справляется с большими ответами. Вот отдельный пример для демонстрации:

// Start a mock HTTP server that returns 2GB of data in the response. Make a
// HTTP request to this server and print the amount of data read from the
// response.
package main

import (
    "fmt"
    "io"
    "log"
    "net/http"
    "strings"
    "time"
)

const oneMB = 1024 * 1024
const oneGB = 1024 * oneMB
const responseSize = 2 * oneGB

const serverAddr = "localhost:9999"

func startServer() {
    // Mock HTTP server that always returns 2GB of data
    go http.ListenAndServe(serverAddr, http.HandlerFunc(func(w http.ResponseWriter, req *http.Request) {
        w.Header().Set("Content-length", fmt.Sprintf("%d", responseSize))

        // 1MB buffer that'll be copied multiple times to the response
        buf := []byte(strings.Repeat("x", oneMB))

        for i := 0; i < responseSize/len(buf); i++ {
            if _, err := w.Write(buf); err != nil {
                log.Fatal("Failed to write to response. Error: ", err.Error())
            }
        }
    }))

    // Some grace period for the server to start
    time.Sleep(100 * time.Millisecond)
}

func main() {
    startServer()

    // HTTP client
    req, err := http.NewRequest("GET", "http://"+serverAddr, nil)
    if err != nil {
        log.Fatal("Error creating HTTP request: ", err.Error())
    }

    client := http.Client{}
    resp, err := client.Do(req)
    if err != nil {
        log.Fatal("Error making HTTP request: ", err.Error())
    }

    // Read the response header
    fmt.Println("Response: Content-length:", resp.Header.Get("Content-length"))

    bytesRead := 0
    buf := make([]byte, oneMB)

    // Read the response body
    for {
        n, err := resp.Body.Read(buf)
        bytesRead += n

        if err == io.EOF {
            break
        }

        if err != nil {
            log.Fatal("Error reading HTTP response: ", err.Error())
        }
    }

    fmt.Println("Response: Read", bytesRead, "bytes")
}

Вы не захотите читать весь ответ в памяти, если он слишком большой. Вместо этого запишите его во временный файл и затем обработайте его.

Если вместо этого вы ищете варианты, позволяющие сделать это надежно, когда сеть не очень надежна, найдите «запросы диапазона HTTP», с помощью которых можно возобновить частично загруженные данные.

Добро пожаловать на сайт PullRequest, где вы можете задавать вопросы и получать ответы от других членов сообщества.
...