Подтвердить что ты не робот

Загрузить/Поток файла из 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();
4b9b3361

Ответ 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();