Completed
Pull Request — master (#186)
by Tom
03:07
created

Value   A

Complexity

Total Complexity 30

Size/Duplication

Total Lines 158
Duplicated Lines 0 %

Coupling/Cohesion

Components 1
Dependencies 6

Importance

Changes 6
Bugs 1 Features 0
Metric Value
wmc 30
c 6
b 1
f 0
lcom 1
cbo 6
dl 0
loc 158
rs 10

14 Methods

Rating   Name   Duplication   Size   Complexity  
A __construct() 0 5 1
A parse() 0 6 1
A parseTokens() 0 15 4
A processComparator() 0 8 3
A processIn() 0 4 1
A processDot() 0 15 3
A hasFunction() 0 3 2
A processSquareBracket() 0 10 3
A processSeparator() 0 3 1
A processScalar() 0 6 2
A processString() 0 3 1
A processBrackets() 0 9 3
A getNewParser() 0 3 1
A callTransphpormFunctions() 0 18 4
1
<?php
2
/* @description     Transformation Style Sheets - Revolutionising PHP templating    *
3
 * @author          Tom Butler [email protected]                                             *
4
 * @copyright       2017 Tom Butler <[email protected]> | https://r.je/                      *
5
 * @license         http://www.opensource.org/licenses/bsd-license.php  BSD License *
6
 * @version         1.2                                                             */
7
namespace Transphporm\Parser;
8
/** Parses "string" and function(args) e.g. data(foo) or iteration(bar) */
9
class Value {
10
	private $baseData;
11
	private $autoLookup;
12
	/*
13
		Stores the last value e.g.
14
			"a" + "b"
15
		Will store "a" before reading the token for the + and perfoming the concatenate operation
16
	*/
17
	private $last;
18
	private $data;
19
	private $result;
20
	private $traversing = false;
21
	private $allowNullResult = false;
22
23
	private $tokenFuncs = [
24
			Tokenizer::NOT => 'processComparator',
25
			Tokenizer::EQUALS => 'processComparator',
26
			Tokenizer::DOT => 'processDot',
27
			Tokenizer::OPEN_SQUARE_BRACKET => 'processSquareBracket',
28
			Tokenizer::ARG => 'processSeparator',
29
			Tokenizer::CONCAT => 'processSeparator',
30
			Tokenizer::SUBTRACT => 'processSeparator',
31
			Tokenizer::MULTIPLY => 'processSeparator',
32
			Tokenizer::DIVIDE => 'processSeparator',
33
			Tokenizer::NAME => 'processScalar',
34
			Tokenizer::NUMERIC => 'processString',
35
			Tokenizer::BOOL => 'processString',
36
			Tokenizer::STRING => 'processString',
37
			Tokenizer::OPEN_BRACKET => 'processBrackets',
38
			Tokenizer::GREATER_THAN => 'processComparator',
39
			Tokenizer::LOWER_THAN => 'processComparator',
40
			Tokenizer::IN => 'processIn',
41
		];
42
43
	public function __construct($data, $autoLookup = false, $allowNullResult = false) {
44
		$this->baseData = $data;
45
		$this->autoLookup = $autoLookup;
46
		$this->allowNullResult = $allowNullResult;
47
	}
48
49
	public function parse($str) {
50
		$tokenizer = new Tokenizer($str);
51
		$tokens = $tokenizer->getTokens();
52
		$this->result = $this->parseTokens($tokens, $this->baseData);
53
		return $this->result;
54
	}
55
56
	public function parseTokens($tokens, $data = null) {
57
		$this->result = new ValueResult();
58
		$this->data = new ValueData($data ? $data : $this->baseData);
59
		$this->last = new Last($this->data, $this->result, $this->autoLookup);
60
		$this->traversing = false;
61
62
		if (count($tokens) <= 0) return [$data];
63
64
		foreach (new TokenFilterIterator($tokens, [Tokenizer::WHITESPACE, Tokenizer::NEW_LINE]) as $name => $token) {
65
			$this->{$this->tokenFuncs[$token['type']]}($token);
66
		}
67
68
		$this->last->process();
69
		return $this->result->getResult();
70
	}
71
72
	private function processComparator($token) {
73
		$this->last->process();
74
75
		if (!(in_array($this->result->getMode(), array_keys($this->tokenFuncs, 'processComparator')) && $token['type'] == Tokenizer::EQUALS)) {
76
			$this->result->setMode($token['type']);
77
			$this->last->clear();
78
		}
79
	}
80
81
	private function processIn($token) {
82
		$this->allowNullResult = false;
83
		$this->processComparator($token);
84
	}
85
86
	//Reads the last selected value from $data regardless if it's an array or object and overrides $this->data with the new value
87
	//Dot moves $data to the next object in $data foo.bar moves the $data pointer from `foo` to `bar`
88
	private function processDot() {
89
		$lastResult = $this->last->traverse();
90
91
		//When . is not preceeded by anything, treat it as part of the string instead of an operator
92
		// foo.bar is treated as looking up `bar` in `foo` whereas .foo is treated as the string ".foo"
93
		if ($lastResult) {
94
			$this->last->makeTraversing();
95
		}
96
		else if ($this->last->isEmpty())  {
97
			$this->processString(['value' => '.']);
98
			$this->result->setMode(Tokenizer::CONCAT);
99
		}
100
101
		$this->last->clear();
102
	}
103
104
	private function hasFunction($name) {
105
		return $this->baseData instanceof \Transphporm\Functionset && $this->baseData->hasFunction($name);
106
	}
107
108
	private function processSquareBracket($token) {
109
		if ($this->hasFunction($this->last->read())) {
110
			$this->callTransphpormFunctions($token);
111
		}
112
		else {
113
			$this->last->traverse();
114
			$this->last->set($this->getNewParser()->parseTokens($token['value'], null)[0]);
115
			if (!is_bool($this->last->read())) $this->last->makeTraversing();
116
		}
117
	}
118
119
	private function processSeparator($token) {
120
		$this->result->setMode($token['type']);
121
	}
122
123
	private function processScalar($token) {
124
		if (is_scalar($this->last->read())) {
125
			$this->result->processValue($this->last->read());
126
		}
127
		$this->last->set($token['value']);
128
	}
129
130
	private function processString($token) {
131
		$this->result->processValue($token['value']);
132
	}
133
134
	private function processBrackets($token) {
135
		if ($this->hasFunction($this->last->read())
136
			&& !$this->data->methodExists($this->last->read())) {
137
			$this->callTransphpormFunctions($token);
138
		}
139
		else {
140
			$this->last->processNested($this->getNewParser(), $token);
141
		}
142
	}
143
144
	private function getNewParser() {
145
		return new Value($this->baseData, $this->autoLookup);
146
	}
147
148
	private function callTransphpormFunctions($token) {
149
		$val = $this->baseData->{$this->last->read()}($token['value']);
150
		$this->result->processValue($val);
151
152
		if ($this->autoLookup) {
153
			if (!is_array($val)) {
154
				$parser = new Value($this->data->getData());
155
				$parsedArr = $parser->parse($val);
156
				$parsedVal = isset($parsedArr[0]) ? $parsedArr[0] : null;
157
			}
158
			else $parsedVal = $val;
159
		}
160
		else $parsedVal = null;
161
162
        $this->result->postProcess($this->data, $val, $parsedVal, $this->allowNullResult);
163
164
		$this->last->clear();
165
	}
166
}
167