巧用Request Range优化下载

最近在做视频相关的项目,上次介绍了 视频的点播,今天来介绍一下视频的切片下载

通常是针对大资源,一次性请求一整个资源会导致响应时间过长,所以进行分批请求,可利用浏览器并发请求的能力

1、前置知识

(注意:应该是Content-Range

HTTP``request Header中,有一个请求头字段Range,用来表示范围请求

  • 即只请求目标文件的一部分
javaScript 复制代码
GET /videoFile.mp4 HTTP/1.1
Host: www.example.re
Range: bytes=0-9999

1.1 正常响应

当后端正常响应时,它对应的状态码为206--- partial content,表示部分内容;并且会在响应头添加Content-Length字段,并返回请求的范围和资源长度

如下:

javaScript 复制代码
HTTP/1.1 206 Partial Content
Content-Length: 10000
Content-Range: bytes 0-9999/25000
Accept-Ranges: bytes

1.2 预请求

客户端可以在发起范围请求之前,先发送一个head请求,看服务端是否支持range请求

  • 若支持,服务端会在响应头中返回一个字段Accept-Range: bytes
  • 如不支持,则返回Accept-Range: none

1.3 超出范围

当请求头中的范围超出资源大小时,服务端端则会返回416--- Range Not Satisfiable

2、后端

后端仍然使用koa来完成,在之前项目的基础上增加两个接口:

  • head请求获取文件的大小
  • get进行范围请求

2.1 获取文件大小

head请求基本与get请求相同(安全且幂等 ),只不过它没有响应体,只有响应头,一般用于请求元信息

这里将用于获取文件的大小,请求结果放在头部的Content-Length字段

javaScript 复制代码
router.head('/file/size', async (ctx) => {
  const { originPath } = ctx.query
  const filePath = path.join(__dirname, '../uploadFiles', originPath)
  // 读取文件信息
  const { size } = fs.statSync(filePath)

  ctx.set('Content-Length', size)
  ctx.status = 200
})

注意: 无需返回响应体信息,即使返回,客户端也不会接收到

2.2 进行范围请求

进行范围请求首先会先检查请求头是否有range字段,如果没有,仍然返回一整个文件

javaScript 复制代码
/**
 * 下载文件
 */
router.get('/file/download', async (ctx) => {
  const { originPath } = ctx.query
  const filePath = path.join(__dirname, '../uploadFiles', originPath)
  
  const { size } = fs.statSync(filePath)

  const range = ctx.headers.range
  
  if (!range) {
    // 告诉客户端 服务端事支持range请求的
    ctx.set('Accept-Range', 'bytes')
    // 流式返回
    ctx.body = fs.createReadStream(filePath);
    return
  }
})

createReadStream接收第二个配置参数,可用于配置读取的起始、终止位置

javaScript 复制代码
router.get('/file/download', async (ctx) => {
  ...

  const range = ctx.headers.range  // 其结构为:bytes=0-999
  
  const [start, end] = gerRange(range)

  // 表示超出范围
  if (start > size || end > size) {
    ctx.status = 416
    ctx.body = ''
    return
  }

  ctx.set('Accept-Ranges', 'bytes')
  ctx.set('Content-Length', end - start + 1)
  ctx.set('Content-Range', `bytes ${start}-${end}/${size}`)
  ctx.status = 206
  
  ctx.body = fs.createReadStream(filePath, {start, end})  

})

3、前端

步骤:

  1. head请求获取文件大小
  2. 请求range文件的blob
  3. 下载文件
javaScript 复制代码
const downloadFile = async (path: string) => {
  try {
    const size = await fetchFileSize(path);
    if (!size) return alert('下载失败');

    const sizeQueue = getFileSlice(+size.headers['content-length']);
    const peomiseQueue = sizeQueue.map(item => {
      const [start, end] = item;
      return fetchFileBlob(path, start, end);
    });

    Promise.all(peomiseQueue)
      .then(res => {
        const blob = new Blob(res, { type: 'video/mp4' });
        const href = URL.createObjectURL(blob);

        // 下载
        const a = document.createElement('a');
        a.download = path.split('/').pop() as string;
        a.href = href;
        a.click();
        URL.revokeObjectURL(blob);
      })
  } catch (error) {
    console.error(error);
  }
};

下载一个大小为130+M的MP4文件,时间大概为1.2s

详细代码已上传github


参考:

相关推荐
隔壁的大叔16 分钟前
Markdown 渲染如何穿插自定义组件
前端·javascript·vue.js
薯老板25 分钟前
JavaScript原型,原型链
javascript
愚者Pro1 小时前
Flutter基础学习
前端·javascript·vue.js
时光足迹1 小时前
Tiptap 简单编辑器模版
前端·javascript·react.js
吴声子夜歌2 小时前
Vue3——使用Mock.js
javascript·vue·mock.js
时光足迹2 小时前
ThreeJS之GUI控制器
前端·javascript·three.js
时光足迹2 小时前
Tiptap编辑器
前端·javascript·react.js
im_AMBER2 小时前
手撕hot100之矩阵!看完这篇就AC~
javascript·数据结构·线性代数·算法·leetcode·矩阵
时光足迹2 小时前
电子书阅读器之笔记高亮(跨段处理)
前端·javascript·react.js
Hello-Mr.Wang2 小时前
【保姆级教程】MasterGo MCP + Cursor 一键实现 UI 设计稿还原
前端·javascript·vue.js·ai编程