Загрузить/Поток файла из URL-адреса - asp.net
Мне нужно передать файл, который приведет к сохранению как подсказка в браузере.
Проблема в том, что каталог, в котором находится файл, фактически сопоставлен, поэтому я не могу использовать Server.MapPath, чтобы определить его фактическое местоположение. Каталог не находится в том же месте (или даже физическом сервере в ящиках) в качестве веб-сайта.
Мне бы хотелось что-то вроде следующего, но это позволит мне передать веб-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();
Ответы
Ответ 1
Вы можете использовать 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 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
Загрузите url в байты и преобразуйте байты в поток:
using (var client = new WebClient())
{
var content = client.DownloadData(url);
using (var stream = new MemoryStream(content))
{
...
}
}
Ответ 3
Я делаю это совсем немного и думал, что могу добавить более простой ответ. Я установил его в качестве простого класса, но каждый вечер я запускаю его для сбора финансовых данных о компаниях, за которыми я слежу.
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");
Ответ 4
Через 2 года я использовал ответ Далласа, но мне пришлось изменить HttpWebRequest
на FileWebRequest
, так как я привязывался к прямым файлам. Не уверен, что это повсюду, но я решил, что добавлю это. Кроме того, я удалил
var resp = Http.Current.Resonse
и просто использовал Http.Current.Response
на месте, где был указан resp
.
Ответ 5
Принятое нами решение от Далласа сработало для нас, если мы используем Load Balancer в Citrix Netscaler (без политики WAF).
Загрузка файла не работает через LB Netscaler, когда он связан с WAF, так как текущий сценарий (длина содержимого неверна) является нарушением RFC, и AppFW сбрасывает соединение, что не происходит, когда WAF политика не связана.
Итак, чего не хватало было:
Response.End();
См. Также: при попытке потоковой передачи файла PDF с помощью asp.net создается "поврежденный файл".
Ответ 6
Попробуйте следующее:
using (var response = WebRequest.Create(url).GetResponse())
using (var read = response.GetResponseStream())
{
// do something
}
Ответ 7
Вы можете попробовать использовать класс DirectoryEntry с префиксом пути IIS:
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();