Files
tysweld.com/wp-content/plugins/duplicator-pro-v4.5.16.2/classes/utilities/class.u.tree.files.php
2025-02-24 22:33:42 +01:00

391 lines
9.7 KiB
PHP

<?php
/**
* Utility class to create a tree structure from paths
*
* Standard: PSR-2
*
* @link http://www.php-fig.org/psr/psr-2
*
* @package DUP_PRO
* @subpackage classes/utilities
* @copyright (c) 2017, Snapcreek LLC
* @license https://opensource.org/licenses/GPL-3.0 GNU Public License
* @since 3.8.1
*/
defined("ABSPATH") or die("");
use Duplicator\Libs\Snap\SnapIO;
/**
* Tree files utility
*/
class DUP_PRO_Tree_files
{
/**
* All props must be public for json encode
*/
/**
*
* @var DUP_PRO_Tree_files_node[]
*/
protected $treeList = array();
/**
* Class contructor
*
* @param string|string[] $rootPaths root paths
* @param bool $addAllChilds if true add all childs
* @param null|false|string|string[] $excludeList if null or false no exclude else exclude list
*/
public function __construct($rootPaths, $addAllChilds = true, $excludeList = array())
{
if (is_null($excludeList) || $excludeList === false) {
$excludeList = array();
} elseif (!is_array($excludeList)) {
$excludeList = array($excludeList);
}
if (!is_array($rootPaths)) {
$rootPaths = array($rootPaths);
}
$rootPaths = array_map(array('\\Duplicator\\Libs\\Snap\\SnapIO', 'safePathUntrailingslashit'), $rootPaths);
foreach ($rootPaths as $path) {
$this->treeList[$path] = new DUP_PRO_Tree_files_node($path);
if ($addAllChilds) {
$this->treeList[$path]->addAllChilds($excludeList);
}
}
}
/**
*
* @param string $path full path
* @param array<string,mixed> $data optiona data associated at node
*
* @return bool|DUP_PRO_Tree_files_node
*/
public function addElement($path, $data = array())
{
foreach ($this->treeList as $rootPath => $tree) {
if (strpos($path, trailingslashit($rootPath)) === 0) {
$newElem = $tree->addChild($path, true, false);
if ($newElem) {
$newElem->data = $data;
}
return $newElem;
}
}
return false;
}
/**
* Sort child list with callback function of all trees root nodes
*
* @param callable $callback function to call
*
* @return void
*/
public function uasort($callback)
{
foreach ($this->treeList as $tree) {
$tree->uasort($callback);
}
}
/**
* traverse tree anche call callback function of all trees root nodes
*
* @param callable $callback function to call
*
* @return void
*/
public function treeTraverseCallback($callback)
{
foreach ($this->treeList as $tree) {
$tree->treeTraverseCallback($callback);
}
}
/**
*
* @return DUP_PRO_Tree_files_node[]
*/
public function getTreeList()
{
return $this->treeList;
}
}
/**
* Tree node data
*/
class DUP_PRO_Tree_files_node
{
const MAX_CHILDS_FOR_FOLDER = 250;
const MAX_TREE_NODES = 5000;
/**
* All props must be public for json encode
*/
/**
*
* @var string unique id l0_l1_l2....
*/
public $id = '';
/**
*
* @var string parent id l0_l1_...
*/
public $parentId = '';
/**
*
* @var string file basename
*/
public $name = '';
/**
*
* @var string full path
*/
public $fullPath = '';
/**
*
* @var bool is directory
*/
public $isDir = false;
/**
*
* @var DUP_PRO_Tree_files_node[] childs nodes
*/
public $childs = array();
/**
*
* @var array<string, mixed> optiona data associated ad node
*/
public $data = array();
/**
*
* @var bool true if folder have a childs
*/
public $haveChildren = null;
/**
*
* @var bool
*/
private $traversed = false;
/**
*
* @var int if can't add a child increment exceeede count
*/
private $nodesExceeded = 0;
/**
*
* @var int
*/
private $numTreeNodes = 1;
/**
*
* @param string $path file path
* @param string $id current level unique id
* @param string $parent_id parent id
*/
public function __construct($path, $id = '0', $parent_id = '')
{
$safePath = SnapIO::safePathUntrailingslashit($path);
$this->id = (strlen($parent_id) == 0 ? '' : $parent_id . '_') . $id;
$this->parentId = $parent_id;
$this->name = basename($safePath);
$this->fullPath = $safePath;
$this->isDir = @is_dir($this->fullPath);
$this->haveChildrenCheck();
}
/**
* create tree tructure until at basename
*
* @param string $path file path
* @param bool $fullPath if true is considered a full path and must be a child of a current node else is a relative path
* @param bool $loadTraverse if true, add the files and folders present at the level of each node
*
* @return boolean|DUP_PRO_Tree_files_node if fails terurn false ellse return the leaf child added
*/
public function addChild($path, $fullPath = true, $loadTraverse = false)
{
if (empty($path)) {
return false;
}
$safePath = SnapIO::safePathUntrailingslashit($path);
if ($fullPath) {
if (strpos($safePath, trailingslashit($this->fullPath)) !== 0) {
throw new Exception('Can\'t add no child on tree; file: "' . $safePath . '" || fullpath: "' . $this->fullPath . '"');
}
$childPath = substr($safePath, strlen($this->fullPath));
} else {
$childPath = $safePath;
}
$tree_list = explode('/', $childPath);
if (empty($tree_list[0])) {
array_shift($tree_list);
}
if (($child = $this->checkAndAddChild($tree_list[0])) === false) {
return false;
}
if ($loadTraverse) {
$child->addAllChilds();
}
if (count($tree_list) > 1) {
array_shift($tree_list);
$nodesBefore = $child->getNumTreeNodes();
$result = $child->addChild(implode('/', $tree_list), false, $loadTraverse);
$this->numTreeNodes += $child->getNumTreeNodes() - $nodesBefore;
return $result;
} else {
return $child;
}
}
/**
* If is dir scan all children files and add on childs list
*
* @param string[] $excludeList child to add
*
* @return void
*/
public function addAllChilds($excludeList = array())
{
if ($this->traversed === false) {
$this->traversed = true;
if ($this->isDir) {
if ($dh = @opendir($this->fullPath)) {
while (($childName = readdir($dh)) !== false) {
if ($childName == '.' || $childName == '..' || in_array($childName, $excludeList)) {
continue;
}
$this->checkAndAddChild($childName);
}
closedir($dh);
}
}
}
}
/**
* check if current dir have children without load nodes
*
* @return void
*/
private function haveChildrenCheck()
{
if ($this->isDir) {
$this->haveChildren = false;
if ($dh = opendir($this->fullPath)) {
while (!$this->haveChildren && ($file = readdir($dh)) !== false) {
$this->haveChildren = $file !== "." && $file !== "..";
}
closedir($dh);
}
}
}
/**
*
* @param string $name child name
*
* @return boolean|DUP_PRO_Tree_files_node if notes exceeded return false
*/
private function checkAndAddChild($name)
{
if (!array_key_exists($name, $this->childs)) {
if ($this->numTreeNodes > self::MAX_TREE_NODES || count($this->childs) >= self::MAX_CHILDS_FOR_FOLDER) {
$this->nodesExceeded++;
return false;
} else {
$child = new DUP_PRO_Tree_files_node($this->fullPath . '/' . $name, (string) count($this->childs), $this->id);
$this->childs[$name] = $child;
$this->numTreeNodes++;
}
}
return $this->childs[$name];
}
/**
* sort child list with callback function
*
* @param callable $value_compare_func function to call
*
* @return void
*/
public function uasort($value_compare_func)
{
if (!is_callable($value_compare_func)) {
return;
}
foreach ($this->childs as $child) {
$child->uasort($value_compare_func);
}
uasort($this->childs, $value_compare_func);
}
/**
* traverse tree anche call callback function
*
* @param callable $callback function to call
*
* @return void
*/
public function treeTraverseCallback($callback)
{
if (!is_callable($callback)) {
return;
}
foreach ($this->childs as $child) {
$child->treeTraverseCallback($callback);
}
call_user_func($callback, $this);
}
/**
* Get the value of nodesExceeded
*
* @return int
*/
public function getNodesExceeded()
{
return $this->nodesExceeded;
}
/**
* Get the value of numTreeNodes
*
* @return int
*/
public function getNumTreeNodes()
{
return $this->numTreeNodes;
}
}