当前位置:首页 > 文章列表 > Golang > Go问答 > Golang 多部分文件表单请求

Golang 多部分文件表单请求

来源:stackoverflow 2024-03-13 09:21:27 0浏览 收藏

从现在开始,努力学习吧!本文《Golang 多部分文件表单请求》主要讲解了等等相关知识点,我会在golang学习网中持续更新相关的系列文章,欢迎大家关注并积极留言建议。下面就先一起来看一下本篇正文内容吧,希望能帮到你!

问题内容

我正在针对 mapbox 编写一个 api 客户端,将一批 svg 图像上传到自定义地图。他们为此提供的 api 已记录在一个可以正常工作的 curl 调用示例中:

curl -f images=@include/mapbox/sprites_dark/aubergine_selected.svg "https://api.mapbox.com/styles/v1///sprite?access_token=$mapbox_api_key" --trace-ascii /开发/stdout

当尝试从 golang 执行相同操作时,我很快发现多形式库非常有限,并编写了一些代码来发出类似于上面提到的 curl 请求的请求。

func createmultipartformdata(filemap map[string]string) (bytes.buffer, *multipart.writer) {
    var b bytes.buffer
    var err error
    w := multipart.newwriter(&b)
    var fw io.writer
    for filename, filepath := range filemap {

        h := make(textproto.mimeheader)
        h.set("content-disposition",
            fmt.sprintf(`form-data; name="%s"; filename="%s"`, "images", filename))
        h.set("content-type", "image/svg+xml")

        if fw, err = w.createpart(h); err != nil {
            fmt.printf("error creating form file %v, %v", filename, err)
            continue
        }

        filecontents, err := ioutil.readfile(filepath)
        filecontents = bytes.replaceall(filecontents, []byte("\n"), []byte("."))

        blocksize := 64
        remainder := len(filecontents) % blocksize
        iterations := (len(filecontents) - remainder) / blocksize

        newbytes := []byte{}
        for i := 0; i < iterations; i++ {
            start := i * blocksize
            end := i*blocksize + blocksize
            newbytes = append(newbytes, filecontents[start:end]...)
            newbytes = append(newbytes, []byte("\n")...)
        }

        if remainder > 0 {
            newbytes = append(newbytes, filecontents[iterations*blocksize:]...)
            newbytes = append(newbytes, []byte("\n")...)
        }

        if err != nil {
            fmt.printf("error reading svg file: %v: %v", filepath, err)
            continue
        }

        _, err = fw.write(newbytes)

        if err != nil {
            log.debugf("could not write file to multipart: %v, %v", filename, err)
            continue
        }
    }

    w.close()

    return b, w
}

在实际请求中设置标头:

    bytes, formWriter := createMultipartFormData(filesMap)

    req, err := http.NewRequest("Post", fmt.Sprintf("https://api.mapbox.com/styles/v1/%v/%v/sprite?access_token=%v", "my_company", styleID, os.Getenv("MAPBOX_API_KEY")), &bytes)

    if err != nil {
        return err
    }

    req.Header.Set("User-Agent", "curl/7.64.1")
    req.Header.Set("Accept", "*/*")
    req.Header.Set("Content-Length", fmt.Sprintf("%v", len(bytes.Bytes())))
    req.Header.Set("Content-Type", formWriter.FormDataContentType())

    byts, _ := httputil.DumpRequest(req, true)
    fmt.Println(string(byts))

    res, err := http.DefaultClient.Do(req)

甚至想要限制行长度并复制 curl 使用的编码,但到目前为止还没有成功。有经验的人知道为什么这在 curl 中有效但在 golang 中无效吗?


解决方案


嗯,我承认解决你的任务的“谜题”的所有部分都可以在网上找到,这有两个问题:

  • 他们经常错过某些有趣的细节。
  • 有时,他们会给出完全错误的建议。

所以,这是一个可行的解决方案。

package main

import (
    "bytes"
    "fmt"
    "io"
    "io/ioutil"
    "mime"
    "mime/multipart"
    "net/http"
    "net/textproto"
    "net/url"
    "os"
    "path/filepath"
    "strconv"
    "strings"
)

func main() {
    const (
        dst   = "https://api.mapbox.com/styles/v1/acmeinc/style_001/sprite"
        fname = "path/to/a/sprite/image.svg"
        token = "an_invalid_token"
    )

    err := post(dst, fname, token)
    if err != nil {
        fmt.fprintln(os.stderr, err)
        os.exit(1)
    }
}

func post(dst, fname, token string) error {
    u, err := url.parse(dst)
    if err != nil {
        return fmt.errorf("failed to parse destination url: %w", err)
    }

    form, err := makerequestbody(fname)
    if err != nil {
        return fmt.errorf("failed to prepare request body: %w", err)
    }

    q := u.query()
    q.set("access_token", token)
    u.rawquery = q.encode()

    hdr := make(http.header)
    hdr.set("content-type", form.contenttype)
    req := http.request{
        method:        "post",
        url:           u,
        header:        hdr,
        body:          ioutil.nopcloser(form.body),
        contentlength: int64(form.contentlen),
    }

    resp, err := http.defaultclient.do(&req)
    if err != nil {
        return fmt.errorf("failed to perform http request: %w", err)
    }
    defer resp.body.close()

    _, _ = io.copy(os.stdout, resp.body)

    return nil
}

type form struct {
    body        *bytes.buffer
    contenttype string
    contentlen  int
}

func makerequestbody(fname string) (form, error) {
    ct, err := getimagecontenttype(fname)
    if err != nil {
        return form{}, fmt.errorf(
            `failed to get content type for image file "%s": %w`,
            fname, err)
    }

    fd, err := os.open(fname)
    if err != nil {
        return form{}, fmt.errorf("failed to open file to upload: %w", err)
    }
    defer fd.close()

    stat, err := fd.stat()
    if err != nil {
        return form{}, fmt.errorf("failed to query file info: %w", err)
    }

    hdr := make(textproto.mimeheader)
    cd := mime.formatmediatype("form-data", map[string]string{
        "name":     "images",
        "filename": fname,
    })
    hdr.set("content-disposition", cd)
    hdr.set("contnt-type", ct)
    hdr.set("content-length", strconv.formatint(stat.size(), 10))

    var buf bytes.buffer
    mw := multipart.newwriter(&buf)

    part, err := mw.createpart(hdr)
    if err != nil {
        return form{}, fmt.errorf("failed to create new form part: %w", err)
    }

    n, err := io.copy(part, fd)
    if err != nil {
        return form{}, fmt.errorf("failed to write form part: %w", err)
    }

    if int64(n) != stat.size() {
        return form{}, fmt.errorf("file size changed while writing: %s", fd.name())
    }

    err = mw.close()
    if err != nil {
        return form{}, fmt.errorf("failed to prepare form: %w", err)
    }

    return form{
        body:        &buf,
        contenttype: mw.formdatacontenttype(),
        contentlen:  buf.len(),
    }, nil
}

var imagecontenttypes = map[string]string{
    "png":  "image/png",
    "jpg":  "image/jpeg",
    "jpeg": "image/jpeg",
    "svg":  "image/svg+xml",
}

func getimagecontenttype(fname string) (string, error) {
    ext := filepath.ext(fname)
    if ext == "" {
        return "", fmt.errorf("file name has no extension: %s", fname)
    }

    ext = strings.tolower(ext[1:])
    ct, found := imagecontenttypes[ext]
    if !found {
        return "", fmt.errorf("unknown file name extension: %s", ext)
    }

    return ct, nil
}

一些关于实现的随机注释可以帮助您理解概念:

  • 为了构造请求的有效负载(主体),我们使用 bytes.buffer 实例。
    它有一个很好的属性,即指向它的指针 (*bytes.buffer) 实现了 io.writerio.reader,因此可以轻松地与处理 i/o 的 go stdlib 的其他部分组合。
  • 在准备发送多部分表单时,我们不会将整个文件的内容放入内存中,而是将它们直接“管道”到“多部分表单编写器”中。
  • 我们有一个查找表,它将要提交的文件名扩展名映射到其 mime 类型;我不知道 api 是否需要这个;如果不是真的需要,准备包含文件的表单字段的代码部分可以简化很多,但 curl 会发送它,我们也是如此。

只是好奇,这是做什么的?

filecontents = bytes.replaceall(filecontents, []byte("\n"), []byte("."))

        blocksize := 64
        remainder := len(filecontents) % blocksize
        iterations := (len(filecontents) - remainder) / blocksize

        newbytes := []byte{}
        for i := 0; i < iterations; i++ {
            start := i * blocksize
            end := i*blocksize + blocksize
            newbytes = append(newbytes, filecontents[start:end]...)
            newbytes = append(newbytes, []byte("\n")...)
        }

        if remainder > 0 {
            newbytes = append(newbytes, filecontents[iterations*blocksize:]...)
            newbytes = append(newbytes, []byte("\n")...)
        }

        if err != nil {
            fmt.printf("error reading svg file: %v: %v", filepath, err)
            continue
        }

将整个文件读入内存很少是一个好主意(ioutil.readfile)。

正如@muffin-top所说,这三行代码怎么样?

for fileName, filePath := range fileMap {

        // h := ...

        fw, _ := w.CreatePart(h) // TODO: handle error

        f, _ := os.Open(filePath) // TODO: handle error

        io.Copy(fw, f) // TODO: handle error

        f.Close() // TODO: handle error
    }

终于介绍完啦!小伙伴们,这篇关于《Golang 多部分文件表单请求》的介绍应该让你收获多多了吧!欢迎大家收藏或分享给更多需要学习的朋友吧~golang学习网公众号也会发布Golang相关知识,快来关注吧!

版本声明
本文转载于:stackoverflow 如有侵犯,请联系study_golang@163.com删除
正确使用dedecms的内容删除功能步骤提示正确使用dedecms的内容删除功能步骤提示
上一篇
正确使用dedecms的内容删除功能步骤提示
详解Discuz系统的介绍和功能
下一篇
详解Discuz系统的介绍和功能
查看更多
最新文章
查看更多
课程推荐
  • 前端进阶之JavaScript设计模式
    前端进阶之JavaScript设计模式
    设计模式是开发人员在软件开发过程中面临一般问题时的解决方案,代表了最佳的实践。本课程的主打内容包括JS常见设计模式以及具体应用场景,打造一站式知识长龙服务,适合有JS基础的同学学习。
    542次学习
  • GO语言核心编程课程
    GO语言核心编程课程
    本课程采用真实案例,全面具体可落地,从理论到实践,一步一步将GO核心编程技术、编程思想、底层实现融会贯通,使学习者贴近时代脉搏,做IT互联网时代的弄潮儿。
    508次学习
  • 简单聊聊mysql8与网络通信
    简单聊聊mysql8与网络通信
    如有问题加微信:Le-studyg;在课程中,我们将首先介绍MySQL8的新特性,包括性能优化、安全增强、新数据类型等,帮助学生快速熟悉MySQL8的最新功能。接着,我们将深入解析MySQL的网络通信机制,包括协议、连接管理、数据传输等,让
    497次学习
  • JavaScript正则表达式基础与实战
    JavaScript正则表达式基础与实战
    在任何一门编程语言中,正则表达式,都是一项重要的知识,它提供了高效的字符串匹配与捕获机制,可以极大的简化程序设计。
    487次学习
  • 从零制作响应式网站—Grid布局
    从零制作响应式网站—Grid布局
    本系列教程将展示从零制作一个假想的网络科技公司官网,分为导航,轮播,关于我们,成功案例,服务流程,团队介绍,数据部分,公司动态,底部信息等内容区块。网站整体采用CSSGrid布局,支持响应式,有流畅过渡和展现动画。
    484次学习
查看更多
AI推荐
  • 毕业宝AIGC检测:AI生成内容检测工具,助力学术诚信
    毕业宝AIGC检测
    毕业宝AIGC检测是“毕业宝”平台的AI生成内容检测工具,专为学术场景设计,帮助用户初步判断文本的原创性和AI参与度。通过与知网、维普数据库联动,提供全面检测结果,适用于学生、研究者、教育工作者及内容创作者。
    16次使用
  • AI Make Song:零门槛AI音乐创作平台,助你轻松制作个性化音乐
    AI Make Song
    AI Make Song是一款革命性的AI音乐生成平台,提供文本和歌词转音乐的双模式输入,支持多语言及商业友好版权体系。无论你是音乐爱好者、内容创作者还是广告从业者,都能在这里实现“用文字创造音乐”的梦想。平台已生成超百万首原创音乐,覆盖全球20个国家,用户满意度高达95%。
    26次使用
  • SongGenerator.io:零门槛AI音乐生成器,快速创作高质量音乐
    SongGenerator
    探索SongGenerator.io,零门槛、全免费的AI音乐生成器。无需注册,通过简单文本输入即可生成多风格音乐,适用于内容创作者、音乐爱好者和教育工作者。日均生成量超10万次,全球50国家用户信赖。
    24次使用
  •  BeArt AI换脸:免费在线工具,轻松实现照片、视频、GIF换脸
    BeArt AI换脸
    探索BeArt AI换脸工具,免费在线使用,无需下载软件,即可对照片、视频和GIF进行高质量换脸。体验快速、流畅、无水印的换脸效果,适用于娱乐创作、影视制作、广告营销等多种场景。
    26次使用
  • SEO标题协启动:AI驱动的智能对话与内容生成平台 - 提升创作效率
    协启动
    SEO摘要协启动(XieQiDong Chatbot)是由深圳协启动传媒有限公司运营的AI智能服务平台,提供多模型支持的对话服务、文档处理和图像生成工具,旨在提升用户内容创作与信息处理效率。平台支持订阅制付费,适合个人及企业用户,满足日常聊天、文案生成、学习辅助等需求。
    28次使用
微信登录更方便
  • 密码登录
  • 注册账号
登录即同意 用户协议隐私政策
返回登录
  • 重置密码