1f62ea8a1Sandi<?php 2f62ea8a1Sandi/** 3f62ea8a1Sandi * DokuWiki media passthrough file 4f62ea8a1Sandi * 5f62ea8a1Sandi * @license GPL 2 (http://www.gnu.org/licenses/gpl.html) 6f62ea8a1Sandi * @author Andreas Gohr <andi@splitbrain.org> 7f62ea8a1Sandi */ 8f62ea8a1Sandi 9b1a72e3dSmatthiasgrimm if(!defined('DOKU_INC')) define('DOKU_INC',realpath(dirname(__FILE__).'/../../').'/'); 10f62ea8a1Sandi require_once(DOKU_INC.'inc/init.php'); 11f62ea8a1Sandi require_once(DOKU_INC.'inc/common.php'); 12f62ea8a1Sandi require_once(DOKU_INC.'inc/pageutils.php'); 13f62ea8a1Sandi require_once(DOKU_INC.'inc/confutils.php'); 14f62ea8a1Sandi require_once(DOKU_INC.'inc/auth.php'); 15f62ea8a1Sandi $mimetypes = getMimeTypes(); 16f62ea8a1Sandi 17f62ea8a1Sandi //get input 18f62ea8a1Sandi $MEDIA = $_REQUEST['media']; 19f62ea8a1Sandi $CACHE = calc_cache($_REQUEST['cache']); 20f62ea8a1Sandi $WIDTH = $_REQUEST['w']; 21f62ea8a1Sandi $HEIGHT = $_REQUEST['h']; 22f62ea8a1Sandi list($EXT,$MIME) = mimetype($MEDIA); 23f62ea8a1Sandi if($EXT === false){ 24f62ea8a1Sandi $EXT = 'unknown'; 25f62ea8a1Sandi $MIME = 'application/octet-stream'; 26f62ea8a1Sandi } 27f62ea8a1Sandi 28f62ea8a1Sandi //media to local file 29f62ea8a1Sandi if(preg_match('#^(https?|ftp)://#i',$MEDIA)){ 30f62ea8a1Sandi //handle external media 31f62ea8a1Sandi $FILE = get_from_URL($MEDIA,$EXT,$CACHE); 32f62ea8a1Sandi if(!$FILE){ 33f62ea8a1Sandi //download failed - redirect to original URL 34f62ea8a1Sandi header('Location: '.$MEDIA); 35f62ea8a1Sandi exit; 36f62ea8a1Sandi } 37f62ea8a1Sandi }else{ 38f62ea8a1Sandi $MEDIA = cleanID($MEDIA); 39f62ea8a1Sandi if(empty($MEDIA)){ 40f62ea8a1Sandi header("HTTP/1.0 400 Bad Request"); 41f62ea8a1Sandi print 'Bad request'; 42f62ea8a1Sandi exit; 43f62ea8a1Sandi } 44f62ea8a1Sandi 45f62ea8a1Sandi //check permissions (namespace only) 46f62ea8a1Sandi if(auth_quickaclcheck(getNS($MEDIA).':X') < AUTH_READ){ 47f62ea8a1Sandi header("HTTP/1.0 401 Unauthorized"); 48f62ea8a1Sandi //fixme add some image for imagefiles 49f62ea8a1Sandi print 'Unauthorized'; 50f62ea8a1Sandi exit; 51f62ea8a1Sandi } 52f62ea8a1Sandi $FILE = mediaFN($MEDIA); 53f62ea8a1Sandi } 54f62ea8a1Sandi 55f62ea8a1Sandi //check file existance 56f62ea8a1Sandi if(!@file_exists($FILE)){ 57f62ea8a1Sandi header("HTTP/1.0 404 Not Found"); 58f62ea8a1Sandi //FIXME add some default broken image 59f62ea8a1Sandi print 'Not Found'; 60f62ea8a1Sandi exit; 61f62ea8a1Sandi } 62f62ea8a1Sandi 63f62ea8a1Sandi //handle image resizing 64f62ea8a1Sandi if((substr($MIME,0,5) == 'image') && $WIDTH){ 65f62ea8a1Sandi $FILE = get_resized($FILE,$EXT,$WIDTH,$HEIGHT); 66f62ea8a1Sandi } 67f62ea8a1Sandi 68f62ea8a1Sandi 69f62ea8a1Sandi //FIXME set sane cachecontrol headers 70f62ea8a1Sandi //FIXME handle conditional and partial requests 71f62ea8a1Sandi 72f62ea8a1Sandi //send file 73f62ea8a1Sandi header("Content-Type: $MIME"); 74f62ea8a1Sandi header('Last-Modified: '.date('r',filemtime($FILE))); 75f62ea8a1Sandi header('Content-Length: '.filesize($FILE)); 76*ea19fb87Smatthiasgrimm header('Cache-Control: must-revalidate, post-check=0, pre-check=0'); 77f62ea8a1Sandi 78f62ea8a1Sandi //application mime type is downloadable 79f62ea8a1Sandi if(substr($MIME,0,11) == 'application'){ 80f62ea8a1Sandi header('Content-Disposition: attachment; filename="'.basename($FILE).'"'); 81f62ea8a1Sandi } 82f62ea8a1Sandi 83f62ea8a1Sandi $fp = @fopen($FILE,"rb"); 84f62ea8a1Sandi if($fp){ 85f62ea8a1Sandi fpassthru($fp); //does a close itself 86f62ea8a1Sandi }else{ 87f62ea8a1Sandi header("HTTP/1.0 500 Internal Server Error"); 88f62ea8a1Sandi print "Could not read $FILE - bad permissions?"; 89f62ea8a1Sandi } 90f62ea8a1Sandi 91f62ea8a1Sandi/* ------------------------------------------------------------------------ */ 92f62ea8a1Sandi 93f62ea8a1Sandi/** 94f62ea8a1Sandi * Resizes the given image to the given size 95f62ea8a1Sandi * 96f62ea8a1Sandi * @author Andreas Gohr <andi@splitbrain.org> 97f62ea8a1Sandi */ 98f62ea8a1Sandifunction get_resized($file, $ext, $w, $h=0){ 99f62ea8a1Sandi global $conf; 100f62ea8a1Sandi 101f62ea8a1Sandi $md5 = md5($file); 102f62ea8a1Sandi $info = getimagesize($file); 103f62ea8a1Sandi if(!$h) $h = round(($w * $info[1]) / $info[0]); 104f62ea8a1Sandi 105f62ea8a1Sandi 106f62ea8a1Sandi //cache 107f62ea8a1Sandi $local = $conf['mediadir'].'/_cache/'.$md5.'.'.$w.'x'.$h.'.'.$ext; 108f62ea8a1Sandi $mtime = @filemtime($local); // 0 if not exists 109f62ea8a1Sandi 110f62ea8a1Sandi if( $mtime > filemtime($file) || resize_image($ext,$file,$info[0],$info[1],$local,$w,$h) ){ 111f62ea8a1Sandi return $local; 112f62ea8a1Sandi } 113f62ea8a1Sandi //still here? resizing failed 114f62ea8a1Sandi return $file; 115f62ea8a1Sandi} 116f62ea8a1Sandi 117f62ea8a1Sandi/** 118f62ea8a1Sandi * Returns the wanted cachetime in seconds 119f62ea8a1Sandi * 120f62ea8a1Sandi * Resolves named constants 121f62ea8a1Sandi * 122f62ea8a1Sandi * @author Andreas Gohr <andi@splitbrain.org> 123f62ea8a1Sandi */ 124f62ea8a1Sandifunction calc_cache($cache){ 125f62ea8a1Sandi global $conf; 126f62ea8a1Sandi 127f62ea8a1Sandi if(strtolower($cache) == 'nocache') return 0; //never cache 128f62ea8a1Sandi if(strtolower($cache) == 'recache') return $conf['cachetime']; //use standard cache 129f62ea8a1Sandi return -1; //cache endless 130f62ea8a1Sandi} 131f62ea8a1Sandi 132f62ea8a1Sandi/** 133f62ea8a1Sandi * Download a remote file and return local filename 134f62ea8a1Sandi * 135f62ea8a1Sandi * returns false if download fails. Uses cached file if available and 136f62ea8a1Sandi * wanted 137f62ea8a1Sandi * 138f62ea8a1Sandi * @author Andreas Gohr <andi@splitbrain.org> 139f62ea8a1Sandi */ 140f62ea8a1Sandifunction get_from_URL($url,$ext,$cache){ 141f62ea8a1Sandi global $conf; 142f62ea8a1Sandi 143f62ea8a1Sandi $url = strtolower($url); 144f62ea8a1Sandi $md5 = md5($url); 145f62ea8a1Sandi 146f62ea8a1Sandi $local = $conf['mediadir']."/_cache/$md5.$ext"; 147f62ea8a1Sandi $mtime = @filemtime($local); // 0 if not exists 148f62ea8a1Sandi 149f62ea8a1Sandi //decide if download needed: 150f62ea8a1Sandi 151f62ea8a1Sandi // never cache exists but no endless cache not exists or expired 152f62ea8a1Sandi if( $cache == 0 || ($mtime != 0 && $cache != -1) || $mtime < time()-$cache ){ 153f62ea8a1Sandi if(io_download($url,$local)){ 154f62ea8a1Sandi return $local; 155f62ea8a1Sandi }else{ 156f62ea8a1Sandi return false; 157f62ea8a1Sandi } 158f62ea8a1Sandi } 159f62ea8a1Sandi 160f62ea8a1Sandi //if cache exists use it else 161f62ea8a1Sandi if($mtime) return $local; 162f62ea8a1Sandi 163f62ea8a1Sandi //else return false 164f62ea8a1Sandi return false; 165f62ea8a1Sandi} 166f62ea8a1Sandi 167f62ea8a1Sandi/** 168f62ea8a1Sandi * resize images 169f62ea8a1Sandi * 170f62ea8a1Sandi * @author Andreas Gohr <andi@splitbrain.org> 171f62ea8a1Sandi */ 172f62ea8a1Sandifunction resize_image($ext,$from,$from_w,$from_h,$to,$to_w,$to_h){ 173f62ea8a1Sandi global $conf; 174f62ea8a1Sandi 175f62ea8a1Sandi if($conf['gdlib'] < 1) return false; //no GDlib available or wanted 176f62ea8a1Sandi 177f62ea8a1Sandi // create an image of the given filetype 178f62ea8a1Sandi if ($ext == 'jpg' || $ext == 'jpeg'){ 179f62ea8a1Sandi if(!function_exists("imagecreatefromjpeg")) return false; 180f62ea8a1Sandi $image = @imagecreatefromjpeg($from); 181f62ea8a1Sandi }elseif($ext == 'png') { 182f62ea8a1Sandi if(!function_exists("imagecreatefrompng")) return false; 183f62ea8a1Sandi $image = @imagecreatefrompng($from); 184f62ea8a1Sandi 185f62ea8a1Sandi }elseif($ext == 'gif') { 186f62ea8a1Sandi if(!function_exists("imagecreatefromgif")) return false; 187f62ea8a1Sandi $image = @imagecreatefromgif($from); 188f62ea8a1Sandi } 189f62ea8a1Sandi if(!$image) return false; 190f62ea8a1Sandi 191f62ea8a1Sandi if(($conf['gdlib']>1) && function_exists("imagecreatetruecolor")){ 192f62ea8a1Sandi $newimg = @imagecreatetruecolor ($to_w, $to_h); 193f62ea8a1Sandi } 194f62ea8a1Sandi if(!$newimg) $newimg = @imagecreate($to_w, $to_h); 195f62ea8a1Sandi if(!$newimg) return false; 196f62ea8a1Sandi 197f62ea8a1Sandi //keep png alpha channel if possible 198f62ea8a1Sandi if($ext == 'png' && $conf['gdlib']>1 && function_exists('imagesavealpha')){ 199f62ea8a1Sandi imagealphablending($newimg, false); 200f62ea8a1Sandi imagesavealpha($newimg,true); 201f62ea8a1Sandi } 202f62ea8a1Sandi 203f62ea8a1Sandi // create cachedir 204f62ea8a1Sandi io_makeFileDir($to); 205f62ea8a1Sandi 206f62ea8a1Sandi //try resampling first 207f62ea8a1Sandi if(function_exists("imagecopyresampled")){ 208f62ea8a1Sandi if(!@imagecopyresampled($newimg, $image, 0, 0, 0, 0, $to_w, $to_h, $from_w, $from_h)) { 209f62ea8a1Sandi imagecopyresized($newimg, $image, 0, 0, 0, 0, $to_w, $to_h, $from_w, $from_h); 210f62ea8a1Sandi } 211f62ea8a1Sandi }else{ 212f62ea8a1Sandi imagecopyresized($newimg, $image, 0, 0, 0, 0, $to_w, $to_h, $from_w, $from_h); 213f62ea8a1Sandi } 214f62ea8a1Sandi 215f62ea8a1Sandi if ($ext == 'jpg' || $ext == 'jpeg'){ 216f62ea8a1Sandi if(!function_exists("imagejpeg")) return false; 217f62ea8a1Sandi return imagejpeg($newimg, $to, 70); 218f62ea8a1Sandi }elseif($ext == 'png') { 219f62ea8a1Sandi if(!function_exists("imagepng")) return false; 220f62ea8a1Sandi return imagepng($newimg, $to); 221f62ea8a1Sandi }elseif($ext == 'gif') { 222f62ea8a1Sandi if(!function_exists("imagegif")) return false; 223f62ea8a1Sandi return imagegif($newimg, $to); 224f62ea8a1Sandi } 225f62ea8a1Sandi 226f62ea8a1Sandi return false; 227f62ea8a1Sandi} 228f62ea8a1Sandi 229f62ea8a1Sandi 230f62ea8a1Sandi//Setup VIM: ex: et ts=2 enc=utf-8 : 231f62ea8a1Sandi?> 232