123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157 |
- <?php
- /**
- * elFinder Plugin Sanitizer
- * Sanitizer of file-name and file-path etc.
- * ex. binding, configure on connector options
- * $opts = array(
- * 'bind' => array(
- * 'upload.pre mkdir.pre mkfile.pre rename.pre archive.pre ls.pre' => array(
- * 'Plugin.Sanitizer.cmdPreprocess'
- * ),
- * 'upload.presave paste.copyfrom' => array(
- * 'Plugin.Sanitizer.onUpLoadPreSave'
- * )
- * ),
- * // global configure (optional)
- * 'plugin' => array(
- * 'Sanitizer' => array(
- * 'enable' => true,
- * 'targets' => array('\\','/',':','*','?','"','<','>','|'), // target chars
- * 'replace' => '_', // replace to this
- * 'callBack' => null // Or @callable sanitize function
- * )
- * ),
- * // each volume configure (optional)
- * 'roots' => array(
- * array(
- * 'driver' => 'LocalFileSystem',
- * 'path' => '/path/to/files/',
- * 'URL' => 'http://localhost/to/files/'
- * 'plugin' => array(
- * 'Sanitizer' => array(
- * 'enable' => true,
- * 'targets' => array('\\','/',':','*','?','"','<','>','|'), // target chars
- * 'replace' => '_', // replace to this
- * 'callBack' => null // Or @callable sanitize function
- * )
- * )
- * )
- * )
- * );
- *
- * @package elfinder
- * @author Naoki Sawada
- * @license New BSD
- */
- class elFinderPluginSanitizer extends elFinderPlugin
- {
- private $replaced = array();
- private $keyMap = array(
- 'ls' => 'intersect',
- 'upload' => 'renames',
- 'mkdir' => array('name', 'dirs')
- );
- public function __construct($opts)
- {
- $defaults = array(
- 'enable' => true, // For control by volume driver
- 'targets' => array('\\', '/', ':', '*', '?', '"', '<', '>', '|'), // target chars
- 'replace' => '_', // replace to this
- 'callBack' => null // Or callable sanitize function
- );
- $this->opts = array_merge($defaults, $opts);
- }
- public function cmdPreprocess($cmd, &$args, $elfinder, $volume)
- {
- $opts = $this->getCurrentOpts($volume);
- if (!$opts['enable']) {
- return false;
- }
- $this->replaced[$cmd] = array();
- $key = (isset($this->keyMap[$cmd])) ? $this->keyMap[$cmd] : 'name';
- if (is_array($key)) {
- $keys = $key;
- } else {
- $keys = array($key);
- }
- foreach ($keys as $key) {
- if (isset($args[$key])) {
- if (is_array($args[$key])) {
- foreach ($args[$key] as $i => $name) {
- if ($cmd === 'mkdir' && $key === 'dirs') {
- // $name need '/' as prefix see #2607
- $name = '/' . ltrim($name, '/');
- $_names = explode('/', $name);
- $_res = array();
- foreach ($_names as $_name) {
- $_res[] = $this->sanitizeFileName($_name, $opts);
- }
- $this->replaced[$cmd][$name] = $args[$key][$i] = join('/', $_res);
- } else {
- $this->replaced[$cmd][$name] = $args[$key][$i] = $this->sanitizeFileName($name, $opts);
- }
- }
- } else if ($args[$key] !== '') {
- $name = $args[$key];
- $this->replaced[$cmd][$name] = $args[$key] = $this->sanitizeFileName($name, $opts);
- }
- }
- }
- if ($cmd === 'ls' || $cmd === 'mkdir') {
- if (!empty($this->replaced[$cmd])) {
- // un-regist for legacy settings
- $elfinder->unbind($cmd, array($this, 'cmdPostprocess'));
- $elfinder->bind($cmd, array($this, 'cmdPostprocess'));
- }
- }
- return true;
- }
- public function cmdPostprocess($cmd, &$result, $args, $elfinder, $volume)
- {
- if ($cmd === 'ls') {
- if (!empty($result['list']) && !empty($this->replaced['ls'])) {
- foreach ($result['list'] as $hash => $name) {
- if ($keys = array_keys($this->replaced['ls'], $name)) {
- if (count($keys) === 1) {
- $result['list'][$hash] = $keys[0];
- } else {
- $result['list'][$hash] = $keys;
- }
- }
- }
- }
- } else if ($cmd === 'mkdir') {
- if (!empty($result['hashes']) && !empty($this->replaced['mkdir'])) {
- foreach ($result['hashes'] as $name => $hash) {
- if ($keys = array_keys($this->replaced['mkdir'], $name)) {
- $result['hashes'][$keys[0]] = $hash;
- }
- }
- }
- }
- }
- // NOTE: $thash is directory hash so it unneed to process at here
- public function onUpLoadPreSave(&$thash, &$name, $src, $elfinder, $volume)
- {
- $opts = $this->getCurrentOpts($volume);
- if (!$opts['enable']) {
- return false;
- }
- $name = $this->sanitizeFileName($name, $opts);
- return true;
- }
- protected function sanitizeFileName($filename, $opts)
- {
- if (!empty($opts['callBack']) && is_callable($opts['callBack'])) {
- return call_user_func_array($opts['callBack'], array($filename, $opts));
- }
- return str_replace($opts['targets'], $opts['replace'], $filename);
- }
- }
|