Code Coverage
 
Lines
Functions and Methods
Classes and Traits
Total
91.19% covered (success)
91.19%
238 / 261
82.14% covered (warning)
82.14%
23 / 28
CRAP
0.00% covered (danger)
0.00%
0 / 1
Scope
91.19% covered (success)
91.19%
238 / 261
82.14% covered (warning)
82.14%
23 / 28
79.95
0.00% covered (danger)
0.00%
0 / 1
 readEntity
100.00% covered (success)
100.00%
11 / 11
100.00% covered (success)
100.00%
1 / 1
4
 readResolvedReferences
100.00% covered (success)
100.00%
4 / 4
100.00% covered (success)
100.00%
1 / 1
2
 readByClusterId
100.00% covered (success)
100.00%
23 / 23
100.00% covered (success)
100.00%
1 / 1
5
 readByProviderId
65.00% covered (warning)
65.00%
13 / 20
0.00% covered (danger)
0.00%
0 / 1
6.07
 readByRequestId
100.00% covered (success)
100.00%
7 / 7
100.00% covered (success)
100.00%
1 / 1
3
 readByDepartmentId
100.00% covered (success)
100.00%
26 / 26
100.00% covered (success)
100.00%
1 / 1
6
 readListBySource
0.00% covered (danger)
0.00%
0 / 7
0.00% covered (danger)
0.00%
0 / 1
6
 testSource
0.00% covered (danger)
0.00%
0 / 2
0.00% covered (danger)
0.00%
0 / 1
6
 readCollection
0.00% covered (danger)
0.00%
0 / 6
0.00% covered (danger)
0.00%
0 / 1
6
 readList
100.00% covered (success)
100.00%
11 / 11
100.00% covered (success)
100.00%
1 / 1
4
 readIsOpened
100.00% covered (success)
100.00%
5 / 5
100.00% covered (success)
100.00%
1 / 1
2
 readIsEnabled
100.00% covered (success)
100.00%
10 / 10
100.00% covered (success)
100.00%
1 / 1
3
 readWaitingNumberUpdated
100.00% covered (success)
100.00%
9 / 9
100.00% covered (success)
100.00%
1 / 1
3
 readIsGivenNumberInContingent
100.00% covered (success)
100.00%
4 / 4
100.00% covered (success)
100.00%
1 / 1
2
 readQueueList
100.00% covered (success)
100.00%
7 / 7
100.00% covered (success)
100.00%
1 / 1
2
 readWithWorkstationCount
100.00% covered (success)
100.00%
9 / 9
100.00% covered (success)
100.00%
1 / 1
2
 readQueueListWithWaitingTime
100.00% covered (success)
100.00%
6 / 6
100.00% covered (success)
100.00%
1 / 1
2
 readListWithScopeAdminEmail
100.00% covered (success)
100.00%
13 / 13
100.00% covered (success)
100.00%
1 / 1
4
 writeEntity
100.00% covered (success)
100.00%
9 / 9
100.00% covered (success)
100.00%
1 / 1
1
 updateEntity
100.00% covered (success)
100.00%
8 / 8
100.00% covered (success)
100.00%
1 / 1
1
 replacePreferences
100.00% covered (success)
100.00%
7 / 7
100.00% covered (success)
100.00%
1 / 1
4
 updateGhostWorkstationCount
100.00% covered (success)
100.00%
6 / 6
100.00% covered (success)
100.00%
1 / 1
1
 updateEmergency
100.00% covered (success)
100.00%
7 / 7
100.00% covered (success)
100.00%
1 / 1
1
 writeImageData
93.33% covered (success)
93.33%
14 / 15
0.00% covered (danger)
0.00%
0 / 1
4.00
 readImageData
100.00% covered (success)
100.00%
10 / 10
100.00% covered (success)
100.00%
1 / 1
2
 deleteImage
100.00% covered (success)
100.00%
4 / 4
100.00% covered (success)
100.00%
1 / 1
1
 deleteEntity
100.00% covered (success)
100.00%
9 / 9
100.00% covered (success)
100.00%
1 / 1
3
 deletePreferences
100.00% covered (success)
100.00%
6 / 6
100.00% covered (success)
100.00%
1 / 1
3
1<?php
2
3namespace BO\Zmsdb;
4
5use BO\Zmsentities\Scope as Entity;
6use BO\Zmsentities\Collection\ScopeList as Collection;
7
8/**
9 *
10 * @SuppressWarnings(Public)
11 * @SuppressWarnings(Coupling)
12 * @SuppressWarnings(Complexity)
13 * @SuppressWarnings(TooManyMethods)
14 *
15 */
16class 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}