ILIAS  release_7 Revision v7.30-3-g800a261c036
All Data Structures Namespaces Files Functions Variables Modules Pages
class.ilCmiXapiContentUploadImporter.php
Go to the documentation of this file.
1 <?php
2 
3 /* Copyright (c) 1998-2019 ILIAS open source, Extended GPL, see docs/LICENSE */
4 
8 
19 {
20  const RELATIVE_CONTENT_DIRECTORY_NAMEBASE = 'lm_data/lm_';
21 
22  const RELATIVE_XSD_DIRECTORY = 'Modules/CmiXapi/xml/contentschema';
23 
24  const IMP_FILE_EXTENSION_XML = 'xml';
25  const IMP_FILE_EXTENSION_ZIP = 'zip';
26 
27  const CMI5_XML = 'cmi5.xml';
28  const CMI5_XSD = 'cmi5_v1_CourseStructure.xsd';
29 
30  const TINCAN_XML = 'tincan.xml';
31  const TINCAN_XSD = 'tincan.xsd';
32 
36  protected static $CONTENT_XML_FILENAMES = [
37  self::CMI5_XML, self::TINCAN_XML
38  ];
39 
43  protected static $CONTENT_XSD_FILENAMES = [
44  self::CMI5_XML => self::CMI5_XSD,
45  self::TINCAN_XML => self::TINCAN_XSD
46  ];
47 
51  protected $object;
52 
57  public function __construct(ilObjCmiXapi $object)
58  {
59  $this->object = $object;
60  }
61 
65  public function ensureCreatedObjectDirectory()
66  {
67  global $DIC; /* @var \ILIAS\DI\Container $DIC */
68 
69  if (!$DIC->filesystem()->web()->has($this->getWebDataDirRelativeObjectDirectory())) {
70  $DIC->filesystem()->web()->createDir($this->getWebDataDirRelativeObjectDirectory());
71  }
72  }
73 
74  protected function sanitizeObjectDirectory()
75  {
76  ilUtil::renameExecutables(implode(DIRECTORY_SEPARATOR, [
78  ]));
79  }
80 
86  public function importServerFile($serverFile)
87  {
89 
90  $this->handleFile($serverFile);
91 
92  $this->sanitizeObjectDirectory();
93  }
94 
99  protected function handleFile(string $serverFile)
100  {
101  $fileInfo = pathinfo($serverFile);
102 
103  switch ($fileInfo['extension']) {
104  case self::IMP_FILE_EXTENSION_XML:
105 
106  $this->handleXmlFile($serverFile);
107  break;
108 
109  case self::IMP_FILE_EXTENSION_ZIP:
110 
111  $this->handleZipContentUpload($serverFile);
112 
113  if ($this->hasStoredContentXml()) {
114  $this->handleXmlFile($this->getStoredContentXml());
115  }
116 
117  break;
118  }
119  }
120 
127  public function importFormUpload(ilFileInputGUI $uploadInput)
128  {
130 
131  $fileData = $_POST[$uploadInput->getPostVar()];
132 
133  $uploadResult = $this->getUpload(
134  $fileData['tmp_name']
135  );
136 
137  $this->handleUpload($uploadResult);
138 
139  $this->sanitizeObjectDirectory();
140  }
141 
148  protected function getUpload($uploadFilePath)
149  {
150  global $DIC; /* @var \ILIAS\DI\Container $DIC */
151 
152  if ($DIC->upload()->hasUploads()) {
153  if (!$DIC->upload()->hasBeenProcessed()) {
154  $DIC->upload()->process();
155  }
156 
157  /* @var FileUploadResult $result */
158 
159  $results = $DIC->upload()->getResults();
160 
161  if (isset($results[$uploadFilePath])) {
162  $result = $results[$uploadFilePath];
163 
164  if ($result->getStatus() == FileUploadProcessingStatus::OK) {
165  return $result;
166  }
167 
169  'upload processing failed with message ' .
170  '"' . $result->getStatus()->getMessage() . '"'
171  );
172  }
173 
174  throw new ilCmiXapiInvalidUploadContentException('upload lost during processing!');
175  }
176 
177  throw new ilCmiXapiInvalidUploadContentException('no upload provided!');
178  }
179 
184  protected function handleUpload(FileUploadResult $uploadResult)
185  {
186  switch ($this->fetchFileExtension($uploadResult)) {
187  case self::IMP_FILE_EXTENSION_XML:
188 
189  $this->handleXmlFileFromUpload($uploadResult->getName(), $uploadResult->getPath());
190  break;
191 
192  case self::IMP_FILE_EXTENSION_ZIP:
193 
194  $this->handleZipContentUpload($uploadResult->getPath());
195 
196  if ($this->hasStoredContentXml()) {
197  $this->handleXmlFile($this->getStoredContentXml());
198  }
199 
200  break;
201  }
202  }
203 
208  protected function handleXmlFile($xmlFilePath)
209  {
210  $dom = new DOMDocument();
211  $dom->load($xmlFilePath);
212 
213  switch (basename($xmlFilePath)) {
214  case self::CMI5_XML:
215 
216  $xsdFilePath = $this->getXsdFilePath(self::CMI5_XSD);
217  $this->validateXmlFile($dom, $xsdFilePath);
218 
219  $this->initObjectFromCmi5Xml($dom);
220 
221  break;
222 
223  case self::TINCAN_XML:
224 
225  $xsdFilePath = $this->getXsdFilePath(self::TINCAN_XSD);
226  $this->validateXmlFile($dom, $xsdFilePath);
227 
228  $this->initObjectFromTincanXml($dom);
229 
230  break;
231  }
232  }
233 
239  protected function handleXmlFileFromUpload($xmlFileName, $xmlFilePath)
240  {
241  $dom = new DOMDocument();
242  $dom->load($xmlFilePath);
243  switch (basename($xmlFileName)) {
244  case self::CMI5_XML:
245 
246  $xsdFilePath = $this->getXsdFilePath(self::CMI5_XSD);
247  $this->validateXmlFile($dom, $xsdFilePath);
248 
249  $this->initObjectFromCmi5Xml($dom);
250 
251  break;
252 
253  case self::TINCAN_XML:
254 
255  $xsdFilePath = $this->getXsdFilePath(self::TINCAN_XSD);
256  $this->validateXmlFile($dom, $xsdFilePath);
257 
258  $this->initObjectFromTincanXml($dom);
259 
260  break;
261  }
262  }
263 
264  protected function validateXmlFile(DOMDocument $dom, $xsdFilePath)
265  {
266  if (!$dom->schemaValidate($xsdFilePath)) {
267  throw new ilCmiXapiInvalidUploadContentException('invalid content xml given!');
268  }
269  }
270 
271  protected function handleZipContentUpload($uploadFilePath)
272  {
273  $targetPath = $this->getAbsoluteObjectDirectory();
274  $zar = new ZipArchive();
275  $zar->open($uploadFilePath);
276  $zar->extractTo($targetPath);
277  $zar->close();
278  }
279 
283  protected function getAbsoluteObjectDirectory()
284  {
285  $dirs = [
286  ILIAS_ABSOLUTE_PATH,
289  ];
290 
291  return implode(DIRECTORY_SEPARATOR, $dirs);
292  }
293 
298  {
299  return self::RELATIVE_CONTENT_DIRECTORY_NAMEBASE . $this->object->getId();
300  }
301 
306  protected function fetchFileExtension(FileUploadResult $uploadResult)
307  {
308  return pathinfo($uploadResult->getName(), PATHINFO_EXTENSION);
309  }
310 
314  protected function hasStoredContentXml()
315  {
316  return $this->getStoredContentXml() !== '';
317  }
318 
322  protected function getStoredContentXml()
323  {
324  global $DIC; /* @var \ILIAS\DI\Container $DIC */
325 
326  foreach (self::$CONTENT_XML_FILENAMES as $xmlFileName) {
327  $xmlFilePath = $this->getWebDataDirRelativeObjectDirectory() . DIRECTORY_SEPARATOR . $xmlFileName;
328 
329  if ($DIC->filesystem()->web()->has($xmlFilePath)) {
330  return $this->getAbsoluteObjectDirectory() . DIRECTORY_SEPARATOR . $xmlFileName;
331  }
332  }
333 
334  return '';
335  }
336 
341  protected function getXsdFilePath($xsdFileName)
342  {
343  return ILIAS_ABSOLUTE_PATH . DIRECTORY_SEPARATOR . self::RELATIVE_XSD_DIRECTORY . DIRECTORY_SEPARATOR . $xsdFileName;
344  }
345 
346  protected function initObjectFromCmi5Xml($dom)
347  {
348  global $DIC;
349  $xPath = new DOMXPath($dom);
350 
351  $courseNode = $xPath->query("//*[local-name()='course']")->item(0);
352  // TODO: multilanguage support
353  $title = $xPath->query("//*[local-name()='title']/*[local-name()='langstring']", $courseNode)->item(0)->nodeValue;
354  $this->object->setTitle(trim($title));
355 
356  $description = $xPath->query("//*[local-name()='description']/*[local-name()='langstring']", $courseNode)->item(0)->nodeValue;
357  $this->object->setDescription(trim($description));
358 
359  $publisherId = trim($courseNode->getAttribute('id'));
360  $this->object->setPublisherId($publisherId);
361 
362  $activityId = $this->generateActivityId($publisherId);
363  $this->object->setActivityId($activityId);
364 
365  foreach ($xPath->query("//*[local-name()='au']") as $assignedUnitNode) {
366  $relativeLaunchUrl = $xPath->query("//*[local-name()='url']", $assignedUnitNode)->item(0)->nodeValue;
367  $launchParameters = $xPath->query("//*[local-name()='launchParameters']", $assignedUnitNode)->item(0)->nodeValue;
368  $moveOn = trim($assignedUnitNode->getAttribute('moveOn'));
369  $entitlementKey = $xPath->query("//*[local-name()='entitlementKey']", $assignedUnitNode)->item(0)->nodeValue;
370  $masteryScore = trim($assignedUnitNode->getAttribute('masteryScore'));
371 
372  if (!empty($relativeLaunchUrl)) {
373  $this->object->setLaunchUrl(trim($relativeLaunchUrl));
374  }
375  if (!empty($launchParameters)) {
376  $this->object->setLaunchParameters(trim($launchParameters));
377  }
378  if (!empty($moveOn)) {
380  $moveOn = ilCmiXapiLP::MOVEON_PASSED;
381  }
382  $this->object->setMoveOn($moveOn);
383  }
384  if (!empty($entitlementKey)) {
385  $this->object->setEntitlementKey($entitlementKey);
386  }
387  if (!empty($masteryScore)) {
388  $this->object->setMasteryScore($masteryScore);
389  } else {
390  $this->object->setMasteryScore(ilObjCmiXapi::LMS_MASTERY_SCORE);
391  }
392 
393  break; // TODO: manage multi au imports
394  }
395  $xml_str = $dom->saveXML();
396  $this->object->setXmlManifest($xml_str);
397  $this->object->update();
398  $this->object->save();
399 
400  $lpSettings = new ilLPObjSettings($this->object->getId());
402  switch ($moveOn) {
405  break;
408  break;
411  break;
412  case ilCmiXapiLP::MOVEON_COMPLETED_AND_PASSED: // ich würde es noch implementieren
414  break;
415  }
416  $lpSettings->setMode($mode);
417  $lpSettings->update();
418  }
419 
420  protected function initObjectFromTincanXml($dom)
421  {
422  $xPath = new DOMXPath($dom);
423 
424  foreach ($xPath->query("//*[local-name()='activity']") as $activityNode) {
425  $title = $xPath->query("//*[local-name()='name']", $activityNode)->item(0)->nodeValue;
426  $this->object->setTitle(trim($title));
427 
428  $description = $xPath->query("//*[local-name()='description']", $activityNode)->item(0)->nodeValue;
429  $this->object->setDescription(trim($description));
430 
431  $activityId = $activityNode->getAttribute('id');
432  $this->object->setActivityId(trim($activityId));
433 
434  $relativeLaunchUrl = $xPath->query("//*[local-name()='launch']", $activityNode)->item(0)->nodeValue;
435  $this->object->setLaunchUrl(trim($relativeLaunchUrl));
436 
437  break; // TODO: manage multi activities imports
438  }
439 
440  $xml_str = $dom->saveXML();
441  $this->object->setXmlManifest($xml_str);
442  $this->object->update();
443  $this->object->save();
444  }
445 
446  private function generateActivityId($publisherId)
447  {
448  global $DIC;
449  $objId = $this->object->getId();
450  $activityId = "https://ilias.de/cmi5/activityid/" . (new \Ramsey\Uuid\UuidFactory())->uuid3(ilCmiXapiUser::getIliasUuid(), $objId . '-' . $publisherId);
451  return $activityId;
452  }
453 }
$result
fetchFileExtension(FileUploadResult $uploadResult)
const MOVEON_COMPLETED_AND_PASSED
getPostVar()
Get Post Variable.
This class represents a file property in a property form.
$objId
Definition: xapitoken.php:39
__construct(ilObjCmiXapi $object)
ilCmiXapiContentUploadImporter constructor.
global $DIC
Definition: goto.php:24
$results
static renameExecutables($a_dir)
Rename uploaded executables for security reasons.
static getWebspaceDir($mode="filesystem")
get webspace directory
const MOVEON_COMPLETED_OR_PASSED
$_POST["username"]