ILIAS  trunk Revision v11.0_alpha-1689-g66c127b4ae8
All Data Structures Namespaces Files Functions Variables Enumerations Enumerator Modules Pages
class.ilExSubmission.php
Go to the documentation of this file.
1 <?php
2 
20 
29 {
30  public const TYPE_FILE = "File";
31  public const TYPE_OBJECT = "Object"; // Blogs in WSP/Portfolio
32  public const TYPE_TEXT = "Text";
33  public const TYPE_REPO_OBJECT = "RepoObject"; // Wikis
34  protected \ILIAS\Exercise\Submission\SubmissionManager $sub_manager;
35  protected \ILIAS\Exercise\InternalDomainService $domain;
36 
37  protected ilObjUser $user;
38  protected ilDBInterface $db;
39  protected ilLanguage $lng;
40  protected ilCtrl $ctrl;
42  protected int $user_id;
45  protected bool $is_tutor;
46  protected bool $public_submissions;
50  private \ilGlobalTemplateInterface $main_tpl;
51 
52  public function __construct(
53  ilExAssignment $a_ass,
54  int $a_user_id,
55  ?ilExAssignmentTeam $a_team = null, // did not find any place that sets this....
56  bool $a_is_tutor = false,
57  bool $a_public_submissions = false
58  ) {
59  global $DIC;
60  $this->main_tpl = $DIC->ui()->mainTemplate();
61 
62  $this->user = $DIC->user();
63  $this->db = $DIC->database();
64  $this->lng = $DIC->language();
65  $this->ctrl = $DIC->ctrl();
66 
67  $this->assignment = $a_ass;
68  $this->ass_type = $this->assignment->getAssignmentType();
69  $this->ass_types = ilExAssignmentTypes::getInstance();
70 
71  $this->user_id = $a_user_id;
72  $this->is_tutor = $a_is_tutor;
73  $this->public_submissions = $a_public_submissions;
74 
75  $this->state = ilExcAssMemberState::getInstanceByIds($a_ass->getId(), $a_user_id);
76 
77  if ($a_ass->hasTeam()) { // ass type uses teams...
78  if (!$a_team) {
79  // this might be a team with no id (since the create on demand parameter is not set)
80  $this->team = ilExAssignmentTeam::getInstanceByUserId($this->assignment->getId(), $this->user_id);
81  } else {
82  $this->team = $a_team;
83  }
84  }
85 
86  if ($this->assignment->getPeerReview()) {
87  $this->peer_review = new ilExPeerReview($this->assignment);
88  }
89  $this->domain = $DIC->exercise()->internal()->domain();
90  $this->sub_manager = $DIC->exercise()->internal()->domain()->submission(
91  $a_ass->getId()
92  );
93  }
94 
95  public function getSubmissionType(): string
96  {
97  return $this->assignment->getAssignmentType()->getSubmissionType();
98  }
99 
100  public function getAssignment(): ilExAssignment
101  {
102  return $this->assignment;
103  }
104 
105  public function getTeam(): ?ilExAssignmentTeam
106  {
107  return $this->team;
108  }
109 
110  public function getPeerReview(): ?ilExPeerReview
111  {
112  return $this->peer_review;
113  }
114 
115  public function validatePeerReviews(): array
116  {
117  $res = array();
118  foreach ($this->getUserIds() as $user_id) {
119  $valid = true;
120 
121  // no peer review == valid
122  if ($this->peer_review) {
123  $valid = $this->peer_review->isFeedbackValidForPassed($user_id);
124  }
125 
126  $res[$user_id] = $valid;
127  }
128  return $res;
129  }
130 
131  public function getUserId(): int
132  {
133  return $this->user_id;
134  }
135 
136  public function getUserIds(): array
137  {
138  if ($this->team &&
139  !$this->hasNoTeamYet()) {
140  return $this->team->getMembers();
141  }
142 
143  // if has no team currently there still might be uploads attached
144  return array($this->user_id);
145  }
146 
150  public function getFeedbackId(): string
151  {
152  if ($this->team) {
153  return "t" . $this->team->getId();
154  } else {
155  return (string) $this->getUserId();
156  }
157  }
158 
159  public function hasSubmitted(): bool
160  {
161  return (bool) $this->sub_manager->getSubmissionsOfUser(
162  $this->getUserId(),
163  null,
164  true
165  )->current();
166  }
167 
168  public function hasSubmittedPrintVersion(): bool
169  {
170  return ($this->getSubmittedEntry(true)?->getRid() != "");
171  }
172 
173  public function getSubmittedEntry(bool $print = false): ?Submission
174  {
175  return $this->sub_manager->getSubmissionsOfUser(
176  $this->getUserId(),
177  null,
178  false,
179  null,
180  $print
181  )->current();
182  }
183 
184  public function getSelectedObject(): ?Submission
185  {
186  return $this->sub_manager->getSubmissionsOfUser($this->getUserId())->current();
187  }
188 
189  public function canSubmit(): bool
190  {
191  return ($this->isOwner() &&
192  $this->state->isSubmissionAllowed());
193  }
194 
195  public function canView(): bool
196  {
197  $ilUser = $this->user;
198 
199  if ($this->canSubmit() ||
200  $this->isTutor() ||
201  $this->isInTeam() ||
202  $this->public_submissions) {
203  return true;
204  }
205 
206  // #16115
207  if ($this->peer_review) {
208  // peer review givers may view peer submissions
209  foreach ($this->peer_review->getPeerReviewsByPeerId($this->getUserId()) as $giver) {
210  if ($giver["giver_id"] == $ilUser->getId()) {
211  return true;
212  }
213  }
214  }
215 
216  return false;
217  }
218 
219  public function isTutor(): bool
220  {
221  return $this->is_tutor;
222  }
223 
224  public function hasNoTeamYet(): bool
225  {
226  if ($this->assignment->hasTeam() &&
227  !$this->team->getId()) {
228  return true;
229  }
230  return false;
231  }
232 
233  public function isInTeam(?int $a_user_id = null): bool
234  {
235  $ilUser = $this->user;
236 
237  if (!$a_user_id) {
238  $a_user_id = $ilUser->getId();
239  }
240  return in_array($a_user_id, $this->getUserIds());
241  }
242 
243  public function isOwner(): bool
244  {
245  $ilUser = $this->user;
246 
247  return ($ilUser->getId() == $this->getUserId());
248  }
249 
250  public function hasPeerReviewAccess(): bool
251  {
252  return ($this->peer_review &&
253  $this->peer_review->hasPeerReviewAccess($this->user_id));
254  }
255 
256  public function canAddFile(): bool
257  {
258  if (!$this->canSubmit()) {
259  return false;
260  }
261 
262  $max = $this->getAssignment()->getMaxFile();
263  $cnt_sub = $this->sub_manager->countSubmissionsOfUser(
264  $this->getUserId()
265  );
266  if ($max &&
267  $max <= $cnt_sub) {
268  return false;
269  }
270 
271  return true;
272  }
273 
274 
275  //
276  // FILES
277  //
278 
279  public function isLate(): bool
280  {
281  $dl = $this->state->getOfficialDeadline();
282  //$dl = $this->assignment->getPersonalDeadline($this->getUserId());
283  return ($dl && $dl < time());
284  }
285 
286  protected function getStorageId(): int
287  {
288  if ($this->ass_type->isSubmissionAssignedToTeam()) {
289  $storage_id = $this->getTeam()->getId();
290  } else {
291  $storage_id = $this->getUserId();
292  }
293  return $storage_id;
294  }
295 
300  public function lookupNewFiles(
301  ?int $a_tutor = null
302  ): array {
303  $ilDB = $this->db;
304  $ilUser = $this->user;
305 
306  $tutor = ($a_tutor)
307  ?: $ilUser->getId();
308 
309  $where = " AND " . $this->getTableUserWhere();
310 
311  $q = "SELECT exc_returned.returned_id AS id " .
312  "FROM exc_usr_tutor, exc_returned " .
313  "WHERE exc_returned.ass_id = exc_usr_tutor.ass_id " .
314  " AND exc_returned.user_id = exc_usr_tutor.usr_id " .
315  " AND exc_returned.ass_id = " . $ilDB->quote($this->getAssignment()->getId(), "integer") .
316  $where .
317  " AND exc_usr_tutor.tutor_id = " . $ilDB->quote($tutor, "integer") .
318  " AND exc_usr_tutor.download_time < exc_returned.ts ";
319 
320  $new_up_set = $ilDB->query($q);
321 
322  $new_up = array();
323  while ($new_up_rec = $ilDB->fetchAssoc($new_up_set)) {
324  $new_up[] = $new_up_rec["id"];
325  }
326 
327  return $new_up;
328  }
329 
333  public static function lookupExerciseIdForReturnedId(
334  int $a_returned_id
335  ): int {
336  global $DIC;
337 
338  $ilDB = $DIC->database();
339 
340  $set = $ilDB->query("SELECT obj_id" .
341  " FROM exc_returned" .
342  " WHERE returned_id = " . $ilDB->quote($a_returned_id, "integer"));
343  $row = $ilDB->fetchAssoc($set);
344  return (int) $row["obj_id"];
345  }
346 
351  public static function findUserFiles(
352  int $a_user_id,
353  string $a_filetitle
354  ): array {
355  global $DIC;
356 
357  $ilDB = $DIC->database();
358 
359  $set = $ilDB->query("SELECT obj_id, ass_id" .
360  " FROM exc_returned" .
361  " WHERE user_id = " . $ilDB->quote($a_user_id, "integer") .
362  " AND filetitle = " . $ilDB->quote($a_filetitle, "text"));
363  $res = array();
364  while ($row = $ilDB->fetchAssoc($set)) {
365  $res[$row["ass_id"]] = $row;
366  }
367  return $res;
368  }
369 
370  public function deleteAllFiles(): void
371  {
372  $this->sub_manager->deleteAllSubmissionsOfUser($this->getUserId());
373  }
374 
379  /* --> use manager deleteSubmissions()
380  public function deleteSelectedFiles(
381  array $file_id_array
382  ): void {
383  $ilDB = $this->db;
384 
385 
386  $where = " AND " . $this->getTableUserWhere(true);
387 
388 
389  if ($file_id_array === []) {
390  return;
391  }
392 
393  if ($file_id_array !== []) {
394  $result = $ilDB->query("SELECT * FROM exc_returned" .
395  " WHERE " . $ilDB->in("returned_id", $file_id_array, false, "integer") .
396  $where);
397 
398  if ($ilDB->numRows($result)) {
399  $result_array = array();
400  while ($row = $ilDB->fetchAssoc($result)) {
401  $row["timestamp"] = $row["ts"];
402  $result_array[] = $row;
403  }
404 
405  // delete the entries in the database
406  $ilDB->manipulate("DELETE FROM exc_returned" .
407  " WHERE " . $ilDB->in("returned_id", $file_id_array, false, "integer") .
408  $where);
409 
410  // delete the files
411  $path = $this->initStorage()->getAbsoluteSubmissionPath();
412  foreach ($result_array as $value) {
413  if ($value["filename"]) {
414  if ($this->team) {
415  $this->team->writeLog(
416  ilExAssignmentTeam::TEAM_LOG_REMOVE_FILE,
417  $value["filetitle"]
418  );
419  }
420 
421  if ($this->getAssignment()->getAssignmentType()->isSubmissionAssignedToTeam()) {
422  $storage_id = $value["team_id"];
423  } else {
424  $storage_id = $value["user_id"];
425  }
426 
427  $filename = $path . "/" . $storage_id . "/" . basename($value["filename"]);
428  if (file_exists($filename)) {
429  unlink($filename);
430  }
431  }
432  }
433  }
434  }
435  }*/
436 
441  public static function deleteUser(
442  int $a_exc_id,
443  int $a_user_id
444  ): void {
445  global $DIC;
446 
447  $db = $DIC->database();
448 
449  foreach (ilExAssignment::getInstancesByExercise($a_exc_id) as $ass) {
450  $submission = new self($ass, $a_user_id);
451  $submission->deleteAllFiles();
452 
453  // remove from any team
454  $team = $submission->getTeam();
455  if ($team) {
456  $team->removeTeamMember($a_user_id);
457  }
458 
459  // #14900
460  $member_status = $ass->getMemberStatus($a_user_id);
461  $member_status->setStatus("notgraded");
462  $member_status->update();
463 
464  $db->manipulateF(
465  "DELETE FROM exc_usr_tutor " .
466  "WHERE ass_id = %s AND usr_id = %s",
467  array("integer", "integer"),
468  array($ass->getId(), $a_user_id)
469  );
470  }
471  }
472 
477  protected function getLastDownloadTime(
478  array $a_user_ids
479  ): string {
480  $ilDB = $this->db;
481  $ilUser = $this->user;
482 
483  $q = "SELECT download_time FROM exc_usr_tutor WHERE " .
484  " ass_id = " . $ilDB->quote($this->getAssignment()->getId(), "integer") . " AND " .
485  $ilDB->in("usr_id", $a_user_ids, "", "integer") . " AND " .
486  " tutor_id = " . $ilDB->quote($ilUser->getId(), "integer") .
487  " ORDER BY download_time DESC";
488  $lu_set = $ilDB->query($q);
489  $lu_rec = $ilDB->fetchAssoc($lu_set);
490  return $lu_rec["download_time"] ?? "";
491  }
492 
493  public function downloadFiles(
494  ?array $a_file_ids = null,
495  bool $a_only_new = false,
496  bool $a_peer_review_mask_filename = false
497  ): bool {
498  $ilUser = $this->user;
499  $lng = $this->lng;
500 
501  $user_ids = $this->getUserIds();
502  $is_team = $this->assignment->hasTeam();
503  // get last download time
504  $download_time = null;
505  if ($a_only_new) {
506  $download_time = $this->getLastDownloadTime($user_ids);
507  }
508 
509  if ($this->is_tutor) {
510  $this->updateTutorDownloadTime();
511  }
512 
513  if ($a_peer_review_mask_filename) {
514  // process peer review sequence id
515  $peer_id = null;
516  foreach ($this->peer_review->getPeerReviewsByGiver($ilUser->getId()) as $idx => $item) {
517  if ($item["peer_id"] == $this->getUserId()) {
518  $peer_id = $idx + 1;
519  break;
520  }
521  }
522  // this will remove personal info from zip-filename
523  $is_team = true;
524  }
525 
526  $subs = iterator_to_array($this->sub_manager->getSubmissionsOfUser(
527  $this->getUserId(),
528  $a_file_ids,
529  false,
530  $download_time
531  ));
532 
533  if (count($subs) > 0) {
534  if (count($subs) == 1) {
536  $sub = current($subs);
537 
538  $title = $sub->getTitle();
539 
540  switch ($this->assignment->getType()) {
543  $name = ilObjUser::_lookupName($sub->getUserId());
544  $title = ilObject::_lookupTitle($this->assignment->getExerciseId()) . " - " .
545  $this->assignment->getTitle() . " - " .
546  $name["firstname"] . " " .
547  $name["lastname"] . " (" .
548  $name["login"] . ").zip";
549  break;
550 
551  // @todo: generalize
553  $title = ilObject::_lookupTitle($this->assignment->getExerciseId()) . " - " .
554  $this->assignment->getTitle() . " (Team " . $this->getTeam()->getId() . ").zip";
555  break;
556 
557  default:
558  break;
559  }
560 
561  if ($a_peer_review_mask_filename) {
562  $title_a = explode(".", $sub->getTitle());
563  $suffix = array_pop($title_a);
564  $title = $this->assignment->getTitle() . "_peer" . $peer_id . "." . $suffix;
565  } elseif ($sub->getLate()) {
566  $title = $lng->txt("exc_late_submission") . " - " .
567  $title;
568  }
569 
570  $this->downloadSingleFile($sub, $title);
571  } else {
572  $this->sub_manager->deliverSubmissions(
573  $subs,
574  $this->getUserId(),
575  $a_peer_review_mask_filename,
576  $peer_id ?? 0
577  );
578  }
579  } else {
580  return false;
581  }
582 
583  return true;
584  }
585 
586  // Update the timestamp of the last download of current user (=tutor)
587  public function updateTutorDownloadTime(): void
588  {
589  $ilUser = $this->user;
590  $ilDB = $this->db;
591 
592  $exc_id = $this->assignment->getExerciseId();
593  $ass_id = $this->assignment->getId();
594 
595  foreach ($this->getUserIds() as $user_id) {
596  $ilDB->manipulateF(
597  "DELETE FROM exc_usr_tutor " .
598  "WHERE ass_id = %s AND usr_id = %s AND tutor_id = %s",
599  array("integer", "integer", "integer"),
600  array($ass_id, $user_id, $ilUser->getId())
601  );
602 
603  $ilDB->manipulateF(
604  "INSERT INTO exc_usr_tutor (ass_id, obj_id, usr_id, tutor_id, download_time) VALUES " .
605  "(%s, %s, %s, %s, %s)",
606  array("integer", "integer", "integer", "integer", "timestamp"),
607  array($ass_id, $exc_id, $user_id, $ilUser->getId(), ilUtil::now())
608  );
609  }
610  }
611 
612  protected function downloadSingleFile(
613  Submission $sub,
614  string $title
615  ): void {
616  $this->domain->submission($this->assignment->getId())->deliverFile(
617  $sub->getUserId(),
618  $sub->getRid(),
619  $title
620  );
621  }
622 
623  // Get user/team where clause
624  public function getTableUserWhere(): string
625  {
626  $ilDB = $this->db;
627 
628  if ($this->getAssignment()->getAssignmentType()->isSubmissionAssignedToTeam()) {
629  $team_id = $this->getTeam()->getId();
630  $where = " team_id = " . $ilDB->quote($team_id, "integer") . " ";
631  } else {
632  $where = " " . $ilDB->in("user_id", $this->getUserIds(), "", "integer") . " ";
633  }
634  return $where;
635  }
636 
637 
642  public function getLastSubmission(): ?string
643  {
644  $ilDB = $this->db;
645 
646  $ilDB->setLimit(1, 0);
647 
648  $q = "SELECT obj_id,user_id,ts FROM exc_returned" .
649  " WHERE ass_id = " . $ilDB->quote($this->assignment->getId(), "integer") .
650  " AND " . $this->getTableUserWhere() .
651  " AND (filename IS NOT NULL OR atext IS NOT NULL)" .
652  " AND ts IS NOT NULL" .
653  " ORDER BY ts DESC";
654  $usr_set = $ilDB->query($q);
655  $array = $ilDB->fetchAssoc($usr_set);
656  return ($array["ts"] ?? null);
657  }
658 
663  public function getLastOpeningHTMLView(): ?string
664  {
665  $this->db->setLimit(1, 0);
666 
667  $q = "SELECT web_dir_access_time FROM exc_returned" .
668  " WHERE ass_id = " . $this->db->quote($this->assignment->getId(), "integer") .
669  " AND (filename IS NOT NULL OR atext IS NOT NULL)" .
670  " AND web_dir_access_time IS NOT NULL" .
671  " AND " . $this->getTableUserWhere() .
672  " ORDER BY web_dir_access_time DESC";
673 
674  $res = $this->db->query($q);
675 
676  $data = $this->db->fetchAssoc($res);
677 
678  return $data["web_dir_access_time"] ?? null;
679  }
680 
681 
682  //
683  // OBJECTS
684  //
685 
691  public function addResourceObject(
692  string $a_wsp_id, // note: text assignments currently call this with "TEXT"
693  ?string $a_text = null
694  ): int {
695  $ilDB = $this->db;
696 
697  if ($this->getAssignment()->getAssignmentType()->isSubmissionAssignedToTeam()) {
698  $user_id = 0;
699  $team_id = $this->getTeam()->getId();
700  } else {
701  $user_id = $this->getUserId();
702  $team_id = 0;
703  }
704 
705  // repository objects must be unique in submissions
706  // the same repo object cannot be used in different submissions or even different assignment/exercises
707  // why? -> the access handling would fail, since the access depends e.g. on teams or even phase of the
708  // assignment
709  if ($this->getAssignment()->getAssignmentType()->getSubmissionType() == ilExSubmission::TYPE_REPO_OBJECT) {
710  $repos_ass_type_ids = $this->ass_types->getIdsForSubmissionType(ilExSubmission::TYPE_REPO_OBJECT);
711  $subs = $this->getSubmissionsForFilename($a_wsp_id, $repos_ass_type_ids);
712  if ($subs !== []) {
713  throw new ilExerciseException("Repository object $a_wsp_id is already assigned to another assignment.");
714  }
715  }
716 
717  $next_id = $ilDB->nextId("exc_returned");
718  $query = sprintf(
719  "INSERT INTO exc_returned " .
720  "(returned_id, obj_id, user_id, filetitle, ass_id, ts, atext, late, team_id) " .
721  "VALUES (%s, %s, %s, %s, %s, %s, %s, %s, %s)",
722  $ilDB->quote($next_id, "integer"),
723  $ilDB->quote($this->assignment->getExerciseId(), "integer"),
724  $ilDB->quote($user_id, "integer"),
725  $ilDB->quote($a_wsp_id, "text"),
726  $ilDB->quote($this->assignment->getId(), "integer"),
727  $ilDB->quote(ilUtil::now(), "timestamp"),
728  $ilDB->quote($a_text, "text"),
729  $ilDB->quote($this->isLate(), "integer"),
730  $ilDB->quote($team_id, "integer")
731  );
732  $ilDB->manipulate($query);
733 
734  return $next_id;
735  }
736 
737  /*
738  * Remove ressource from assignement (and delete
739  * its submission): Note: The object itself will not be deleted.
740  */
741  public function deleteResourceObject(): void
742  {
743  $this->deleteAllFiles();
744  }
745 
751  public function updateTextSubmission(string $a_text): ?int
752  {
753  $ilDB = $this->db;
754 
755  // no text = remove submission
756  if (!trim($a_text)) {
757  $this->sub_manager->deleteAllSubmissionsOfUser($this->getUserId());
758  return null;
759  }
760 
761  $sub = $this->sub_manager->getSubmissionsOfUser($this->getUserId())->current();
762 
763  if (!$sub) {
764  return $this->addResourceObject("TEXT", $a_text);
765  } else {
766  $id = $sub->getId();
767  if ($id) {
768  $ilDB->manipulate("UPDATE exc_returned" .
769  " SET atext = " . $ilDB->quote($a_text, "text") .
770  ", ts = " . $ilDB->quote(ilUtil::now(), "timestamp") .
771  ", late = " . $ilDB->quote($this->isLate(), "integer") .
772  " WHERE returned_id = " . $ilDB->quote($id, "integer"));
773  return $id;
774  }
775  }
776  return null;
777  }
778 
779  //
780  // GUI helper
781  //
782 
786  public function getDownloadedFilesInfoForTableGUIS(): array
787  {
788  $lng = $this->lng;
789  $ilCtrl = $this->ctrl;
790 
791  $result = array();
792  $result["files"]["count"] = "---";
793 
794  // submission:
795  // see if files have been resubmmited after solved
796  $last_sub = $this->getLastSubmission();
797  if ($last_sub) {
798  $last_sub = ilDatePresentation::formatDate(new ilDateTime($last_sub, IL_CAL_DATETIME));
799  } else {
800  $last_sub = "---";
801  }
802  $result["last_submission"]["txt"] = $lng->txt("exc_last_submission");
803  $result["last_submission"]["value"] = $last_sub;
804 
805  // #16994
806  $ilCtrl->setParameterByClass("ilexsubmissionfilegui", "member_id", $this->getUserId());
807 
808  // assignment type specific
809  switch ($this->assignment->getType()) {
811  // data is merged by team - see above
812  // fallthrough
813 
815  $subs = $this->sub_manager->getSubmissionsOfUser($this->getUserId());
816  $late_files = 0;
817  $cnt_all = 0;
818  foreach ($subs as $sub) {
819  if ($sub->getLate()) {
820  $late_files++;
821  }
822  $cnt_all++;
823  }
824 
825  // nr of submitted files
826  $result["files"]["txt"] = $lng->txt("exc_files_returned");
827  if ($late_files !== 0) {
828  $result["files"]["txt"] .= ' - <span class="warning">' . $lng->txt("exc_late_submission") . " (" . $late_files . ")</span>";
829  }
830  $sub_cnt = $cnt_all;
831  $new = $this->lookupNewFiles();
832  if ($new !== []) {
833  $sub_cnt .= " " . sprintf($lng->txt("cnt_new"), count($new));
834  }
835 
836  $result["files"]["count"] = $sub_cnt;
837 
838  // download command
839  if ($sub_cnt > 0) {
840  $result["files"]["download_url"] =
841  $ilCtrl->getLinkTargetByClass("ilexsubmissionfilegui", "downloadReturned");
842 
843  if (count($new) <= 0) {
844  $result["files"]["download_txt"] = $lng->txt("exc_tbl_action_download_files");
845  } else {
846  $result["files"]["download_txt"] = $lng->txt("exc_tbl_action_download_all_files");
847  }
848 
849  // download new files only
850  if ($new !== []) {
851  $result["files"]["download_new_url"] =
852  $ilCtrl->getLinkTargetByClass("ilexsubmissionfilegui", "downloadNewReturned");
853 
854  $result["files"]["download_new_txt"] = $lng->txt("exc_tbl_action_download_new_files");
855  }
856  }
857  break;
858 
860  $result["files"]["txt"] = $lng->txt("exc_blog_returned");
862  $sub = $this->sub_manager->getSubmissionsOfUser($this->getUserId())->current();
863  if ($sub) {
864  if ($sub->getRid() != "") {
865  if ($sub->getLate()) {
866  $result["files"]["txt"] .= ' - <span class="warning">' . $lng->txt("exc_late_submission") . "</span>";
867  }
868 
869  $result["files"]["count"] = 1;
870 
871  $result["files"]["download_url"] =
872  $ilCtrl->getLinkTargetByClass("ilexsubmissionfilegui", "downloadReturned");
873 
874  $result["files"]["download_txt"] = $lng->txt("exc_tbl_action_download_files");
875  }
876  }
877  break;
878 
880  $result["files"]["txt"] = $lng->txt("exc_portfolio_returned");
882  $sub = $this->sub_manager->getSubmissionsOfUser($this->getUserId())->current();
883  if ($sub) {
884  if ($sub->getRid() != "") {
885  if ($sub->getLate()) {
886  $result["files"]["txt"] .= ' - <span class="warning">' . $lng->txt("exc_late_submission") . "</span>";
887  }
888 
889  $result["files"]["count"] = 1;
890 
891  $result["files"]["download_url"] =
892  $ilCtrl->getLinkTargetByClass("ilexsubmissionfilegui", "downloadReturned");
893 
894  $result["files"]["download_txt"] = $lng->txt("exc_tbl_action_download_files");
895  }
896  }
897  break;
898 
900  $result["files"]["txt"] = $lng->txt("exc_files_returned_text");
901  $sub = $this->sub_manager->getSubmissionsOfUser($this->getUserId())->current();
902  if ($sub) {
903  $result["files"]["count"] = 1;
904 
905  if (trim($sub->getText()) !== '' && trim($sub->getText()) !== '0') {
906  if ($sub->getLate()) {
907  $result["files"]["txt"] .= ' - <span class="warning">' . $lng->txt("exc_late_submission") . "</span>";
908  }
909 
910  $result["files"]["download_url"] =
911  $ilCtrl->getLinkTargetByClass("ilexsubmissiontextgui", "showAssignmentText");
912 
913  $result["files"]["download_txt"] = $lng->txt("exc_tbl_action_text_assignment_show");
914  }
915  }
916  break;
917 
919  $result["files"]["txt"] = $lng->txt("exc_wiki_returned");
920  $sub = $this->sub_manager->getSubmissionsOfUser($this->getUserId())->current();
921  if ($sub) {
922  if ($sub->getRid() != "") {
923  if ($sub->getLate()) {
924  $result["files"]["txt"] .= ' - <span class="warning">' . $lng->txt("exc_late_submission") . "</span>";
925  }
926 
927  $result["files"]["count"] = 1;
928 
929  $result["files"]["download_url"] =
930  $ilCtrl->getLinkTargetByClass("ilexsubmissionfilegui", "downloadReturned");
931 
932  $result["files"]["download_txt"] = $lng->txt("exc_tbl_action_download_files");
933  }
934  }
935  break;
936  }
937 
938  $ilCtrl->setParameterByClass("ilexsubmissionfilegui", "member_id", "");
939 
940  return $result;
941  }
942 
946  public static function getSubmissionsForFilename(
947  string $a_filename,
948  array $a_assignment_types = array()
949  ): array {
950  global $DIC;
951 
952  $db = $DIC->database();
953 
954  $query = "SELECT * FROM exc_returned r LEFT JOIN exc_assignment a" .
955  " ON (r.ass_id = a.id) " .
956  " WHERE r.filetitle = " . $db->quote($a_filename, "string");
957 
958  if (is_array($a_assignment_types) && $a_assignment_types !== []) {
959  $query .= " AND " . $db->in("a.type", $a_assignment_types, false, "integer");
960  }
961 
962  $set = $db->query($query);
963  $rets = array();
964  while ($rec = $db->fetchAssoc($set)) {
965  $rets[] = $rec;
966  }
967 
968 
969  return $rets;
970  }
971 
975  public static function getDirectoryNameFromUserData(int $a_user_id): string
976  {
977  $userName = ilObjUser::_lookupName($a_user_id);
979  trim($userName["lastname"]) . "_" .
980  trim($userName["firstname"]) . "_" .
981  trim($userName["login"]) . "_" .
982  $userName["user_id"]
983  );
984  }
985 
986  public static function getAssignmentParticipants(
987  int $a_exercise_id,
988  int $a_ass_id
989  ): array {
990  global $DIC;
991 
992  $ilDB = $DIC->database();
993 
994  $participants = array();
995  $query = "SELECT user_id FROM exc_returned WHERE ass_id = " .
996  $ilDB->quote($a_ass_id, "integer") .
997  " AND obj_id = " .
998  $ilDB->quote($a_exercise_id, "integer");
999 
1000  $res = $ilDB->query($query);
1001 
1002  while ($row = $ilDB->fetchAssoc($res)) {
1003  $participants[] = $row['user_id'];
1004  }
1005 
1006  return $participants;
1007  }
1008 
1009  public static function processZipFile(
1010  string $a_directory,
1011  string $a_file,
1012  bool $structure
1013  ): void {
1014  global $DIC;
1015 
1016  $lng = $DIC->language();
1017 
1018  $pathinfo = pathinfo($a_file);
1019  $file = $pathinfo["basename"];
1020 
1021  // see 22727
1022  if (($pathinfo["extension"] ?? '') === '') {
1023  $file .= ".zip";
1024  }
1025 
1026  // Copy zip-file to new directory, unzip and remove it
1027  // TODO: check archive for broken file
1028  //copy ($a_file, $a_directory . "/" . $file);
1029  ilFileUtils::moveUploadedFile($a_file, $file, $a_directory . "/" . $file);
1030  $DIC->legacyArchives()->unzip(
1031  $a_directory . "/" . $file,
1032  null,
1033  false,
1034  true,
1035  false
1036  );
1037  unlink($a_directory . "/" . $file);
1038  //echo "-".$a_directory . "/" . $file."-";
1039  // Stores filename and paths into $filearray to check for viruses
1040  // Checks if filenames can be read, else -> throw exception and leave
1041  $filearray = [];
1042  ilFileUtils::recursive_dirscan($a_directory, $filearray);
1043 
1044  // if there are no files unziped (->broken file!)
1045  if (empty($filearray)) {
1046  throw new ilFileUtilsException(
1047  $lng->txt("archive_broken"),
1049  );
1050  }
1051 
1052  // virus handling
1053  foreach ($filearray["file"] as $key => $value) {
1054  // remove "invisible" files
1055  if (substr($value, 0, 1) == "." || stristr(
1056  $filearray["path"][$key],
1057  "/__MACOSX/"
1058  )) {
1059  unlink($filearray["path"][$key] . $value);
1060  unset($filearray["path"][$key]);
1061  unset($filearray["file"][$key]);
1062  continue;
1063  }
1064 
1065  $vir = ilVirusScanner::virusHandling($filearray["path"][$key], $value);
1066  if (!$vir[0]) {
1067  // Unlink file and throw exception
1068  unlink($filearray['path'][$key]);
1069  throw new ilFileUtilsException(
1070  $lng->txt("file_is_infected") . "<br />" . $vir[1],
1072  );
1073  } elseif ($vir[1] != "") {
1074  throw new ilFileUtilsException(
1075  $vir[1],
1077  );
1078  }
1079  }
1080 
1081  // If archive is to be used "flat"
1082  $doublettes = '';
1083  if (!$structure) {
1084  foreach (array_count_values($filearray["file"]) as $key => $value) {
1085  // Archive contains same filenames in different directories
1086  if ($value != "1") {
1087  $doublettes .= " '" . ilFileUtils::utf8_encode($key) . "'";
1088  }
1089  }
1090  if (strlen($doublettes) > 0) {
1091  throw new ilFileUtilsException(
1092  $lng->txt("exc_upload_error") . "<br />" . $lng->txt(
1093  "zip_structure_error"
1094  ) . $doublettes,
1096  );
1097  }
1098  } else {
1099  $mac_dir = $a_directory . "/__MACOSX";
1100  if (file_exists($mac_dir)) {
1101  ilFileUtils::delDir($mac_dir);
1102  }
1103  }
1104  }
1105 }
lookupNewFiles(?int $a_tutor=null)
Check how much files have been uploaded by the learner after the last download of the tutor...
$res
Definition: ltiservices.php:66
updateTextSubmission(string $a_text)
Handle text assignment submissions.
removeTeamMember(int $a_user_id, ?int $a_exc_ref_id=null)
Exercise assignment.
addResourceObject(string $a_wsp_id, ?string $a_text=null)
Add personal resource or repository object (ref_id) to assigment.
getFeedbackId()
used for the legacy storage path of feedbacks only
const IL_CAL_DATETIME
manipulateF(string $query, array $types, array $values)
txt(string $a_topic, string $a_default_lang_fallback_mod="")
gets the text for a given topic if the topic is not in the list, the topic itself with "-" will be re...
fetchAssoc(ilDBStatement $statement)
static utf8_encode(string $string)
static virusHandling(string $a_file, string $a_orig_name='', bool $a_clean=true)
$valid
ilGlobalTemplateInterface $main_tpl
ilExcAssMemberState $state
static _lookupName(int $a_user_id)
lookup user name
ilExPeerReview $peer_review
This file is part of ILIAS, a powerful learning management system published by ILIAS open source e-Le...
isInTeam(?int $a_user_id=null)
quote($value, string $type)
static getInstanceByUserId(int $a_assignment_id, int $a_user_id, bool $a_create_on_demand=false)
static processZipFile(string $a_directory, string $a_file, bool $structure)
const TYPE_UPLOAD
direct checks against const should be avoided, use type objects instead
static now()
Return current timestamp in Y-m-d H:i:s format.
getLastDownloadTime(array $a_user_ids)
static getAssignmentParticipants(int $a_exercise_id, int $a_ass_id)
This file is part of ILIAS, a powerful learning management system published by ILIAS open source e-Le...
getSubmittedEntry(bool $print=false)
downloadSingleFile(Submission $sub, string $title)
static getASCIIFilename(string $a_filename)
while($session_entry=$r->fetchRow(ilDBConstants::FETCHMODE_ASSOC)) return null
ILIAS Exercise InternalDomainService $domain
Exercise peer review.
static _lookupTitle(int $obj_id)
ILIAS Exercise Submission SubmissionManager $sub_manager
static recursive_dirscan(string $dir, array &$arr)
Recursively scans a given directory and writes path and filename into referenced array.
static delDir(string $a_dir, bool $a_clean_only=false)
removes a dir and all its content (subdirs and files) recursively
ilExAssignment $assignment
global $DIC
Definition: shib_login.php:22
static getInstanceByIds(int $a_ass_id, int $a_user_id=0)
getLastSubmission()
TODO -> get rid of getTableUserWhere and move to repository class Get the date of the last submission...
$structure
TOTAL STRUCTURE.
static lookupExerciseIdForReturnedId(int $a_returned_id)
Get exercise from submission id (used in ilObjMediaObject)
query(string $query)
Run a (read-only) Query on the database.
static moveUploadedFile(string $a_file, string $a_name, string $a_target, bool $a_raise_errors=true, string $a_mode="move_uploaded")
move uploaded file
ilExAssignmentTeam $team
__construct(ilExAssignment $a_ass, int $a_user_id, ?ilExAssignmentTeam $a_team=null, bool $a_is_tutor=false, bool $a_public_submissions=false)
getLastOpeningHTMLView()
TODO -> get rid of getTableUserWhere and move to repository class Get a mysql timestamp from the last...
static getDirectoryNameFromUserData(int $a_user_id)
in(string $field, array $values, bool $negate=false, string $type="")
static getInstancesByExercise(int $a_exc_id)
$id
plugin.php for ilComponentBuildPluginInfoObjectiveTest::testAddPlugins
Definition: plugin.php:23
static formatDate(ilDateTime $date, bool $a_skip_day=false, bool $a_include_wd=false, bool $include_seconds=false, ?ilObjUser $user=null,)
$q
Definition: shib_logout.php:21
ilExAssignmentTypes $ass_types
Exercise submission //TODO: This class has many static methods related to delivered "files"...
ilExAssignmentTypeInterface $ass_type
static deleteUser(int $a_exc_id, int $a_user_id)
Deletes already delivered files.
static findUserFiles(int $a_user_id, string $a_filetitle)
Check if given file was assigned Used in Blog/Portfolio.
This file is part of ILIAS, a powerful learning management system published by ILIAS open source e-Le...
This file is part of ILIAS, a powerful learning management system published by ILIAS open source e-Le...
This file is part of ILIAS, a powerful learning management system published by ILIAS open source e-Le...
static getSubmissionsForFilename(string $a_filename, array $a_assignment_types=array())
Get assignment return entries for a filename.