Как определить длину содержимого gzip файла?

Сейчас я пытаюсь обслуживать файлы CSS и JS с сервера, которые не позволят мне включить mod_gzip или mod_deflate. Поэтому я написал небольшой PHP скрипт для сжатия с помощью GZIP и возврата к пользователю.

Пример кода:

$filename = "style.css";

if (!file_exists($filename) || !($info = stat($filename))) {
  header("HTTP/1.1 404 Not Found");
  die();
}

header("Date: ".gmdate("D, j M Y H:i:s e", time()));
header("Cache-Control: max-age=2592000");
header("Last-Modified: ".gmdate("D, j M Y H:i:s e", $info['mtime']));
header("Etag: ".sprintf("\"%x-%x-%x\"", $info['ino'], $info['size'], $info['mtime']));
header("Accept-Ranges: bytes");
header("Cache-Control: Expires ".gmdate("D, j M Y H:i:s e", $info['mtime']+2592000));
header("Content-Type: text/html");

ob_start("ob_gzhandler");
echo file_get_contents($filename);
ob_end_flush();

У меня сейчас две проблемы. Во-первых, у меня возникли проблемы с определением результирующего размера сжатого файла, чтобы сообщить браузеру о содержании. Обычно я бы включил эту строку:

header("Content-Length: ".$info["size"]);

Но если я это сделаю, браузер зависает, пытаясь ждать большего количества данных. Есть ли способ рассчитать общий размер? Или я должен игнорировать эту директиву заголовка.

Другая проблема заключается в том, что всякий раз, когда я просматриваю этот файл PHP в Firefox, он пытается загрузить результат. В Chrome он просто отображает его, как я ожидал. Любые предложения?

Изменить: Благодаря SoapBox я заменил конец кода следующим:

header("Content-Encoding: gzip");
$compressed = gzencode(file_get_contents($filename), 5);
header("Content-Length: ".strlen($compressed));
die($compressed);

Это отлично подходит для контента! Но я все еще получаю Firefox, чтобы загрузить файл, а не отображать его.: (

Изменить снова:. Здесь приведен модифицированный код конца кода, любезно предоставленный Cletus.

// Start buffered output
ob_start();
// Check for gzip capability
if (stripos($_SERVER['HTTP_ACCEPT_ENCODING'], "gzip") !== false) {
  ob_start("ob_gzhandler");
  echo file_get_contents($filename);
  ob_end_flush();
} else
  echo file_get_contents($filename);

// Write the content length
header('Content-Length: '.ob_get_length());
ob_end_flush();

Я собираюсь начать новый вопрос, чтобы выяснить, почему Firefox продолжает пытаться загрузить файл.

Ответы

Ответ 1

Проблема заключается в том, что для того, чтобы узнать длину содержимого, вам нужно знать, поддерживает ли клиент gzip-кодировку, и вы делегировали это решение с помощью ob_gzhandler. Из HTTP-заголовки:

ob_start();
ob_start('ob_gzhandler');

  ... output the page content...

ob_end_flush();  // The ob_gzhandler one

header('Content-Length: '.ob_get_length());

ob_end_flush();  // The main one

Полная версия:

$filename = "style.css";

if (!file_exists($filename) || !($info = stat($filename))) {
  header("HTTP/1.1 404 Not Found");
  die();
}

header("Date: ".gmdate("D, j M Y H:i:s e", time()));
header("Cache-Control: max-age=2592000");
header("Last-Modified: ".gmdate("D, j M Y H:i:s e", $info['mtime']));
header("ETag: ".sprintf("\"%x-%x-%x\"", $info['ino'], $info['size'], $info['mtime']));
header("Accept-Ranges: bytes");
header("Expires: ".gmdate("D, j M Y H:i:s e", $info['mtime']+2592000));
header("Content-Type: text/css"); // note: this was text/html for some reason?

ob_start();
ob_start("ob_gzhandler");
echo file_get_contents($filename);
ob_end_flush();
header('Content-Length: '.ob_get_length());
ob_end_flush();

Это много лучше, чем сама проблема с кодировкой gzip.

Ответ 2

Вам нужно сначала выполнить весь gzip и измерить результат (либо сохранить содержимое в памяти, либо записать их на диск при сжатии, а затем статизировать gzipped файл), а затем написать заголовок Content-Length, а затем отправьте содержимое файла.

Или используйте chunked transfer encoding.

Ответ 3

Чтобы решить проблему с firefox, я думаю, вам нужно включить header( "Content-Encoding: gzip" );, чтобы браузер знал, что нужно распаковать содержимое.

Что касается длины содержимого, вы можете попробовать просто оставить это значение выключенным или попытаться выяснить способ использования "Transfer-Encoding: chunked" (вы не можете отправить этот заголовок, вам нужно отформатировать данные специально для него). Возможно, что ob_end_flush автоматически включает блокировку.

Я рекомендую вам получить wirehark и захватить то, что отправляет ваш php script, и сравнить его с правильно работающим сервером, чтобы увидеть, какие заголовки и т.д. отсутствуют.