ILIAS  trunk Revision v11.0_alpha-1689-g66c127b4ae8
All Data Structures Namespaces Files Functions Variables Enumerations Enumerator Modules Pages
class.ilTestArchiver.php
Go to the documentation of this file.
1 <?php
2 
19 declare(strict_types=1);
20 
21 use ILIAS\Test\Results\Presentation\TitlesBuilder as ResultsTitleBuilder;
31 
36 {
37  public const DIR_SEP = DIRECTORY_SEPARATOR;
38 
39  public const EXPORT_DIRECTORY = 'archive_exports';
40 
41  private const PASS_MATERIALS_PATH_COMPONENT = 'materials';
42  private const QUESTION_PATH_COMPONENT_PREFIX = 'q_';
43 
44  private const TEST_BEST_SOLUTION_PATH_COMPONENT = 'best_solution';
45  private const HTML_BEST_SOLUTION_FILENAME = 'best_solution.html';
46  private const TEST_MATERIALS_PATH_COMPONENT = 'materials';
47 
48  private const TEST_RESULT_FILENAME = 'test_result.html';
49 
50  private const LOG_DTSGROUP_FORMAT = 'D M j G:i:s T Y';
51  private const LOG_ADDITION_STRING = ' Adding ';
52 
53  private const TEST_LOG_FILENAME = 'test_log.xlsx';
54  private const DATA_INDEX_FILENAME = 'data_index.csv';
55  private const ARCHIVE_LOG = 'archive.log';
56 
57  private string $external_directory_path;
58  private string $client_id = CLIENT_ID;
60 
62 
64 
65  public function __construct(
66  private readonly ilLanguage $lng,
67  private readonly ilDBInterface $db,
68  private readonly ilObjUser $user,
69  private readonly UIFactory $ui_factory,
70  private readonly UIRenderer $ui_renderer,
71  private readonly IRSS $irss,
72  private readonly ServerRequestInterface $request,
73  private readonly ilObjectDataCache $obj_cache,
74  private readonly ilTestParticipantAccessFilterFactory $participant_access_filter_factory,
75  private readonly TestLogViewer $log_viewer,
76  private readonly int $test_obj_id,
77  private ?int $test_ref_id = null
78  ) {
80  global $DIC;
81  $ilias = $DIC['ilias'];
82 
83  $this->html_generator = new ilTestHTMLGenerator();
84  $this->external_directory_path = $ilias->ini_ilias->readVariable('clients', 'datadir');
85  $this->archive_data_index = $this->readArchiveDataIndex();
86  }
87 
89  {
91  }
92 
93  public function setParticipantData(ilTestParticipantData $participant_data): void
94  {
95  $this->participant_data = $participant_data;
96  }
97 
99  int $active_fi,
100  int $pass,
101  int $question_fi,
102  string $original_filename,
103  string $file_path
104  ): void {
106  $this->ensurePassDataDirectoryIsAvailable($active_fi, $pass);
107 
108  $pass_question_directory = $this->getPassDataDirectory($active_fi, $pass)
109  . DIRECTORY_SEPARATOR . self::QUESTION_PATH_COMPONENT_PREFIX . $question_fi;
110  if (!is_dir($pass_question_directory)) {
111  mkdir($pass_question_directory, 0777, true);
112  }
113 
114  copy($file_path, $pass_question_directory . DIRECTORY_SEPARATOR . $original_filename);
115 
116  $this->logArchivingProcess(
117  date(self::LOG_DTSGROUP_FORMAT) . self::LOG_ADDITION_STRING
118  . $pass_question_directory . DIRECTORY_SEPARATOR . $original_filename
119  );
120  }
121 
122  public function handInParticipantMisc(
123  int $active_fi,
124  int $pass,
125  string $original_filename,
126  string $file_path
127  ): void {
129  $this->ensurePassDataDirectoryIsAvailable($active_fi, $pass);
130  $new_path = $this->getPassDataDirectory($active_fi, $pass) . DIRECTORY_SEPARATOR . $original_filename;
131  copy($file_path, $new_path);
132  $this->logArchivingProcess(date(self::LOG_DTSGROUP_FORMAT) . self::LOG_ADDITION_STRING . $new_path);
133  }
134 
135  public function handInTestBestSolution(string $best_solution): void
136  {
138 
139  $best_solution_path = $this->getTestArchive() . DIRECTORY_SEPARATOR . self::TEST_BEST_SOLUTION_PATH_COMPONENT;
140  if (!is_dir($best_solution_path)) {
141  mkdir($best_solution_path, 0777, true);
142  }
143 
144  $this->html_generator->generateHTML(
145  $best_solution,
146  $best_solution_path . DIRECTORY_SEPARATOR . self::HTML_BEST_SOLUTION_FILENAME
147  );
148 
149  $this->logArchivingProcess(
150  date(self::LOG_DTSGROUP_FORMAT) . self::LOG_ADDITION_STRING
151  . $best_solution_path . DIRECTORY_SEPARATOR . self::HTML_BEST_SOLUTION_FILENAME
152  );
153 
154  $this->logArchivingProcess(
155  date(self::LOG_DTSGROUP_FORMAT) . self::LOG_ADDITION_STRING . $best_solution_path
156  );
157  }
158 
160  int $active_fi,
161  int $pass,
162  ilObjTest $tst_obj
163  ): void {
164  $questions = $tst_obj->getQuestionsOfPass($active_fi, $pass);
165  foreach ($questions as $question) {
166  $question = $tst_obj->getQuestionDataset($question['question_fi']);
167  if ($question->type_tag === 'assFileUpload') {
169  $this->ensurePassDataDirectoryIsAvailable($active_fi, $pass);
170  $this->ensurePassMaterialsDirectoryIsAvailable($active_fi, $pass);
171  $pass_material_directory = $this->getPassMaterialsDirectory($active_fi, $pass);
172  $archive_folder = $pass_material_directory . DIRECTORY_SEPARATOR . $question->question_id . DIRECTORY_SEPARATOR;
173  if (!file_exists($archive_folder)) {
174  mkdir($archive_folder, 0777, true);
175  }
176  $resource_id = $tst_obj->getTextAnswer($active_fi, $question->question_id, $pass);
177  if ($resource_id === '') {
178  continue;
179  }
180  $irss_unique_id = $this->irss->manage()->find($resource_id);
181  if ($irss_unique_id != null) {
182  $resource = $this->irss->manage()->getResource($irss_unique_id);
183  $information = $resource->getCurrentRevision()->getInformation();
184  $stream = $this->irss->consume()->stream($irss_unique_id);
185  // this feels unnecessary..
186  $file_stream = fopen($stream->getStream()->getMetadata('uri'), 'r');
187  $file_content = stream_get_contents($file_stream);
188  fclose($file_stream);
189  $target_destination = $archive_folder . $information->getTitle();
190  file_put_contents($target_destination, $file_content);
191  }
192  }
193  }
194  }
195 
197  int $question_fi,
198  string $orginial_filename,
199  string $file_path
200  ): void {
202 
203  $best_solution_path = $this->getTestArchive() . DIRECTORY_SEPARATOR . self::TEST_BEST_SOLUTION_PATH_COMPONENT;
204  if (!is_dir($best_solution_path)) {
205  mkdir($best_solution_path, 0777, true);
206  }
207 
208  $materials_path = $best_solution_path . DIRECTORY_SEPARATOR . self::TEST_MATERIALS_PATH_COMPONENT;
209  if (!is_dir($materials_path)) {
210  mkdir($materials_path, 0777, true);
211  }
212 
213  $question_materials_path = $materials_path . DIRECTORY_SEPARATOR . self::QUESTION_PATH_COMPONENT_PREFIX . $question_fi;
214  if (!is_dir($question_materials_path)) {
215  mkdir($question_materials_path, 0777, true);
216  }
217 
218  copy($file_path, $question_materials_path . DIRECTORY_SEPARATOR . $orginial_filename);
219 
220  $this->logArchivingProcess(
221  date(self::LOG_DTSGROUP_FORMAT) . self::LOG_ADDITION_STRING
222  . $question_materials_path . DIRECTORY_SEPARATOR . $orginial_filename
223  );
224  }
225 
226  public function handInTestResult(int $active_fi, int $pass, string $pdf_path): void
227  {
229  $this->ensurePassDataDirectoryIsAvailable($active_fi, $pass);
230  $new_path = $this->getPassDataDirectory($active_fi, $pass) . DIRECTORY_SEPARATOR . self::TEST_RESULT_FILENAME;
231  copy($pdf_path, $new_path);
232  $this->logArchivingProcess(date(self::LOG_DTSGROUP_FORMAT) . self::LOG_ADDITION_STRING . $new_path);
233  }
234 
235  protected function hasTestArchive(): bool
236  {
237  return is_dir($this->getTestArchive());
238  }
239 
240  protected function createArchiveForTest(): void
241  {
243  }
244 
245  protected function getTestArchive(): string
246  {
247  $test_archive_directory = $this->external_directory_path . DIRECTORY_SEPARATOR . $this->client_id . DIRECTORY_SEPARATOR . 'tst_data'
248  . DIRECTORY_SEPARATOR . 'archive' . DIRECTORY_SEPARATOR . 'tst_' . $this->test_obj_id;
249  return $test_archive_directory;
250  }
251 
252  protected function ensureTestArchiveIsAvailable(): void
253  {
254  if (!$this->hasTestArchive()) {
255  $this->createArchiveForTest();
256  }
257  return;
258  }
259 
260  public function updateTestArchive(): void
261  {
262  $this->log_viewer->getLogExportForRefjId(
263  $this->test_ref_id
264  )->writeToFile(
265  $this->getTestArchive() . DIRECTORY_SEPARATOR . self::TEST_LOG_FILENAME
266  );
267 
268  // Generate test pass overview
269  $test = new ilObjTest($this->test_obj_id, false);
270  if ($this->test_ref_id !== null) {
271  $test->setRefId($this->test_ref_id);
272  }
273 
274  $array_of_actives = [];
275  $participants = $test->getParticipants();
276 
277  foreach (array_keys($participants) as $key) {
278  $array_of_actives[] = $key;
279  }
280 
281  $filename = realpath($this->getTestArchive()) . DIRECTORY_SEPARATOR . 'participant_attempt_overview.html';
282  $this->html_generator->generateHTML(
284  $test,
285  $array_of_actives
286  ),
287  $filename
288  );
289  }
290 
291  public function ensureZipExportDirectoryExists(): void
292  {
293  if (!$this->hasZipExportDirectory()) {
294  $this->createZipExportDirectory();
295  }
296  }
297 
298  public function hasZipExportDirectory(): bool
299  {
300  return is_dir($this->getZipExportDirectory());
301  }
302 
303  protected function createZipExportDirectory(): void
304  {
305  mkdir($this->getZipExportDirectory(), 0777, true);
306  }
307 
308  public function getZipExportDirectory(): string
309  {
310  return $this->external_directory_path . DIRECTORY_SEPARATOR . $this->client_id . DIRECTORY_SEPARATOR . 'tst_data'
311  . DIRECTORY_SEPARATOR . self::EXPORT_DIRECTORY . DIRECTORY_SEPARATOR . 'tst_' . $this->test_obj_id;
312  }
313 
314  public function compressTestArchive(): void
315  {
316  $this->updateTestArchive();
318 
319  $zip_output_path = $this->getZipExportDirectory();
320  $zip_output_filename = 'test_archive_obj_' . $this->test_obj_id . '_' . time() . '.zip';
321 
322  ilFileUtils::zip($this->getTestArchive(), $zip_output_path . DIRECTORY_SEPARATOR . $zip_output_filename, true);
323  return;
324  }
325 
326  protected function hasPassDataDirectory(int $active_fi, int $pass): bool
327  {
328  return is_dir($this->getPassDataDirectory($active_fi, $pass));
329  }
330 
331  protected function createPassDataDirectory(int $active_fi, int $pass): void
332  {
333  mkdir($this->getPassDataDirectory($active_fi, $pass), 0777, true);
334  return;
335  }
336 
337  private function buildPassDataDirectory($active_fi, $pass): ?string
338  {
339  foreach ($this->archive_data_index as $data_index_entry) {
340  if ($data_index_entry != null && $data_index_entry['identifier'] == $active_fi . '|' . $pass) {
341  array_shift($data_index_entry);
342  return $this->getTestArchive() . DIRECTORY_SEPARATOR . implode(DIRECTORY_SEPARATOR, $data_index_entry);
343  }
344  }
345 
346  return null;
347  }
348 
349  protected function getPassDataDirectory(int $active_fi, int $pass): ?string
350  {
351  $pass_data_dir = $this->buildPassDataDirectory($active_fi, $pass);
352 
353  if ($pass_data_dir !== null) {
354  return $pass_data_dir;
355  }
356 
357  $test_obj = new ilObjTest($this->test_obj_id, false);
358  if ($test_obj->getAnonymity()) {
359  $firstname = $this->lng->txt('anonymous');
360  $lastname = '';
361  $matriculation = '';
362  } elseif ($this->getParticipantData()) {
363  $usr_data = $this->getParticipantData()->getUserDataByActiveId($active_fi);
364  $firstname = $usr_data['firstname'] ?? $this->lng->txt('deleted_user');
365  $lastname = $usr_data['lastname'] ?? '';
366  $matriculation = $usr_data['matriculation'] ?? '';
367  } else {
368  $firstname = $this->user->getFirstname();
369  $lastname = $this->user->getLastname();
370  $matriculation = $this->user->getMatriculation();
371  }
372 
374  date(DATE_ISO8601),
375  $active_fi,
376  $pass,
377  $firstname,
378  $lastname,
379  $matriculation
380  );
381 
382  return $this->buildPassDataDirectory($active_fi, $pass);
383  }
384 
385  protected function ensurePassDataDirectoryIsAvailable(int $active_fi, int $pass): void
386  {
387  if (!$this->hasPassDataDirectory($active_fi, $pass)) {
388  $this->createPassDataDirectory($active_fi, $pass);
389  }
390  return;
391  }
392 
393  protected function hasPassMaterialsDirectory(int $active_fi, int $pass): bool
394  {
395  if (is_dir($this->getPassMaterialsDirectory($active_fi, $pass))) {
396  return true;
397  }
398  return false;
399  }
400 
401  protected function createPassMaterialsDirectory(int $active_fi, int $pass): string
402  {
408  $user = $this->user;
409 
410  if ($this->getParticipantData()) {
411  $usrData = $this->getParticipantData()->getUserDataByActiveId($active_fi);
412  $user = new ilObjUser();
413  $user->setFirstname($usrData['firstname'] ?? $this->lng->txt('deleted_user'));
414  $user->setLastname($usrData['lastname'] ?? '');
415  $user->setMatriculation($usrData['matriculation'] ?? '');
416  }
417 
419  date('c'),
420  $active_fi,
421  $pass,
422  $user->getFirstname(),
423  $user->getLastname(),
424  $user->getMatriculation()
425  );
426  $material_directory = $this->getPassMaterialsDirectory($active_fi, $pass);
427  mkdir($material_directory, 0777, true);
428  return $material_directory;
429  }
430 
431  protected function getPassMaterialsDirectory(int $active_fi, int $pass): string
432  {
433  $pass_data_directory = $this->getPassDataDirectory($active_fi, $pass);
434  return $pass_data_directory . DIRECTORY_SEPARATOR . self::PASS_MATERIALS_PATH_COMPONENT;
435  }
436 
437  protected function ensurePassMaterialsDirectoryIsAvailable(int $active_fi, int $pass): void
438  {
439  if (!$this->hasPassMaterialsDirectory($active_fi, $pass)) {
440  $this->createPassMaterialsDirectory($active_fi, $pass);
441  }
442  }
443 
444  protected function readArchiveDataIndex(): array
445  {
450  $data_index_file = $this->getTestArchive() . DIRECTORY_SEPARATOR . self::DATA_INDEX_FILENAME;
451 
452  $contents = [];
453 
455  if (@file_exists($data_index_file)) {
456  $lines = explode("\n", file_get_contents($data_index_file));
457  foreach ($lines as $line) {
458  if (strlen($line) === 0) {
459  continue;
460  }
461  $line_items = explode('|', $line);
462  $line_data = [];
463  $line_data['identifier'] = $line_items[0] . '|' . $line_items[1];
464  $line_data['yyyy'] = $line_items[2];
465  $line_data['mm'] = $line_items[3];
466  $line_data['dd'] = $line_items[4];
467  $line_data['directory'] = $line_items[5];
468  $contents[] = $line_data;
469  }
470  }
471  return $contents;
472  }
473 
474  protected function appendToArchiveDataIndex(
475  string $date,
476  int $active_fi,
477  int $pass,
478  string $user_firstname,
479  string $user_lastname,
480  string $matriculation
481  ): void {
482  $line = $this->determinePassDataPath($date, $active_fi, $pass, $user_firstname, $user_lastname, $matriculation);
483 
484  $this->archive_data_index[] = $line;
485  $output_contents = '';
486 
487  foreach ($this->archive_data_index as $line_data) {
488  if ($line_data['identifier'] == "|") {
489  continue;
490  }
491  $output_contents .= implode('|', $line_data) . "\n";
492  }
493 
494  file_put_contents($this->getTestArchive() . DIRECTORY_SEPARATOR . self::DATA_INDEX_FILENAME, $output_contents);
495  $this->readArchiveDataIndex();
496  return;
497  }
498 
499  private function determinePassDataPath(
500  string $date,
501  int $active_fi,
502  int $pass,
503  string $user_firstname,
504  string $user_lastname,
505  string $matriculation
506  ): array {
507  $parsed_date = date_create_from_format('Y-m-d\TH:i:sP', $date);
508  if (!$parsed_date) {
509  throw new Exception('Invalid date format. Expected ISO 8601 format.');
510  }
511 
512  $line = [
513  'identifier' => $active_fi . '|' . $pass,
514  'yyyy' => date_format($parsed_date, 'Y'),
515  'mm' => date_format($parsed_date, 'm'),
516  'dd' => date_format($parsed_date, 'd'),
517  'directory' => $active_fi . '_' . $pass . '_' . $user_firstname . '_' . $user_lastname . '_' . $matriculation
518  ];
519  return $line;
520  }
521 
522  private function createUserResultsForArchive(
523  \ilObjTest $test_obj,
524  array $active_ids,
525  ): string {
526  $template = new ilTemplate('tpl.il_as_tst_participants_result_output.html', true, true, 'components/ILIAS/Test');
527 
528  $participant_data = new ilTestParticipantData($this->db, $this->lng);
529  $participant_data->setParticipantAccessFilter(
530  $this->participant_access_filter_factory->getAccessResultsUserFilter($test_obj->getRefId())
531  );
532  $participant_data->setActiveIdsFilter($active_ids);
533  $participant_data->load($test_obj->getTestId());
534 
535  $test_session_factory = new ilTestSessionFactory($test_obj, $this->db, $this->user);
536 
537  $count = 0;
538  foreach ($active_ids as $active_id) {
539  if (!in_array($active_id, $participant_data->getActiveIds())) {
540  continue;
541  }
542 
543  $count++;
544  $results = '';
545  if ($active_id > 0) {
547  $test_obj,
548  $test_session_factory->getSession($active_id),
549  $participant_data->getUserDataByActiveId($active_id),
550  (int) $active_id,
551  ilObjTest::_getResultPass($active_id)
552  );
553  }
554  if ($count < count($active_ids)) {
555  $template->touchBlock('break');
556  }
557  $template->setCurrentBlock('user_result');
558  $template->setVariable('USER_RESULT', $results);
559  $template->parseCurrentBlock();
560  }
561 
562  return $template->get();
563  }
564 
565  public function getResultsOfUserOutput(
566  \ilObjTest $test_obj,
567  ilTestSession $test_session,
568  array $participant_data,
569  int $active_id,
570  int $attempt
571  ): string {
572  $template = new ilTemplate('tpl.il_as_tst_results_participant.html', true, true, 'components/ILIAS/Test');
573 
574  $uname = "{$participant_data['firstname']} {$participant_data['lastname']}";
575  if ($test_obj->getAnonymity()) {
576  $uname = $this->lng->txt('anonymous');
577  }
578 
579  $test_result_title_builder = new ResultsTitleBuilder($this->lng, $this->obj_cache);
580 
581  $result_array = $test_obj->getTestResult(
582  $active_id,
583  $attempt,
584  false,
585  true
586  );
587 
588  $table = $this->ui_factory->table()->data(
589  $this->getDataRetrievalForAttemptOverviewTable($result_array),
590  $test_result_title_builder->getPassDetailsHeaderLabel($attempt + 1),
592  )->withRequest($this->request);
593  $template->setVariable(
594  'PASS_DETAILS',
595  $this->ui_renderer->render($table)
596  );
597 
598  if ($test_obj->isShowExamIdInTestResultsEnabled()) {
599  $template->setCurrentBlock('exam_id_footer');
600  $template->setVariable('EXAM_ID_VAL', ilObjTest::lookupExamId(
601  $test_session->getActiveId(),
602  $attempt
603  ));
604  $template->setVariable('EXAM_ID_TXT', $this->lng->txt('exam_id'));
605  $template->parseCurrentBlock();
606  }
607 
608  $template->setCurrentBlock('participant_block_id');
609  $template->setVariable('PARTICIPANT_BLOCK_ID', "participant_active_{$active_id}");
610  $template->parseCurrentBlock();
611 
612  $template->setVariable('TEXT_HEADING', sprintf($this->lng->txt('tst_result_user_name'), $uname));
613 
614  if ($participant_data['matriculation'] !== '') {
615  $template->setVariable('USER_DATA', "{$this->lng->txt('matriculation')}: {$participant_data['matriculation']}");
616  }
617 
618  $results = $test_obj->getResultsForActiveId($active_id);
619  $status = $this->lng->txt($results['passed'] ? 'passed_official' : 'failed_official');
620  $template->setVariable(
621  'GRADING_MESSAGE',
622  "{$this->lng->txt('passed_status')}: {$status}<br>"
623  . "{$this->lng->txt('tst_mark')}: {$results['mark_official']}"
624  );
625 
626  $template->setVariable('PASS_FINISH_DATE_LABEL', $this->lng->txt('tst_pass_finished_on'));
627  $template->setVariable(
628  'PASS_FINISH_DATE_VALUE',
629  (new \DateTimeImmutable('@' . ilObjTest::lookupLastTestPassAccess($active_id, $attempt)))
630  ->setTimezone(new DateTimeZone($this->user->getTimeZone()))
631  ->format($this->user->getDateTimeFormat()->toString())
632  );
633 
634  return $template->get();
635  }
636 
638  bool $show_requested_hints_info
639  ): array {
640  $cf = $this->ui_factory->table()->column();
641  $columns = [
642  'order' => $cf->number($this->lng->txt('order')),
643  'question_id' => $cf->number($this->lng->txt('question_id')),
644  'title' => $cf->text($this->lng->txt('tst_question_title')),
645  'reachable_points' => $cf->number($this->lng->txt('tst_maximum_points')),
646  'reached_points' => $cf->number($this->lng->txt('tst_reached_points'))
647  ];
648  if ($show_requested_hints_info) {
649  $columns['hints'] = $cf->number($this->lng->txt('tst_question_hints_requested_hint_count_header'));
650  }
651  $columns['solved'] = $cf->text($this->lng->txt('tst_percent_solved'));
652  return $columns;
653  }
654 
655  private function getDataRetrievalForAttemptOverviewTable(array $result_data): DataRetrieval
656  {
657  return new class ($result_data) implements DataRetrieval {
658  public function __construct(
659  private readonly array $result_data
660  ) {
661  }
662 
663  public function getRows(
664  DataRowBuilder $row_builder,
665  array $visible_column_ids,
666  Range $range,
667  Order $order,
668  ?array $filter_data,
669  ?array $additional_parameters
670  ): \Generator {
671  $i = 1;
672  foreach ($this->result_data as $result) {
673  if (!isset($result['qid'])) {
674  continue;
675  }
676  yield $row_builder->buildDataRow(
677  (string) $result['qid'],
678  [
679  'order' => $i++,
680  'question_id' => $result['qid'],
681  'title' => $result['title'],
682  'reachable_points' => $result['max'],
683  'reached_points' => $result['reached'],
684  'hints' => $result['requested_hints'] ?? 0,
685  'solved' => $result['percent']
686  ]
687  );
688  }
689  }
690 
691  public function getTotalRowCount(
692  ?array $filter_data,
693  ?array $additional_parameters
694  ): ?int {
695  return count($this->result_data);
696  }
697  };
698  }
699 
700  private function logArchivingProcess(string $message): void
701  {
702  $archive = $this->getTestArchive() . DIRECTORY_SEPARATOR . self::ARCHIVE_LOG;
703  if (file_exists($archive)) {
704  $content = file_get_contents($archive) . "\n" . $message;
705  } else {
706  $content = $message;
707  }
708 
709  file_put_contents($archive, $content);
710  }
711 }
ilTestParticipantData $participant_data
createUserResultsForArchive(\ilObjTest $test_obj, array $active_ids,)
handInParticipantQuestionMaterial(int $active_fi, int $pass, int $question_fi, string $original_filename, string $file_path)
getPassDataDirectory(int $active_fi, int $pass)
ensurePassDataDirectoryIsAvailable(int $active_fi, int $pass)
getTestId()
Gets the database id of the additional test data.
createPassMaterialsDirectory(int $active_fi, int $pass)
isShowExamIdInTestResultsEnabled()
getQuestionsOfPass(int $active_id, int $pass)
hasPassMaterialsDirectory(int $active_fi, int $pass)
static makeDirParents(string $a_dir)
Create a new directory and all parent directories.
getTestResult(int $active_id, ?int $pass=null, bool $ordered_sequence=false, bool $consider_hidden_questions=true, bool $consider_optional_questions=true)
Calculates the results of a test for a given user and returns an array with all test results...
Both the subject and the direction need to be specified when expressing an order. ...
Definition: Order.php:28
buildDataRow(string $id, array $record)
while($session_entry=$r->fetchRow(ilDBConstants::FETCHMODE_ASSOC)) return null
setParticipantAccessFilter(Closure $participantAccessFilter)
const QUESTION_PATH_COMPONENT_PREFIX
This file is part of ILIAS, a powerful learning management system published by ILIAS open source e-Le...
setActiveIdsFilter(array $active_ids_filter)
createPassDataDirectory(int $active_fi, int $pass)
static _getResultPass($active_id)
Retrieves the pass number that should be counted for a given user.
buildPassDataDirectory($active_fi, $pass)
getPassMaterialsDirectory(int $active_fi, int $pass)
handInTestBestSolution(string $best_solution)
const CLIENT_ID
Definition: constants.php:41
global $DIC
Definition: shib_login.php:22
getResultsForActiveId(int $active_id)
handInParticipantUploadedResults(int $active_fi, int $pass, ilObjTest $tst_obj)
isOfferingQuestionHintsEnabled()
$results
determinePassDataPath(string $date, int $active_fi, int $pass, string $user_firstname, string $user_lastname, string $matriculation)
getResultsOfUserOutput(\ilObjTest $test_obj, ilTestSession $test_session, array $participant_data, int $active_id, int $attempt)
$filename
Definition: buildRTE.php:78
Class that handles PDF generation for test and assessment.
getQuestionDataset($question_id)
Returns the dataset for a given question id.
getTextAnswer($active_id, $question_id, $pass=null)
Returns the text answer of a given user for a given question.
handInParticipantMisc(int $active_fi, int $pass, string $original_filename, string $file_path)
__construct(Container $dic, ilPlugin $plugin)
global $lng
Definition: privfeed.php:31
static zip(string $a_dir, string $a_file, bool $compress_content=false)
appendToArchiveDataIndex(string $date, int $active_fi, int $pass, string $user_firstname, string $user_lastname, string $matriculation)
const TEST_BEST_SOLUTION_PATH_COMPONENT
$message
Definition: xapiexit.php:31
handInTestResult(int $active_fi, int $pass, string $pdf_path)
setParticipantData(ilTestParticipantData $participant_data)
logArchivingProcess(string $message)
handInBestSolutionQuestionMaterial(int $question_fi, string $orginial_filename, string $file_path)
A simple class to express a naive range of whole positive numbers.
Definition: Range.php:28
ensurePassMaterialsDirectoryIsAvailable(int $active_fi, int $pass)
ilTestHTMLGenerator $html_generator
getDataRetrievalForAttemptOverviewTable(array $result_data)
static lookupExamId($active_id, $pass)
getColumnsForAttemptOverviewTable(bool $show_requested_hints_info)
hasPassDataDirectory(int $active_fi, int $pass)