Passed
Push — master ( e5139b...14b6ee )
by Pol
02:21
created

FilesystemNode::add()   A

Complexity

Conditions 6
Paths 6

Size

Total Lines 29
Code Lines 13

Duplication

Lines 0
Ratio 0 %

Code Coverage

Tests 13
CRAP Score 6.0131

Importance

Changes 1
Bugs 0 Features 0
Metric Value
cc 6
eloc 13
nc 6
nop 1
dl 0
loc 29
ccs 13
cts 14
cp 0.9286
crap 6.0131
rs 9.2222
c 1
b 0
f 0
1
<?php
2
3
declare(strict_types = 1);
4
5
namespace drupol\phpvfs\Node;
6
7
use drupol\phptree\Node\AttributeNode;
8
use drupol\phptree\Node\NodeInterface;
9
use drupol\phpvfs\Utils\Path;
10
11
/**
12
 * Class Vfs.
13
 */
14
abstract class FilesystemNode extends AttributeNode implements FilesystemNodeInterface
15
{
16
    /**
17
     * FilesystemNode constructor.
18
     *
19
     * @param array $attributes
20
     * @param null|int $capacity
21
     */
22 20
    public function __construct(
23
        array $attributes = [],
24
        ?int $capacity = 0
25
    ) {
26 20
        $time = \time();
27
28
        $attributes = [
29 20
            'uid' => \function_exists('posix_getuid') ? \posix_getuid() : 0,
30 20
            'gid' => \function_exists('posix_getgid') ? \posix_getgid() : 0,
31 20
            'atime' => $time,
32 20
            'mtime' => $time,
33 20
            'ctime' => $time,
34 20
        ] + $attributes;
35
36 20
        parent::__construct($attributes, $capacity);
37 20
    }
38
39
    /**
40
     * {@inheritdoc}
41
     *
42
     * @throws \Exception
43
     *
44
     * @return \drupol\phpvfs\Node\DirectoryInterface
45
     */
46 16
    public function add(NodeInterface ...$nodes): NodeInterface
47
    {
48
        /** @var \drupol\phpvfs\Node\FilesystemNodeInterface $node */
49 16
        foreach ($nodes as $node) {
50 16
            $node = $node->root();
0 ignored issues
show
Bug introduced by
The method root() does not exist on drupol\phptree\Node\NodeInterface. It seems like you code against a sub-type of drupol\phptree\Node\NodeInterface such as drupol\phpvfs\Node\FilesystemNode or drupol\phpvfs\Node\FilesystemNodeInterface or drupol\phpvfs\Node\FilesystemNode or drupol\phpvfs\Node\FilesystemNode. ( Ignorable by Annotation )

If this is a false-positive, you can also ignore this issue in your code via the ignore-call  annotation

50
            /** @scrutinizer ignore-call */ 
51
            $node = $node->root();
Loading history...
51
52 16
            if (!($node instanceof FilesystemNodeInterface)) {
53
                throw new \Exception('Invalid filesystem node type.');
54
            }
55
56 16
            if ($this->getAttribute('id') === $node->getAttribute('id')) {
57 8
                $this->add($node[0]->setParent(null));
58
59 8
                continue;
60
            }
61
62
            // If the $cwd contains the nodechild.
63 16
            if (null !== $child = $this->contains($node)) {
64 3
                if (0 !== $node->degree()) {
65 3
                    $child->add($node[0]->setParent(null));
66
                }
67
68 3
                continue;
69
            }
70
71 16
            parent::add($node->setParent(null));
72
        }
73
74 16
        return $this;
0 ignored issues
show
Bug Best Practice introduced by
The expression return $this returns the type drupol\phpvfs\Node\FilesystemNode which is incompatible with the documented return type drupol\phpvfs\Node\DirectoryInterface.
Loading history...
75
    }
76
77
    /**
78
     * {@inheritdoc}
79
     */
80 1
    public function getPath(): Path
81
    {
82
        $paths = [
83 1
            $this->getAttribute('id'),
84
        ];
85
86 1
        foreach ($this->getAncestors() as $ancestor) {
87 1
            \array_unshift($paths, $ancestor->getAttribute('id'));
0 ignored issues
show
Bug introduced by
The method getAttribute() does not exist on drupol\phptree\Node\NodeInterface. It seems like you code against a sub-type of drupol\phptree\Node\NodeInterface such as drupol\phptree\Node\AttributeNode or drupol\phptree\Node\AttributeNodeInterface or drupol\phptree\Node\AttributeNode. ( Ignorable by Annotation )

If this is a false-positive, you can also ignore this issue in your code via the ignore-call  annotation

87
            \array_unshift($paths, $ancestor->/** @scrutinizer ignore-call */ getAttribute('id'));
Loading history...
88
        }
89
90 1
        return Path::fromString(\str_replace('//', '/', \implode('/', $paths)));
91
    }
92
93
    /**
94
     * {@inheritdoc}
95
     */
96 17
    public function root(): FilesystemNodeInterface
97
    {
98 17
        $root = $this;
99
100 17
        foreach ($this->getAncestors() as $ancestor) {
101 14
            $root = $ancestor;
102
        }
103
104 17
        return $root;
0 ignored issues
show
Bug Best Practice introduced by
The expression return $root could return the type drupol\phptree\Node\NodeInterface which includes types incompatible with the type-hinted return drupol\phpvfs\Node\FilesystemNodeInterface. Consider adding an additional type-check to rule them out.
Loading history...
105
    }
106
107
    /**
108
     * @param \drupol\phpvfs\Node\FilesystemNodeInterface $node
109
     *
110
     * @return null|\drupol\phpvfs\Node\FilesystemNodeInterface
111
     */
112 16
    protected function contains(FilesystemNodeInterface $node): ?FilesystemNodeInterface
113
    {
114
        /** @var \drupol\phpvfs\Node\FilesystemNodeInterface $child */
115 16
        foreach ($this->children() as $child) {
116 4
            if ($node->getAttribute('id') === $child->getAttribute('id')) {
0 ignored issues
show
Bug introduced by
The method getAttribute() does not exist on ArrayObject. ( Ignorable by Annotation )

If this is a false-positive, you can also ignore this issue in your code via the ignore-call  annotation

116
            if ($node->getAttribute('id') === $child->/** @scrutinizer ignore-call */ getAttribute('id')) {

This check looks for calls to methods that do not seem to exist on a given type. It looks for the method on the type itself as well as in inherited classes or implemented interfaces.

This is most likely a typographical error or the method has been renamed.

Loading history...
117 4
                return $child;
0 ignored issues
show
Bug Best Practice introduced by
The expression return $child returns the type ArrayObject which is incompatible with the type-hinted return drupol\phpvfs\Node\FilesystemNodeInterface|null.
Loading history...
118
            }
119
        }
120
121 16
        return null;
122
    }
123
}
124