创建调整大小的图像缓存,但防止滥用

时间:2012-06-23 11:16:37

标签: php image .htaccess caching

好的,所以我对如何服务和缓存图片有了一个想法。 我不知道这是否是正确的做法,但如果确实如此,我想知道如何防止滥用。

情况:

的index.php

<img src="images/cache/200x150-picture_001.jpg" />

图片/高速缓存/ htaccess的

RewriteEngine On
RewriteCond %{REQUEST_FILENAME} !-f
RewriteCond %{REQUEST_FILENAME} !-d
RewriteRule ^(.*)$ images/image.php?f=$1 [L]

以上检查图像是否存在,如果不存在则将其重写为image.php

image.php PSEUDO代码

  

从文件名

获取高度和宽度      

调整大小并将图像保存到缓存文件夹

     

使用content-type和readfile为图像提供服务。

这样我就试图用readfiles减少HTTP请求和PHP加载。 浏览器获取image / jpeg(如果存在),如果不存在则生成。

最终所有图片都会被缓存并以正确的高度和宽度提供,因此浏览器不会下载超大尺寸的图片。

唯一的问题..如果您将图片网址更改为不同的尺寸,则可以填满服务器。

我做得对,做正确的方法是什么。有没有机会阻止这种情况发生?

我知道这个缓存的整个概念已经被提炼了一百万次,请赐教。

3 个答案:

答案 0 :(得分:1)

一些方法:

  • 维护一组允许的分辨率,并检查所请求的分辨率是否在该阵列中。缺点:如果不编辑阵列,则无法快速添加分辨率。

  • 如果这是在CMS上下文中:仅允许经过身份验证的用户创建图像(尚未在缓存中);否则拒绝请求。当经过身份验证的用户在CMS中添加图像时,他们会预览图像,然后执行此操作会生成调整大小的图像。缺点:实施起来并不容易。

答案 1 :(得分:1)

缓存是一个非常重要的问题 - 您的解决方案似乎是合理的,但确实对有意和无意的拒绝服务攻击持开放态度。它也没有解决图像更改时发生的情况 - 如何从缓存中删除所有已调整大小的图像?它不会设置缓存标头以允许“下游”缓存。它没有处理同时刷新整个缓存的风险,要求在HTTP请求的上下文中重新生成所有图像,这可能是主要的性能消耗。

您是否看过Apache's caching module等“现成”解决方案?

答案 2 :(得分:0)

我只是这样做:

<img src="/thumbnail.php?thumb=mybigpicture.ext" ... />

以下是我的表现方式。请注意,此实现可以扩展图像,无论它们是否比较高,反之亦然,并且与大多数PHP尝试不同,它可以进行大规模的缩放。

<?php

function thumb_image($request = "") {
  $cfgthumb['folder'] = "/images/cache";
  $cfgthumb['height'] = 150;
  $cfgthumb['width'] = 200;
  $cfgthumb['error'] = "/images/error.jpg";
  $cfgthumb['default'] = "/images/notfound.jpg";

  $thumb = $cfgthumb['folder'] . "/" . md5($request);
  header("Content-Type: image/jpeg");
  if (is_readable($thumb)) echo file_get_contents($thumb);
  elseif (is_readable($request)) {
    $extension = strtolower(end(explode(".", $request)));
    switch ($extension) {
    case "gif":
      $simage = imagecreatefromgif($request);
      break;
    case "jpeg":
    case "jpg":
      $simage = imagecreatefromjpeg($request);
      break;
    case "png":
      $simage = imagecreatefrompng($request);
      break;
    }
    if ($simage) {
      $simage_width = imagesx($simage);
      $simage_height = imagesy($simage);
      if (($simage_width > $cfgthumb['width']) || ($simage_height > $cfgthumb['height'])) {
        if ($simage_width > $simage_height) {
          $dimage_width = $cfgthumb['width'];
          $dimage_height = floor($simage_height * ($cfgthumb['width'] / $simage_width));
        } else {
          $dimage_width = floor($simage_width * ($cfgthumb['height'] / $simage_height));
          $dimage_height = $cfgthumb['height'];
        }
      } else {
        $dimage_width = $simage_width;
        $dimage_height = $simage_height;
      }
      $dimage = imagecreatetruecolor($dimage_width, $dimage_height);
      imagegammacorrect($simage, 2.2, 1.0);
      imagecopyresampled($dimage, $simage, 0, 0, 0, 0, $dimage_width, $dimage_height, $simage_width, $simage_height);
      imagegammacorrect($dimage, 1.0, 2.2);
      imagejpeg($dimage, $thumb, 100);
      imagejpeg($dimage, NULL, 100);
      imagedestroy($simage);
      imagedestroy($dimage);
    } else echo file_get_contents($cfgthumb['error']);
  } else echo file_get_contents($cfgthumb['default']);
}

?>