web-dev-qa-db-ja.com

URLからのファイルのダウンロード/ストリーミング-asp.net

ブラウザでプロンプトとして保存するファイルをストリーミングする必要があります。問題は、ファイルのあるディレクトリが仮想的にマップされているため、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();
65
mp3duck

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 = "application/octet-stream";

        //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();
        }
    }
95
Dallas

URLをバイトにダウンロードし、バイトをストリームに変換します。

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

私はこれをかなり行い、より簡単な答えを追加できると考えました。ここでは簡単なクラスとして設定しましたが、これを毎晩実行して、フォローしている会社の財務データを収集します。

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");
12
CodeChops

2年後、ダラスの回答を使用しましたが、直接ファイルにリンクしていたため、HttpWebRequestFileWebRequestに変更する必要がありました。これがどこにでも当てはまるかどうかはわかりませんが、追加したいと思いました。また、私は削除しました

var resp = Http.Current.Resonse

respが参照された場所でHttp.Current.Responseを使用しただけです。

4
Tingo

ダラスの受け入れられたソリューションは、Citrix NetscalerでLoad Balancerを使用する場合(WAFポリシーなし)に機能していました。

現在のシナリオ(コンテンツ長が正しくない)はRFC違反であり、AppFWは接続をリセットするため、WAFに関連付けられている場合、NetscalerのLBを介したファイルのダウンロードは機能しません。ポリシーは関連付けられていません。

不足していたのは:

Response.End();

参照: asp.netでPDFファイルをストリーミングしようとすると、「破損したファイル」が生成されます

0
Oliver Fink