ILIAS  trunk Revision v11.0_alpha-1689-g66c127b4ae8
All Data Structures Namespaces Files Functions Variables Enumerations Enumerator Modules Pages
class.ilTestEvaluationFactory.php
Go to the documentation of this file.
1 <?php
2 
19 declare(strict_types=1);
20 
22 
27 {
28  public function __construct(
29  protected ilDBInterface $db,
30  protected ilObjTest $test_obj
31  ) {
32  }
33 
37  private function getAccessFilteredActiveIds(): array
38  {
39  if (($participants_list = $this->test_obj->getAccessFilteredParticipantList()) !== null) {
40  return $participants_list->getAllActiveIds();
41  }
42  return array_keys($this->test_obj->getTestParticipants());
43  }
44 
48  private function retrieveEvaluationData(array $active_ids): \Generator
49  {
50  $query = '
51  SELECT tst_test_result.question_fi,
52  tst_test_result.points result_points,
53  tst_test_result.answered,
54  tst_test_result.manual,
55 
56  qpl_questions.original_id,
57  qpl_questions.title questiontitle,
58  qpl_questions.points qpl_maxpoints,
59 
60  tst_active.active_id,
61  tst_active.submitted,
62  tst_active.last_finished_pass,
63  tst_pass_result.*,
64 
65  usr_data.usr_id,
66  usr_data.firstname,
67  usr_data.lastname,
68  usr_data.title,
69  usr_data.login
70 
71  FROM tst_active
72 
73  LEFT JOIN tst_pass_result ON tst_active.active_id = tst_pass_result.active_fi
74  LEFT JOIN tst_test_result ON tst_active.active_id = tst_test_result.active_fi AND tst_test_result.pass = tst_pass_result.pass
75  LEFT JOIN qpl_questions ON qpl_questions.question_id = tst_test_result.question_fi
76  LEFT JOIN usr_data ON tst_active.user_fi = usr_data.usr_id
77 
78  WHERE tst_active.test_fi = %s
79  AND %s
80 
81  ORDER BY tst_active.active_id ASC, tst_pass_result.pass ASC, tst_test_result.tstamp DESC
82  ';
83 
84  $ret = [];
85  $result = $this->db->query(
86  sprintf(
87  $query,
88  $this->db->quote($this->test_obj->getTestId(), 'integer'),
89  $this->db->in('tst_active.active_id', $active_ids, false, 'integer'),
90  )
91  );
92  while ($row = $this->db->fetchAssoc($result)) {
93  yield $row;
94  }
95  }
96 
98  {
99  $participants = [];
100  $current_user = null;
101  $current_attempt = null;
102 
103  foreach ($this->retrieveEvaluationData($this->getAccessFilteredActiveIds()) as $row) {
104  $active_id = $row['active_id'];
105  $pass = $row['pass'];
106 
107  if ($current_user !== $active_id) {
108  $current_user = $active_id;
109  $current_attempt = null;
110  $user_eval_data = $this->buildBasicUserEvaluationDataFromDB($row);
111  }
112 
113  if ($current_attempt !== $pass) {
114  $current_attempt = $pass;
115  $attempt = $this->buildBasicAttemptEvaluationDataFromDB($row);
116  }
117 
118  $attempt = $this->addQuestionToAttempt($attempt, $row);
119  $user_eval_data->addPass($pass, $attempt);
120  $participants[$active_id] = $user_eval_data;
121  }
122  return new ilTestEvaluationData($participants);
123  }
124 
126  {
127  $eval_data_rows = $this->retrieveEvaluationData($this->getAccessFilteredActiveIds());
128  $participants = [];
129  $current_user = null;
130  $current_attempt = null;
131 
132  foreach ($eval_data_rows as $row) {
133  if($row['pass'] === null) {
134  continue;
135  }
136 
137  if ($current_user !== $row['active_id']) {
138  $current_user = $row['active_id'];
139  $current_attempt = null;
140  $user_eval_data = $this->addVisitingTimeToUserEvalData(
142  $row['active_id']
143  );
144  }
145 
146  if ($row['pass'] !== null && $current_attempt !== $row['pass']) {
147  $current_attempt = $row['pass'];
148  $attempt = $this->addPointsAndQuestionCountToAttempt(
150  $row
151  );
152 
153  $start_time = $this->getFirstVisitForActiveIdAndAttempt($row['active_id'], $current_attempt);
154  if ($start_time !== null) {
155  $attempt->setStartTime($start_time);
156  }
157  $attempt->setStatusOfAttempt(
158  $this->buildFinalizedBy($current_attempt, $row['last_finished_pass'], $row['finalized_by'])
159  );
160  }
161 
162  $user_eval_data->addPass($row['pass'], $this->addQuestionToAttempt($attempt, $row));
163  $participants[$row['active_id']] = $user_eval_data;
164  }
165 
166  $evaluation_data = $this->addQuestionsToParticipantPasses(
167  new ilTestEvaluationData($participants)
168  );
169  return $this->addMarksToParticipants($evaluation_data);
170  }
171 
173  {
174  $user_data = new ilTestEvaluationUserData($this->test_obj->getPassScoring());
175 
176  $user_data->setName(
177  $this->test_obj->buildName($row['usr_id'], $row['firstname'], $row['lastname'])
178  );
179 
180  if ($row['login'] !== null) {
181  $user_data->setLogin($row['login']);
182  }
183  if ($row['usr_id'] !== null) {
184  $user_data->setUserID($row['usr_id']);
185  }
186  $user_data->setSubmitted((bool) $row['submitted']);
187  $user_data->setLastFinishedPass($row['last_finished_pass']);
188  return $user_data;
189  }
190 
192  {
193  $attempt = new \ilTestEvaluationPassData();
194  $attempt->setPass($row['pass']);
195  $attempt->setReachedPoints($row['points']);
196  $attempt->setNrOfAnsweredQuestions($row['answeredquestions']);
197  $attempt->setWorkingTime($row['workingtime']);
198  $attempt->setExamId((string) $row['exam_id']);
199  $attempt->setRequestedHintsCount($row['hint_count']);
200  $attempt->setDeductedHintPoints($row['hint_points']);
201  return $attempt;
202  }
203 
205  ilTestEvaluationUserData $user_data,
206  int $active_id
208  $visiting_time = $this->test_obj->getVisitingTimeOfParticipant($active_id);
209  $user_data->setFirstVisit($visiting_time['first_access']);
210  $user_data->setLastVisit($visiting_time['last_access']);
211  return $user_data;
212  }
213 
215  ilTestEvaluationPassData $attempt,
216  array $row
218  if ($row['questioncount'] !== 0) {
219  $attempt->setMaxPoints($row['maxpoints']);
220  $attempt->setQuestionCount($row['questioncount']);
221  return $attempt;
222  }
223 
224  list($count, $points) = array_values(
225  $this->test_obj->getQuestionCountAndPointsForPassOfParticipant($row['active_id'], $row['pass'])
226  );
227  $attempt->setMaxPoints($points);
228  $attempt->setQuestionCount($count);
229  return $attempt;
230  }
231 
232  private function addQuestionToAttempt(
233  ilTestEvaluationPassData $attempt,
234  array $row
236  if ($row['question_fi'] === null) {
237  return $attempt;
238  }
239 
240  $attempt->addAnsweredQuestion(
241  $row["question_fi"],
242  $row["qpl_maxpoints"],
243  $row["result_points"],
244  (bool) $row['answered'],
245  null,
246  $row['manual']
247  );
248  return $attempt;
249  }
250 
252  {
253  foreach ($evaluation_data->getParticipantIds() as $active_id) {
254  $user_eval_data = $evaluation_data->getParticipant($active_id);
255  $add_user_questions = $this->test_obj->isRandomTest() ?
257  $active_id,
258  $user_eval_data,
259  $this->test_obj->getQuestionCountWithoutReloading()
260  ) :
262 
263  foreach ($add_user_questions as $q) {
264  $user_eval_data->addQuestion(
265  $q['original_id'],
266  $q['question_id'],
267  $q['max_points'],
268  $q['sequence'],
269  $q['pass']
270  );
271  $evaluation_data->addQuestionTitle($q['question_id'], $q['title']);
272  }
273  }
274  return $evaluation_data;
275  }
276 
278  int $active_id,
279  ilTestEvaluationUserData $user_eval_data,
280  int $question_count
281  ): array {
282  $ret = [];
283  for ($testpass = 0; $testpass <= $user_eval_data->getLastPass(); $testpass++) {
284  $this->db->setLimit($question_count, 0);
285  $query = '
286  SELECT tst_test_rnd_qst.sequence, tst_test_rnd_qst.question_fi, qpl_questions.original_id,
287  tst_test_rnd_qst.pass, qpl_questions.points, qpl_questions.title
288  FROM tst_test_rnd_qst, qpl_questions
289  WHERE tst_test_rnd_qst.question_fi = qpl_questions.question_id
290  AND tst_test_rnd_qst.pass = %s
291  AND tst_test_rnd_qst.active_fi = %s ORDER BY tst_test_rnd_qst.sequence
292  ';
293 
294  $result = $this->db->queryF(
295  $query,
296  ['integer','integer'],
297  [$testpass, $active_id]
298  );
299 
300  if ($result->numRows()) {
301  while ($row = $this->db->fetchAssoc($result)) {
302  $tpass = array_key_exists('pass', $row) ? $row['pass'] : 0;
303 
304  if (
305  !isset($row['question_fi'], $row['points'], $row['sequence']) ||
306  !is_numeric($row['question_fi']) || !is_numeric($row['points']) || !is_numeric($row['sequence'])
307  ) {
308  continue;
309  }
310 
311  $ret[] = [
312  'original_id' => (int) $row['original_id'],
313  'question_id' => (int) $row['question_fi'],
314  'max_points' => (float) $row['points'],
315  'sequence' => (int) $row['sequence'],
316  'pass' => $tpass,
317  'title' => $row['title']
318  ];
319  }
320  }
321  }
322  return $ret;
323  }
324 
326  int $active_id
327  ): array {
328  $ret = [];
329 
330  $query = '
331  SELECT tst_test_question.sequence, tst_test_question.question_fi,
332  qpl_questions.points, qpl_questions.title, qpl_questions.original_id
333  FROM tst_test_question, tst_active, qpl_questions
334  WHERE tst_test_question.question_fi = qpl_questions.question_id
335  AND tst_active.active_id = %s
336  AND tst_active.test_fi = tst_test_question.test_fi
337  ORDER BY tst_test_question.sequence
338  ';
339 
340  $result = $this->db->queryF(
341  $query,
342  ['integer'],
343  [$active_id]
344  );
345 
346  if ($result->numRows()) {
347  $questionsbysequence = [];
348  while ($row = $this->db->fetchAssoc($result)) {
349  $questionsbysequence[$row['sequence']] = $row;
350  }
351 
352  $seqresult = $this->db->queryF(
353  'SELECT * FROM tst_sequence WHERE active_fi = %s',
354  ['integer'],
355  [$active_id]
356  );
357 
358  while ($seqrow = $this->db->fetchAssoc($seqresult)) {
359  $questionsequence = unserialize($seqrow["sequence"]);
360  foreach ($questionsequence as $sidx => $seq) {
361  if (!isset($questionsbysequence[$seq])) {
362  continue;
363  }
364  $ret[] = [
365  'original_id' => $questionsbysequence[$seq]['original_id'] ?? 0,
366  'question_id' => $questionsbysequence[$seq]['question_fi'],
367  'max_points' => $questionsbysequence[$seq]['points'],
368  'sequence' => $sidx + 1,
369  'pass' => $seqrow['pass'],
370  'title' => $questionsbysequence[$seq]["title"]
371  ];
372  }
373  }
374  }
375  return $ret;
376  }
377 
379  {
380  $mark_schema = $this->test_obj->getMarkSchema();
381 
382  foreach ($evaluation_data->getParticipantIds() as $active_id) {
383  $user_eval_data = $evaluation_data->getParticipant($active_id);
384 
385  $mark = $mark_schema->getMatchingMark(
386  $user_eval_data->getReachedPointsInPercent()
387  );
388 
389  if ($mark === null) {
390  continue;
391  }
392 
393  $user_eval_data->setMark($mark);
394  for ($i = 0; $i < $user_eval_data->getPassCount(); $i++) {
395  $pass_data = $user_eval_data->getPass($i);
396  if ($pass_data === null) {
397  continue;
398  }
399  $mark = $mark_schema->getMatchingMark(
400  $pass_data->getReachedPointsInPercent()
401  );
402  if ($mark !== null) {
403  $pass_data->setMark($mark);
404  }
405  }
406  }
407 
408  return $evaluation_data;
409  }
410 
411  public function getAllActivesPasses(): array
412  {
413  $query = '
414  SELECT active_fi, pass
415  FROM tst_active actives
416  INNER JOIN tst_pass_result passes
417  ON active_fi = active_id
418  WHERE test_fi = %s
419  ';
420 
421  $res = $this->db->queryF($query, ['integer'], [$this->test_obj->getTestId()]);
422 
423  $passes = [];
424  while ($row = $this->db->fetchAssoc($res)) {
425  if (!isset($passes[$row['active_fi']])) {
426  $passes[$row['active_fi']] = [];
427  }
428 
429  $passes[$row['active_fi']][] = $row['pass'];
430  }
431 
432  return $passes;
433  }
434 
435  public function getFirstVisitForActiveIdAndAttempt(int $active_id, int $attempt): ?string
436  {
437  $times = $this->db->fetchAssoc(
438  $this->db->queryF(
439  'SELECT MIN(started) AS first_access '
440  . 'FROM tst_times WHERE active_fi = %s AND pass = %s',
441  ['integer', 'integer'],
442  [$active_id, $attempt]
443  )
444  );
445 
446  return $times['first_access'];
447  }
448 
449  private function buildFinalizedBy(int $current_attempt, ?int $last_finished_attempt, ?string $finalized_by): StatusOfAttempt
450  {
451  if ($last_finished_attempt === null || $current_attempt > $last_finished_attempt) {
452  return StatusOfAttempt::RUNNING;
453  }
454 
455  if ($finalized_by === null) {
456  return StatusOfAttempt::FINISHED_BY_UNKNOWN;
457  }
458 
459  return StatusOfAttempt::tryFrom($finalized_by);
460  }
461 }
$res
Definition: ltiservices.php:66
retrieveQuestionsForParticipantPassesForSequencedTests(int $active_id)
setFirstVisit(?\DateTimeImmutable $time)
addPointsAndQuestionCountToAttempt(ilTestEvaluationPassData $attempt, array $row)
addQuestionsToParticipantPasses(ilTestEvaluationData $evaluation_data)
while($session_entry=$r->fetchRow(ilDBConstants::FETCHMODE_ASSOC)) return null
addAnsweredQuestion(int $question_id, float $max_points, float $reached_points, bool $is_answered, ?int $sequence=null, int $manual=0)
getFirstVisitForActiveIdAndAttempt(int $active_id, int $attempt)
retrieveQuestionsForParticipantPassesForRandomTests(int $active_id, ilTestEvaluationUserData $user_eval_data, int $question_count)
setLastVisit(?\DateTimeImmutable $time)
addVisitingTimeToUserEvalData(ilTestEvaluationUserData $user_data, int $active_id)
$q
Definition: shib_logout.php:21
addMarksToParticipants(ilTestEvaluationData $evaluation_data)
addQuestionToAttempt(ilTestEvaluationPassData $attempt, array $row)
addQuestionTitle(int $question_id, string $question_title)
__construct(protected ilDBInterface $db, protected ilObjTest $test_obj)
buildFinalizedBy(int $current_attempt, ?int $last_finished_attempt, ?string $finalized_by)