Node.JS段点续传:Nginx配置文件分段下载功能实现


发布者 ourjs  发布时间 1516698299844
关键字 JS学习  Node.JS 
Html5 提供了一个新的 Range 标签来实现文件的分段下载。在Node.JS中可以配置这个标签来实现文件的分段下载。

Header标签


请求 Request Header: 下载 3744 以后的文件内容

range: bytes=3744-


返回 Response Header: 文件总长 15522643 个字节

accept-ranges': 'bytes'
content-range': 'bytes */15522643'


Nginx配置


首先要配置Nginx支持range标签返回,很简单添加 add_header Accept-Ranges bytes; 这一行即可

server {
  listen 80;
  server_name adksdf.com;
  location ~ ^/(img/|js/|css/|upload/|font/|fonts/|res/|icon) {
    add_header Access-Control-Allow-Origin *;
    add_header Accept-Ranges bytes;
    root /var/www/...;
    access_log off;
    expires max;
  }
  ...
}

启用以后,如果node.js端发送了含有 range header的请求信息 (req.headers),

{ host: '192.168.0.105:8064',
  connection: 'keep-alive',
  range: 'bytes=260052520-',
  'if-range': '1974',
  'accept-encoding': 'identity',
  'user-agent': 'Mozilla/5.0 (Windows NT 6.1; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/69.0.3497.100 Safari/537.36',
  'accept-language': 'zh-CN,zh;q=0.9',
  cookie: 't0=newghost; t1=ddcade01e4d3b527d852f7e0f435e5f4f79faf668f7920fd3635e5405bf1f64b; t2=fdbf030f55a600e2772fe157590e3005; _wsid=jmipa1vw-1pvdtjg5oa4rjwp8ym2nicmwx3i' }


那么nginx会返回含有range相关的信息:

这是一段完整的Response Header,注意这里的 content-length 不是文件的总长度,而是当前 range 的长度。

{ server: 'nginx',
  date: 'Wed, 24 Jan 2018 02:43:20 GMT',
  'content-type': 'application/zip',
  'content-length': '12420187',
  'last-modified': 'Tue, 16 Jan 2018 12:09:47 GMT',
  connection: 'close',
  etag: '"5a5deb8b-ecdb53"',
  expires: 'Thu, 31 Dec 2037 23:55:55 GMT',
  'cache-control': 'max-age=315360000',
  'access-control-allow-origin': '*',
  'accept-ranges': 'bytes',
  'content-range': 'bytes 3102456-15522642/15522643' }

可以根据这个header中的content-range来获取文件的总大小。

Node.JS实现


这段示例先检测本地下了一半的文件,然后以 'r+' 读写模式创建文件流,并将response流写入文件。
这里将表态文件添加 range 的支持。

var reqOptions  = { url: packageUrl, headers: {} }
var filepath    = '/path/to/your/part/file'
var fileOptions = {}
fs.stat(filepath, function(err, states) {
  if (states) {
    //Range: bytes=3744-
    reqOptions.headers['range'] = 'bytes=' + states.size + '-'
    fileOptions = { start: states.size, flags: 'r+' }
  }
  //创建 http 对象方法
  var reqUrl  = reqOptions.url
  var urlObj  = url.parse(reqUrl)
  var options = {
      hostname  : urlObj.hostname
    , port      : urlObj.port
    , path      : urlObj.pathname
    , headers   : reqOptions.headers || {}
  }
  var req = http.request(options, function(res) {
    var receives    = []
    var err         = null
    var statusCode  = res.statusCode
    var headers     = res.headers
    var ws = fs.createWriteStream(filepath, fileOptions)
    ws.on('error', function(e) {
      console.log('ws error', e)
    })
    res.on('data', function(chrunk) {
      ws.write(chrunk)
    })
    res.on('error', function(err) {
      ws.end()
    })
    res.on('end', function() {
      ws.end()
    })
  })
  req.on('error', function(e) {
    cb && cb(e, null, {})
  })
  req.end()
  ...
})

返回Header

在请求nginx可能会返回其他status code,比如说 206或416,含意如下:

206 Partial Content 
返回的是部分文件内容

416 Requested Range Not Satisfiable
请求的range超过文件尺寸




回复 (1)
微信扫码 立即评论




 热门文章 - 分享最多
  1. redis、memcache和mongodb各自的优缺点是什么,怎么选择呢?
  2. Node.JS 8.x和9.x新特性:N-API,NPM5,ERROR CODE
  3. 移动端开发框架哪个好?jQuery/Vue/AngularJS有哪些区别和优缺点?
  4. 比特币最近为何会暴跌?大资金如何靠做空比特币获利
  5. 红衣教主周鸿祎会不会成为中国首富
  6. Node.JS中UDP打洞穿透内网路由,架设内网服务器技术详解及源码
  7. OnceAir顽石企业私有云网盘使用介绍
  8. 马化腾创办腾讯的第一桶金是怎么来的:炒股10万炒到70万
  9. Node.JS读取中文TXT编码文件显示乱码问题解决方案
  10. node.js用fs.rename强制重命名或移动文件夹

 相关阅读
  1. Node.JS如何查看本地MAC/IP地址、计算cpu使用率和内存容量
  2. JavaScript数组从头开始的位置插入新元素或删除第一个元素
  3. Node.JS中UDP打洞穿透内网路由,架设内网服务器技术详解及源码
  4. 移动端开发框架哪个好?jQuery/Vue/AngularJS有哪些区别和优缺点?
  5. Node.JS 8.x和9.x新特性:N-API,NPM5,ERROR CODE
  6. Node.JS读取中文TXT编码文件显示乱码问题解决方案
  7. Node.JS与USB接口通信:检测U盘/移动硬盘插拔事件和发送接数据
  8. NodeJS动态传参特性:不定个数参数的省略,默认值与解构
  9. 从 Node 到 Go:一个粗略的比较—GO平均性能比JavaScript快十几倍
  10. Node.JS中如何快速扫描端口并发现局域网内的Web服务器地址(80)

  开源的 OurJS
OurJS开源博客已经迁移到 OnceOA 平台。

  关注我们
扫一扫即可关注我们:
OnceJS

OnceOA