Привет товарищи кодировщики!
Я ищу решение для отображения html-страницы, в то время как мой php-код подготавливает .zip, который затем загружается. Причина в том, что иногда молнии намного больше, и на их изготовление требуется время.
HTML-страница будет простой: «Подождите, пока готовятся ваши файлы .zip». Используемая сторона PHP - это Symfony. Итак, я перехожу к своей функции getInboxExportAction, вызывая https://myapi.com/orders/orderid/inbox/export.
Функция загрузки (makeExportDownloadRequestResponse) работает нормально. Но если я добавлю флеш после того, как сделаю свой $ response, .zip будет распечатан в html, а не загружен ...
public function getInboxExportAction(Request $request, $orderId)
{
$response = new Response($this->twig->render('base.html.twig',
['content' => '
<h1>Download</h1>
<p>Your zip is being prepared for download, please wait...</p>
']));
$response->headers->set('Content-Type', 'text/html');
//Here I would like to echo + flush my html.
//Then stop the flush and continue my code
$receivedOrder = $this->fetchRequestedReceivedOrder($orderId);
if (!$receivedOrder instanceof ReceivedOrder) {
return $receivedOrder;
}
if (!$receivedOrder->getSentorder()) {
$this->makeErrorResponse('Sent order was not found', Response::HTTP_BAD_REQUEST);
}
return $this->makeExportDownloadRequestResponse($receivedOrder->getSentorder(), $request->get('format'));
}
Я также очень открыт для любых других идей, которые кому-то придется решить эту проблему :)
Спасибо,
Обновлено: моя функция $ this-> makeExportDownloadRequestResponse () возвращает ответ, а не путь. Мы отключаем файл на сервере из соображений хранения.
$content = file_get_contents($zipTmpFile);
unlink($zipTmpFile);
$response = new Response($content);
$dispositionHeader = $response->headers->makeDisposition(ResponseHeaderBag::DISPOSITION_ATTACHMENT,
'myFile_' . date('Y_m_d_h_i_s') . '.zip');
$response->headers->set('Content-Disposition', $dispositionHeader);
return $response;
Второе редактирование: я понимаю, что то, что я пытаюсь сделать (изменить тип содержимого за один вызов), я обычно не одобрял. Сейчас я пытаюсь придумать более элегантное решение.






В вашем getInboxExportAction просто верните ответ со своим текстом.
Затем в этом ответе добавьте тег <meta http-equiv = "refresh" content = "1;url=xxx">, чтобы перенаправить пользователя к другому действию, которое будет генерировать zip-файл.
Вы также можете обработать это перенаправление с помощью javascript.
Обратите внимание, что вы можете использовать StreamedResponse для обработки загрузки zip-архива: https://symfony.com/doc/current/components/http_foundation.html#streaming-a-response
Самое главное, что вы установили $response->headers->set('Content-Type', 'text/html');, браузер не будет скачивать файл .zip.
Если ваш метод makeExportDownloadRequestResponse может вернуть абсолютный путь к файлу .zip (или вы можете его вычислить), вы можете попробовать:
public function getInboxExportAction(Request $request, $orderId)
{
$response = new Response($this->twig->render('base.html.twig',
['content' => '
<h1>Download</h1>
<p>Your zip is being prepared for download, please wait...</p>
']));
$response->headers->set('Content-Type', 'text/html');
//Here I would like to echo + flush my html.
//Then stop the flush and continue my code
$receivedOrder = $this->fetchRequestedReceivedOrder($orderId);
if (!$receivedOrder instanceof ReceivedOrder) {
return $receivedOrder;
}
if (!$receivedOrder->getSentorder()) {
$this->makeErrorResponse('Sent order was not found', Response::HTTP_BAD_REQUEST);
}
// prepare the to be downloaded zip file
// then, do sth to get the .zip file path
$zipFilePath = $this->makeExportDownloadRequestResponse($receivedOrder->getSentorder(), $request->get('format'));
// re-set header
header('Content-Type: application/zip');
header('Content-disposition: attachment; filename=whateverName.zip');
header('Content-Length: ' . filesize($zipFilePath));
readfile($zipFilePath);
}
Это позволит браузеру загрузить zip-файл.
Наконец, если ваша функция работает как API, и для ее выполнения вам нужен интерфейс JS, попробуйте использовать класс Blob на JS.
Вы действительно можете использовать StreamedResponse, чтобы избавиться от этой проблемы. Насчет того, что из-за флеша отображается весь контент .wip файла, попробуйте вызвать ob_flush перед
Я не мог изменить тип содержимого в StreamedResponse, но спасибо за отправку :)
Спасибо за решение, которое звучит довольно изящно. Хотя я не мог использовать его в своем проекте, я оставлю его на следующий раз.