ブラウザでプロンプトとして保存するファイルをストリーミングする必要があります。問題は、ファイルのあるディレクトリが仮想的にマップされているため、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();
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();
}
}
URLをバイトにダウンロードし、バイトをストリームに変換します。
using (var client = new WebClient())
{
var content = client.DownloadData(url);
using (var stream = new MemoryStream(content))
{
...
}
}
私はこれをかなり行い、より簡単な答えを追加できると考えました。ここでは簡単なクラスとして設定しましたが、これを毎晩実行して、フォローしている会社の財務データを収集します。
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");
2年後、ダラスの回答を使用しましたが、直接ファイルにリンクしていたため、HttpWebRequest
をFileWebRequest
に変更する必要がありました。これがどこにでも当てはまるかどうかはわかりませんが、追加したいと思いました。また、私は削除しました
var resp = Http.Current.Resonse
resp
が参照された場所でHttp.Current.Response
を使用しただけです。
ダラスの受け入れられたソリューションは、Citrix NetscalerでLoad Balancerを使用する場合(WAFポリシーなし)に機能していました。
現在のシナリオ(コンテンツ長が正しくない)はRFC違反であり、AppFWは接続をリセットするため、WAFに関連付けられている場合、NetscalerのLBを介したファイルのダウンロードは機能しません。ポリシーは関連付けられていません。
不足していたのは:
Response.End();