4

Content-Range快速版本:不知道正文长度时如何发送正确的标题?

我有一个 FLAC 文件。我想将其转码为 MP3 并立即流式传输给用户。到目前为止,我有这样的事情:

function transcode(file) {
  var spawn = require('child_process').spawn

  var decode = spawn('flac', [
    '--decode',
    '--stdout',
    file
  ])

  var encode = spawn('lame', [
    '-V0',
    '-',
    '-'
  ])

  decode.stdout.pipe(encode.stdin)

  return encode
}

var express = require('express')
var app = express()

app.get('/somefile.mp3', function (req, res) {
  res.setHeader('Accept-Ranges', 'bytes')
  res.setHeader('Content-Range', 'bytes')
  res.setHeader('Content-Type', 'audio/mpeg')
  transcode(file).stdout.pipe(res)
})

这按预期工作,但它是“流式传输”,所以我不能跳过。显然我需要做点什么Content-Range。使用:https ://github.com/visionmedia/node-range-parser

function sliceStream(start, writeStream, readStream) {
  var length = 0
  var passed = false

  readStream.on('data', function (buf) {
    if (passed) return writeStream.write(buf);

    length += buf.length

    if (length < start) return;

    passed = true
    writeStream.write(buf.slice(length - start))
  })

  readStream.on('end', function () {
    writeStream.end()
  })
}

var parseRange = require('range-parser')

app.get('/somefile.mp3', function (req, res) {
  var ranges = parseRange(Infinity, req.headers['range'])

  if (ranges === -1 || ranges === -2) return res.send(400);

  var start = ranges[0].start

  res.setHeader('Accept-Ranges', 'bytes')
  res.setHeader('Content-Type', 'audio/mpeg')

  if (!start) {
    res.setHeader('Content-Range', 'bytes')
    transcode(file).stdout.pipe(res)
    return
  }

  res.setHeader('Content-Range', 'bytes ' + start + '-')
  sliceStream(start, transcode(file).stdout, res)
})

这就是我卡住的地方。由于我不会等到整首歌曲被编码,所以我不知道歌曲的大小。由于我刚刚在 Chrome 中获得“取消”,我假设Content-Range标题格式不正确,没有大小。

另外,我目前只是在浏览器中打开这首歌,所以我假设它使用了该<audio>元素。

建议?

4

1 回答 1

6

是的,您的Content-Range标题格式不正确,没有大小。但是,您可以尝试发送您的服务器已经转码的当前大小。虽然我怀疑 Chrome 是否会优雅地处理不断变化的大小……</p>

您没有处理许多事情

  1. 您似乎没有发送206 Partial Content状态(也许这是由图书馆处理的,不确定)。
  2. 看起来您甚至没有检查范围请求的结尾部分。Chrome 通常不会发送除 之外的任何内容0-,但其他浏览器会发送。事实上,有些人可能会在一个请求中发送多个范围(支持的皇家痛苦)。
  3. 您没有发送正确的Content-Range响应标头,因为您也没有包含您正在发送的内容的结束索引。它应该如下所示:
    Content-Range: bytes 0-2048/3980841
  4. 最后,如果客户端发出一个超出范围的范围请求——也就是说,没有一个范围值与资源的范围重叠——服务应该响应一个416 Requested Range Not Satisfiable状态。

编辑:我没有测试过这种特殊情况,但是如果您从 FLAC 转码为 192kbps CBR MP3,我想如果您发送的内容长度稍微不准确(关闭小于 1000 位):

  • 音频的最后会被播放器剪掉。大约 1000 位会剪辑大约5ms的音频(对人类来说并不明显)。
  • 浏览器将忽略结束索引或内容长度,并简单地继续接受和/或请求超出Content-Range您最初响应的范围,直到您关闭连接或发送 416 状态。
  • The missing/erroneous end of the audio may cause the <audio> to throw a MEDIA_ERR_NETWORK or a MEDIA_ERR_DECODE error that you'd simply have to handle gracefully. (The audio would still be clipped in this case.)
于 2013-01-13T18:13:12.993 回答