Completed
Push — master ( e8cf0b...dd193e )
by Maxim
03:10
created

ConditionsRunner::reduce()   A

Complexity

Conditions 4
Paths 3

Size

Total Lines 10
Code Lines 7

Duplication

Lines 0
Ratio 0 %

Code Coverage

Tests 8
CRAP Score 4

Importance

Changes 0
Metric Value
cc 4
eloc 7
nc 3
nop 3
dl 0
loc 10
ccs 8
cts 8
cp 1
crap 4
rs 9.2
c 0
b 0
f 0
1
<?php
2
/**
3
 * This file is a part of "Axessors" library.
4
 *
5
 * @author <[email protected]>
6
 * @license GPL
7
 */
8
9
namespace NoOne4rever\Axessors;
10
11
use NoOne4rever\Axessors\Exceptions\TypeError;
12
13
/**
14
 * Class ConditionsSuit.
15
 * 
16
 * Processes Axessors conditions.
17
 * 
18
 * @package NoOne4rever\Axessors
19
 */
20
class ConditionsRunner extends RunningSuit
21
{
22
    /** @var string method name */
23
    private $method;
24
25
    /**
26
     * ConditionsSuit constructor.
27
     * 
28
     * @param int $mode mode of execution
29
     * @param PropertyData $data property data
30
     * @param string $class class
31
     * @param string $method method name
32
     * @param object|null $object object
33
     */
34 62
    public function __construct(int $mode, PropertyData $data, string $class, string $method, $object = null)
35
    {
36 62
        parent::__construct($mode, $data, $class, $object);
37 62
        $this->method = $method;
38 62
    }
39
40
    /**
41
     * Checks the conditions defined in the Axessors comment.
42
     *
43
     * Creates logical tree of the conditions and then checks if the general result is true.
44
     *
45
     * @param $value mixed value of the property
46
     * @return bool result of checking of the conditions
47
     */
48 62
    public function processConditions($value): bool
49
    {
50 62
        $conditions = $this->calculateConditions($value);
51 60
        if (empty($conditions)) {
52 22
            return true;
53
        } else {
54 38
            return array_reduce($conditions, function ($carry, $item) {
55 38
                return $this->reduce($carry, $item, true);
56 38
            });
57
        }
58
    }
59
60
    /**
61
     * Reduces conditions tree and returns the result.
62
     * 
63
     * @param mixed $carry previous item
64
     * @param mixed $item item
65
     * @param bool $mode mode
66
     * @return bool the result of reducing
67
     */
68 38
    private function reduce($carry, $item, bool $mode): bool 
69
    {
70 38
        if ($item === $mode || $carry === $mode) {
71 21
            return $mode;
72 19
        } elseif (is_array($item)) {
73 4
            return array_reduce($item, function ($carry, $item) {
74 4
                return $this->reduce($carry, $item, false);
75 4
            });
76
        } else {
77 19
            return !$mode;
78
        }
79
    }
80
81
    /**
82
     * Calculates every condition defined in the Axessors comment.
83
     *
84
     * @param $value mixed value of the property
85
     * @return bool[] results of the conditions
86
     */
87 62
    private function calculateConditions($value): array
88
    {
89 62
        $calculatedConditions = [];
90 62
        $conditions = $this->mode == RunningSuit::OUTPUT_MODE ? $this->propertyData->getOutputConditions() : $this->propertyData->getInputConditions();
91 62
        foreach ($conditions as $number => $complexCondition) {
92 40
            if (is_array($complexCondition)) {
93 4
                foreach ($complexCondition as $condition) {
94 4
                    $calculatedConditions[$number][] = $this->executeCondition($condition, $value);
95
                }
96
            } else {
97 36
                $calculatedConditions[$number] = $this->executeCondition($complexCondition, $value);
98
            }
99
        }
100 60
        return $calculatedConditions;
101
    }
102
103
    /**
104
     * Calculates a condition defined in the Axessors comment.
105
     *
106
     * @param $condition string the condition
107
     * @param $value mixed value of the property
108
     * @return bool result of the condition
109
     */
110 40
    private function executeCondition(string $condition, $value): bool
111
    {
112 40
        if (strpos($condition, '`') !== false) {
113 13
            return $this->executeInjectedString($condition, $value, true);
114
        } else {
115 29
            return $this->runStandardCondition($condition, $value);
116
        }
117
    }
118
119
    /**
120
     * Runs Axessors condition.
121
     * 
122
     * @param string $condition condition
123
     * @param mixed $value value to check 
124
     * @return bool the result of the checkout
125
     */
126 29
    private function runStandardCondition(string $condition, $value): bool 
127
    {
128 29
        $value = $this->count($value);
129 28
        if (strpos($condition, '..') !== false) {
130 12
            $condition = explode('..', $condition);
131 12
            $condition = "<= {$condition[1]} && $value >= {$condition[0]}";
132
        }
133 28
        return eval("return $value $condition;");
134
    }
135
136
    /**
137
     * Casts the property to integer.
138
     *
139
     * If the property is string or array returns it's length.
140
     * If the property is integer of float returns the property itself.
141
     *
142
     * @param $value mixed value of the property
143
     * @return int integer value of the property
144
     * @throws TypeError if the property can't be turned into integer
145
     */
146 29
    private function count($value): int
147
    {
148 29
        switch (gettype($value)) {
149 29
            case 'integer':
150 7
            case 'float':
151 22
                break;
152 7
            case 'string':
153 5
                $value = strlen($value);
154 5
                break;
155 2
            case 'array':
156 1
                $value = count($value);
157 1
                break;
158
            default:
159 1
                throw new TypeError('value "' . var_export($value, true) . "\" passed to {$this->class}::{$this->method}() is not countable");
160
        }
161 28
        return $value;
162
    }
163
}