Code Coverage |
||||||||||
Lines |
Functions and Methods |
Classes and Traits |
||||||||
Total | |
61.07% |
80 / 131 |
|
55.56% |
5 / 9 |
CRAP | |
0.00% |
0 / 1 |
ProcessStatusFree | |
61.07% |
80 / 131 |
|
55.56% |
5 / 9 |
92.42 | |
0.00% |
0 / 1 |
prepareCalendarAndDays | |
100.00% |
11 / 11 |
|
100.00% |
1 / 1 |
3 | |||
getProcessDataHandle | |
100.00% |
12 / 12 |
|
100.00% |
1 / 1 |
4 | |||
readFreeProcesses | |
100.00% |
20 / 20 |
|
100.00% |
1 / 1 |
3 | |||
readFreeProcessesMinimalDeduplicated | |
0.00% |
0 / 12 |
|
0.00% |
0 / 1 |
20 | |||
extractProcessInfo | |
0.00% |
0 / 18 |
|
0.00% |
0 / 1 |
42 | |||
generateUniqueKey | |
0.00% |
0 / 1 |
|
0.00% |
0 / 1 |
2 | |||
createMinimalProcess | |
0.00% |
0 / 20 |
|
0.00% |
0 / 1 |
2 | |||
readReservedProcesses | |
100.00% |
15 / 15 |
|
100.00% |
1 / 1 |
4 | |||
writeEntityReserved | |
100.00% |
22 / 22 |
|
100.00% |
1 / 1 |
6 |
1 | <?php |
2 | |
3 | namespace BO\Zmsdb; |
4 | |
5 | use BO\Zmsentities\Process as Entity; |
6 | use BO\Zmsentities\Collection\ProcessList as Collection; |
7 | |
8 | /** |
9 | * @SuppressWarnings(Coupling) |
10 | */ |
11 | class ProcessStatusFree extends Process |
12 | { |
13 | private function prepareCalendarAndDays( |
14 | \BO\Zmsentities\Calendar $calendar, |
15 | \DateTimeInterface $now, |
16 | $slotsRequired = null |
17 | ) { |
18 | $calendar = (new Calendar())->readResolvedEntity($calendar, $now, true); |
19 | $dayquery = new Day(); |
20 | $dayquery->writeTemporaryScopeList($calendar, $slotsRequired); |
21 | $selectedDate = $calendar->getFirstDay(); |
22 | $days = [$selectedDate]; |
23 | if ($calendar->getLastDay(false)) { |
24 | $days = []; |
25 | while ($selectedDate <= $calendar->getLastDay(false)) { |
26 | $days[] = $selectedDate; |
27 | $selectedDate = $selectedDate->modify('+1 day'); |
28 | } |
29 | } |
30 | return [$calendar, $dayquery, $days]; |
31 | } |
32 | |
33 | private function getProcessDataHandle( |
34 | array $days, |
35 | $slotType, |
36 | $slotsRequired, |
37 | $groupData |
38 | ) { |
39 | return $this->fetchHandle( |
40 | sprintf( |
41 | Query\ProcessStatusFree::QUERY_SELECT_PROCESSLIST_DAYS, |
42 | Query\ProcessStatusFree::buildDaysCondition($days) |
43 | ) |
44 | . ($groupData ? Query\ProcessStatusFree::GROUPBY_SELECT_PROCESSLIST_DAY : ''), |
45 | [ |
46 | 'slotType' => $slotType, |
47 | 'forceRequiredSlots' => |
48 | ($slotsRequired === null || $slotsRequired < 1) ? 1 : intval($slotsRequired), |
49 | ] |
50 | ); |
51 | } |
52 | |
53 | public function readFreeProcesses( |
54 | \BO\Zmsentities\Calendar $calendar, |
55 | \DateTimeInterface $now, |
56 | $slotType = 'public', |
57 | $slotsRequired = null, |
58 | $groupData = false |
59 | ) { |
60 | list($calendar, $dayquery, $days) = $this->prepareCalendarAndDays($calendar, $now, $slotsRequired); |
61 | $processData = $this->getProcessDataHandle($days, $slotType, $slotsRequired, $groupData); |
62 | $processList = new Collection(); |
63 | $scopeList = []; |
64 | while ($item = $processData->fetch(\PDO::FETCH_ASSOC)) { |
65 | $process = new \BO\Zmsentities\Process($item); |
66 | $process->requests = $calendar->requests; |
67 | $process->appointments->getFirst()->setDateByString( |
68 | $process->appointments->getFirst()->date, |
69 | 'Y-m-d H:i:s' |
70 | ); |
71 | |
72 | if (! isset($scopeList[$process->scope->id])) { |
73 | $scopeList[$process->scope->id] = $calendar->scopes->getEntity($process->scope->id); |
74 | } |
75 | |
76 | $process->scope = $scopeList[$process->scope->id]; |
77 | $process->queue['withAppointment'] = 1; |
78 | $process->appointments->getFirst()->scope = $process->scope; |
79 | $processList->addEntity($process); |
80 | } |
81 | $processData->closeCursor(); |
82 | unset($dayquery); |
83 | return $processList; |
84 | } |
85 | |
86 | public function readFreeProcessesMinimalDeduplicated( |
87 | \BO\Zmsentities\Calendar $calendar, |
88 | \DateTimeInterface $now, |
89 | string $slotType = 'public', |
90 | ?int $slotsRequired = null, |
91 | bool $groupData = false |
92 | ): array { |
93 | list($calendar, $dayquery, $days) = $this->prepareCalendarAndDays($calendar, $now, $slotsRequired); |
94 | $processData = $this->getProcessDataHandle($days, $slotType, $slotsRequired, $groupData); |
95 | |
96 | $unique = []; |
97 | while ($item = $processData->fetch(\PDO::FETCH_ASSOC)) { |
98 | $processInfo = $this->extractProcessInfo($item, $calendar); |
99 | if ($processInfo) { |
100 | $key = $this->generateUniqueKey($processInfo['providerId'], $processInfo['date']); |
101 | if (!isset($unique[$key])) { |
102 | $unique[$key] = $this->createMinimalProcess($processInfo); |
103 | } |
104 | } |
105 | } |
106 | |
107 | $processData->closeCursor(); |
108 | unset($dayquery); |
109 | |
110 | return array_values($unique); |
111 | } |
112 | |
113 | private function extractProcessInfo(array $item, \BO\Zmsentities\Calendar $calendar): ?array |
114 | { |
115 | $scopeId = $item['scope__id'] ?? null; |
116 | $dateString = $item['appointments__0__date'] ?? null; |
117 | |
118 | if (!$scopeId || !$dateString) { |
119 | return null; |
120 | } |
121 | |
122 | $date = strtotime($dateString); |
123 | if (!$date) { |
124 | return null; |
125 | } |
126 | |
127 | $scope = $calendar->scopes->getEntity($scopeId); |
128 | if (!$scope) { |
129 | return null; |
130 | } |
131 | |
132 | $providerId = $scope->getProviderId(); |
133 | if (!$providerId) { |
134 | return null; |
135 | } |
136 | |
137 | return [ |
138 | 'scopeId' => $scopeId, |
139 | 'providerId' => $providerId, |
140 | 'date' => $date |
141 | ]; |
142 | } |
143 | |
144 | private function generateUniqueKey(string $providerId, int $date): string |
145 | { |
146 | return $providerId . '_' . $date; |
147 | } |
148 | |
149 | private function createMinimalProcess(array $processInfo): array |
150 | { |
151 | return [ |
152 | '$schema' => 'https://schema.berlin.de/queuemanagement/process.json', |
153 | 'scope' => [ |
154 | 'id' => $processInfo['scopeId'], |
155 | 'provider' => [ |
156 | 'id' => $processInfo['providerId'] |
157 | ] |
158 | ], |
159 | 'appointments' => [ |
160 | [ |
161 | 'date' => (string)$processInfo['date'], |
162 | 'scope' => [ |
163 | 'id' => $processInfo['scopeId'], |
164 | 'provider' => [ |
165 | 'id' => $processInfo['providerId'] |
166 | ] |
167 | ] |
168 | ] |
169 | ] |
170 | ]; |
171 | } |
172 | |
173 | public function readReservedProcesses($resolveReferences = 2) |
174 | { |
175 | $processList = new Collection(); |
176 | $query = new Query\Process(Query\Base::SELECT); |
177 | $query |
178 | ->addResolvedReferences($resolveReferences) |
179 | ->addEntityMapping() |
180 | ->addConditionAssigned() |
181 | ->addConditionIsReserved(); |
182 | $resultData = $this->fetchList($query, new Entity()); |
183 | foreach ($resultData as $process) { |
184 | if (2 == $resolveReferences) { |
185 | $process['requests'] = (new Request())->readRequestByProcessId($process->id, $resolveReferences); |
186 | $process['scope'] = (new Scope())->readEntity($process->getScopeId(), $resolveReferences); |
187 | } |
188 | if ($process instanceof Entity) { |
189 | $processList->addEntity($process); |
190 | } |
191 | } |
192 | return $processList; |
193 | } |
194 | |
195 | /** |
196 | * Insert a new process if there are free slots |
197 | * |
198 | * @param \BO\Zmsentities\Process $process |
199 | * @param \DateTimeInterface $now |
200 | * @param String $slotType |
201 | * @param Int $slotsRequired we cannot use process.appointments.0.slotCount, because setting slotsRequired is |
202 | * a priviliged operation. Just using the input would be a security flaw to get a wider selection of times |
203 | * If slotsRequired = 0, readFreeProcesses() uses the slotsRequired based on request-provider relation |
204 | */ |
205 | public function writeEntityReserved( |
206 | \BO\Zmsentities\Process $process, |
207 | \DateTimeInterface $now, |
208 | $slotType = "public", |
209 | $slotsRequired = 0, |
210 | $resolveReferences = 0, |
211 | $userAccount = null |
212 | ) { |
213 | $process = clone $process; |
214 | $process->status = 'reserved'; |
215 | $appointment = $process->getAppointments()->getFirst(); |
216 | $slotList = (new Slot())->readByAppointment( |
217 | $appointment, |
218 | $slotsRequired, |
219 | (null !== $userAccount), |
220 | true |
221 | ); |
222 | $freeProcessList = $this->readFreeProcesses($process->toCalendar(), $now, $slotType, $slotsRequired); |
223 | |
224 | if (!$freeProcessList->getAppointmentList()->hasAppointment($appointment) || ! $slotList) { |
225 | throw new Exception\Process\ProcessReserveFailed(); |
226 | } |
227 | |
228 | foreach ($slotList as $slot) { |
229 | if ($process->id > 99999) { |
230 | $newProcess = clone $process; |
231 | $newProcess->getFirstAppointment()->setTime($slot->time); |
232 | $this->writeNewProcess($newProcess, $now, $process->id, 0, true, $userAccount); |
233 | } elseif ($process->id === 0) { |
234 | $process = $this->writeNewProcess($process, $now, 0, count($slotList) - 1, true, $userAccount); |
235 | } else { |
236 | throw new \Exception("SQL UPDATE error on inserting new $process on $slot"); |
237 | } |
238 | } |
239 | $this->writeRequestsToDb($process); |
240 | return $this->readEntity($process->getId(), new Helper\NoAuth(), $resolveReferences); |
241 | } |
242 | } |