js模拟发送 FormData数据

摘要:
后台express要求connect multipart模块接收formData类的数据类型ourFormData{constructor(data,rs){returnnewString((function(data,r){letdata_string=“”for(let[k,v]of Object.entries(data)){if(({}))。toString。调用(v)=='

后台express需要connect-multiparty模块接收formData的数据类型

 class ourFormData {
      constructor(data, rs) {
        return new String((function (data, rs) {
          let data_string = '
'
          for (let [k, v] of Object.entries(data)) {
            if (({}).toString.call(v) === '[object Array]') {
              for (let el of v) {
                data_string +=
                  `------WebKitFormBoundary${rs}
Content-Disposition: form-data; name="${k}"

${el}
`;
              }
            } else {
              data_string +=
                `------WebKitFormBoundary${rs}
Content-Disposition: form-data; name="${k}"

${v}
`;
            }
          }
          data_string += `------WebKitFormBoundary${rs}--`
          return data_string;

        })(data, rs));
      }
    }


    function random(a, b) {
      return Math.floor(Math.random() * (b - a + 1) + a);
    }

    function randomString32(len) {
      const loopn = len || 32;
      const c = 'ABCDEFGHJKMNPQRSTWXYZabcdefhijkmnprstwxyz2345678';
      const c_len = c.length;
      let res = '';
      for (let i = 0; i < loopn; i++) {
        res += c.charAt(random(0, c_len - 1));
      }
      return res;
    }


    let xhr = new XMLHttpRequest;
    xhr.open('post', 'http://localhost:3000/');
    let rs = randomString32(16);
    xhr.setRequestHeader('Content-Type', `multipart/form-data; boundary=----WebKitFormBoundary${rs}`);
   
   // let rs = Date.now().toString(16) // Docs: 
    xhr.send(new ourFormData({
      name: ['ajanuw', 'alone'],
      age: 11
    }, rs));

    xhr.onload = e => {
      console.log(xhr.response);
    }

js模拟发送 FormData数据第1张

router.post('/', function (req, res, next) {
  l(req.body)
  res
     .set({
       'access-control-allow-origin': '*'
     })
     .send('hello');
});

上传文件 Docs

<body>
  <input type="file" name="file" id="file">
  <script>
    let l = console.log
    class OurFormData {
      constructor(data, rs) {
        let data_string = '
'
        this.segments = []
        this.rs = rs
        this.status = 0
        let resolve
        let result = new Promise(res => resolve = res)

        let k, v
        let getTag = v => ({}).toString.call(v)
        for ([k, v] of Object.entries(data)) {
          let tag = getTag(v)
          if (tag === '[object File]') {
            // 单文件
            let render = new FileReader()
            render.readAsBinaryString(v);
            render.index = this.segments.length

            render.onload = ({
              target: {
                result
              }
            }) => {
              this.segments[render.index] += `${result}
`
              this.status--

              // 所有异步全部完成
              if (this.status === 0) {
                resolve(this.handleResData(this.segments))
              }
            }

            this.segments.push(
              `------WebKitFormBoundary${this.rs}
Content-Disposition: form-data; name="${k}"; filename="${v.name}"
Content-Type: "${v.type}"

`
            )
            this.status++
          } else if (tag === '[obejct Array]' && v.length > 0 && getTag(v[0]) === '[object File]') {
            // 多文件
            let file, render
            for (file of v) {
              render = new FileReader()
              render.readAsBinaryString(file);
              render.index = this.segments.length

              render.onload = ({
                target: {
                  result
                }
              }) => {
                this.segments[render.index] += `${result}
`
                this.status--

                // 所有异步全部完成
                if (this.status === 0) {
                  resolve(this.handleResData(this.segments))
                }
              }

              this.segments.push(
                `------WebKitFormBoundary${this.rs}
Content-Disposition: form-data; name="${k}"; filename="${v.name}"
Content-Type: "${v.type}"

`
              )
              this.status++
            }
          } else if (tag === '[object Array]') {
            // 处理数组, age: [12, 14]
            let $_
            for ($_ of v) {
              this.segments.push(
                `------WebKitFormBoundary${this.rs}
Content-Disposition: form-data; name="${k}"

${$_}
`
              )
            }
          } else {
            // string and number
            this.segments.push(
              `------WebKitFormBoundary${this.rs}
Content-Disposition: form-data; name="${k}"

${v}
`
            )
          }
        }
        if (this.status === 0) {
          resolve(this.handleResData(this.segments))
        }
        return result
      }

      handleResData(segments) {
        segments.unshift(`
`)
        segments.push(`------WebKitFormBoundary${this.rs}--`)
        let data = segments.join('')

        let bytes = data.length
        // let view = new Uint8Array(bytes)
        // for (let i = 0; i < bytes; i++) {
        //   view[i] = data.charCodeAt(i) & 0xff
        // }
        let buffer = new ArrayBuffer(bytes)
        let view = new DataView(buffer, 0)
        let i
        for (i = 0; i < bytes; i++) {
          // 丢弃高位,只要1byte
          // view.setUint8(i, data.charCodeAt(i) & 0xff)
          view.setUint8(i, data.codePointAt(i) & 0xff)
        }
        return view
      }
    }

    (async function main() {
      let data = {
        name: 'ajanuw',
        age: [1,2]
      }
      document.querySelector('#file').onchange = async ({
        target: {
          files
        }
      }) => {
        if (files.length === 0) return;
        let file = files[0]
        Object.assign(data, {
          file
        })

        let rs = Date.now().toString(16)
        let sendData = await new OurFormData(data, rs)

        let formdata = new FormData()
        formdata.append('file', file)
        let res = await post('http://localhost:5000/test3', sendData, rs)
        l(res)
      }
    })()

    function post(url, data, rs) {
      return new Promise(resolve => {
        let xhr = new XMLHttpRequest()
        xhr.open('post', url)
        xhr.setRequestHeader('Content-Type', `multipart/form-data; boundary=----WebKitFormBoundary${rs}`)
        xhr.send(data)
        xhr.onload = e => {
          resolve(xhr.response)
        }
      })
    }
  </script>

</body>
// 后台
  @Post('test3')
  @UseInterceptors(FileInterceptor('file'))
  @Header('Access-Control-Allow-Origin', '*')
  test3(@UploadedFile() file, @Body() body) {
    l(file)
    l(body)
    const writeFile = createWriteStream(join(__dirname, '..', 'upload', `${Date.now().toString(16) +'-'+ file.originalname}`))
    writeFile.write(file.buffer, () => {
      l('文件已保存~')
    })
    return body
  }


  @Options('test3')
  @Header('Access-Control-Allow-Origin', '*')
  testOption() {

  }
// 打印信息

{ fieldname: 'file',
  originalname: 'kishin-sagume-touhou-wings-wall-red-eyes-24385.jpg',
  encoding: '7bit',
  mimetype: 'image/jpeg',
  buffer:
   <Buffer ff d8 ff e0 00 10 4a 46 49 46 00 01 01 00 00 01 00 01 00 00 ff fe 00 3c 43 52 45 41 54 4f 52 3a 20 67 64 2d 6a 70 65 67 20 76 31 2e 30 20 28 75 73 69 ... >,
  size: 983894 }
{ name: 'ajanuw', age: [ '1', '2' ] }
文件已保存~

免责声明:文章转载自《js模拟发送 FormData数据》仅用于学习参考。如对内容有疑问,请及时联系本站处理。

上篇Vue中引入css文件sqlserver 查询当天/本周/本月/本季度/本年的数据下篇

宿迁高防,2C2G15M,22元/月;香港BGP,2C5G5M,25元/月 雨云优惠码:MjYwNzM=

相关文章

js获取当前时间戳以及前一天时间戳

 js获取当前时间戳以及前一天时间戳(毫秒) var timestamp = (new Date()).getTime(); console.log(timestamp);//打印当前时间戳 console.log(timestamp-24*60*60*1000);//当前时间戳(毫秒) - 1天毫秒数 = 前一天时间戳...

js中访问SqlServer数据库

1 <script language="JavaScript"> 2 //创建数据库对象 3 var objdbConn = new ActiveXObject("ADODB.Connection"); 4 //DSN字符串 5 var strdsn = "Driver={SQL Server};...

js使用toFixed遇到的问题以及由此引发的小数精度问题

原文链接: https://www.cnblogs.com/yalong/p/15762637.html 项目中使用 toFixed 出现的问题: 一. js报错 Uncaught SyntaxError: Invalid or unexpected token 如下图所示: 就是说对 整数 和 字符串 使用toFixed() 会报错 二. 四舍五入不...

JS散度(Jensen-Shannon)

JS散度相似度衡量指标。 https://blog.csdn.net/wateryouyo/article/details/52831115 https://blog.csdn.net/FrankieHello/article/details/80614422?utm_source=copy  KL散度、JS散度和交叉熵 三者都是用来衡量两个概率分布之间的...

(转)Arcgis for Js之鼠标经过显示对象名的实现

http://blog.csdn.net/gisshixisheng/article/details/41889345 在浏览地图时,移动鼠标经过某个对象或者POI的时候,能够提示该对象的名称对用户来说是很实用的,本文讲述在Arcgis for Js中,用两种不同的方式来实现该效果。 为了有个直观的概念,先给大家看看实现后的效果: 百度地图的效果 效果...

2020前端面试题常问集锦

以下为常备面试题集锦,面好多家公司大都问的如此(后续更新补);还有一些算法和手写代码后面整理; js陈述类型1、Es6的class和构造函数的区别: class xx { }(1)不存在变量提升(2)方法默认是不可枚举的,class所有方法没有原型对象prototype也没有构造器不能用new来调用; 2、普通函数和箭头函数的区别?(1)this指向不...