= 0) { $limits = get_config('system', 'imagick_limits', false); if ($limits) foreach ($limits as $k => $v) IMagick::setResourceLimit($k, $v); $ph = new PhotoImagick($data, $type); } else { // earlier imagick versions have issues with scaling png's // don't log this because it will just fill the logfile. // leave this note here so those who are looking for why // we aren't using imagick can find it } } if(! $ph) { $ph = new PhotoGd($data, $type); } return $ph; } /** * @brief Guess image mimetype from filename or from Content-Type header. * * @param string $filename * Image filename * @param string $data (optional) * Data array fetched from cURL with z_fetch_url * @return null|string Guessed mimetype */ function guess_image_type($filename, $data = '') { if($data) $headers = (is_array($data) ? $data['header'] : $data); // logger('Photo: guess_image_type: '.$filename . ($headers?' from curl headers':''), LOGGER_DEBUG); $type = null; $m = null; $ph = photo_factory(''); $types = $ph->supportedTypes(); if($headers) { $hdrs = []; $h = explode("\n", $headers); foreach ($h as $l) { if (strpos($l, ':') === false) { continue; } list($k, $v) = array_map('trim', explode(':', trim($l), 2)); $hdrs[strtolower($k)] = $v; } logger('Curl headers: ' .var_export($hdrs, true), LOGGER_DEBUG); if(array_key_exists('content-type', $hdrs) && array_key_exists($hdrs['content-type'], $types)) $type = $hdrs['content-type']; } if(is_null($type)){ $ignore_imagick = get_config('system', 'ignore_imagick'); // Guessing from extension? Isn't that... dangerous? if(class_exists('Imagick') && ! $ignore_imagick) { $v = Imagick::getVersion(); preg_match('/ImageMagick ([0-9]+\.[0-9]+\.[0-9]+)/', $v['versionString'], $m); if(version_compare($m[1], '6.6.7') >= 0) { /** * Well, this not much better, * but at least it comes from the data inside the image, * we won't be tricked by a manipulated extension */ $body = false; if (strpos($filename, 'http') === false && file_exists($filename) && is_readable($filename)) $body == file_get_contents($filename); elseif (is_array($data) && array_key_exists('body', $data)) $body = $data['body']; if ($body) { $image = new Imagick(); $image->readImageBlob($body); $r = $image->identifyImage(); if ($r && is_array($r) && array_key_exists($r['mimetype'], $types)) $type = $r['mimetype']; } } else { // earlier imagick versions have issues with scaling png's // don't log this because it will just fill the logfile. // leave this note here so those who are looking for why // we aren't using imagick can find it } } if(is_null($type)) { $ext = pathinfo($filename, PATHINFO_EXTENSION); foreach($types as $m => $e) { if($ext === $e) { $type = $m; } } } if(is_null($type) && strpos($filename, 'http') === 0) { $size = getimagesize($filename); if (array_key_exists($size['mime'], $types)) $type = $size['mime']; } if(is_null($type)) { if(strpos(strtolower($filename),'jpg') !== false) $type = 'image/jpeg'; elseif(strpos(strtolower($filename),'jpeg') !== false) $type = 'image/jpeg'; elseif(strpos(strtolower($filename),'gif') !== false) $type = 'image/gif'; elseif(strpos(strtolower($filename),'png') !== false) $type = 'image/png'; elseif(strpos(strtolower($filename),'webp') !== false) $type = 'image/webp'; } } logger('Photo: guess_image_type: filename = ' . $filename . ' type = ' . $type, LOGGER_DEBUG); return $type; } /** * @brief Delete thing photo from database. * * @param string $url * @param string $ob_hash * @return void */ function delete_thing_photo($url, $ob_hash) { $hash = basename($url); $hash = substr($hash, 0, strpos($hash, '-')); // hashes should be 32 bytes. if((! $ob_hash) || (strlen($hash) < 16)) return; q("delete from photo where xchan = '%s' and photo_usage = %d and resource_id = '%s'", dbesc($ob_hash), intval(PHOTO_THING), dbesc($hash) ); } /** * @brief Fetches a photo from external site and prepares its miniatures. * * @param string $photo * external URL to fetch base image * @param string $xchan * channel unique hash * @param boolean $thing * TRUE if this is a thing URL * @param boolean $force * TRUE if ignore image modification date check (force fetch) * * @return array of results * * \e string \b 0 => local URL to full image * * \e string \b 1 => local URL to standard thumbnail * * \e string \b 2 => local URL to micro thumbnail * * \e string \b 3 => image type * * \e boolean \b 4 => TRUE if fetch failure * * \e string \b 5 => modification date */ function import_xchan_photo($photo, $xchan, $thing = false, $force = false) { $modified = ''; $o = null; $flags = (($thing) ? PHOTO_THING : PHOTO_XCHAN); $album = (($thing) ? 'Things' : 'Contact Photos'); logger('Updating channel photo from ' . $photo . ' for ' . $xchan, LOGGER_DEBUG); if($thing) $hash = photo_new_resource(); else { $r = q("SELECT resource_id, edited, mimetype, expires, description FROM photo WHERE xchan = '%s' AND photo_usage = %d AND imgscale = %d LIMIT 1", dbesc($xchan), intval(PHOTO_XCHAN), intval(PHOTO_RES_PROFILE_300) ); if($r) { $hash = $r[0]['resource_id']; $modified = $r[0]['edited']; $type = $r[0]['mimetype']; $exp = strtotime($r[0]['expires']); $etag = $r[0]['description']; } else $hash = photo_new_resource(); } $photo_failure = false; $img_str = ''; if($photo && strpos($photo, z_root() . '/' . get_default_profile_photo()) === false) { if($force || empty($modified)) $result = z_fetch_url($photo, true); else { $h = []; $h[] = "If-Modified-Since: " . gmdate("D, d M Y H:i:s", $exp) . " GMT"; if(! empty($etag)) $h[] = "If-None-Match: " . $etag; $result = z_fetch_url($photo, true, 0, [ 'headers' => $h ]); } if(isset($result)) { $hdrs = []; $h = explode("\n", $result['header']); foreach ($h as $l) { if (strpos($l, ':') === false) { continue; } list($t,$v) = array_map("trim", explode(":", trim($l), 2)); $hdrs[strtolower($t)] = $v; } if(array_key_exists('expires', $hdrs)) $expires = strtotime($hdrs['expires']); if($expires - 60 < time()) $expires = time() + 60; else { $cc = ''; if(array_key_exists('cache-control', $hdrs)) $cc = $hdrs['cache-control']; if(strpos($cc, 'no-cache')) $expires = time() + 60; else { $ttl = (preg_match('/max-age=(\d+)/i', $cc, $o) ? intval($o[1]) : 86400); $expires = time() + $ttl; } } $modified = gmdate('Y-m-d H:i:s', (array_key_exists('last-modified', $hdrs) ? strtotime($hdrs['last-modified']) : time())); if($result['success']) { $img_str = $result['body']; $type = guess_image_type($photo, $result); if(is_null($type)) $photo_failure = true; } else $photo_failure = true; } if(! isset($result) || $result['return_code'] == 304) { $photo = z_root() . '/photo/' . $hash . '-4'; $thumb = z_root() . '/photo/' . $hash . '-5'; $micro = z_root() . '/photo/' . $hash . '-6'; if(isset($result)) q("UPDATE photo SET expires = '%s' WHERE xchan = '%s' and photo_usage = %d and imgscale IN (4, 5, 6)", dbescdate(gmdate('Y-m-d H:i:s', (isset($expires) ? $expires : time() + 86400))), dbesc($xchan), intval(PHOTO_XCHAN) ); $photo_notmodified = true; $photo_failure = true; } } else $photo_failure = true; if(! $photo_failure) { $img = photo_factory($img_str, $type); if($img->is_valid()) { $width = $img->getWidth(); $height = $img->getHeight(); if($width && $height) { if(($width / $height) > 1.2) { // crop out the sides $margin = $width - $height; $img->cropImage(300, ($margin / 2), 0, $height, $height); } elseif(($height / $width) > 1.2) { // crop out the bottom $margin = $height - $width; $img->cropImage(300, 0, 0, $width, $width); } else $img->scaleImageSquare(300); } else $photo_failure = true; $p = [ 'xchan' => $xchan, 'resource_id' => $hash, 'filename' => basename($photo), 'album' => $album, 'photo_usage' => $flags, 'edited' => $modified, 'description' => (array_key_exists('etag', $hdrs) ? $hdrs['etag'] : ''), 'expires' => gmdate('Y-m-d H:i:s', (isset($expires) ? $expires : time() + 86400)) ]; $r1 = $img->storeThumbnail($p, PHOTO_RES_PROFILE_300); $img->scaleImage(80); $r2 = $img->storeThumbnail($p, PHOTO_RES_PROFILE_80); $img->scaleImage(48); $r3 = $img->storeThumbnail($p, PHOTO_RES_PROFILE_48); if($r1 === false || $r2 === false || $r3 === false) $photo_failure = true; $photo = z_root() . '/photo/' . $hash . '-4'; $thumb = z_root() . '/photo/' . $hash . '-5'; $micro = z_root() . '/photo/' . $hash . '-6'; } else { logger('Invalid image from ' . $photo); $photo_failure = true; } } if($photo_failure && ! isset($photo_notmodified)) { $default = get_default_profile_photo(); $photo = z_root() . '/' . $default; $thumb = z_root() . '/' . get_default_profile_photo(80); $micro = z_root() . '/' . get_default_profile_photo(48); $type = 'image/png'; $modified = gmdate('Y-m-d H:i:s', filemtime($default)); } logger('xchan: ' . $xchan . '; HTTP code: ' . (isset($result) && array_key_exists('return_code', $result) ? $result['return_code'] : 'none') . '; modified: ' . $modified . '; URL: ' . $photo, LOGGER_DEBUG); return([ $photo, $thumb, $micro, $type, $photo_failure, $modified ]); } /** * @brief Import channel photo from a URL. * * @param string $photo URL to a photo * @param int $aid * @param int $uid channel_id * @return null|string Guessed image mimetype or null. */ function import_channel_photo_from_url($photo, $aid, $uid) { $type = null; if($photo) { $result = z_fetch_url($photo, true); if($result['success']) { $img_str = $result['body']; $type = guess_image_type($photo, $result); import_channel_photo($img_str, $type, $aid, $uid); } } return $type; } /** * @brief Import a channel photo and prepare its miniatures. * * @param string $photo Image data * @param string $type * @param int $aid * @param int $uid channel_id * @return boolean|string false on failure, otherwise resource_id of photo */ function import_channel_photo($photo, $type, $aid, $uid) { logger('Importing channel photo for ' . $uid, LOGGER_DEBUG); $r = q("SELECT resource_id FROM photo WHERE uid = %d AND photo_usage = %d AND imgscale = %d", intval($uid), intval(PHOTO_PROFILE), intval(PHOTO_RES_PROFILE_300) ); if ($r) $hash = $r[0]['resource_id']; else $hash = photo_new_resource(); $photo_failure = false; $filename = $hash; $img = photo_factory($photo, $type); if($img->is_valid()) { // config array for image save method $p = [ 'aid' => $aid, 'uid' => $uid, 'resource_id' => $hash, 'filename' => $filename, 'album' => t('Profile Photos'), 'photo_usage' => PHOTO_PROFILE ]; // photo size $img->scaleImageSquare(300); $r1 = $img->storeThumbnail($p, PHOTO_RES_PROFILE_300); // thumb size $img->scaleImage(80); $r2 = $img->storeThumbnail($p, PHOTO_RES_PROFILE_80); // micro size $img->scaleImage(48); $r3 = $img->storeThumbnail($p, PHOTO_RES_PROFILE_48); if($r1 === false || $r2 === false || $r3 === false) $photo_failure = true; } else { logger('Invalid image.'); $photo_failure = true; } if($photo_failure) return false; else return $hash; }