Chrome неверно интерпретирует прикрепление файла как документ

У меня есть ссылка, которая идет на PHP script, который генерирует CSV файл, который будет загружен и сохранен. Я генерирую заголовки Content-Disposition и Content-Type в ответе. Каждый браузер отлично загружает файл, кроме Chrome (v19).

Ссылка выглядит примерно так:

http://hostname.com/controller/action/export

Заголовки, возвращенные из этого запроса:

Cache-control:no-cache, must-revalidate
Connection:Keep-Alive
Content-Disposition:attachment; filename=2012-03-14.csv
Content-Encoding:gzip
Content-Length:521
Content-Type:application/vnd.ms-excel
Date:Thu, 15 Mar 2012 05:17:55 GMT
Expires:Mon, 26 Jul 1997 05:00:00 GMT
Keep-Alive:timeout=5, max=92
P3P:CP="NOI ADM DEV PSAi COM NAV OUR OTRo STP IND DEM"
Pragma:no-cache
Server:Apache/2.2.11 (Win32) mod_ssl/2.2.11 OpenSSL/0.9.8i mod_autoindex_color PHP/5.2.9
Vary:Accept-Encoding
X-Powered-By:PHP/5.2.9

Инструменты разработчика Chrome показывают, что сетевое соединение было Canceled, а в окне консоли отображается следующая ошибка:

Resource interpreted as Document but transferred with MIME type application/vnd.ms-excel:

Я попытался использовать разные значения для Content-Type, я отключил заголовки Cache-Control и Content-Type. Я тестировал, используя Javascript location.href=, тег <a>, <form action="POST">, отключил сжатие Gzip и другие другие способы, чтобы заставить Chrome фактически загрузить файл.

Каждый другой браузер отлично загружает файл, поэтому мой вопрос: что заставляет Chrome интерпретировать запрос как "Документ" для отображения вместо вложения? Есть ли другой заголовок, который я пропускаю, или заголовок в списке, который его запутывает?

EDIT: Здесь PHP-код по запросу, хотя он немного длинный:

function renderHeaders($filename = null) {
  header("Content-Type: application/vnd.ms-excel");
  header("Cache-Control: public, must-revalidate, max-age=0");
  if (is_string($filename)) {
    $this->setFilename($filename);
  }
  if ($this->filename === null) {
    $this->filename = 'Data.csv';
  }
  if ($this->filename) {
    header("Content-disposition: attachment; filename=".$this->filename);
  }
}

function render($outputHeaders = true, $to_encoding = null, $from_encoding = "auto") {
  if ($outputHeaders) {
    $this->renderHeaders();
  }
  if ($this->_tmpFile) {
    rewind($this->buffer);
    $output = '';
    while (!feof($this->buffer)) {
      $output .= fread($this->buffer, 8192);
    }
    fclose($this->buffer);
  } else {
    rewind($this->buffer);
    $output = stream_get_contents($this->buffer);
  }
  // get around excel bug (http://support.microsoft.com/kb/323626/)
  if (substr($output,0,2) == 'ID') {
    $pos = strpos($output, $this->delimiter);
    if ($pos === false) {
      $pos = strpos($output, "\n");
    }
    if ($pos !== false) {
      $output = $this->enclosure . substr($output, 0, $pos) . $this->enclosure . substr($output, $pos);
    }
  }
  if ($to_encoding) {
    $output = mb_convert_encoding($output, $to_encoding, $from_encoding);
  }
  $this->clear();
  return $this->output($output);
}

Ответ 1

Тип контента неверен, он должен быть text/csv, а не application/vnd.ms-excel.

Источник: Wikipedia

Ответ 2

Это из-за вашего типа контента.

Вы должны изменить: Content-Type: применение/vnd.ms-первенствует

Попробуйте следующее:

<?php

header("Content-Disposition: attachment; filename=$file");
header("Content-Type: application/zip");
...

Ответ 3

Попробуйте это, он должен принудительно загрузить все браузеры.

    header ("Content-type: octet/stream");
    header ("Content-disposition: attachment; filename=\"".$file."\";");
    header("Content-Length: ".filesize($file));
    readfile($file);

Ответ 4

Тип MIME "application/vnd.ms-excel" специально предназначен для двоичных файлов Microsoft Excel. Если вы сообщите Chrome, что вы отправляете двоичный файл, а затем передаете его текстовому файлу, это может вызвать проблемы. Вместо этого используйте утвержденный тип значений, разделенных запятой, "application/csv".

Заменить:

header("Content-Type: application/vnd.ms-excel");

с:

header("Content-Type: application/csv");