zoukankan      html  css  js  c++  java
  • js模拟发送 FormData数据

    后台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);
        }
    

    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' ] }
    文件已保存~
    
  • 相关阅读:
    【高并发】面试官问我:为什么局部变量是线程安全的?
    Java中的String到底占用多大的内存空间?你所了解的可能都是错误的!!
    【高并发】学好并发编程,关键是要理解这三个核心问题
    【高并发】高并发分布式锁架构解密,不是所有的锁都是分布式锁!!
    12张图带你彻底理解分布式事务产生的场景和解决方案!!
    【高并发】面试官:讲讲高并发场景下如何优化加锁方式?
    【高并发】秒杀系统架构解密,不是所有的秒杀都是秒杀(升级版)!!
    十一长假我肝了这本超硬核PDF,现决定开源!!
    一文搞懂PV、UV、VV、IP及其关系与计算
    我把这个贼好用的Excel导出工具开源了!!
  • 原文地址:https://www.cnblogs.com/ajanuw/p/9371288.html
Copyright © 2011-2022 走看看