从Controller symfony2返回图像

时间:2013-07-01 16:18:08

标签: symfony controller response

我想知道如何在没有任何模板的情况下从控制器返回图像。 我想将它用于时事通讯中的像素跟踪。

我从这段代码开始

    $image = "1px.png";
    $file =    readfile("/path/to/my/image/1px.png");
    $headers = array(
        'Content-Type'     => 'image/png',
        'Content-Disposition' => 'inline; filename="'.$file.'"');
    return new Response($image, 200, $headers);

但是在导航器上我有一个断开的链接(找不到文件......)

6 个答案:

答案 0 :(得分:24)

根据serving files BinaryFileResponse时的Symfony文档,您可以使用AccessLogApacheMiddleware

use Symfony\Component\HttpFoundation\BinaryFileResponse;
$file = 'path/to/file.txt';
$response = new BinaryFileResponse($file);
// you can modify headers here, before returning
return $response;

此BinaryFileResponse会自动处理一些HTTP请求标头,并使您免于使用readfile()或其他文件功能。

答案 1 :(得分:10)

现在您将文件名作为响应正文返回,并将文件内容写入Content-Disposition的filename属性。

return new Response($image, 200, $headers);

应该是:

return new Response($file, 200, $headers);

......和......

'Content-Disposition' => 'inline; filename="'.$file.'"');

应该是......

'Content-Disposition' => 'inline; filename="'.$image.'"');

正确?

进一步查看this question

答案 2 :(得分:3)

这对我来说很好。

$filepath = "/path/to/my/image/chart.png";
$filename = "chart.png";

$response = new Response();
$disposition = $response->headers->makeDisposition(ResponseHeaderBag::DISPOSITION_INLINE, $filename);
$response->headers->set('Content-Disposition', $disposition);
$response->headers->set('Content-Type', 'image/png');
$response->setContent(file_get_contents($filepath));

return $response;

答案 3 :(得分:2)

另外,我不得不改变:

$file =    readfile("/path/to/my/image/1px.png");

到此:

$file =    file_get_contents("/path/to/my/image/1px.png");

似乎像readfile一样回显内容,因为它读取它强制标头提前输出并取消强制Content-Type标头。

答案 4 :(得分:2)

快速浏览器

use Symfony\Bundle\FrameworkBundle\Controller\Controller;
use Symfony\Component\HttpFoundation\Response;
use Symfony\Component\HttpFoundation\ResponseHeaderBag;

class AnotherController extends Controller {

  public function imagesAction($img = 'my_pixel_tracking.png'){

    $filepath = '/path/to/images/'.$img;

      if(file_exists($filepath)){
        $response = new Response();
        $disposition = $response->headers->makeDisposition(ResponseHeaderBag::DISPOSITION_INLINE, $img);
        $response->headers->set('Content-Disposition', $disposition);
        $response->headers->set('Content-Type', 'image/png');
        $response->setContent(file_get_contents($filepath));
        return $response;
      }
      else{
        return $this->redirect($this->generateUrl('my_url_to_site_index'));
      }
  }
}

答案 5 :(得分:1)

file_get_contents是个坏主意。通过file_get_contents读取大量图像会杀死我的小服务器。我必须找到另一个解决方案,这对我来说现在非常完美和快速。

关键是使用readfile($ sFileName)而不是file_get_contents。 Symfony Stream Response能够采用将在发送时执行的回调函数($ oResponse-> send())。所以这是一个使用readfile()的好地方。

作为一个小小的好处,我写下了一种缓存方式。

use Symfony\Component\HttpFoundation\Request;
use Symfony\Component\HttpFoundation\Response;
use Symfony\Component\HttpFoundation\StreamedResponse;

class ImageController
{

    public function indexAction(Request $oRequest, Response $oResponse)
    {
        // Get the filename from the request 
        // e.g. $oRequest->get("imagename")
        $sFileName = "/images_directory/demoimage.jpg";
        if( ! is_file($sFileName)){
            $oResponse->setStatusCode(404);

            return $oResponse;
        }

        // Caching...
        $sLastModified = filemtime($sFileName);
        $sEtag = md5_file($sFileName);

        $sFileSize = filesize($sFileName);
        $aInfo = getimagesize($sFileName);

        if(in_array($sEtag, $oRequest->getETags()) || $oRequest->headers->get('If-Modified-Since') === gmdate("D, d M Y H:i:s", $sLastModified)." GMT" ){
            $oResponse->headers->set("Content-Type", $aInfo['mime']);
            $oResponse->headers->set("Last-Modified", gmdate("D, d M Y H:i:s", $sLastModified)." GMT");
            $oResponse->setETag($sEtag);
            $oResponse->setPublic();
            $oResponse->setStatusCode(304);

            return $oResponse;
        }

        $oStreamResponse = new StreamedResponse();
        $oStreamResponse->headers->set("Content-Type", $aInfo['mime']);
        $oStreamResponse->headers->set("Content-Length", $sFileSize);
        $oStreamResponse->headers->set("ETag", $sEtag);
        $oStreamResponse->headers->set("Last-Modified", gmdate("D, d M Y H:i:s", $sLastModified)." GMT");

        $oStreamResponse->setCallback(function() use ($sFileName) {
            readfile($sFileName);
        });

        return $oStreamResponse;
    }
}