如果状态为“ 206部分内容”,则Chrome无法加载视频

时间:2019-07-27 14:29:09

标签: java google-chrome video-streaming html5-video httpserver

我正在制作一个简单的Java com.sun.net.httpserver.HttpServer来提供静态视频文件。如果我返回状态码206(部分内容),则当我尝试通过浏览器访问该视频时,该视频无法播放(该视频在状态码为200的情况下可以正常播放,但我希望能够搜索和循环播放该视频),这是我的HttpHandler

final String path = StaticHandler.toPathSafe(httpExchange.getRequestURI().getPath());
System.out.println(path);
final File file = new File(path);

if (file.isFile())
{
    int code = 200;
    long position = 0L;
    long end = file.length();

    if (httpExchange.getRequestHeaders().containsKey("Range"))
    {
        try
        {
            long[] range = StaticHandler.parseRange(httpExchange.getRequestHeaders().get("Range").get(0));

            position = range[0];
            if (range[1] != -1)
                end = range[1];

            // the video loads fine when code = 200;
            code = 206;

            httpExchange.getResponseHeaders().set("Content-Range", "bytes " + position + "-" + end + "/" + file.length());
        }
        catch (Exception e)
        {
            e.printStackTrace();
        }
    }

    httpExchange.getResponseHeaders().set("Accept-Range", "bytes");
    httpExchange.getResponseHeaders().set("Content-Type", "video/mp4");
    httpExchange.getResponseHeaders().set("Content-Length", String.valueOf(end - position));

    System.out.println("Response: " + position + ", " + end);

    httpExchange.sendResponseHeaders(code, 0L);

    final FileChannel fileChannel = new FileInputStream(file).getChannel();
    final WritableByteChannel responseChannel = Channels.newChannel(response.getOutputStream());
    fileChannel.transferTo(position, end - position, responseChannel);

    responseChannel.close();
    fileChannel.close();
}
else
{
    System.out.println("404");
    httpExchange.sendResponseHeaders(404, -1);
}

以上代码无法在chrome上加载,但在Firefox中工作正常,这是我在Chrome浏览器中获得的标头:

Response Headers:

Accept-range: bytes
Content-length: 31491166
Content-range: bytes 0-31491166/31491166
Content-type: video/mp4
Date: Sat, 27 Jul 2019 14:32:55 GMT
Transfer-encoding: chunked


Request Headers:

Accept: */*
Accept-Encoding: identity;q=1, *;q=0
Accept-Language: en-US,en;q=0.9
Cache-Control: no-cache
Connection: keep-alive
Host: 192.168.56.1:5000
Pragma: no-cache
Range: bytes=0-
Referer: http://192.168.56.1:5000/30MB.mp4
User-Agent: Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/75.0.3770.142 Safari/537.36

我是否缺少某些内容和/或代码中存在错误?

2 个答案:

答案 0 :(得分:1)

范围是包含范围的开始和结束。不是起始长度

例如:

Content-range: bytes 0-0/31491166

返回1个字节(从零字节到零字节)

您的代码无效,因为浏览器正在等待一个永远不会发送的字节。

答案 1 :(得分:0)

使用以下代码段。它在所有 Safari,Firefox和Chrome 浏览器上都能正常工作并经过测试。

protected byte[] prepareContent(final HttpHeaders headers, byte[] media,
        String range) throws IOException {
    
    long rangeStart = 0;
    long rangeEnd;

    long fileSize = media.length;
    String[] ranges = range.split("-");
    rangeStart = Long.parseLong(ranges[0].substring(6));

    if (ranges.length > 1) {
        rangeEnd = Long.parseLong(ranges[1]);
    } else {
        rangeEnd = fileSize - 1;
    }
    if (fileSize < rangeEnd) {
        rangeEnd = fileSize - 1;
    }
    
    String contentLength = String.valueOf((rangeEnd - rangeStart) + 1);
    
    headers.add("Content-Length", contentLength);
    headers.add("Content-Range", "bytes " + rangeStart + "-" + rangeEnd + "/" + fileSize);
    headers.add("Content-Type", "video/mp4");
    headers.add("Accept-Ranges","bytes");
    return readByteRange(media, rangeStart, rangeEnd);
}

private byte[] readByteRange(byte[] media, long start, long end) throws IOException {
    
    try (InputStream inputStream = new ByteArrayInputStream(media);
         ByteArrayOutputStream bufferedOutputStream = new ByteArrayOutputStream()) {
        
        int nRead;
        while ((nRead = inputStream.read(media, 0, media.length)) != -1) {
            bufferedOutputStream.write(media, 0, nRead);
        }
        bufferedOutputStream.flush();
        byte[] result = new byte[(int) (end - start) + 1];
        System.arraycopy(bufferedOutputStream.toByteArray(), (int) start, result, 0, result.length);
        return result;
    }
}