HttpServer - HttpExchange - Seekable Stream

时间:2013-08-05 15:35:24

标签: java httpserver

我在一个示例java http服务器和.Net客户端(在平板电脑上)上工作。 使用我的http服务器,.Net客户端必须能够下载文件。

它运行正常,但现在我必须能够在连接中断后恢复下载。

这里有一些代码:

Java服务器:(它是在一个单独的线程中启动的,因此是run方法)。

public void run() {

    try {
        server = com.sun.net.httpserver.HttpServer.create(
                new InetSocketAddress(
                        portNumber), this.maximumConnexion);

        server.setExecutor(executor);
        server.createContext("/", new ConnectionHandler(this.rootPath));
        server.start();


    } catch (IOException e1) {
        //For debugging
        e1.printStackTrace();
    }

}

我的HttpHandler:(只有处理GET请求的部分)

/**
 * handleGetMethod : handle GET request. If the file specified in the URI is
 * available, send it to the client.
 * 
 * @param httpExchange
 * @throws IOException
 */
private void handleGetMethod(HttpExchange httpExchange) throws IOException {

File file = new File(this.rootPath + this.fileRef).getCanonicalFile();

if (!file.isFile()) {
    this.handleError(httpExchange, 404);
} else if (!file.getPath().startsWith(this.rootPath.replace('/', '\\'))) { // windows work with anti-slash!
    // Suspected path traversal attack.
    System.out.println(file.getPath());
    this.handleError(httpExchange, 403);
} else {
    //Send the document.


    httpExchange.sendResponseHeaders(200, file.length());       
    System.out.println("file length : "+ file.length() + " bytes.");


    OutputStream os = httpExchange.getResponseBody();

    FileInputStream fs = new FileInputStream(file);

    final byte[] buffer = new byte[1024];
    int count = 0;
    while ((count = fs.read(buffer)) >= 0) {
        os.write(buffer, 0, count);
    }
    os.flush();
    fs.close();
    os.close();
}

}

现在我的.Net客户端:(简化)

  try{

        Stream response = await httpClient.GetStreamAsync(URI + this.fileToDownload.Text);


    FileSavePicker savePicker = new FileSavePicker();
    savePicker.SuggestedStartLocation = PickerLocationId.DocumentsLibrary;
    // Dropdown of file types the user can save the file as
    savePicker.FileTypeChoices.Add("Application/pdf", new List<string>() { ".pdf" });
    // Default file name if the user does not type one in or select a file to replace
    savePicker.SuggestedFileName = "new doc";

    StorageFile file = await savePicker.PickSaveFileAsync();

    if (file != null)
    {

        const int BUFFER_SIZE = 1024*1024;
        using (Stream outputFileStream = await file.OpenStreamForWriteAsync())
        {
            using (response)
            {
                var buffer = new byte[BUFFER_SIZE];
                int bytesRead;                          
                do
                {
                    bytesRead = response.Read(buffer, 0, BUFFER_SIZE);
                    outputFileStream.Write(buffer, 0, bytesRead);                                
                } while (bytesRead > 0);
            } 
            outputFileStream.Flush();
       }

   }

}
catch (HttpRequestException hre)
{   //For debugging
    this.Display.Text += hre.Message;
    this.Display.Text += hre.Source;
}
catch (Exception ex)
{
    //For debugging
    this.Display.Text += ex.Message;
    this.Display.Text += ex.Source;
}

因此,为了恢复下载,我想在.Net客户端部分中使用一些搜索操作。 但每次我尝试像response.Seek(offset, response.Position);这样的事情时,都会发生错误,通知Stream不支持搜索操作。 是的,它没有,但我如何指定(在我的服务器端)使用可搜索的流? 方法HttpExchange.setStreams是否有用? 或者,我不需要修改流但是需要配置我的HttpServer实例吗?

感谢。

1 个答案:

答案 0 :(得分:1)

使用Range,Accept-Range和Content-Range字段可以正常工作。只需要做一些工作就可以发送文件的正确部分并设置响应的标题。

服务器可以通过设置Accept-Range字段来通知客户端它支持Range字段:

responseHeader.set("Accept-Ranges", "bytes");

然后在发送部分文件时设置Content-range字段:

responseHeader.set("Content-range", "bytes " + this.offSet + "-" + this.range + "/" + this.fileLength);

最后,返回代码必须设置为206(部分内容)。

有关范围,接受范围和内容范围字段的详细信息,请参阅http://www.w3.org/Protocols/rfc2616/rfc2616-sec14.html

注意:Opera 12.16使用字段“Range”恢复下载,但似乎IE 10和Firefox 22不使用此字段。可能是我最初寻找的一些可寻找的溪流。如果有人对此有答案,我会很高兴看到它=)。