Completed
Push — master ( 1c2b41...01a392 )
by Michael
02:44
created

StatsJsonView::processSingleSource()   C

Complexity

Conditions 10
Paths 8

Size

Total Lines 48
Code Lines 20

Duplication

Lines 0
Ratio 0 %

Code Coverage

Tests 0
CRAP Score 110

Importance

Changes 3
Bugs 1 Features 0
Metric Value
c 3
b 1
f 0
dl 0
loc 48
ccs 0
cts 30
cp 0
rs 5.3454
cc 10
eloc 20
nc 8
nop 1
crap 110

How to fix   Complexity   

Long Method

Small methods make your code easier to understand, in particular if combined with a good name. Besides, if your method is small, finding a good name is usually much easier.

For example, if you find yourself adding comments to a method's body, this is usually a good sign to extract the commented part to a new method, and use the comment as a starting point when coming up with a good name for this new method.

Commonly applied refactorings include:

1
<?php
2
3
namespace Stats\Views\Stats;
4
5
use Joomla\View\BaseJsonView;
6
7
/**
8
 * JSON response for requesting the stats data.
9
 *
10
 * @property-read  \Stats\Models\StatsModel  $model  The model object.
11
 *
12
 * @since          1.0
13
 */
14
class StatsJsonView extends BaseJsonView
15
{
16
	/**
17
	 * Flag if the response should return the raw data.
18
	 *
19
	 * @var    boolean
20
	 * @since  1.0
21
	 */
22
	private $authorizedRaw = false;
23
24
	/**
25
	 * Array holding the valid data sources.
26
	 *
27
	 * @var    array
28
	 * @since  1.0
29
	 */
30
	private $dataSources = ['php_version', 'db_type', 'db_version', 'cms_version', 'server_os'];
31
32
	/**
33
	 * The data source to return.
34
	 *
35
	 * @var    string
36
	 * @since  1.0
37
	 */
38
	private $source;
39
40
	/**
41
	 * Count of the number of items.
42
	 *
43
	 * @var    integer
44
	 * @since  1.0
45
	 */
46
	private $totalItems = 0;
47
48
	/**
49
	 * Set whether the raw data should be returned.
50
	 *
51
	 * @param   boolean  $authorizedRaw  Flag if the response should return the raw data.
52
	 *
53
	 * @return  void
54
	 *
55
	 * @since   1.0
56
	 */
57 1
	public function isAuthorizedRaw($authorizedRaw)
58
	{
59 1
		$this->authorizedRaw = $authorizedRaw;
60 1
	}
61
62
	/**
63
	 * Method to render the view.
64
	 *
65
	 * @return  string  The rendered view.
66
	 *
67
	 * @since   1.0
68
	 * @throws  \InvalidArgumentException
69
	 */
70 2
	public function render()
71
	{
72 2
		$items = $this->model->getItems($this->source);
73
74
		// Null out the model now to free some memory
75 2
		$this->model = null;
76
77 2
		if ($this->source)
78 2
		{
79
			return $this->processSingleSource($items);
80
		}
81
82 2
		$php_version = [];
83 2
		$db_type     = [];
84 2
		$db_version  = [];
85 2
		$cms_version = [];
86 2
		$server_os   = [];
87
88
		// If we have the entire database, we have to loop within each group to put it all together
89 2
		foreach ($items as $group)
90
		{
91 2
			$this->totalItems += count($group);
92
93 2
			foreach ($group as $item)
94
			{
95 2
				foreach ($this->dataSources as $source)
96
				{
97 2
					if (isset($item[$source]) && !is_null($item[$source]))
98 2
					{
99
						// Special case, if the server is empty then change the key to "unknown"
100 2
						if ($source === 'server_os' && empty($item[$source]))
101 2
						{
102 2
							if (!isset(${$source}['unknown']))
103 2
							{
104 2
								${$source}['unknown'] = 0;
105 2
							}
106
107 2
							${$source}['unknown']++;
108 2
						}
109
						else
110
						{
111 2
							if (!isset(${$source}[$item[$source]]))
112 2
							{
113 2
								${$source}[$item[$source]] = 0;
114 2
							}
115
116 2
							${$source}[$item[$source]]++;
117
						}
118 2
					}
119 2
				}
120 2
			}
121 2
		}
122
123 2
		unset($items);
124
125
		$data = [
126 2
			'php_version' => $php_version,
127 2
			'db_type'     => $db_type,
128 2
			'db_version'  => $db_version,
129 2
			'cms_version' => $cms_version,
130 2
			'server_os'   => $server_os,
131 2
		];
132
133 2
		$responseData = $this->buildResponseData($data);
134
135 2
		$responseData['total'] = $this->totalItems;
136
137 2
		$this->addData('data', $responseData);
138
139 2
		return parent::render();
140
	}
141
142
	/**
143
	 * Set the data source.
144
	 *
145
	 * @param   string  $source  Data source to return.
146
	 *
147
	 * @return  void
148
	 *
149
	 * @since   1.0
150
	 */
151 1
	public function setSource($source)
152
	{
153 1
		$this->source = $source;
154 1
	}
155
156
	/**
157
	 * Process the raw data into the response data format.
158
	 *
159
	 * @param   array  $data  The raw data array.
160
	 *
161
	 * @return  array
162
	 *
163
	 * @since   1.0
164
	 */
165 2
	private function buildResponseData(array $data)
166
	{
167 2
		$responseData = [];
168
169 2
		foreach ($data as $key => $value)
170
		{
171 2
			foreach ($value as $name => $count)
172
			{
173
				if ($name)
174 2
				{
175 2
					$responseData[$key][] = [
176 2
						'name'  => $name,
177
						'count' => $count
178 2
					];
179 2
				}
180 2
			}
181 2
		}
182
183 2
		unset($data);
184
185 2
		if (!$this->authorizedRaw)
186 2
		{
187 1
			$responseData = $this->sanitizeData($responseData);
188 1
		}
189
190 2
		return $responseData;
191
	}
192
193
	/**
194
	 * Process the response for a single data source.
195
	 *
196
	 * @param   \Generator  $generator  The source items to process.
197
	 *
198
	 * @return  string  The rendered view.
199
	 *
200
	 * @since   1.0
201
	 */
202
	private function processSingleSource(\Generator $generator)
203
	{
204
		$data = [
205
			${$this->source} = [],
206
		];
207
208
		foreach ($generator as $items)
209
		{
210
			$this->totalItems = count($items);
211
212
			foreach ($items as $item)
213
			{
214
				foreach ($this->dataSources as $source)
215
				{
216
					if (isset($item[$source]) && !is_null($item[$source]))
217
					{
218
						// Special case, if the server is empty then change the key to "unknown"
219
						if ($source === 'server_os' && empty($item[$source]))
220
						{
221
							if (!isset($data[$source]['unknown']))
222
							{
223
								$data[$source]['unknown'] = 0;
224
							}
225
226
							$data[$source]['unknown']++;
227
						}
228
						else
229
						{
230
							if (!isset($data[$source][$item[$source]]))
231
							{
232
								$data[$source][$item[$source]] = 0;
233
							}
234
235
							$data[$source][$item[$source]]++;
236
						}
237
					}
238
				}
239
			}
240
		}
241
242
		$responseData = $this->buildResponseData($data);
243
244
		$responseData['total'] = $this->totalItems;
245
246
		$this->addData('data', $responseData);
247
248
		return parent::render();
0 ignored issues
show
Comprehensibility Bug introduced by
It seems like you call parent on a different method (render() instead of processSingleSource()). Are you sure this is correct? If so, you might want to change this to $this->render().

This check looks for a call to a parent method whose name is different than the method from which it is called.

Consider the following code:

class Daddy
{
    protected function getFirstName()
    {
        return "Eidur";
    }

    protected function getSurName()
    {
        return "Gudjohnsen";
    }
}

class Son
{
    public function getFirstName()
    {
        return parent::getSurname();
    }
}

The getFirstName() method in the Son calls the wrong method in the parent class.

Loading history...
249
	}
250
251
	/**
252
	 * Sanitize the response data into summarized groups.
253
	 *
254
	 * @param   array  $responseData  The response data to sanitize.
255
	 *
256
	 * @return  array
257
	 *
258
	 * @since   1.0
259
	 */
260 1
	private function sanitizeData(array $responseData)
261
	{
262 1
		foreach ($responseData as $key => $dataGroup)
263
		{
264
			switch ($key)
265
			{
266 1
				case 'php_version':
267 1
				case 'db_version':
268
					// We're going to group by minor version branch here and convert to a percentage
269 1
					$counts = [];
270
271 1
					foreach ($dataGroup as $row)
272
					{
273 1
						$exploded = explode('.', $row['name']);
274 1
						$version  = $exploded[0] . '.' . (isset($exploded[1]) ? $exploded[1] : '0');
275
276
						// If the container does not exist, add it
277 1
						if (!isset($counts[$version]))
278 1
						{
279 1
							$counts[$version] = 0;
280 1
						}
281
282 1
						$counts[$version] += $row['count'];
283 1
					}
284
285 1
					$sanitizedData = [];
286
287 1 View Code Duplication
					foreach ($counts as $version => $count)
0 ignored issues
show
Duplication introduced by
This code seems to be duplicated across your project.

Duplicated code is one of the most pungent code smells. If you need to duplicate the same code in three or more different places, we strongly encourage you to look into extracting the code into a single class or operation.

You can also find more detailed suggestions in the “Code” section of your repository.

Loading history...
288
					{
289 1
						$sanitizedData[$version] = round(($count / $this->totalItems) * 100, 2);
290 1
					}
291
292 1
					$responseData[$key] = $sanitizedData;
293
294 1
					break;
295
296 1
				case 'server_os':
297
					// We're going to group by operating system here
298 1
					$counts = [];
299
300 1
					foreach ($dataGroup as $row)
301
					{
302 1
						$fullOs = explode(' ', $row['name']);
303 1
						$os     = $fullOs[0];
304
305
						// If the container does not exist, add it
306 1
						if (!isset($counts[$os]))
307 1
						{
308 1
							$counts[$os] = 0;
309 1
						}
310
311 1
						$counts[$os] += $row['count'];
312 1
					}
313
314 1
					$sanitizedData = [];
315
316 1 View Code Duplication
					foreach ($counts as $os => $count)
0 ignored issues
show
Duplication introduced by
This code seems to be duplicated across your project.

Duplicated code is one of the most pungent code smells. If you need to duplicate the same code in three or more different places, we strongly encourage you to look into extracting the code into a single class or operation.

You can also find more detailed suggestions in the “Code” section of your repository.

Loading history...
317
					{
318 1
						$sanitizedData[$os] = round(($count / $this->totalItems) * 100, 2);
319 1
					}
320
321 1
					$responseData[$key] = $sanitizedData;
322
323 1
					break;
324
325 1
				case 'db_type':
326 1
				case 'cms_version':
327 1
				default:
328
					// For now, group by the object name and figure out the percentages
329 1
					$sanitizedData = [];
330
331 1
					foreach ($dataGroup as $row)
332
					{
333 1
						$sanitizedData[$row['name']] = round(($row['count'] / $this->totalItems) * 100, 2);
334 1
					}
335
336 1
					$responseData[$key] = $sanitizedData;
337
338 1
					break;
339
			}
340 1
		}
341
342 1
		return $responseData;
343
	}
344
}
345