使用curl使用GoLang发送Multipart请求

时间:2018-01-06 11:37:15

标签: curl go multipartform-data

我想将下面的代码转换为golang。

curl -X POST
  'https://api.documo.com/v1/fax/send' \
  -H 'Authorization: Basic YOUR_API_KEY' \
  -H 'content-type: multipart/form-data' \
  -F 'recipientFax=12345678900' \
  -F 'coverPage=false' \
  -F 'recipientName=John' \
  -F 'subject=test' \
  -F 'notes=test' \
  -F '=@/home/user/Documents/Sample.pdf'

请帮忙

以下是我试过的代码。

func SendFaxDocumo(files []*multipart.FileHeader, params map[string]string) {

    request, err := SendMultipartRequest("https://api.documo.com/v1/fax/send", params, files)
    if err != nil {
        fmt.Println(err.Error())

    }
    client := &http.Client{}
    resp, err := client.Do(request)
    if err != nil {
        fmt.Println(err.Error())
    } else {
        body := &bytes.Buffer{}
        _, err := body.ReadFrom(resp.Body)
        if err != nil {
            fmt.Println(err.Error())
        }
        resp.Body.Close()
        fmt.Println(resp.StatusCode)
        fmt.Println(resp.Header)
        fmt.Println(body)
    }
}

func SendMultipartRequest(uri string, params map[string]string, files []*multipart.FileHeader) (*http.Request, error) {

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

    for i, _ := range files {
        // string_index := strconv.Itoa(i)
        file, err := files[i].Open()
        defer file.Close()
        if err != nil {
            fmt.Println(err)
        }

        doc , err := ioutil.ReadAll(file)
        if err != nil {
            fmt.Println(err,980)
        }

        part, err := writer.CreateFormFile("file",files[i].Filename)

        if err != nil {
            fmt.Println(err)
        }

        part.Write([]byte(string(doc)))
        if _, err = io.Copy(part, file); err != nil {
            fmt.Println(err)
        }

        if err != nil {
            fmt.Println(err)
        }
    }

    for key, val := range params {
        _ = writer.WriteField(key, val)
    }

    err := writer.Close()
    if err != nil {
        return nil, err
    }

    req, err := http.NewRequest("POST", uri, body)
    req.Header.Set("Content-Type", writer.FormDataContentType())
    req.Header.Set("Authorization", "Basic my_key")

    return req, err
}

我得到的错误如下: {"错误":{"名称":"错误","消息":"不允许文件扩展名。允许的格式:tiff,gif,png,jpeg,pdf,doc,xls,ppt,pcl,eps,ps,txt,rtf,xml,fxc,docx,pptx,xlsx"}}

即使我是通过pdf尝试,这是从请求的网址获得的响应。

1 个答案:

答案 0 :(得分:1)

我花了很长时间研究相同的问题。使用httputil.DumpRequest查看我实际传递给Documo的数据后,我看到文件字段的content-type标头设置为Content-Type: application/octet-stream。这是由您使用的CreateFormFile方法设置的。有关源代码,请参见multipart pkg

octet-stream子类型用作二进制文件或需要在应用程序中打开的文件的默认设置。参见MDNIANAthis awesome explanation

据我所知,问题是 Documo希望我们明确告诉他们我们将向他们发送哪种文件,以便他们知道如何处理渲染。因此,对于我来说,octet-stream没有告诉他们任何事情。将文件内容类型切换为application/pdf就可以了。

解决方案

我基本上没有复制writer.CreateFormField,而是从CreateFormFile的源代码复制了三行,并将Content-Type硬编码为标头到application/pdf而不是{{1 }}(注意:我的服务目前仅处理PDF,因此如果文件类型不同,则可能需要进行调整)。

代码:

application/octet-stream

这不是最性感的解决方案,但我现在终于可以通过Documo发送完整的文档。

希望这会有所帮助。