PageRenderTime 28ms CodeModel.GetById 19ms RepoModel.GetById 0ms app.codeStats 0ms

/Gallary/modules/gallery/helpers/legal_file.php

https://bitbucket.org/JakePratt/kupcakz.com
PHP | 310 lines | 182 code | 18 blank | 110 comment | 38 complexity | f745b0f5c50c55af536739e3061a78e4 MD5 | raw file
  1. <?php defined("SYSPATH") or die("No direct script access.");
  2. /**
  3. * Gallery - a web based photo album viewer and editor
  4. * Copyright (C) 2000-2013 Bharat Mediratta
  5. *
  6. * This program is free software; you can redistribute it and/or modify
  7. * it under the terms of the GNU General Public License as published by
  8. * the Free Software Foundation; either version 2 of the License, or (at
  9. * your option) any later version.
  10. *
  11. * This program is distributed in the hope that it will be useful, but
  12. * WITHOUT ANY WARRANTY; without even the implied warranty of
  13. * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
  14. * General Public License for more details.
  15. *
  16. * You should have received a copy of the GNU General Public License
  17. * along with this program; if not, write to the Free Software
  18. * Foundation, Inc., 51 Franklin Street - Fifth Floor, Boston, MA 02110-1301, USA.
  19. */
  20. class legal_file_Core {
  21. private static $photo_types_by_extension;
  22. private static $movie_types_by_extension;
  23. private static $photo_extensions;
  24. private static $movie_extensions;
  25. private static $photo_types;
  26. private static $movie_types;
  27. private static $blacklist = array("php", "php3", "php4", "php5", "phtml", "phtm", "shtml", "shtm",
  28. "pl", "cgi", "asp", "sh", "py", "c", "js");
  29. /**
  30. * Create a default list of allowed photo MIME types paired with their extensions and then let
  31. * modules modify it. This is an ordered map, mapping extensions to their MIME types.
  32. * Extensions cannot be duplicated, but MIMEs can (e.g. jpeg and jpg both map to image/jpeg).
  33. *
  34. * @param string $extension (opt.) - return MIME of extension; if not given, return complete array
  35. */
  36. static function get_photo_types_by_extension($extension=null) {
  37. if (empty(self::$photo_types_by_extension)) {
  38. $types_by_extension_wrapper = new stdClass();
  39. $types_by_extension_wrapper->types_by_extension = array(
  40. "jpg" => "image/jpeg", "jpeg" => "image/jpeg", "gif" => "image/gif", "png" => "image/png");
  41. module::event("photo_types_by_extension", $types_by_extension_wrapper);
  42. foreach (self::$blacklist as $key) {
  43. unset($types_by_extension_wrapper->types_by_extension[$key]);
  44. }
  45. self::$photo_types_by_extension = $types_by_extension_wrapper->types_by_extension;
  46. }
  47. if ($extension) {
  48. // return matching MIME type
  49. $extension = strtolower($extension);
  50. if (isset(self::$photo_types_by_extension[$extension])) {
  51. return self::$photo_types_by_extension[$extension];
  52. } else {
  53. return null;
  54. }
  55. } else {
  56. // return complete array
  57. return self::$photo_types_by_extension;
  58. }
  59. }
  60. /**
  61. * Create a default list of allowed movie MIME types paired with their extensions and then let
  62. * modules modify it. This is an ordered map, mapping extensions to their MIME types.
  63. * Extensions cannot be duplicated, but MIMEs can (e.g. jpeg and jpg both map to image/jpeg).
  64. *
  65. * @param string $extension (opt.) - return MIME of extension; if not given, return complete array
  66. */
  67. static function get_movie_types_by_extension($extension=null) {
  68. if (empty(self::$movie_types_by_extension)) {
  69. $types_by_extension_wrapper = new stdClass();
  70. $types_by_extension_wrapper->types_by_extension = array(
  71. "flv" => "video/x-flv", "mp4" => "video/mp4", "m4v" => "video/x-m4v");
  72. module::event("movie_types_by_extension", $types_by_extension_wrapper);
  73. foreach (self::$blacklist as $key) {
  74. unset($types_by_extension_wrapper->types_by_extension[$key]);
  75. }
  76. self::$movie_types_by_extension = $types_by_extension_wrapper->types_by_extension;
  77. }
  78. if ($extension) {
  79. // return matching MIME type
  80. $extension = strtolower($extension);
  81. if (isset(self::$movie_types_by_extension[$extension])) {
  82. return self::$movie_types_by_extension[$extension];
  83. } else {
  84. return null;
  85. }
  86. } else {
  87. // return complete array
  88. return self::$movie_types_by_extension;
  89. }
  90. }
  91. /**
  92. * Create a merged list of all allowed photo and movie MIME types paired with their extensions.
  93. *
  94. * @param string $extension (opt.) - return MIME of extension; if not given, return complete array
  95. */
  96. static function get_types_by_extension($extension=null) {
  97. $types_by_extension = legal_file::get_photo_types_by_extension();
  98. if (movie::allow_uploads()) {
  99. $types_by_extension = array_merge($types_by_extension,
  100. legal_file::get_movie_types_by_extension());
  101. }
  102. if ($extension) {
  103. // return matching MIME type
  104. $extension = strtolower($extension);
  105. if (isset($types_by_extension[$extension])) {
  106. return $types_by_extension[$extension];
  107. } else {
  108. return null;
  109. }
  110. } else {
  111. // return complete array
  112. return $types_by_extension;
  113. }
  114. }
  115. /**
  116. * Create a default list of allowed photo extensions and then let modules modify it.
  117. *
  118. * @param string $extension (opt.) - return true if allowed; if not given, return complete array
  119. */
  120. static function get_photo_extensions($extension=null) {
  121. if (empty(self::$photo_extensions)) {
  122. $extensions_wrapper = new stdClass();
  123. $extensions_wrapper->extensions = array_keys(legal_file::get_photo_types_by_extension());
  124. module::event("legal_photo_extensions", $extensions_wrapper);
  125. self::$photo_extensions = array_diff($extensions_wrapper->extensions, self::$blacklist);
  126. }
  127. if ($extension) {
  128. // return true if in array, false if not
  129. return in_array(strtolower($extension), self::$photo_extensions);
  130. } else {
  131. // return complete array
  132. return self::$photo_extensions;
  133. }
  134. }
  135. /**
  136. * Create a default list of allowed movie extensions and then let modules modify it.
  137. *
  138. * @param string $extension (opt.) - return true if allowed; if not given, return complete array
  139. */
  140. static function get_movie_extensions($extension=null) {
  141. if (empty(self::$movie_extensions)) {
  142. $extensions_wrapper = new stdClass();
  143. $extensions_wrapper->extensions = array_keys(legal_file::get_movie_types_by_extension());
  144. module::event("legal_movie_extensions", $extensions_wrapper);
  145. self::$movie_extensions = array_diff($extensions_wrapper->extensions, self::$blacklist);
  146. }
  147. if ($extension) {
  148. // return true if in array, false if not
  149. return in_array(strtolower($extension), self::$movie_extensions);
  150. } else {
  151. // return complete array
  152. return self::$movie_extensions;
  153. }
  154. }
  155. /**
  156. * Create a merged list of all allowed photo and movie extensions.
  157. *
  158. * @param string $extension (opt.) - return true if allowed; if not given, return complete array
  159. */
  160. static function get_extensions($extension=null) {
  161. $extensions = legal_file::get_photo_extensions();
  162. if (movie::allow_uploads()) {
  163. $extensions = array_merge($extensions, legal_file::get_movie_extensions());
  164. }
  165. if ($extension) {
  166. // return true if in array, false if not
  167. return in_array(strtolower($extension), $extensions);
  168. } else {
  169. // return complete array
  170. return $extensions;
  171. }
  172. }
  173. /**
  174. * Create a merged list of all photo and movie filename filters,
  175. * (e.g. "*.gif"), based on allowed extensions.
  176. */
  177. static function get_filters() {
  178. $filters = array();
  179. foreach (legal_file::get_extensions() as $extension) {
  180. array_push($filters, "*." . $extension, "*." . strtoupper($extension));
  181. }
  182. return $filters;
  183. }
  184. /**
  185. * Create a default list of allowed photo MIME types and then let modules modify it.
  186. * Can be used to add legal alternatives for default MIME types.
  187. * (e.g. flv maps to video/x-flv by default, but video/flv is still legal).
  188. */
  189. static function get_photo_types() {
  190. if (empty(self::$photo_types)) {
  191. $types_wrapper = new stdClass();
  192. // Need array_unique since types_by_extension can be many-to-one (e.g. jpeg and jpg).
  193. $types_wrapper->types = array_unique(array_values(legal_file::get_photo_types_by_extension()));
  194. module::event("legal_photo_types", $types_wrapper);
  195. self::$photo_types = $types_wrapper->types;
  196. }
  197. return self::$photo_types;
  198. }
  199. /**
  200. * Create a default list of allowed movie MIME types and then let modules modify it.
  201. * Can be used to add legal alternatives for default MIME types.
  202. * (e.g. flv maps to video/x-flv by default, but video/flv is still legal).
  203. */
  204. static function get_movie_types() {
  205. if (empty(self::$movie_types)) {
  206. $types_wrapper = new stdClass();
  207. // Need array_unique since types_by_extension can be many-to-one (e.g. jpeg and jpg).
  208. $types_wrapper->types = array_unique(array_values(legal_file::get_movie_types_by_extension()));
  209. $types_wrapper->types[] = "video/flv";
  210. module::event("legal_movie_types", $types_wrapper);
  211. self::$movie_types = $types_wrapper->types;
  212. }
  213. return self::$movie_types;
  214. }
  215. /**
  216. * Change the extension of a filename. If the original filename has no
  217. * extension, add the new one to the end.
  218. */
  219. static function change_extension($filename, $new_ext) {
  220. $filename_no_ext = preg_replace("/\.[^\.\/]*?$/", "", $filename);
  221. return "{$filename_no_ext}.{$new_ext}";
  222. }
  223. /**
  224. * Reduce the given file to having a single extension.
  225. */
  226. static function smash_extensions($filename) {
  227. if (!$filename) {
  228. // It's harmless, so return it before it causes issues with pathinfo.
  229. return $filename;
  230. }
  231. $parts = pathinfo($filename);
  232. $result = "";
  233. if ($parts["dirname"] != ".") {
  234. $result .= $parts["dirname"] . "/";
  235. }
  236. $parts["filename"] = str_replace(".", "_", $parts["filename"]);
  237. $parts["filename"] = preg_replace("/[_]+/", "_", $parts["filename"]);
  238. $parts["filename"] = trim($parts["filename"], "_");
  239. $result .= isset($parts["extension"]) ? "{$parts['filename']}.{$parts['extension']}" : $parts["filename"];
  240. return $result;
  241. }
  242. /**
  243. * Sanitize a filename for a given type (given as "photo" or "movie") and a target file format
  244. * (given as an extension). This returns a completely legal and valid filename,
  245. * or throws an exception if the type or extension given is invalid or illegal. It tries to
  246. * maintain the filename's original extension even if it's not identical to the given extension
  247. * (e.g. don't change "JPG" or "jpeg" to "jpg").
  248. *
  249. * Note: it is not okay if the extension given is legal but does not match the type (e.g. if
  250. * extension is "mp4" and type is "photo", it will throw an exception)
  251. *
  252. * @param string $filename (with no directory)
  253. * @param string $extension (can be uppercase or lowercase)
  254. * @param string $type (as "photo" or "movie")
  255. * @return string sanitized filename (or null if bad extension argument)
  256. */
  257. static function sanitize_filename($filename, $extension, $type) {
  258. // Check if the type is valid - if so, get the mime types of the
  259. // original and target extensions; if not, throw an exception.
  260. $original_extension = pathinfo($filename, PATHINFO_EXTENSION);
  261. switch ($type) {
  262. case "photo":
  263. $mime_type = legal_file::get_photo_types_by_extension($extension);
  264. $original_mime_type = legal_file::get_photo_types_by_extension($original_extension);
  265. break;
  266. case "movie":
  267. $mime_type = legal_file::get_movie_types_by_extension($extension);
  268. $original_mime_type = legal_file::get_movie_types_by_extension($original_extension);
  269. break;
  270. default:
  271. throw new Exception("@todo INVALID_TYPE");
  272. }
  273. // Check if the target extension is blank or invalid - if so, throw an exception.
  274. if (!$extension || !$mime_type) {
  275. throw new Exception("@todo ILLEGAL_EXTENSION");
  276. }
  277. // Check if the mime types of the original and target extensions match - if not, fix it.
  278. if (!$original_extension || ($mime_type != $original_mime_type)) {
  279. $filename = legal_file::change_extension($filename, $extension);
  280. }
  281. // It should be a filename without a directory - remove all slashes (and backslashes).
  282. $filename = str_replace("/", "_", $filename);
  283. $filename = str_replace("\\", "_", $filename);
  284. // Remove extra dots from the filename. This will also remove extraneous underscores.
  285. $filename = legal_file::smash_extensions($filename);
  286. // It's possible that the filename has no base (e.g. ".jpg") - if so, give it a generic one.
  287. if (empty($filename) || (substr($filename, 0, 1) == ".")) {
  288. $filename = $type . $filename; // e.g. "photo.jpg" or "movie.mp4"
  289. }
  290. return $filename;
  291. }
  292. }