巧用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


参考:

相关推荐
echome8881 小时前
JavaScript Promise 与 async/await 实战:5 个高频异步编程场景的优雅解决方案
开发语言·javascript·ecmascript
摸鱼仙人~2 小时前
Math.js 使用教程
开发语言·javascript·ecmascript
weixin199701080162 小时前
《中控网商品详情页前端性能优化实战》
前端·性能优化
wuhen_n3 小时前
LangChain Agents 实战:构建智能文件管理助手
前端·javascript·人工智能·langchain·ai编程
. . . . .3 小时前
抽象语法树 AST
javascript
zero15974 小时前
TypeScript 快速实战系列:基础入门|TypeScript 核心语法 1 小时吃透(必备基础)
javascript·typescript·大模型编程语言
zzginfo5 小时前
JavaScript 解构赋值
开发语言·javascript·ecmascript
邂逅星河浪漫5 小时前
【JavaScript】==和===区别详解
java·javascript·==·===
Dxy12393102166 小时前
JavaScript 如何捕获异常:从基础到进阶的完整指南
开发语言·javascript·udp
弹简特7 小时前
【JavaSE-网络部分06】TCP 纯高性能优化机制:延迟应答・捎带应答【传输层】
网络·tcp/ip·性能优化·捎带应答·延迟应答