从URL下载/流式传输文件 - asp.net

82

我需要流式传输一个文件,以便在浏览器中弹出“另存为”对话框。

问题在于,该文件所在的目录是虚拟映射的,因此我无法使用Server.MapPath确定它的实际位置。该目录与网站不在同一位置(甚至不在实际服务器上)。

我希望有类似以下的方案,但可以传递web URL而不是服务器文件路径。

我可能最终不得不通过配置基路径来构建我的文件路径,然后附加其余路径,但希望能以这种方式完成。

var filePath = Server.MapPath(DOCUMENT_PATH);

if (!File.Exists(filePath))
    return;

var fileInfo = new System.IO.FileInfo(filePath);
Response.ContentType = "application/octet-stream";
Response.AddHeader("Content-Disposition", String.Format("attachment;filename=\"{0}\"", filePath));
Response.AddHeader("Content-Length", fileInfo.Length.ToString());
Response.WriteFile(filePath);
Response.End();

你能详细阐述一下“虚拟映射”的意思吗?是指可以通过 URL 访问的虚拟 IIS 文件夹吗? - EventHorizon
它使用的是VPP路径,这是EpiServer CMS使用的一个概念。我们设置虚拟路径名称(例如"/documents/"),然后指定其映射到哪个物理路径(例如"//servername/documents")。系统会在运行时创建对该目录的引用。您可以通过Web URL浏览文件,没有问题。 - mp3duck
文件名确实可以通过URL访问。我需要使用这个URL来流式传输文件,而不是服务器路径,因为我无法从URL(使用MapPath)确定这个路径。 - mp3duck
如果您知道URL和文件扩展名在浏览器中打开“另存为”提示(或者这是您的问题之一?),也许您可以将请求重定向到要下载的文件?否则,user97970的建议似乎是正确的方法。 - EventHorizon
9个回答

113

你可以使用 HttpWebRequest 获取文件并将其流式传输回客户端。这允许你通过URL获取文件。我找到的一个例子是(但不记得在哪里看到的)

    //Create a stream for the file
    Stream stream = null;

    //This controls how many bytes to read at a time and send to the client
    int bytesToRead = 10000;

    // Buffer to read bytes in chunk size specified above
    byte[] buffer = new Byte[bytesToRead];

    // The number of bytes read
    try
    {
      //Create a WebRequest to get the file
      HttpWebRequest fileReq = (HttpWebRequest) HttpWebRequest.Create(url);

      //Create a response for this request
      HttpWebResponse fileResp = (HttpWebResponse) fileReq.GetResponse();

      if (fileReq.ContentLength > 0)
        fileResp.ContentLength = fileReq.ContentLength;

        //Get the Stream returned from the response
        stream = fileResp.GetResponseStream();

        // prepare the response to the client. resp is the client Response
        var resp = HttpContext.Current.Response;

        //Indicate the type of data being sent
        resp.ContentType = MediaTypeNames.Application.Octet;
    
        //Name the file 
        resp.AddHeader("Content-Disposition", "attachment; filename=\"" + fileName + "\"");
        resp.AddHeader("Content-Length", fileResp.ContentLength.ToString());
    
        int length;
        do
        {
            // Verify that the client is connected.
            if (resp.IsClientConnected)
            {
                // Read data into the buffer.
                length = stream.Read(buffer, 0, bytesToRead);

                // and write it out to the response's output stream
                resp.OutputStream.Write(buffer, 0, length);

                // Flush the data
                resp.Flush();

                //Clear the buffer
                buffer = new Byte[bytesToRead];
            }
            else
            {
                // cancel the download if client has disconnected
                length = -1;
            }
        } while (length > 0); //Repeat until no data is read
    }
    finally
    {
        if (stream != null)
        {
            //Close the input stream
            stream.Close();
        }
    }

2
刷新响应并在每次运行时重新创建缓冲区是不必要的,它只会减慢传输速度。同时,我认为您应该将一些变量重命名(fileReq -> urlRequest,fileResp -> urlResponse)。另外,没有理由在开头创建Stream变量,您应该在使用它的地方创建它(靠近GetResponseStream()调用的位置)。 - data
3
在这个例子中,resp未定义:代码无法编译。 - Case
1
正如@Sharon所提到的,这段代码无法编译。你需要做的是,在注释// prepare the response to the client. resp is the client Response处添加以下代码:var resp = HttpContext.Current.Response; - Jeremy Wiggins
将流添加到电子邮件附件为我的 MP3 文件完成了工作 :) 太棒了! - Fahad Abid Janjua
1
我知道我来晚了,但是非常感谢你的分享 :) 我一直在寻找从远程位置/URL流式传输文件的解决方案,这个方法对我非常有效。唯一的改变是我用SafeWebRequest替换了HttpWebRequest - Terry
显示剩余10条评论

33

将下载链接转换为字节并将字节转换为流:

using (var client = new WebClient())
{
    var content = client.DownloadData(url);
    using (var stream = new MemoryStream(content))
    {
        ...
    }
}   

7
虽然这个解决方案看起来很简单且运行良好,但需要记住的是它会将文件加载到您的计算机内存中。因此,当数据过大或同时进行多个请求时,由于内存不足而导致异常的可能性很高。 - Benjamin
3
Benjamin,可以通过使用HttpClient来缓解这个问题。 - jpaugh

15

我经常做这件事,并且认为我可以提供一个更简单的答案。我在这里设置了一个简单的类,但是我每天晚上都会运行它来收集我关注的公司的财务数据。

class WebPage
{
    public static string Get(string uri)
    {
        string results = "N/A";

        try
        {
            HttpWebRequest req = (HttpWebRequest)WebRequest.Create(uri);
            HttpWebResponse resp = (HttpWebResponse)req.GetResponse();

            StreamReader sr = new StreamReader(resp.GetResponseStream());
            results = sr.ReadToEnd();
            sr.Close();
        }
        catch (Exception ex)
        {
            results = ex.Message;
        }
        return results;
    }
}

在这种情况下,我传入一个URL,它将返回HTML页面。如果你想对流进行不同的操作,你可以轻松地改变它。

使用方法如下:

string page = WebPage.Get("http://finance.yahoo.com/q?s=yhoo");

我的情况有所不同,有一个文档管理系统,以 .tif 格式返回文档,其中没有在查询字符串中发送编号。例如:链接,但这种方法不起作用,欢迎提出任何想法。谢谢。 - kayhan yüksel
@kayhanyüksel:我尝试了那个链接,但它超时了。抱歉。 - CodeChops
这只是我的两分之一解决方案:也许您需要通过 req.Credentials = credential; 添加一些凭据到请求中,然后您可以直接使用 resp.GetResponseStream() 中的 Stream - dns_nx

7

两年后,我使用了Dallas的答案,但是由于我链接到的是直接文件,所以我不得不将HttpWebRequest更改为FileWebRequest。不确定这是否在任何地方都适用,但我想我会添加它。另外,我删除了

var resp = Http.Current.Resonse

并且只要有地方引用了resp,我就用Http.Current.Response来替换它。


1
完美。我稍微修改了Dalla的答案。我在下面发布我的版本。 - Ricardo Appleton
但是你需要注意你正在下载的文件类型。使用HttpWebRequest无法下载PDF文件,但如果你要下载消息文件(.msg),则无法使用FileWebRequest - dns_nx

7

如果您正在寻找.NET Core版本的@Dallas答案,请使用以下内容。

        Stream stream = null;

        //This controls how many bytes to read at a time and send to the client
        int bytesToRead = 10000;

        // Buffer to read bytes in chunk size specified above
        byte[] buffer = new Byte[bytesToRead];

        // The number of bytes read
        try
        {
            //Create a WebRequest to get the file
            HttpWebRequest fileReq = (HttpWebRequest)HttpWebRequest.Create(@"file url");

            //Create a response for this request
            HttpWebResponse fileResp = (HttpWebResponse)fileReq.GetResponse();

            if (fileReq.ContentLength > 0)
                fileResp.ContentLength = fileReq.ContentLength;

            //Get the Stream returned from the response
            stream = fileResp.GetResponseStream();

            // prepare the response to the client. resp is the client Response
            var resp = HttpContext.Response;

            //Indicate the type of data being sent
            resp.ContentType = "application/octet-stream";

            //Name the file 
            resp.Headers.Add("Content-Disposition", "attachment; filename=test.zip");
            resp.Headers.Add("Content-Length", fileResp.ContentLength.ToString());

            int length;
            do
            {
                // Verify that the client is connected.
                if (!HttpContext.RequestAborted.IsCancellationRequested)
                {
                    // Read data into the buffer.
                    length = stream.Read(buffer, 0, bytesToRead);

                    // and write it out to the response's output stream
                    resp.Body.Write(buffer, 0, length);


                    //Clear the buffer
                    buffer = new Byte[bytesToRead];
                }
                else
                {
                    // cancel the download if client has disconnected
                    length = -1;
                }
            } while (length > 0); //Repeat until no data is read
        }
        finally
        {
            if (stream != null)
            {
                //Close the input stream
                stream.Close();
            }
        }

5

我认为在 .Net Core 中最简单的方法是:

using (MemoryStream ms = new MemoryStream())
    using (HttpClient client = new HttpClient())
    {
      client.GetStreamAsync(url).Result.CopyTo(ms);
      // use ms in what you want
    }
  }

现在您已经将文件作为流下载到MS中。


是的,HttpClient也被微软推荐用来替代HttpWebRequest。"我们不建议您在新的开发中使用HttpWebRequest。相反,请使用System.Net.Http.HttpClient类。" -- https://learn.microsoft.com/en-us/dotnet/api/system.net.httpwebrequest?view=net-8.0&viewFallbackFrom=dotnet-aspire-8.0 - undefined

0
你可以尝试使用带有IIS路径前缀的DirectoryEntry类:
using(DirectoryEntry de = new DirectoryEntry("IIS://Localhost/w3svc/1/root" + DOCUMENT_PATH))
{
    filePath = de.Properties["Path"].Value;
}

if (!File.Exists(filePath))
        return;

var fileInfo = new System.IO.FileInfo(filePath);
Response.ContentType = "application/octet-stream";
Response.AddHeader("Content-Disposition", String.Format("attachment;filename=\"{0}\"", filePath));
Response.AddHeader("Content-Length", fileInfo.Length.ToString());
Response.WriteFile(filePath);
Response.End();

0

虽然我也不赞成覆盖用户偏好设置的想法,但这是我在.net6中实现的方法。

将以下代码添加到Startup.Configure中。本质上,我们只需更改PDF内容类型并在下载时添加内容-分发。您可以进一步细化以仅针对特定文件等。

var extensionProvider = new FileExtensionContentTypeProvider();
extensionProvider.Mappings[".pdf"] = "application/octet-stream";

app.UseStaticFiles(new StaticFileOptions
{
    ContentTypeProvider = extensionProvider,
    OnPrepareResponse = response =>
    {
        if (response.File.Name.EndsWith(".pdf", StringComparison.InvariantCultureIgnoreCase))
        {
            response.Context.Response.Headers.Remove("content-disposition");
            response.Context.Response.Headers.Add("content-disposition","attachment; filename=\"" + response.File.Name + "\"");
        }
    }
}); 

0

达拉斯提供的解决方案对我们有效,如果我们在 Citrix Netscaler 上使用负载均衡器(不带 WAF 策略)。

当与 WAF 关联时,通过 Netscaler 的 LB 下载文件无法正常工作,因为当前情况下 Content-length 不正确,这是 RFC 违规行为,AppFW 会重置连接,而当未关联 WAF 策略时则不会发生此问题。

所以缺少的是:

Response.End();

参见: 尝试使用asp.net流式传输PDF文件会产生“损坏的文件”


网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接