Skip to content
Snippets Groups Projects
helper.php 28.8 KiB
Newer Older
 * ownCloud
 *
 * @author Frank Karlitschek
 * @author Jakob Sack
Frank Karlitschek's avatar
Frank Karlitschek committed
 * @copyright 2012 Frank Karlitschek frank@owncloud.org
 *
 * This library is free software; you can redistribute it and/or
 * modify it under the terms of the GNU AFFERO GENERAL PUBLIC LICENSE
 * License as published by the Free Software Foundation; either
 * version 3 of the License, or any later version.
 *
 * This library is distributed in the hope that it will be useful,
 * but WITHOUT ANY WARRANTY; without even the implied warranty of
 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
 * GNU AFFERO GENERAL PUBLIC LICENSE for more details.
 *
 * You should have received a copy of the GNU Affero General Public
 * License along with this library.  If not, see <http://www.gnu.org/licenses/>.
 *
 */
 * Collection of useful functions
Jakob Sack's avatar
Jakob Sack committed
class OC_Helper {
Robin Appelman's avatar
Robin Appelman committed
	private static $mimetypeIcons = array();
	private static $templateManager;
	 * Creates an url using a defined route
	 * @param string $route
	 * @param array $parameters
	 * @return
	 * @internal param array $args with param=>value, will be appended to the returned url
	 * @return string the url
	 *
	 * Returns a url to the given app and file.
	 */
	public static function linkToRoute($route, $parameters = array()) {
		return OC::$server->getURLGenerator()->linkToRoute($route, $parameters);
Robin Appelman's avatar
Robin Appelman committed
	 * @param string $app app
	 * @param string $file file
	 * @param array $args array with param=>value, will be appended to the returned url
Robin Appelman's avatar
Robin Appelman committed
	 * @return string the url
	 * Returns a url to the given app and file.
	public static function linkTo( $app, $file, $args = array() ) {
		return OC::$server->getURLGenerator()->linkTo($app, $file, $args);
	/**
	 * @param $key
	 * @return string url to the online documentation
	 */
	public static function linkToDocs($key) {
		$theme = new OC_Defaults();
	 * Creates an absolute url
Robin Appelman's avatar
Robin Appelman committed
	 * @param string $app app
	 * @param string $file file
	 * @param array $args array with param=>value, will be appended to the returned url
Robin Appelman's avatar
Robin Appelman committed
	 * @return string the url
	 *
	 * Returns a absolute url to the given app and file.
	 */
	public static function linkToAbsolute($app, $file, $args = array()) {
Thomas Tanghus's avatar
Thomas Tanghus committed
		return OC::$server->getURLGenerator()->getAbsoluteURL(
			self::linkTo($app, $file, $args)
		);
	 * Makes an $url absolute
Robin Appelman's avatar
Robin Appelman committed
	 * @param string $url the url
	 * @return string the absolute url
	 *
	 * Returns a absolute url to the given app and file.
	 */
	public static function makeURLAbsolute($url) {
		return OC::$server->getURLGenerator()->getAbsoluteURL($url);
	 * Creates an url for remote use
	 * @param string $service id
	 * @return string the url
	 *
	 * Returns a url to the given service.
	 */
	public static function linkToRemoteBase($service) {
		return self::linkTo('', 'remote.php') . '/' . $service;
	 * Creates an absolute url for remote use
Robin Appelman's avatar
Robin Appelman committed
	 * @param string $service id
Robin Appelman's avatar
Robin Appelman committed
	 * @return string the url
	 *
	 * Returns a absolute url to the given service.
	 */
	public static function linkToRemote($service, $add_slash = true) {
Thomas Tanghus's avatar
Thomas Tanghus committed
		return OC::$server->getURLGenerator()->getAbsoluteURL(
			self::linkToRemoteBase($service)
				. (($add_slash && $service[strlen($service) - 1] != '/') ? '/' : '')
		);
	 * Creates an absolute url for public use
Robin Appelman's avatar
Robin Appelman committed
	 * @param string $service id
Robin Appelman's avatar
Robin Appelman committed
	 * @return string the url
	 *
	 * Returns a absolute url to the given service.
	 */
	public static function linkToPublic($service, $add_slash = false) {
		return OC::$server->getURLGenerator()->getAbsoluteURL(
			self::linkTo(
				'', 'public.php') . '?service=' . $service
				. (($add_slash && $service[strlen($service) - 1] != '/') ? '/' : ''
			)
		);
	 * Creates path to an image
Robin Appelman's avatar
Robin Appelman committed
	 * @param string $app app
	 * @param string $image image name
	 * @return string the url
	 * Returns the path to the image.
	public static function imagePath($app, $image) {
		return OC::$server->getURLGenerator()->imagePath($app, $image);
Frank Karlitschek's avatar
Frank Karlitschek committed
	}
	 * get path to icon of file type
Robin Appelman's avatar
Robin Appelman committed
	 * @param string $mimetype mimetype
	 * @return string the url
	 * Returns the path to the image of this file type.
Michael Gapczynski's avatar
Michael Gapczynski committed
	public static function mimetypeIcon($mimetype) {
Georg Ehrke's avatar
Georg Ehrke committed
		$alias = array(
			'application/octet-stream' => 'file', // use file icon as fallback

			'application/illustrator' => 'image',
			'application/coreldraw' => 'image',
			'application/x-gimp' => 'image',
			'application/x-photoshop' => 'image',

			'application/x-font-ttf' => 'font',
			'application/font-woff' => 'font',
			'application/vnd.ms-fontobject' => 'font',

			'application/json' => 'text/code',
			'application/x-perl' => 'text/code',
			'text/x-shellscript' => 'text/code',
			'application/xml' => 'text/html',
			'text/css' => 'text/code',
			'application/x-tex' => 'text',

			'application/x-compressed' => 'package/x-generic',
			'application/x-7z-compressed' => 'package/x-generic',
			'application/x-deb' => 'package/x-generic',
			'application/x-gzip' => 'package/x-generic',
			'application/x-rar-compressed' => 'package/x-generic',
			'application/x-tar' => 'package/x-generic',
			'application/zip' => 'package/x-generic',

Georg Ehrke's avatar
Georg Ehrke committed
			'application/msword' => 'x-office/document',
			'application/vnd.openxmlformats-officedocument.wordprocessingml.document' => 'x-office/document',
			'application/vnd.openxmlformats-officedocument.wordprocessingml.template' => 'x-office/document',
			'application/vnd.ms-word.document.macroEnabled.12' => 'x-office/document',
			'application/vnd.ms-word.template.macroEnabled.12' => 'x-office/document',
			'application/vnd.oasis.opendocument.text' => 'x-office/document',
			'application/vnd.oasis.opendocument.text-template' => 'x-office/document',
			'application/vnd.oasis.opendocument.text-web' => 'x-office/document',
			'application/vnd.oasis.opendocument.text-master' => 'x-office/document',
			'application/mspowerpoint' => 'x-office/presentation',
Georg Ehrke's avatar
Georg Ehrke committed
			'application/vnd.ms-powerpoint' => 'x-office/presentation',
			'application/vnd.openxmlformats-officedocument.presentationml.presentation' => 'x-office/presentation',
			'application/vnd.openxmlformats-officedocument.presentationml.template' => 'x-office/presentation',
			'application/vnd.openxmlformats-officedocument.presentationml.slideshow' => 'x-office/presentation',
			'application/vnd.ms-powerpoint.addin.macroEnabled.12' => 'x-office/presentation',
			'application/vnd.ms-powerpoint.presentation.macroEnabled.12' => 'x-office/presentation',
			'application/vnd.ms-powerpoint.template.macroEnabled.12' => 'x-office/presentation',
			'application/vnd.ms-powerpoint.slideshow.macroEnabled.12' => 'x-office/presentation',
			'application/vnd.oasis.opendocument.presentation' => 'x-office/presentation',
			'application/vnd.oasis.opendocument.presentation-template' => 'x-office/presentation',
			'application/msexcel' => 'x-office/spreadsheet',
Georg Ehrke's avatar
Georg Ehrke committed
			'application/vnd.ms-excel' => 'x-office/spreadsheet',
			'application/vnd.openxmlformats-officedocument.spreadsheetml.sheet' => 'x-office/spreadsheet',
			'application/vnd.openxmlformats-officedocument.spreadsheetml.template' => 'x-office/spreadsheet',
			'application/vnd.ms-excel.sheet.macroEnabled.12' => 'x-office/spreadsheet',
			'application/vnd.ms-excel.template.macroEnabled.12' => 'x-office/spreadsheet',
			'application/vnd.ms-excel.addin.macroEnabled.12' => 'x-office/spreadsheet',
			'application/vnd.ms-excel.sheet.binary.macroEnabled.12' => 'x-office/spreadsheet',
			'application/vnd.oasis.opendocument.spreadsheet' => 'x-office/spreadsheet',
			'application/vnd.oasis.opendocument.spreadsheet-template' => 'x-office/spreadsheet',
			'text/csv' => 'x-office/spreadsheet',
			'application/msaccess' => 'database',
Michael Gapczynski's avatar
Michael Gapczynski committed
		if (isset($alias[$mimetype])) {
			$mimetype = $alias[$mimetype];
Robin Appelman's avatar
Robin Appelman committed
		if (isset(self::$mimetypeIcons[$mimetype])) {
			return self::$mimetypeIcons[$mimetype];
		}
		// Replace slash and backslash with a minus
Michael Gapczynski's avatar
Michael Gapczynski committed
		$icon = str_replace('/', '-', $mimetype);
Michael Gapczynski's avatar
Michael Gapczynski committed
		if ($mimetype === 'dir') {
			self::$mimetypeIcons[$mimetype] = OC::$WEBROOT . '/core/img/filetypes/folder.png';
			return OC::$WEBROOT . '/core/img/filetypes/folder.png';
		if ($mimetype === 'dir-shared') {
			self::$mimetypeIcons[$mimetype] = OC::$WEBROOT . '/core/img/filetypes/folder-shared.png';
			return OC::$WEBROOT . '/core/img/filetypes/folder-shared.png';
		}
		if ($mimetype === 'dir-external') {
			self::$mimetypeIcons[$mimetype] = OC::$WEBROOT . '/core/img/filetypes/folder-external.png';
			return OC::$WEBROOT . '/core/img/filetypes/folder-external.png';
		}
		if (file_exists(OC::$SERVERROOT . '/core/img/filetypes/' . $icon . '.png')) {
			self::$mimetypeIcons[$mimetype] = OC::$WEBROOT . '/core/img/filetypes/' . $icon . '.png';
			return OC::$WEBROOT . '/core/img/filetypes/' . $icon . '.png';
Michael Gapczynski's avatar
Michael Gapczynski committed

		// Try only the first part of the filetype
		$mimePart = substr($icon, 0, strpos($icon, '-'));
		if (file_exists(OC::$SERVERROOT . '/core/img/filetypes/' . $mimePart . '.png')) {
			self::$mimetypeIcons[$mimetype] = OC::$WEBROOT . '/core/img/filetypes/' . $mimePart . '.png';
			return OC::$WEBROOT . '/core/img/filetypes/' . $mimePart . '.png';
Michael Gapczynski's avatar
Michael Gapczynski committed
		} else {
			self::$mimetypeIcons[$mimetype] = OC::$WEBROOT . '/core/img/filetypes/file.png';
			return OC::$WEBROOT . '/core/img/filetypes/file.png';
	 * get path to preview of file
	 * @param string $path path
	 * @return string the url
	 *
	 * Returns the path to the preview of the file.
	 */
	public static function previewIcon($path) {
		return self::linkToRoute( 'core_ajax_preview', array('x' => 36, 'y' => 36, 'file' => $path ));
Georg Ehrke's avatar
Georg Ehrke committed
	public static function publicPreviewIcon( $path, $token ) {
		return self::linkToRoute( 'core_ajax_public_preview', array('x' => 36, 'y' => 36, 'file' => $path, 't' => $token));
	 * Make a human file size
Robin Appelman's avatar
Robin Appelman committed
	 * @param int $bytes file size in bytes
	 * @return string a human readable file size
	public static function humanFileSize($bytes) {
		if ($bytes < 0) {
		$bytes = round($bytes / 1024, 0);
		$bytes = round($bytes / 1024, 1);
		if ($bytes < 1024) {
		$bytes = round($bytes / 1024, 1);
		if ($bytes < 1024) {
			return "$bytes GB";
		}
		$bytes = round($bytes / 1024, 1);
		if ($bytes < 1024) {
			return "$bytes TB";
		}
		return "$bytes PB";
	 * @param int $bytes file size in bytes
	 * @return string a php parseable file size
	 *
	 * Makes 2048 to 2k and 2^41 to 2048G
	 */
	public static function phpFileSize($bytes) {
		if ($bytes < 0) {
			return "?";
		}
		if ($bytes < 1024) {
			return $bytes . "B";
		}
		$bytes = round($bytes / 1024, 1);
		if ($bytes < 1024) {
			return $bytes . "K";
		}
		$bytes = round($bytes / 1024, 1);
		if ($bytes < 1024) {
			return $bytes . "M";
		}
		$bytes = round($bytes / 1024, 1);
		return $bytes . "G";
	}

	 * Make a computer file size
	 * @param string $str file size in human readable format
Robin Appelman's avatar
Robin Appelman committed
	 * @return int a file size in bytes
	 *
	 * Makes 2kB to 2048.
	 *
	 * Inspired by: http://www.php.net/manual/en/function.filesize.php#92418
	 */
	public static function computerFileSize($str) {
		$str = strtolower($str);
			'b' => 1,
			'k' => 1024,
			'kb' => 1024,
			'mb' => 1024 * 1024,
			'gb' => 1024 * 1024 * 1024,
			'tb' => 1024 * 1024 * 1024 * 1024,
			'pb' => 1024 * 1024 * 1024 * 1024 * 1024,
		if (preg_match('#([kmgtp]?b?)$#si', $str, $matches) && !empty($bytes_array[$matches[1]])) {
			$bytes *= $bytes_array[$matches[1]];
		}

Robin Appelman's avatar
Robin Appelman committed
	/**
	 * Recursive copying of folders
Robin Appelman's avatar
Robin Appelman committed
	 * @param string $src source folder
	 * @param string $dest target folder
	 *
	 */
	static function copyr($src, $dest) {
Robin Appelman's avatar
Robin Appelman committed
				mkdir($dest);
			}
			$files = scandir($src);
			foreach ($files as $file) {
				if ($file != "." && $file != "..") {
Robin Appelman's avatar
Robin Appelman committed
					self::copyr("$src/$file", "$dest/$file");
				}
			}
		} elseif (file_exists($src) && !\OC\Files\Filesystem::isFileBlacklisted($src)) {
Robin Appelman's avatar
Robin Appelman committed
			copy($src, $dest);
		}
	}
Robin Appelman's avatar
Robin Appelman committed
	/**
	 * Recursive deletion of folders
Robin Appelman's avatar
Robin Appelman committed
	 * @param string $dir path to the folder
Robin Appelman's avatar
Robin Appelman committed
	 * @return bool
Robin Appelman's avatar
Robin Appelman committed
	 */
	static function rmdirr($dir) {
		if (is_dir($dir)) {
			$files = scandir($dir);
			foreach ($files as $file) {
				if ($file !== '' && $file !== "." && $file !== "..") {
Robin Appelman's avatar
Robin Appelman committed
					self::rmdirr("$dir/$file");
				}
			}
			rmdir($dir);
Robin Appelman's avatar
Robin Appelman committed
			unlink($dir);
		}
Robin Appelman's avatar
Robin Appelman committed
			return true;
	/**
	 * @return \OC\Files\Type\Detection
	 */
	static public function getMimetypeDetector() {
		if (!self::$mimetypeDetector) {
			self::$mimetypeDetector = new \OC\Files\Type\Detection();
			self::$mimetypeDetector->registerTypeArray(include 'mimetypes.list.php');
		}
		return self::$mimetypeDetector;
	}

	/**
	 * @return \OC\Files\Type\TemplateManager
	 */
	static public function getFileTemplateManager() {
		if (!self::$templateManager) {
			self::$templateManager = new \OC\Files\Type\TemplateManager();
		}
		return self::$templateManager;
	}

	 * Try to guess the mimetype based on filename
	 *
Robin Appelman's avatar
Robin Appelman committed
	 * @param string $path
	 * @return string
	 */
	static public function getFileNameMimeType($path) {
		return self::getMimetypeDetector()->detectPath($path);
	}

	/**
	 * get the mimetype form a local file
	 * @param string $path
	 * @return string
	 * does NOT work for ownClouds filesystem, use OC_FileSystem::getMimeType instead
	 */
	static function getMimeType($path) {
		return self::getMimetypeDetector()->detect($path);
	/**
	 * Get a secure mimetype that won't expose potential XSS.
	 *
	 * @param string $mimeType
	 * @return string
	 */
	static function getSecureMimeType($mimeType) {
		return self::getMimetypeDetector()->getSecureMimeType($mimeType);
	}

	/**
	 * get the mimetype form a data string
Robin Appelman's avatar
Robin Appelman committed
	 * @param string $data
	 * @return string
	 */
	static function getStringMimeType($data) {
		return self::getMimetypeDetector()->detectString($data);
François Kubler's avatar
François Kubler committed
	/**
	 * Checks $_REQUEST contains a var for the $s key. If so, returns the html-escaped value of this var; otherwise returns the default value provided by $d.
Robin Appelman's avatar
Robin Appelman committed
	 * @param string $s name of the var to escape, if set.
	 * @param string $d default value.
	 * @return string the print-safe value.
François Kubler's avatar
François Kubler committed
	 *
	 */
	 * detect if a given program is found in the search PATH
	 *
	 * @param string $name
	 * @param bool $path
	 * @internal param string $program name
	 * @internal param string $optional search path, defaults to $PATH
	 * @return bool    true if executable program found in path
	 */
	public static function canExecute($name, $path = false) {
		// path defaults to PATH from environment if not set
		if ($path === false) {
			$path = getenv("PATH");
		}
		// check method depends on operating system
		if (!strncmp(PHP_OS, "WIN", 3)) {
			// on Windows an appropriate COM or EXE file needs to exist
			$exts = array(".exe", ".com");
			$check_fn = "file_exists";
		} else {
			// anywhere else we look for an executable file of that name
			$exts = array("");
			$check_fn = "is_executable";
		}
		// Default check will be done with $path directories :
		$dirs = explode(PATH_SEPARATOR, $path);
		// WARNING : We have to check if open_basedir is enabled :
		$obd = ini_get('open_basedir');
			if (count($obd_values) > 0 and $obd_values[0]) {
Robin Appelman's avatar
Robin Appelman committed
				// open_basedir is in effect !
				// We need to check if the program is in one of these dirs :
				$dirs = $obd_values;
			}
		}
		foreach ($dirs as $dir) {
			foreach ($exts as $ext) {
				if ($check_fn("$dir/$name" . $ext))
	/**
	 * copy the contents of one stream to another
Robin Appelman's avatar
Robin Appelman committed
	 * @param resource $source
	 * @param resource $target
	 * @return array the number of bytes copied and result
Felix Moeller's avatar
Felix Moeller committed
	public static function streamCopy($source, $target) {
			return array(0, false);
		while (!feof($source)) {
			if (($c = fwrite($target, fread($source, 8192))) === false) {
	/**
	 * create a temporary file with an unique filename
Robin Appelman's avatar
Robin Appelman committed
	 * @param string $postfix
	 * @return string
	 *
	 * temporary files are automatically cleaned up after the script is finished
	 */
	public static function tmpFile($postfix = '') {
		$file = get_temp_dir() . '/' . md5(time() . rand()) . $postfix;
		$fh = fopen($file, 'w');
		fclose($fh);
		return $file;
	}
	 * move a file to oc-noclean temp dir
	 * @param string $filename
	 * @return mixed
	public static function moveToNoClean($filename = '') {
		if ($filename == '') {
			return false;
		}
		$tmpDirNoClean = get_temp_dir() . '/oc-noclean/';
		if (!file_exists($tmpDirNoClean) || !is_dir($tmpDirNoClean)) {
			if (file_exists($tmpDirNoClean)) {
				unlink($tmpDirNoClean);
			mkdir($tmpDirNoClean);
		}
		$newname = $tmpDirNoClean . basename($filename);
		if (rename($filename, $newname)) {
			return $newname;
		} else {
			return false;
		}
Bart Visscher's avatar
Bart Visscher committed

	/**
	 * create a temporary folder with an unique filename
	 * @return string
	 *
	 * temporary files are automatically cleaned up after the script is finished
	 */
	public static function tmpFolder() {
		$path = get_temp_dir() . '/' . md5(time() . rand());
		mkdir($path);
		self::$tmpFiles[] = $path;
		return $path . '/';
	/**
	 * remove all files created by self::tmpFile
	 */
	public static function cleanTmp() {
		$leftoversFile = get_temp_dir() . '/oc-not-deleted';
		if (file_exists($leftoversFile)) {
			$leftovers = file($leftoversFile);
			foreach ($leftovers as $file) {
				self::rmdirr($file);
			}
			unlink($leftoversFile);
		}

		foreach (self::$tmpFiles as $file) {
			if (file_exists($file)) {
				if (!self::rmdirr($file)) {
					file_put_contents($leftoversFile, $file . "\n", FILE_APPEND);
	 * remove all files in PHP /oc-noclean temp dir
	 */
	public static function cleanTmpNoClean() {
		$tmpDirNoCleanName=get_temp_dir() . '/oc-noclean/';
		if(file_exists($tmpDirNoCleanName) && is_dir($tmpDirNoCleanName)) {
			$files=scandir($tmpDirNoCleanName);
			foreach($files as $file) {
				$fileName = $tmpDirNoCleanName . $file;
				if (!\OC\Files\Filesystem::isIgnoredDir($file) && filemtime($fileName) + 600 < time()) {
					unlink($fileName);
				}
			}
			// if oc-noclean is empty delete it
			$isTmpDirNoCleanEmpty = true;
			$tmpDirNoClean = opendir($tmpDirNoCleanName);
			if(is_resource($tmpDirNoClean)) {
				while (false !== ($file = readdir($tmpDirNoClean))) {
					if (!\OC\Files\Filesystem::isIgnoredDir($file)) {
						$isTmpDirNoCleanEmpty = false;
					}
				}
			}
			if ($isTmpDirNoCleanEmpty) {
				rmdir($tmpDirNoCleanName);
			}
Thomas Tanghus's avatar
Thomas Tanghus committed
	/**
	 * Adds a suffix to the name in case the file exists
	 *
	 * @param string $path
	 * @param string $filename
	public static function buildNotExistingFileName($path, $filename) {
		$view = \OC\Files\Filesystem::getView();
		return self::buildNotExistingFileNameForView($path, $filename, $view);
	}

	/**
	 * Adds a suffix to the name in case the file exists
	 *
	 * @param string $path
	 * @param string $filename
	public static function buildNotExistingFileNameForView($path, $filename, \OC\Files\View $view) {
Thomas Tanghus's avatar
Thomas Tanghus committed
		}
		if ($pos = strrpos($filename, '.')) {
			$name = substr($filename, 0, $pos);
			$ext = substr($filename, $pos);
		} else {
			$name = $filename;
Robin Appelman's avatar
Robin Appelman committed
			$ext = '';
Thomas Tanghus's avatar
Thomas Tanghus committed
		}

		$newpath = $path . '/' . $filename;
		if ($view->file_exists($newpath)) {
			if (preg_match_all('/\((\d+)\)/', $name, $matches, PREG_OFFSET_CAPTURE)) {
				//Replace the last "(number)" with "(number+1)"
				$last_match = count($matches[0]) - 1;
				$counter = $matches[1][$last_match][0] + 1;
				$offset = $matches[0][$last_match][1];
				$match_length = strlen($matches[0][$last_match][0]);
			} else {
				$counter = 2;
				$offset = false;
			}
			do {
					//Replace the last "(number)" with "(number+1)"
					$newname = substr_replace($name, '(' . $counter . ')', $offset, $match_length);
				} else {
					$newname = $name . ' (' . $counter . ')';
				}
				$newpath = $path . '/' . $newname . $ext;
				$counter++;
			} while ($view->file_exists($newpath));
Thomas Tanghus's avatar
Thomas Tanghus committed
		}

		return $newpath;
	}
	 * Checks if $sub is a subdirectory of $parent
Robin Appelman's avatar
Robin Appelman committed
	 * @param string $sub
	 * @param string $parent
Georg Ehrke's avatar
Georg Ehrke committed
	 * @return bool
	 */
	public static function isSubDirectory($sub, $parent) {
Lukas Reschke's avatar
Lukas Reschke committed
		$realpathSub = realpath($sub);
		$realpathParent = realpath($parent);

		// realpath() may return false in case the directory does not exist
		// since we can not be sure how different PHP versions may behave here
		// we do an additional check whether realpath returned false
		if($realpathSub === false ||  $realpathParent === false) {
			return false;
		}

		// Check whether $sub is a subdirectory of $parent
		if (strpos($realpathSub, $realpathParent) === 0) {
			return true;
Georg Ehrke's avatar
Georg Ehrke committed
	}
	 * Returns an array with all keys from input lowercased or uppercased. Numbered indices are left as is.
	 *
	 * @param array $input The array to work on
	 * @param int $case Either MB_CASE_UPPER or MB_CASE_LOWER (default)
	 * @param string $encoding The encoding parameter is the character encoding. Defaults to UTF-8
	 * @return array
	 *
	 * Returns an array with all keys from input lowercased or uppercased. Numbered indices are left as is.
	 * based on http://www.php.net/manual/en/function.array-change-key-case.php#107715
	 *
	 */
	public static function mb_array_change_key_case($input, $case = MB_CASE_LOWER, $encoding = 'UTF-8') {
		$case = ($case != MB_CASE_UPPER) ? MB_CASE_LOWER : MB_CASE_UPPER;
		$ret = array();
		foreach ($input as $k => $v) {
			$ret[mb_convert_case($k, $case, $encoding)] = $v;
		}
		return $ret;
	}

	/**
	 * replaces a copy of string delimited by the start and (optionally) length parameters with the string given in replacement.
	 * @param string $string
	 * @param string $replacement The replacement string.
	 * @param int $start If start is positive, the replacing will begin at the start'th offset into string. If start is negative, the replacing will begin at the start'th character from the end of string.
	 * @param int $length Length of the part to be replaced
	 * @param string $encoding The encoding parameter is the character encoding. Defaults to UTF-8
	 * @internal param string $input The input string. .Opposite to the PHP build-in function does not accept an array.
	 * @return string
	 */
	public static function mb_substr_replace($string, $replacement, $start, $length = null, $encoding = 'UTF-8') {
		$start = intval($start);
		$length = intval($length);
		$string = mb_substr($string, 0, $start, $encoding) .
Thomas Müller's avatar
Thomas Müller committed
			$replacement .
			mb_substr($string, $start + $length, mb_strlen($string, 'UTF-8') - $start, $encoding);
	 * Replace all occurrences of the search string with the replacement string
	 *
	 * @param string $search The value being searched for, otherwise known as the needle.
	 * @param string $replace The replacement
	 * @param string $subject The string or array being searched and replaced on, otherwise known as the haystack.
	 * @param string $encoding The encoding parameter is the character encoding. Defaults to UTF-8
	 * @param int $count If passed, this will be set to the number of replacements performed.
	 * @return string
	 *
	 */
	public static function mb_str_replace($search, $replace, $subject, $encoding = 'UTF-8', &$count = null) {
		$offset = -1;
Robin Appelman's avatar
Robin Appelman committed
		$length = mb_strlen($search, $encoding);
		while (($i = mb_strrpos($subject, $search, $offset, $encoding)) !== false) {
			$subject = OC_Helper::mb_substr_replace($subject, $replace, $i, $length);
			$offset = $i - mb_strlen($subject, $encoding);
	 * performs a search in a nested array
	 * @param array $haystack the array to be searched
	 * @param string $needle the search string
	 * @param string $index optional, only search this key name
	 * @return mixed the key of the matching field, otherwise false
	 *
	 * performs a search in a nested array
	 *
	 * taken from http://www.php.net/manual/en/function.array-search.php#97645
	 */
	public static function recursiveArraySearch($haystack, $needle, $index = null) {
		$aIt = new RecursiveArrayIterator($haystack);
		$it = new RecursiveIteratorIterator($aIt);

			if (((isset($index) AND ($it->key() == $index)) OR (!isset($index))) AND ($it->current() == $needle)) {
				return $aIt->key();
			}

			$it->next();
		}

		return false;
	}
	/**
	 * Shortens str to maxlen by replacing characters in the middle with '...', eg.
	 * ellipsis('a very long string with lots of useless info to make a better example', 14) becomes 'a very ...example'
	 * @param string $str the string
	 * @param string $maxlen the maximum length of the result
	 * @return string with at most maxlen characters
	 */
	public static function ellipsis($str, $maxlen) {
		if (strlen($str) > $maxlen) {
			$characters = floor($maxlen / 2);
			return substr($str, 0, $characters) . '...' . substr($str, -1 * $characters);
		}
		return $str;
	}
Thomas Müller's avatar
Thomas Müller committed
	/**
	 * calculates the maximum upload size respecting system settings, free space and user quota
Thomas Müller's avatar
Thomas Müller committed
	 *
	 * @param string $dir the current folder where the user currently operates
	 * @param int $freeSpace the number of bytes free on the storage holding $dir, if not set this will be received from the storage directly
	 * @return int number of bytes representing
Thomas Müller's avatar
Thomas Müller committed
	 */
	public static function maxUploadFilesize($dir, $freeSpace = null) {
		if (is_null($freeSpace) || $freeSpace < 0){
			$freeSpace = self::freeSpace($dir);
		return min($freeSpace, self::uploadLimit());
	/**
	 * Calculate free space left within user quota
	 * @param string $dir the current folder where the user currently operates
	 * @return int number of bytes representing
	 */
	public static function freeSpace($dir) {
		$freeSpace = \OC\Files\Filesystem::free_space($dir);
		if ($freeSpace !== \OC\Files\SPACE_UNKNOWN) {
			return $freeSpace;
		} else {
			return INF;
		}
	}
	/**
	 * Calculate PHP upload limit
	 *
	 * @return PHP upload file size limit
	 */
	public static function uploadLimit() {
		$upload_max_filesize = OCP\Util::computerFileSize(ini_get('upload_max_filesize'));
		$post_max_size = OCP\Util::computerFileSize(ini_get('post_max_size'));
		if ((int)$upload_max_filesize === 0 and (int)$post_max_size === 0) {
			return INF;
		} elseif ((int)$upload_max_filesize === 0 or (int)$post_max_size === 0) {
			return max($upload_max_filesize, $post_max_size); //only the non 0 value counts
			return min($upload_max_filesize, $post_max_size);
Thomas Müller's avatar
Thomas Müller committed
	}
	 * @param string $function_name
	 * @return bool
	 */
	public static function is_function_enabled($function_name) {
		if (!function_exists($function_name)) {
			return false;
		}
		$disabled = explode(',', ini_get('disable_functions'));
		$disabled = array_map('trim', $disabled);
		if (in_array($function_name, $disabled)) {
			return false;
		}
		$disabled = explode(',', ini_get('suhosin.executor.func.blacklist'));
		$disabled = array_map('trim', $disabled);
		if (in_array($function_name, $disabled)) {
			return false;
		}
		return true;
	}
	 * Calculate the disc space for the given path
	 *
	 * @param string $path
	 * @param \OCP\Files\FileInfo $rootInfo (optional)
	 * @return array
	public static function getStorageInfo($path, $rootInfo = null) {
		// return storage info without adding mount points
		if (is_null($rootInfo)) {
			$rootInfo = \OC\Files\Filesystem::getFileInfo($path, false);
		}
		$used = $rootInfo->getSize();
		$quota = 0;
		// TODO: need a better way to get total space from storage
		$storage = $rootInfo->getStorage();
		if ($storage->instanceOfStorage('\OC\Files\Storage\Wrapper\Quota')) {
			$quota = $storage->getQuota();
		}
		$free = $storage->free_space('');
			$total = $free + $used;
		} else {
			$total = $free; //either unknown or unlimited
		}
		if ($total > 0) {
			if ($quota > 0 && $total > $quota) {
				$total = $quota;
			}
			// prevent division by zero or error codes (negative values)
			$relative = round(($used / $total) * 10000) / 100;
		} else {
			$relative = 0;
		}

		return array('free' => $free, 'used' => $used, 'total' => $total, 'relative' => $relative);
	}