Code Coverage |
||||||||||
Lines |
Functions and Methods |
Classes and Traits |
||||||||
Total | |
91.19% |
238 / 261 |
|
82.14% |
23 / 28 |
CRAP | |
0.00% |
0 / 1 |
Scope | |
91.19% |
238 / 261 |
|
82.14% |
23 / 28 |
79.95 | |
0.00% |
0 / 1 |
readEntity | |
100.00% |
11 / 11 |
|
100.00% |
1 / 1 |
4 | |||
readResolvedReferences | |
100.00% |
4 / 4 |
|
100.00% |
1 / 1 |
2 | |||
readByClusterId | |
100.00% |
23 / 23 |
|
100.00% |
1 / 1 |
5 | |||
readByProviderId | |
65.00% |
13 / 20 |
|
0.00% |
0 / 1 |
6.07 | |||
readByRequestId | |
100.00% |
7 / 7 |
|
100.00% |
1 / 1 |
3 | |||
readByDepartmentId | |
100.00% |
26 / 26 |
|
100.00% |
1 / 1 |
6 | |||
readListBySource | |
0.00% |
0 / 7 |
|
0.00% |
0 / 1 |
6 | |||
testSource | |
0.00% |
0 / 2 |
|
0.00% |
0 / 1 |
6 | |||
readCollection | |
0.00% |
0 / 6 |
|
0.00% |
0 / 1 |
6 | |||
readList | |
100.00% |
11 / 11 |
|
100.00% |
1 / 1 |
4 | |||
readIsOpened | |
100.00% |
5 / 5 |
|
100.00% |
1 / 1 |
2 | |||
readIsEnabled | |
100.00% |
10 / 10 |
|
100.00% |
1 / 1 |
3 | |||
readWaitingNumberUpdated | |
100.00% |
9 / 9 |
|
100.00% |
1 / 1 |
3 | |||
readIsGivenNumberInContingent | |
100.00% |
4 / 4 |
|
100.00% |
1 / 1 |
2 | |||
readQueueList | |
100.00% |
7 / 7 |
|
100.00% |
1 / 1 |
2 | |||
readWithWorkstationCount | |
100.00% |
9 / 9 |
|
100.00% |
1 / 1 |
2 | |||
readQueueListWithWaitingTime | |
100.00% |
6 / 6 |
|
100.00% |
1 / 1 |
2 | |||
readListWithScopeAdminEmail | |
100.00% |
13 / 13 |
|
100.00% |
1 / 1 |
4 | |||
writeEntity | |
100.00% |
9 / 9 |
|
100.00% |
1 / 1 |
1 | |||
updateEntity | |
100.00% |
8 / 8 |
|
100.00% |
1 / 1 |
1 | |||
replacePreferences | |
100.00% |
7 / 7 |
|
100.00% |
1 / 1 |
4 | |||
updateGhostWorkstationCount | |
100.00% |
6 / 6 |
|
100.00% |
1 / 1 |
1 | |||
updateEmergency | |
100.00% |
7 / 7 |
|
100.00% |
1 / 1 |
1 | |||
writeImageData | |
93.33% |
14 / 15 |
|
0.00% |
0 / 1 |
4.00 | |||
readImageData | |
100.00% |
10 / 10 |
|
100.00% |
1 / 1 |
2 | |||
deleteImage | |
100.00% |
4 / 4 |
|
100.00% |
1 / 1 |
1 | |||
deleteEntity | |
100.00% |
9 / 9 |
|
100.00% |
1 / 1 |
3 | |||
deletePreferences | |
100.00% |
6 / 6 |
|
100.00% |
1 / 1 |
3 |
1 | <?php |
2 | |
3 | namespace BO\Zmsdb; |
4 | |
5 | use BO\Zmsentities\Scope as Entity; |
6 | use BO\Zmsentities\Collection\ScopeList as Collection; |
7 | |
8 | /** |
9 | * |
10 | * @SuppressWarnings(Public) |
11 | * @SuppressWarnings(Coupling) |
12 | * @SuppressWarnings(Complexity) |
13 | * @SuppressWarnings(TooManyMethods) |
14 | * |
15 | */ |
16 | class Scope extends Base |
17 | { |
18 | public static $cache = [ ]; |
19 | |
20 | public function readEntity($scopeId, $resolveReferences = 0, $disableCache = false) |
21 | { |
22 | $cacheKey = "$scopeId-$resolveReferences"; |
23 | if ($disableCache || ! array_key_exists($cacheKey, self::$cache)) { |
24 | $query = new Query\Scope(Query\Base::SELECT); |
25 | $query->addEntityMapping() |
26 | ->addResolvedReferences($resolveReferences) |
27 | ->addConditionScopeId($scopeId); |
28 | $scope = $this->fetchOne($query, new Entity()); |
29 | if (! $scope->hasId()) { |
30 | return null; |
31 | } |
32 | self::$cache[$cacheKey] = $this->readResolvedReferences($scope, $resolveReferences); |
33 | } |
34 | return self::$cache[$cacheKey]; |
35 | } |
36 | |
37 | public function readResolvedReferences( |
38 | \BO\Zmsentities\Schema\Entity $scope, |
39 | $resolveReferences |
40 | ) { |
41 | if (0 < $resolveReferences) { |
42 | $scope['dayoff'] = (new DayOff())->readByScopeId($scope->id); |
43 | $scope['closure'] = (new Closure())->readByScopeId($scope->id); |
44 | } |
45 | return $scope; |
46 | } |
47 | |
48 | public function readByClusterId( |
49 | $clusterId, |
50 | $resolveReferences = 0 |
51 | ) { |
52 | $scopeList = new Collection(); |
53 | if ($resolveReferences > 0) { |
54 | $query = new Query\Scope(Query\Base::SELECT); |
55 | $query->addEntityMapping() |
56 | ->addResolvedReferences($resolveReferences - 1) |
57 | ->addConditionClusterId($clusterId); |
58 | $result = $this->fetchList($query, new Entity()); |
59 | } else { |
60 | $result = $this->getReader()->perform( |
61 | (new Query\Scope(Query\Base::SELECT))->getQuerySimpleClusterMatch(), |
62 | [$clusterId] |
63 | ); |
64 | } |
65 | if ($result) { |
66 | foreach ($result as $entity) { |
67 | if (0 == $resolveReferences) { |
68 | $entity = new Entity( |
69 | array( |
70 | 'id' => $entity['id'], |
71 | '$ref' => '/scope/' . $entity['id'] . '/' |
72 | ) |
73 | ); |
74 | $scopeList->addEntity($entity); |
75 | } else { |
76 | $scopeList->addEntity($this->readResolvedReferences($entity, $resolveReferences - 1)); |
77 | } |
78 | } |
79 | } |
80 | return $scopeList; |
81 | } |
82 | |
83 | public function readByProviderId($providerId, $resolveReferences = 0) |
84 | { |
85 | $scopeList = new Collection(); |
86 | $query = new Query\Scope(Query\Base::SELECT); |
87 | $query->addEntityMapping() |
88 | ->addResolvedReferences($resolveReferences) |
89 | ->addConditionProviderId($providerId); |
90 | $result = $this->fetchList($query, new Entity()); |
91 | if (count($result)) { |
92 | foreach ($result as $entity) { |
93 | if (0 == $resolveReferences) { |
94 | $entity = new Entity( |
95 | array( |
96 | 'id' => $entity->id, |
97 | '$ref' => '/scope/' . $entity->id . '/' |
98 | ) |
99 | ); |
100 | $scopeList->addEntity($entity); |
101 | } else { |
102 | if ($entity instanceof Entity) { |
103 | $entity = $this->readResolvedReferences($entity, $resolveReferences - 1); |
104 | $scopeList->addEntity($entity); |
105 | } |
106 | } |
107 | } |
108 | } |
109 | return $scopeList; |
110 | } |
111 | |
112 | public function readByRequestId($requestId, $source, $resolveReferences = 0) |
113 | { |
114 | $scopeList = new Collection(); |
115 | $providerList = (new Provider())->readListBySource($source, 0, true, $requestId); |
116 | |
117 | foreach ($providerList as $provider) { |
118 | $scopeListByProvider = $this->readByProviderId($provider->getId(), $resolveReferences); |
119 | if ($scopeListByProvider->count()) { |
120 | $scopeList->addList($scopeListByProvider); |
121 | } |
122 | } |
123 | return $scopeList->withUniqueScopes(); |
124 | } |
125 | |
126 | public function readByDepartmentId($departmentId, $resolveReferences = 0) |
127 | { |
128 | $scopeList = new Collection(); |
129 | if ($resolveReferences > 0) { |
130 | $query = new Query\Scope(Query\Base::SELECT); |
131 | $query->addEntityMapping() |
132 | ->addResolvedReferences($resolveReferences) |
133 | ->addConditionDepartmentId($departmentId); |
134 | $result = $this->fetchList($query, new Entity()); |
135 | } else { |
136 | $result = $this->getReader()->perform( |
137 | (new Query\Scope(Query\Base::SELECT))->getQuerySimpleDepartmentMatch(), |
138 | [$departmentId] |
139 | ); |
140 | } |
141 | if ($result) { |
142 | foreach ($result as $entity) { |
143 | if (0 == $resolveReferences) { |
144 | $entity = new Entity( |
145 | array( |
146 | 'id' => $entity['id'], |
147 | 'contact' => ['name' => $entity['contact__name']], |
148 | '$ref' => '/scope/' . $entity['id'] . '/' |
149 | ) |
150 | ); |
151 | $scopeList->addEntity($entity); |
152 | } else { |
153 | if ($entity instanceof Entity) { |
154 | $entity = $this->readResolvedReferences($entity, $resolveReferences); |
155 | $scopeList->addEntity($entity); |
156 | } |
157 | } |
158 | } |
159 | } |
160 | return $scopeList; |
161 | } |
162 | public function readListBySource($source, $resolveReferences = 0) |
163 | { |
164 | $this->testSource($source); |
165 | $query = new Query\Request(Query\Base::SELECT); |
166 | $query->setResolveLevel($resolveReferences); |
167 | $query->addConditionRequestSource($source); |
168 | $query->addEntityMapping(); |
169 | $requestList = $this->readCollection($query); |
170 | return ($requestList->count()) ? $requestList->sortByCustomKey('id') : $requestList; |
171 | } |
172 | |
173 | protected function testSource($source) |
174 | { |
175 | if (! (new Source())->readEntity($source)) { |
176 | throw new Exception\Source\UnknownDataSource(); |
177 | } |
178 | } |
179 | |
180 | protected function readCollection($query) |
181 | { |
182 | $requestList = new Collection(); |
183 | $statement = $this->fetchStatement($query); |
184 | while ($requestData = $statement->fetch(\PDO::FETCH_ASSOC)) { |
185 | $request = new Entity($query->postProcessJoins($requestData)); |
186 | $requestList->addEntity($request); |
187 | } |
188 | return $requestList; |
189 | } |
190 | |
191 | public function readList($resolveReferences = 0) |
192 | { |
193 | $scopeList = new Collection(); |
194 | $query = new Query\Scope(Query\Base::SELECT); |
195 | $query->addEntityMapping() |
196 | ->addResolvedReferences($resolveReferences); |
197 | $result = $this->fetchList($query, new Entity()); |
198 | if (count($result)) { |
199 | foreach ($result as $entity) { |
200 | if ($entity instanceof Entity) { |
201 | $entity = $this->readResolvedReferences($entity, $resolveReferences); |
202 | $scopeList->addEntity($entity); |
203 | } |
204 | } |
205 | } |
206 | return $scopeList; |
207 | } |
208 | |
209 | /** |
210 | * get a scope and return true if it is opened |
211 | * |
212 | * * @param |
213 | * scopeId |
214 | * now |
215 | * |
216 | * @return Bool |
217 | */ |
218 | public function readIsOpened($scopeId, $now) |
219 | { |
220 | $isOpened = false; |
221 | $availabilityList = (new Availability())->readOpeningHoursListByDate($scopeId, $now, 2); |
222 | if ($availabilityList->isOpened($now)) { |
223 | $isOpened = true; |
224 | } |
225 | return $isOpened; |
226 | } |
227 | |
228 | public function readIsEnabled($scopeId, $now) |
229 | { |
230 | $query = new Query\Scope(Query\Base::SELECT); |
231 | $query->addEntityMapping() |
232 | ->setResolveLevel(0) |
233 | ->addConditionScopeId($scopeId); |
234 | $scope = $this->fetchOne($query, new Entity()); |
235 | return ( |
236 | $this->readIsOpened($scopeId, $now) && |
237 | $this->readIsGivenNumberInContingent($scopeId) && |
238 | ! $scope->getStatus('ticketprinter', 'deactivated') |
239 | ); |
240 | } |
241 | |
242 | /** |
243 | * get last given waitingnumer and return updated (+1) waitingnumber |
244 | * |
245 | * * @param |
246 | * scopeId |
247 | * now |
248 | * |
249 | * @return Bool |
250 | */ |
251 | public function readWaitingNumberUpdated($scopeId, $dateTime, $respectContingent = true) |
252 | { |
253 | if (! $this->readIsGivenNumberInContingent($scopeId) && $respectContingent) { |
254 | throw new Exception\Scope\GivenNumberCountExceeded(); |
255 | } |
256 | $this->perform( |
257 | (new Query\Scope(Query\Base::SELECT))->getQueryLastWaitingNumber(), |
258 | ['scope_id' => $scopeId] |
259 | ); |
260 | $entity = $this->readEntity($scopeId, 0, true)->updateStatusQueue($dateTime); |
261 | $scope = $this->updateEntity($scopeId, $entity); |
262 | return $scope->getStatus('queue', 'lastGivenNumber'); |
263 | } |
264 | |
265 | /** |
266 | * get last given waitingnumer and return updated (+1) waitingnumber |
267 | * |
268 | * * @param |
269 | * scopeId |
270 | * now |
271 | * |
272 | * @return Bool |
273 | */ |
274 | public function readIsGivenNumberInContingent($scopeId) |
275 | { |
276 | $isInContingent = $this->getReader() |
277 | ->fetchValue((new Query\Scope(Query\Base::SELECT)) |
278 | ->getQueryGivenNumbersInContingent(), ['scope_id' => $scopeId]); |
279 | return ($isInContingent) ? true : false; |
280 | } |
281 | |
282 | /** |
283 | * get list of queues on scope by daytime |
284 | * |
285 | * * @param |
286 | * scopeId |
287 | * now |
288 | * |
289 | * @return number |
290 | */ |
291 | public function readQueueList($scopeId, $dateTime, $resolveReferences = 0) |
292 | { |
293 | if ($resolveReferences > 0) { |
294 | // resolveReferences > 0 is only necessary for a resolved process |
295 | $queueList = (new Process()) |
296 | ->readProcessListByScopeAndTime($scopeId, $dateTime, $resolveReferences - 1) |
297 | ->toQueueList($dateTime); |
298 | } else { |
299 | $queueList = (new Queue()) |
300 | ->readListByScopeAndTime($scopeId, $dateTime, $resolveReferences); |
301 | } |
302 | return $queueList->withSortedArrival(); |
303 | } |
304 | |
305 | /** |
306 | * get waitingtime of scope |
307 | * |
308 | * * @param |
309 | * scopeId |
310 | * now |
311 | * |
312 | * @return \BO\Zmsentities\Scope |
313 | */ |
314 | public function readWithWorkstationCount($scopeId, $dateTime, $resolveReferences = 0) |
315 | { |
316 | //get scope |
317 | $query = new Query\Scope(Query\Base::SELECT); |
318 | $query |
319 | ->addEntityMapping() |
320 | ->addConditionScopeId($scopeId) |
321 | ->addResolvedReferences($resolveReferences) |
322 | ->addSelectWorkstationCount($dateTime); |
323 | $scope = $this->fetchOne($query, new Entity()); |
324 | $scope = $this->readResolvedReferences($scope, $resolveReferences); |
325 | return ($scope->hasId()) ? $scope : null; |
326 | } |
327 | |
328 | public function readQueueListWithWaitingTime($scope, $dateTime, $resolveReferences = 0) |
329 | { |
330 | $timeAverage = $scope->getPreference('queue', 'processingTimeAverage'); |
331 | $scope = (! $timeAverage) ? (new Scope())->readEntity($scope->id) : $scope; |
332 | $queueList = $this->readQueueList($scope->id, $dateTime, $resolveReferences); |
333 | $timeAverage = $scope->getPreference('queue', 'processingTimeAverage'); |
334 | $workstationCount = $scope->getCalculatedWorkstationCount(); |
335 | return $queueList->withEstimatedWaitingTime($timeAverage, $workstationCount, $dateTime); |
336 | } |
337 | |
338 | /** |
339 | * get list of scopes with admin |
340 | * |
341 | * * @param |
342 | * scopeId |
343 | * now |
344 | * |
345 | * @return number |
346 | */ |
347 | public function readListWithScopeAdminEmail($resolveReferences = 0) |
348 | { |
349 | $scopeList = new Collection(); |
350 | $query = new Query\Scope(Query\Base::SELECT); |
351 | $query |
352 | ->addEntityMapping() |
353 | ->addConditionWithAdminEmail() |
354 | ->addResolvedReferences($resolveReferences); |
355 | $result = $this->fetchList($query, new Entity()); |
356 | if (count($result)) { |
357 | foreach ($result as $entity) { |
358 | if ($entity instanceof Entity) { |
359 | $entity = $this->readResolvedReferences($entity, $resolveReferences); |
360 | $scopeList->addEntity($entity); |
361 | } |
362 | } |
363 | } |
364 | return $scopeList; |
365 | } |
366 | |
367 | |
368 | |
369 | |
370 | |
371 | /** |
372 | * write a scope |
373 | * |
374 | * @return Entity |
375 | */ |
376 | public function writeEntity(\BO\Zmsentities\Scope $entity, $parentId) |
377 | { |
378 | self::$cache = []; |
379 | $query = new Query\Scope(Query\Base::INSERT); |
380 | $values = $query->reverseEntityMapping($entity, $parentId); |
381 | $query->addValues($values); |
382 | $this->writeItem($query); |
383 | $lastInsertId = $this->getWriter() |
384 | ->lastInsertId(); |
385 | $this->replacePreferences($entity); |
386 | return $this->readEntity($lastInsertId); |
387 | } |
388 | |
389 | /** |
390 | * update a scope |
391 | * |
392 | * @param |
393 | * scopeId |
394 | * |
395 | * @return Entity |
396 | */ |
397 | public function updateEntity($scopeId, \BO\Zmsentities\Scope $entity, $resolveReferences = 0) |
398 | { |
399 | self::$cache = []; |
400 | $query = new Query\Scope(Query\Base::UPDATE); |
401 | $query->addConditionScopeId($scopeId); |
402 | $values = $query->reverseEntityMapping($entity); |
403 | $query->addValues($values); |
404 | $this->writeItem($query); |
405 | $this->replacePreferences($entity); |
406 | return $this->readEntity($scopeId, $resolveReferences); |
407 | } |
408 | |
409 | public function replacePreferences(\BO\Zmsentities\Scope $entity) |
410 | { |
411 | if (isset($entity['preferences'])) { |
412 | $preferenceQuery = new Preferences(); |
413 | $entityName = 'scope'; |
414 | $entityId = $entity['id']; |
415 | foreach ($entity['preferences'] as $groupName => $groupValues) { |
416 | foreach ($groupValues as $name => $value) { |
417 | $preferenceQuery->replaceProperty($entityName, $entityId, $groupName, $name, $value); |
418 | } |
419 | } |
420 | } |
421 | } |
422 | |
423 | /** |
424 | * update ghostWorkstationCount |
425 | * |
426 | * @param |
427 | * scopeId |
428 | * entity |
429 | * dateTime (now) |
430 | * |
431 | * @return Entity |
432 | */ |
433 | public function updateGhostWorkstationCount(\BO\Zmsentities\Scope $entity, \DateTimeInterface $dateTime) |
434 | { |
435 | $query = new Query\Scope(Query\Base::UPDATE); |
436 | $query->addConditionScopeId($entity->id); |
437 | $values = $query->setGhostWorkstationCountEntityMapping($entity, $dateTime); |
438 | $query->addValues($values); |
439 | $this->writeItem($query); |
440 | return $entity; |
441 | } |
442 | |
443 | /** |
444 | * update emergency |
445 | * |
446 | * @param |
447 | * scopeId |
448 | * entity |
449 | * |
450 | * @return Entity |
451 | */ |
452 | public function updateEmergency($scopeId, \BO\Zmsentities\Scope $entity) |
453 | { |
454 | self::$cache = []; |
455 | $query = new Query\Scope(Query\Base::UPDATE); |
456 | $query->addConditionScopeId($scopeId); |
457 | $values = $query->setEmergencyEntityMapping($entity); |
458 | $query->addValues($values); |
459 | $this->writeItem($query); |
460 | return $this->readEntity($scopeId); |
461 | } |
462 | |
463 | /** |
464 | * update image data for call display image |
465 | * |
466 | * @param |
467 | * scopeId |
468 | * Mimepart entity |
469 | * |
470 | * @return Mimepart entity |
471 | */ |
472 | public function writeImageData($scopeId, \BO\Zmsentities\Mimepart $entity) |
473 | { |
474 | if ($entity->mime && $entity->content) { |
475 | $this->deleteImage($scopeId); |
476 | $extension = $entity->getExtension(); |
477 | if ($extension == 'jpeg') { |
478 | $extension = 'jpg'; //compatibility ZMS1 |
479 | } |
480 | $imageName = 's_' . $scopeId . '_bild.' . $extension; |
481 | $this->getWriter()->perform( |
482 | (new Query\Scope(Query\Base::REPLACE))->getQueryWriteImageData(), |
483 | array( |
484 | 'imagename' => $imageName, |
485 | 'imagedata' => $entity->content |
486 | ) |
487 | ); |
488 | } |
489 | $entity->id = $scopeId; |
490 | return $entity; |
491 | } |
492 | |
493 | /** |
494 | * read image data |
495 | * |
496 | * @param |
497 | * scopeId |
498 | * |
499 | * @return Mimepart entity |
500 | */ |
501 | public function readImageData($scopeId) |
502 | { |
503 | $imageName = 's_' . $scopeId . '_bild'; |
504 | $imageData = new \BO\Zmsentities\Mimepart(); |
505 | $fileData = $this->getReader()->fetchAll( |
506 | (new Query\Scope(Query\Base::SELECT))->getQueryReadImageData(), |
507 | ['imagename' => "$imageName%"] |
508 | ); |
509 | if ($fileData) { |
510 | $imageData->content = $fileData[0]['imagecontent']; |
511 | $imageData->mime = pathinfo($fileData[0]['imagename'])['extension']; |
512 | } |
513 | return $imageData; |
514 | } |
515 | |
516 | /** |
517 | * delete image data for call display image |
518 | * |
519 | * @param |
520 | * scopeId |
521 | * |
522 | * @return Status |
523 | */ |
524 | public function deleteImage($scopeId) |
525 | { |
526 | $imageName = 's_' . $scopeId . '_bild'; |
527 | return $this->perform((new Query\Scope(Query\Base::DELETE))->getQueryDeleteImage(), array( |
528 | 'imagename' => "$imageName%" |
529 | )); |
530 | } |
531 | |
532 | /** |
533 | * remove a scope |
534 | * |
535 | * @param |
536 | * scopeId |
537 | * |
538 | * @return Resource Status |
539 | */ |
540 | public function deleteEntity($scopeId) |
541 | { |
542 | $processListCount = (new Process())->readProcessListCountByScope($scopeId); |
543 | if (0 < $processListCount) { |
544 | throw new Exception\Scope\ScopeHasProcesses(); |
545 | } |
546 | self::$cache = []; |
547 | $entity = $this->readEntity($scopeId); |
548 | $query = new Query\Scope(Query\Base::DELETE); |
549 | $query->addConditionScopeId($scopeId); |
550 | $this->deletePreferences($entity); |
551 | return ($this->deleteItem($query)) ? $entity : null; |
552 | } |
553 | |
554 | public function deletePreferences(\BO\Zmsentities\Scope $entity) |
555 | { |
556 | $preferenceQuery = new Preferences(); |
557 | $entityName = 'scope'; |
558 | $entityId = $entity['id']; |
559 | foreach ($entity['preferences'] as $groupName => $groupValues) { |
560 | foreach (array_keys($groupValues) as $name) { |
561 | $preferenceQuery->deleteProperty($entityName, $entityId, $groupName, $name); |
562 | } |
563 | } |
564 | } |
565 | } |