3 declare(strict_types=1);
36 public const CONVERT_XSL =
'./Modules/Scorm2004/templates/xsl/op/scorm12To2004.xsl';
37 public const WRAPPER_HTML =
'./Modules/Scorm2004/scripts/converter/GenericRunTimeWrapper1.0_aadlc/GenericRunTimeWrapper.htm';
38 public const WRAPPER_JS =
'./Modules/Scorm2004/scripts/converter/GenericRunTimeWrapper1.0_aadlc/SCOPlayerWrapper.js';
45 public function __construct(
int $a_id = 0,
bool $a_call_by_reference =
true)
49 $this->
lng = $DIC->language();
50 $this->error = $DIC[
"ilErr"];
51 $this->db = $DIC->database();
53 $this->
user = $DIC->user();
54 $this->
tabs = $DIC->tabs();
66 $this->import_sequencing = $a_val;
89 if (!function_exists(
'json_encode') || !function_exists(
'json_decode')) {
93 $needs_convert =
false;
100 $check_for_manifest_file = is_file($manifest_file);
105 if (!$check_for_manifest_file) {
106 $ilErr->raiseError($this->
lng->txt(
"Manifestfile $manifest_file not found!"),
$ilErr->MESSAGE);
111 if ($check_for_manifest_file) {
112 $manifest_file_array = file($manifest_file);
114 foreach ($manifest_file_array as $mfa) {
116 if (@iconv(
'UTF-8',
'UTF-8', $mfa) != $mfa) {
117 $needs_convert =
true;
125 $estimated_manifest_filesize = filesize($manifest_file) * 2;
129 $check_disc_free = 2;
134 if ($needs_convert) {
136 if ($check_for_manifest_file && ($check_disc_free > 1)) {
139 if (!copy($manifest_file, $manifest_file .
".old")) {
140 echo
"Failed to copy $manifest_file...<br>\n";
145 $f_write_handler = fopen($manifest_file .
".new",
"w");
146 $f_read_handler = fopen($manifest_file .
".old",
"r");
147 while (!feof($f_read_handler)) {
148 $zeile = fgets($f_read_handler);
150 fwrite($f_write_handler, utf8_encode($zeile));
152 fclose($f_read_handler);
153 fclose($f_write_handler);
156 if (!copy($manifest_file .
".new", $manifest_file)) {
157 echo
"Failed to copy $manifest_file...<br>\n";
160 if (!@is_file($manifest_file)) {
161 $ilErr->raiseError($this->
lng->txt(
"cont_no_manifest"),
$ilErr->WARNING);
166 if (!($check_disc_free > 1)) {
167 $ilErr->raiseError($this->
lng->txt(
"Not enough space left on device!"),
$ilErr->MESSAGE);
173 $hmani = fopen($manifest_file,
"r");
174 $start = fread($hmani, 3);
175 if (strtolower(bin2hex($start)) ===
"efbbbf") {
176 $f_write_handler = fopen($manifest_file .
".new",
"w");
177 while (!feof($hmani)) {
178 $n = fread($hmani, 900);
179 fwrite($f_write_handler, $n);
181 fclose($f_write_handler);
185 if (!copy($manifest_file .
".new", $manifest_file)) {
186 echo
"Failed to copy $manifest_file...<br>\n";
202 $out = file_get_contents($this->imsmanifestFile);
203 $check =
'/xmlns="http:\/\/www.imsglobal.org\/xsd\/imscp_v1p1"/';
204 $replace =
"xmlns=\"http://www.imsproject.org/xsd/imscp_rootv1p1p2\"";
206 file_put_contents($this->imsmanifestFile,
$out);
214 ##check manifest-file for version. Check for schemaversion as this is a required element for SCORM 2004 215 ##accept 2004 3rd Edition an CAM 1.3 as valid schemas 219 $this->imsmanifestFile = $manifest;
220 $doc =
new DomDocument();
224 $doc->load($this->imsmanifestFile);
225 $elements = $doc->getElementsByTagName(
"schemaversion");
227 if (isset($elements->item(0)->nodeValue)) {
228 $schema = $elements->item(0)->nodeValue;
230 if (strtolower(trim($schema)) ===
"cam 1.3" || strtolower(trim($schema)) ===
"2004 3rd edition" || strtolower(trim($schema)) ===
"2004 4th edition") {
238 $organizations = $doc->getElementsByTagName(
"organizations");
240 if ($organizations->item(0) == null) {
241 die(
"organizations missing in manifest");
243 $default = $organizations->item(0)->getAttribute(
"default");
244 if ($default ==
"" || $default == null) {
246 $organization = $doc->getElementsByTagName(
"organization");
247 $item = $organization->item(0);
248 if ($item !== null) {
249 $ident = $item->getAttribute(
"identifier");
250 $organizations->item(0)->setAttribute(
"default", $ident);
258 $wrapperdir = $packageFolder .
"/GenericRunTimeWrapper1.0_aadlc";
259 if (!mkdir($wrapperdir) && !is_dir($wrapperdir)) {
260 throw new \RuntimeException(sprintf(
'Directory "%s" was not created', $wrapperdir));
262 copy(self::WRAPPER_HTML, $wrapperdir .
"/GenericRunTimeWrapper.htm");
263 copy(self::WRAPPER_JS, $wrapperdir .
"/SCOPlayerWrapper.js");
266 $backupManifest = $packageFolder .
"/imsmanifest.xml.back";
267 $ret = copy($this->imsmanifestFile, $backupManifest);
271 $ilLog->debug(
"SCORM: about to transform to SCORM 2004");
275 $xsl->load(self::CONVERT_XSL);
276 $prc =
new XSLTProcessor();
277 $r = @$prc->importStyleSheet($xsl);
279 file_put_contents($this->imsmanifestFile, $prc->transformToXML($totransform));
281 $ilLog->debug(
"SCORM: Transformation completed");
291 $ilDB = $DIC->database();
293 $result =
$ilDB->queryF(
295 SELECT MAX(c_timestamp) last_access 296 FROM cmi_node, cp_node 297 WHERE cmi_node.cp_node_id = cp_node.cp_node_id 298 AND cp_node.slm_id = %s 300 GROUP BY c_timestamp',
301 array(
'integer',
'integer'),
302 array($a_obj_id, $a_usr_id)
304 if (
$ilDB->numRows($result)) {
305 $row =
$ilDB->fetchAssoc($result);
306 return (
string) $row[
"last_access"];
317 foreach ($a_users as $user) {
334 $sco_set =
$ilDB->queryF(
336 SELECT DISTINCT cmi_node.cp_node_id id 337 FROM cp_node, cmi_node 339 AND cp_node.cp_node_id = cmi_node.cp_node_id 340 ORDER BY cmi_node.cp_node_id ',
342 array($this->
getId())
347 while ($sco_rec =
$ilDB->fetchAssoc($sco_set)) {
348 $item[
'id'] = $sco_rec[
"id"];
349 $item[
'title'] = self::_lookupItemTitle((
int) $sco_rec[
"id"]);
367 $val_set =
$ilDB->queryF(
368 'SELECT cp_node_id FROM cp_node 370 AND cp_node.slm_id = %s',
371 array(
'text',
'integer'),
372 array(
'item',$this->
getId())
374 while ($val_rec =
$ilDB->fetchAssoc($val_set)) {
375 $scos[] = $val_rec[
'cp_node_id'];
378 foreach ($scos as $sco) {
379 $data_set =
$ilDB->queryF(
381 SELECT c_timestamp last_access, total_time, success_status, completion_status, 382 c_raw, scaled, cp_node_id 384 WHERE cp_node_id = %s 386 array(
'integer',
'integer'),
387 array($sco,$a_user_id)
390 while ($data_rec =
$ilDB->fetchAssoc($data_set)) {
391 if ($data_rec[
"success_status"] !=
"" && $data_rec[
"success_status"] !==
"unknown") {
392 $status = $data_rec[
"success_status"];
394 if ($data_rec[
"completion_status"] ==
"") {
397 $status = $data_rec[
"completion_status"];
403 if ($data_rec[
"c_raw"] != null) {
404 $score = $data_rec[
"c_raw"];
405 if ($data_rec[
"scaled"] != null) {
409 if ($data_rec[
"scaled"] != null) {
410 $score .= ($data_rec[
"scaled"] * 100) .
"%";
412 $title = self::_lookupItemTitle((
int) $data_rec[
"cp_node_id"]);
414 $data[] = array(
"sco_id" => $data_rec[
"cp_node_id"],
415 "score" => $score,
"time" => $time,
"status" => $status,
"last_access" => $last_access,
"title" =>
$title);
417 $data_rec[
"total_time"] = self::_ISODurationToCentisec($data_rec[
"total_time"]) / 100;
418 $data[$data_rec[
"cp_node_id"]] = $data_rec;
432 $val_set =
$ilDB->queryF(
433 'SELECT package_attempts FROM sahs_user WHERE user_id = %s AND obj_id = %s',
434 array(
'integer',
'integer'),
435 array($a_user_id, $this->
getId())
438 $val_rec =
$ilDB->fetchAssoc($val_set);
440 if ($val_rec[
"package_attempts"] == null) {
441 $val_rec[
"package_attempts"] = 0;
444 return (
int) $val_rec[
"package_attempts"];
453 $val_set =
$ilDB->queryF(
454 'SELECT module_version FROM sahs_user WHERE user_id = %s AND obj_id = %s',
455 array(
'integer',
'integer'),
456 array($a_user_id, $this->
getId())
459 $val_rec =
$ilDB->fetchAssoc($val_set);
461 if ($val_rec[
"module_version"] == null) {
462 $val_rec[
"module_version"] =
"";
464 return $val_rec[
"module_version"];
473 $collection = $olp->getCollectionInstance();
475 $scos = $collection->getItems();
478 $fhandle = fopen($a_file,
"rb");
480 $obj_id = $this->getID();
482 $usersToDelete = array();
483 $fields = fgetcsv($fhandle, 4096,
';');
484 while (($csv_rows = fgetcsv($fhandle, 4096,
";")) !==
false) {
486 $data = array_combine($fields, $csv_rows);
488 if (isset(
$data[
"Login"])) {
491 if (isset(
$data[
"login"])) {
495 if (isset(
$data[
"user"]) && is_numeric(
$data[
"user"])) {
500 if (isset(
$data[
'Date'])) {
501 $date_ex = explode(
'.',
$data[
'Date']);
502 $last_access = implode(
'-', array($date_ex[2], $date_ex[1], $date_ex[0]));
504 if (isset(
$data[
'LastAccess'])) {
505 $last_access =
$data[
'LastAccess'];
510 if (isset(
$data[
"Status"])) {
511 if (is_numeric(
$data[
"Status"])) {
512 $status =
$data[
"Status"];
522 if (isset(
$data[
"Attempts"])) {
523 $attempts = (
int)
$data[
"Attempts"];
526 $percentage_completed = 0;
528 $percentage_completed = 100;
529 } elseif (isset(
$data[
'percentageCompletedSCOs'])) {
530 $percentage_completed = (
int)
$data[
'percentageCompletedSCOs'];
533 $sco_total_time_sec = null;
534 if (isset(
$data[
'SumTotal_timeSeconds'])) {
535 $sco_total_time_sec = (
int)
$data[
'SumTotal_timeSeconds'];
539 $usersToDelete[] = $user_id;
541 $this->
importSuccessForSahsUser($user_id, $last_access, $status, $attempts, $percentage_completed, $sco_total_time_sec);
546 foreach ($scos as $sco_id) {
549 SELECT completion_status, success_status, user_id FROM cmi_node WHERE cp_node_id = %s AND user_id = %s',
550 array(
'integer',
'integer'),
551 array($sco_id,$user_id)
555 $nextId =
$ilDB->nextId(
'cmi_node');
556 $val_set =
$ilDB->manipulateF(
557 'INSERT INTO cmi_node 558 (cp_node_id,user_id,completion_status,c_timestamp,cmi_node_id) 559 VALUES(%s,%s,%s,%s,%s)',
560 array(
'integer',
'integer',
'text',
'timestamp',
'integer'),
561 array($sco_id,$user_id,
'completed',$last_access,$nextId)
566 if (($row[
"completion_status"] ===
"completed" && $row[
"success_status"] !==
"failed") || $row[
"success_status"] ===
"passed") {
567 if ($doUpdate !=
true) {
574 if ($doUpdate ==
true) {
578 'completion_status' => array(
'text',
'completed'),
579 'success_status' => array(
'text',
''),
580 'suspend_data' => array(
'text',
''),
581 'c_timestamp' => array(
'timestamp', $last_access)
584 'user_id' => array(
'integer', $user_id),
585 'cp_node_id' => array(
'integer', $sco_id)
597 if (count($usersToDelete) > 0) {
612 $aV = array(0, 0, 0, 0, 0, 0);
615 if (strpos($str,
"P") != 0) {
619 $aT = array(
"Y",
"M",
"D",
"H",
"M",
"S");
622 $str = substr($str, 1);
623 for (
$i = 0, $max = count($aT);
$i < $max;
$i++) {
624 if (strpos($str,
"T") === 0) {
625 $str = substr($str, 1);
629 $p = strpos($str, $aT[
$i]);
632 if ($i == 1 && strpos($str,
"T") > -1 && strpos($str,
"T") < $p) {
635 if ($aT[$i] ===
"S") {
636 $aV[
$i] = substr($str, 0, $p);
638 $aV[
$i] = intval(substr($str, 0, $p));
640 if (!is_numeric($aV[$i])) {
645 if ($i > 2 && !$bTFound) {
649 $str = substr($str, $p + 1);
652 if (!$bErr && strlen($str) != 0) {
660 return $aV[0] * 3_155_760_000 + $aV[1] * 262_980_000 + $aV[2] * 8_640_000 + $aV[3] * 360000 + $aV[4] * 6000 + round($aV[5] * 100);
666 $val_set = $DIC->database()->queryF(
668 SELECT distinct(cp_node.cp_node_id) FROM cp_node,cp_resource,cp_item 669 WHERE cp_item.cp_node_id = cp_node.cp_node_id 670 AND cp_item.resourceid = cp_resource.id 673 AND cp_node.slm_id = %s ',
674 array(
'text',
'text',
'integer'),
675 array(
'sco',
'item',$a_slm_id)
677 return $DIC->database()->numRows($val_set);
688 $ilDB = $DIC->database();
693 $val_set =
$ilDB->queryF(
695 SELECT cp_node.cp_node_id FROM cp_node,cp_resource,cp_item 696 WHERE cp_item.cp_node_id = cp_node.cp_node_id 697 AND cp_item.resourceid = cp_resource.id 700 AND cp_node.slm_id = %s',
701 array(
'text',
'text',
'integer'),
702 array(
'sco' ,
'item',$a_id)
704 while ($val_rec =
$ilDB->fetchAssoc($val_set)) {
705 $scos[] = $val_rec[
'cp_node_id'];
711 foreach ($scos as
$i => $value) {
712 $val_set =
$ilDB->queryF(
714 SELECT * FROM cmi_node 717 AND (completion_status = %s OR success_status = %s))',
718 array(
'integer',
'integer',
'text',
'text'),
719 array($a_user, $value,
'completed',
'passed')
722 if (
$ilDB->numRows($val_set) > 0) {
724 $key = array_search($value, $scos_c);
725 unset($scos_c[
$key]);
729 if (count($scos_c) == 0) {
746 $ilDB = $DIC->database();
750 $val_set =
$ilDB->queryF(
751 "SELECT cp_node.cp_node_id FROM cp_node,cp_resource,cp_item WHERE" .
752 " cp_item.cp_node_id=cp_node.cp_node_id AND cp_item.resourceId = cp_resource.id AND scormType='sco' AND nodeName='item' AND cp_node.slm_id = %s GROUP BY cp_node.cp_node_id",
756 while ($val_rec =
$ilDB->fetchAssoc($val_set)) {
757 $scos[] = $val_rec[
'cp_node_id'];
761 foreach ($scos as
$i => $iValue) {
762 $val_set =
$ilDB->queryF(
763 "SELECT scaled FROM cmi_node WHERE (user_id = %s AND cp_node_id = %s)",
764 array(
'integer',
'integer'),
765 array($a_user, $scos[
$i])
767 if ($val_set->numRows() > 0) {
768 $val_rec =
$ilDB->fetchAssoc($val_set);
769 if ($val_rec[
'scaled'] != null) {
771 $scaled = $val_rec[
'scaled'];
775 return ($set == 1) ? $scaled : -1;
787 $ilDB = $DIC->database();
790 $item_set =
$ilDB->queryF(
792 SELECT cp_item.* FROM cp_node, cp_item WHERE slm_id = %s 793 AND cp_node.cp_node_id = cp_item.cp_node_id 794 ORDER BY cp_node.cp_node_id ',
800 while ($item_rec =
$ilDB->fetchAssoc($item_set)) {
803 SELECT cp_resource.* FROM cp_node, cp_resource 805 AND cp_node.cp_node_id = cp_resource.cp_node_id 806 AND cp_resource.id = %s ',
807 array(
'integer',
'text'),
808 array($a_obj_id,$item_rec[
"resourceid"])
813 if (
$res[
"scormtype"] ===
"sco") {
814 $items[] = array(
"id" => $item_rec[
"cp_node_id"],
815 "title" => $item_rec[
"title"]);
826 public static function _getStatus(
int $a_obj_id,
int $a_user_id)
830 $ilDB = $DIC->database();
832 $status_set =
$ilDB->queryF(
834 SELECT * FROM cmi_gobjective 836 AND objective_id = %s 838 array(
'integer',
'text',
'integer'),
839 array($a_obj_id,
'-course_overall_status-',$a_user_id)
842 if ($status_rec =
$ilDB->fetchAssoc($status_set)) {
843 return $status_rec[
"status"];
856 $ilDB = $DIC->database();
859 $status_set =
$ilDB->queryF(
861 SELECT * FROM cmi_gobjective 863 AND objective_id = %s 865 array(
'integer',
'text',
'integer'),
866 array($a_obj_id,
'-course_overall_status-',$a_user_id)
869 if ($status_rec =
$ilDB->fetchAssoc($status_set)) {
870 return $status_rec[
"satisfied"];
883 $ilDB = $DIC->database();
885 $status_set =
$ilDB->queryF(
887 SELECT * FROM cmi_gobjective 889 AND objective_id = %s 891 array(
'integer',
'text',
'integer'),
892 array($a_obj_id,
'-course_overall_status-',$a_user_id)
895 if ($status_rec =
$ilDB->fetchAssoc($status_set)) {
896 return (
float) $status_rec[
"measure"];
906 $ilDB = $DIC->database();
910 SELECT * FROM cp_item 911 WHERE cp_node_id = %s',
916 if (
$i =
$ilDB->fetchAssoc($r)) {
929 $ilDB = $DIC->database();
933 $result =
$ilDB->query(
934 'SELECT cp_node.cp_node_id ' 935 .
'FROM cp_node, cp_resource, cp_item ' 936 .
'WHERE cp_item.cp_node_id = cp_node.cp_node_id ' 937 .
'AND cp_item.resourceId = cp_resource.id ' 938 .
'AND scormType = ' .
$ilDB->quote(
'sco',
'text') .
' ' 939 .
'AND nodeName = ' .
$ilDB->quote(
'item',
'text') .
' ' 940 .
'AND cp_node.slm_id = ' .
$ilDB->quote($a_id,
'integer') .
' ' 941 .
'GROUP BY cp_node.cp_node_id' 944 while ($row =
$ilDB->fetchAssoc($result)) {
945 $scos[] = $row[
'cp_node_id'];
950 foreach ($scos as
$i => $value) {
952 'SELECT c_max FROM cmi_node WHERE (user_id = %s AND cp_node_id = %s)',
953 array(
'integer',
'integer'),
954 array($a_user, $value)
959 if ($row[
'c_max'] != null) {
961 $max = $row[
'c_max'];
965 return ($set == 1) ? $max : null;
975 $ilDB = $DIC->database();
976 $retAr = array(
"raw" => null,
"max" => null,
"scaled" => null);
977 $val_set =
$ilDB->queryF(
978 "SELECT c_raw, c_max, scaled FROM cmi_node WHERE (user_id = %s AND cp_node_id = %s)",
979 array(
'integer',
'integer'),
980 array($a_user, $a_cp_node_id)
982 if ($val_set->numRows() > 0) {
983 $val_rec =
$ilDB->fetchAssoc($val_set);
984 $retAr[
"raw"] = $val_rec[
'c_raw'];
985 $retAr[
"max"] = $val_rec[
'c_max'];
986 $retAr[
"scaled"] = $val_rec[
'scaled'];
987 if ($val_rec[
'scaled'] == null && $val_rec[
'c_raw'] != null && $val_rec[
'c_max'] != null) {
988 $retAr[
"scaled"] = ($val_rec[
'c_raw'] / $val_rec[
'c_max']);
const LP_STATUS_COMPLETED_NUM
get_user_id(string $a_login)
This file is part of ILIAS, a powerful learning management system published by ILIAS open source e-Le...
importSuccess(string $a_file)
static getLogger(string $a_component_id)
Get component logger.
This file is part of ILIAS, a powerful learning management system published by ILIAS open source e-Le...
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...
static _getStatus(int $a_obj_id, int $a_user_id)
const LP_STATUS_NOT_ATTEMPTED
importSuccessForSahsUser(int $user_id, string $last_access, int $status, ?int $attempts=null, ?int $percentage_completed=null, ?int $sco_total_time_sec=null)
static _getSatisfied(int $a_obj_id, int $a_user_id)
const LP_STATUS_IN_PROGRESS_NUM
static formatDate(ilDateTime $date, bool $a_skip_day=false, bool $a_include_wd=false, bool $include_seconds=false)
deleteTrackingDataOfUsers(array $a_users)
static _ISODurationToCentisec(string $str)
convert ISO 8601 Timeperiods to centiseconds
static _lookupLastAccess(int $a_obj_id, int $a_usr_id)
Return the last access timestamp for a given user.
setImportSequencing(bool $a_val)
Set import sequencing.
static secondsToString(int $seconds, bool $force_with_seconds=false, ?ilLanguage $a_lng=null)
converts seconds to string: Long: 7 days 4 hour(s) ...
static now()
Return current timestamp in Y-m-d H:i:s format.
getAttemptsForUser(int $a_user_id)
get number of atttempts for a certain user and package
const LP_STATUS_IN_PROGRESS
static _getMeasure(int $a_obj_id, int $a_user_id)
getTrackedItems()
get all tracked items of current user
static _getUniqueScaledScoreForUser(int $a_id, int $a_user)
Get the Unique Scaled Score of a course Conditions: Only one SCO may set cmi.score.scaled.
static getQuantityOfSCOs(int $a_slm_id)
static _lookupItemTitle(int $a_node_id)
static _refreshStatus(int $a_obj_id, ?array $a_users=null)
__construct(int $a_id=0, bool $a_call_by_reference=true)
Constructor.
static _getScores2004ForUser(int $a_cp_node_id, int $a_user)
const LP_STATUS_NOT_ATTEMPTED_NUM
getDataDirectory(?string $mode="filesystem")
get data directory of lm
getModuleVersionForUser(int $a_user_id)
get module version that tracking data for a user was recorded on
__construct(Container $dic, ilPlugin $plugin)
This file is part of ILIAS, a powerful learning management system published by ILIAS open source e-Le...
static _getCourseCompletionForUser(int $a_id, int $a_user)
Get the completion of a SCORM module for a given user.
static _getTrackingItems(int $a_obj_id)
get all tracking items of scorm object currently a for learning progress only
static _deleteReadEventsForUsers(int $a_obj_id, array $a_user_ids)
getTrackingDataAgg(int $a_user_id, ?bool $raw=false)
convert_1_2_to_2004(string $manifest)
readObject()
read manifest file
static getInstance(int $obj_id)
This file is part of ILIAS, a powerful learning management system published by ILIAS open source e-Le...
static removeCMIDataForUserAndPackage(int $user_id, int $packageId)
static _getMaxScoreForUser(int $a_id, int $a_user)
Returns score.max for the learning module, refered to the last sco where score.max is set...
const LP_STATUS_FAILED_NUM
static _updateStatus(int $a_obj_id, int $a_usr_id, ?object $a_obj=null, bool $a_percentage=false, bool $a_force_raise=false)
getImportSequencing()
Get import sequencing.