При отправке файла в Google drive API из кода golang выдается ошибка: неподдерживаемый контент с типом: image / jpeg - PullRequest
0 голосов
/ 08 апреля 2019

на основе Google Drive API документы правильный способ загрузки файла:

curl -v -H 'Authorization: Bearer mytoken' -F 'metadata={"name": "test3.jpeg"};type=application/json' -F file=@jpeg_image.jpeg 'https://www.googleapis.com/upload/drive/v3/files?uploadType=multipart'

Теперь мне нужно выполнить тот же запрос из кода golang, но мне трудно преобразовать его в golang, вот код, который я использую после нескольких попыток:

// fileBytes are of type []byte
buffer := &bytes.Buffer{}
    multipartWriter := multipart.NewWriter(buffer)
    multipartWriter.WriteField("metadata", "{\"name\": \"test3.jpef\"};type=application/json")
    partHeader := textproto.MIMEHeader{}
    partHeader.Add("Content-Type", "image/jpeg")
    filePartWriter, _ := multipartWriter.CreatePart(partHeader)
    io.Copy(filePartWriter, bytes.NewReader(fileBytes))
    multipartWriter.Close()

    googleDriveRequest, _ := http.NewRequest(http.MethodPost, "https://www.googleapis.com/upload/drive/v3/files?uploadType=multipart", buffer)
    googleDriveRequest.Header.Add("Authorization", "Bearer "+accessToken)
    googleDriveRequest.Header.Add("Content-Type", multipartWriter.FormDataContentType())

    googleDriveAPIResponse, googleDriveAPIError := lib.HttpClient.Do(googleDriveRequest)

С помощью curl запрос успешно выполняется:

{
 "kind": "drive#file",
 "id": "1DwsbQGP3-QtS5p0iQqRtAjcCCsAfxzGD",
 "name": "test3.jpeg",
 "mimeType": "image/jpeg"
}

С Голангом я получаю 400:

{\n \"error\": {\n  \"errors\": [\n   {\n    \"domain\": \"global\",\n    \"reason\": \"badContent\",\n    \"message\": \"Unsupported content with type: image/jpeg \"\n   }\n  ],\n  \"code\": 400,\n  \"message\": \"Unsupported content with type: image/jpeg\"\n }\n}\n

Я также пытался с другими значениями для partHeader.Add("Content-Type", "image/jpeg"), я также пытался с application/x-www-form-urlencoded, я также закомментировал эту строку и позволил golang выполнить обнаружение, но все равно получил ту же ошибку.

Есть предложения или идеи?

С уважением,

1 Ответ

1 голос
/ 08 апреля 2019

Проблема в отправляемых вами метаданных JSON. Вот пример кода (используйте правильную проверку ошибок), который будет работать,

import (
    "bytes"
    "fmt"
    "io"
    "io/ioutil"
    "log"
    "mime/multipart"
    "net/http"
    "net/textproto"
)

func main() {
    accessToken := "YOUR-TOKEN"

    client := http.Client{}
    mediaData, _ := ioutil.ReadFile("test.png")

    body := &bytes.Buffer{}
    writer := multipart.NewWriter(body)

    // JSON Metadata (part-1)
    jsonMetadata := textproto.MIMEHeader{}
    metadata := `{"name": "test.png"}`
    jsonMetadata.Set("Content-Type", "application/json")
    part, _ := writer.CreatePart(jsonMetadata)
    part.Write([]byte(metadata))

    // Image bytes (part-2)
    imageData := textproto.MIMEHeader{}
    partAttach, _ := writer.CreatePart(imageData)
    io.Copy(partAttach, bytes.NewReader(mediaData))
    writer.Close()

    // Request Content Type with boundary
    contentType := fmt.Sprintf("multipart/related; boundary=%s", writer.Boundary())

    // HTTP Request with auth and content type headers
    req, _ := http.NewRequest(http.MethodPost, "https://www.googleapis.com/upload/drive/v3/files?uploadType=multipart", bytes.NewReader(body.Bytes()))
    req.Header.Add("Authorization", "Bearer "+accessToken)
    req.Header.Add("Content-Type", contentType)

    // Send request
    resp, err := client.Do(req)
    if err != nil {
        log.Fatalf("failed to send request: %v", err)
    }
    defer resp.Body.Close()
    content, _ := ioutil.ReadAll(resp.Body)

    log.Printf("http status: %d", resp.StatusCode)
    log.Printf("response: %s", string(content))
}

Ссылка: https://developers.google.com/drive/api/v3/manage-uploads?authuser=1#send_a_multipart_upload_request

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