loophp /
phptree
| 1 | <?php |
||||
| 2 | |||||
| 3 | /** |
||||
| 4 | * For the full copyright and license information, please view |
||||
| 5 | * the LICENSE file that was distributed with this source code. |
||||
| 6 | */ |
||||
| 7 | |||||
| 8 | declare(strict_types=1); |
||||
| 9 | |||||
| 10 | namespace loophp\phptree\Node; |
||||
| 11 | |||||
| 12 | use drupol\phpmerkle\Hasher\DoubleSha256; |
||||
| 13 | use drupol\phpmerkle\Hasher\HasherInterface; |
||||
| 14 | use loophp\phptree\Modifier\FulfillCapacity; |
||||
| 15 | use loophp\phptree\Modifier\ModifierInterface; |
||||
| 16 | use loophp\phptree\Modifier\RemoveNullNode; |
||||
| 17 | |||||
| 18 | /** |
||||
| 19 | * Class MerkleNode. |
||||
| 20 | */ |
||||
| 21 | class MerkleNode extends ValueNode implements MerkleNodeInterface |
||||
| 22 | { |
||||
| 23 | /** |
||||
| 24 | * @var HasherInterface |
||||
| 25 | */ |
||||
| 26 | private $hasher; |
||||
| 27 | |||||
| 28 | /** |
||||
| 29 | * @var ModifierInterface[] |
||||
| 30 | */ |
||||
| 31 | private $modifiers = []; |
||||
| 32 | |||||
| 33 | /** |
||||
| 34 | * MerkleNode constructor. |
||||
| 35 | * |
||||
| 36 | * @param mixed $value |
||||
| 37 | * @param HasherInterface $hasher |
||||
| 38 | */ |
||||
| 39 | 8 | public function __construct( |
|||
| 40 | $value, |
||||
| 41 | int $capacity = 2, |
||||
| 42 | ?HasherInterface $hasher = null |
||||
| 43 | ) { |
||||
| 44 | 8 | parent::__construct($value, $capacity, null, null); |
|||
| 45 | |||||
| 46 | 8 | $this->hasher = $hasher ?? new DoubleSha256(); |
|||
| 47 | 8 | $this->modifiers = [ |
|||
| 48 | 8 | new RemoveNullNode(), |
|||
| 49 | 8 | new FulfillCapacity(), |
|||
| 50 | ]; |
||||
| 51 | 8 | } |
|||
| 52 | |||||
| 53 | 6 | public function hash(): string |
|||
| 54 | { |
||||
| 55 | 6 | return $this->hasher->unpack($this->doHash($this->normalize())); |
|||
| 56 | } |
||||
| 57 | |||||
| 58 | 1 | public function label(): string |
|||
| 59 | { |
||||
| 60 | 1 | return $this->isLeaf() ? |
|||
| 61 | 1 | $this->getValue() : |
|||
| 62 | 1 | $this->hash(); |
|||
| 63 | } |
||||
| 64 | |||||
| 65 | 6 | public function normalize(): MerkleNodeInterface |
|||
| 66 | { |
||||
| 67 | 6 | return array_reduce( |
|||
| 68 | 6 | $this->modifiers, |
|||
| 69 | static function (MerkleNodeInterface $tree, ModifierInterface $modifier): MerkleNodeInterface { |
||||
| 70 | 6 | return $modifier->modify($tree); |
|||
|
0 ignored issues
–
show
Bug
Best Practice
introduced
by
Loading history...
|
|||||
| 71 | 6 | }, |
|||
| 72 | 6 | $this->clone() |
|||
| 73 | ); |
||||
| 74 | } |
||||
| 75 | |||||
| 76 | 6 | private function doHash(MerkleNodeInterface $node): string |
|||
| 77 | { |
||||
| 78 | // If node is a leaf, then compute its hash from its value. |
||||
| 79 | 6 | if ($node->isLeaf()) { |
|||
| 80 | 6 | return $this->hasher->hash($node->getValue()); |
|||
|
0 ignored issues
–
show
It seems like
$node->getValue() can also be of type null; however, parameter $data of drupol\phpmerkle\Hasher\HasherInterface::hash() does only seem to accept string, maybe add an additional type check?
(
Ignorable by Annotation
)
If this is a false-positive, you can also ignore this issue in your code via the
Loading history...
|
|||||
| 81 | } |
||||
| 82 | |||||
| 83 | 6 | $hash = ''; |
|||
| 84 | /** @var MerkleNodeInterface $child */ |
||||
| 85 | 6 | foreach ($node->children() as $child) { |
|||
| 86 | 6 | $hash .= $this->doHash($child); |
|||
| 87 | } |
||||
| 88 | |||||
| 89 | 6 | return $this->hasher->hash($hash); |
|||
| 90 | } |
||||
| 91 | } |
||||
| 92 |