aicc.class.php 46 KB

12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091929394959697989910010110210310410510610710810911011111211311411511611711811912012112212312412512612712812913013113213313413513613713813914014114214314414514614714814915015115215315415515615715815916016116216316416516616716816917017117217317417517617717817918018118218318418518618718818919019119219319419519619719819920020120220320420520620720820921021121221321421521621721821922022122222322422522622722822923023123223323423523623723823924024124224324424524624724824925025125225325425525625725825926026126226326426526626726826927027127227327427527627727827928028128228328428528628728828929029129229329429529629729829930030130230330430530630730830931031131231331431531631731831932032132232332432532632732832933033133233333433533633733833934034134234334434534634734834935035135235335435535635735835936036136236336436536636736836937037137237337437537637737837938038138238338438538638738838939039139239339439539639739839940040140240340440540640740840941041141241341441541641741841942042142242342442542642742842943043143243343443543643743843944044144244344444544644744844945045145245345445545645745845946046146246346446546646746846947047147247347447547647747847948048148248348448548648748848949049149249349449549649749849950050150250350450550650750850951051151251351451551651751851952052152252352452552652752852953053153253353453553653753853954054154254354454554654754854955055155255355455555655755855956056156256356456556656756856957057157257357457557657757857958058158258358458558658758858959059159259359459559659759859960060160260360460560660760860961061161261361461561661761861962062162262362462562662762862963063163263363463563663763863964064164264364464564664764864965065165265365465565665765865966066166266366466566666766866967067167267367467567667767867968068168268368468568668768868969069169269369469569669769869970070170270370470570670770870971071171271371471571671771871972072172272372472572672772872973073173273373473573673773873974074174274374474574674774874975075175275375475575675775875976076176276376476576676776876977077177277377477577677777877978078178278378478578678778878979079179279379479579679779879980080180280380480580680780880981081181281381481581681781881982082182282382482582682782882983083183283383483583683783883984084184284384484584684784884985085185285385485585685785885986086186286386486586686786886987087187287387487587687787887988088188288388488588688788888989089189289389489589689789889990090190290390490590690790890991091191291391491591691791891992092192292392492592692792892993093193293393493593693793893994094194294394494594694794894995095195295395495595695795895996096196296396496596696796896997097197297397497597697797897998098198298398498598698798898999099199299399499599699799899910001001100210031004100510061007100810091010101110121013101410151016101710181019102010211022102310241025102610271028102910301031103210331034103510361037103810391040104110421043104410451046104710481049105010511052105310541055105610571058105910601061106210631064106510661067106810691070107110721073107410751076107710781079108010811082108310841085108610871088108910901091109210931094109510961097109810991100110111021103110411051106110711081109111011111112
  1. <?php
  2. /* For licensing terms, see /license.txt */
  3. /**
  4. * Defines the AICC class, which is meant to contain the aicc items (nuclear elements)
  5. * @package chamilo.learnpath
  6. * @author Yannick Warnier <ywarnier@beeznest.org>
  7. * @license GNU/GPL
  8. */
  9. /**
  10. * Code
  11. */
  12. require_once 'aiccItem.class.php';
  13. //require_once 'aiccMetadata.class.php';
  14. //require_once 'aiccOrganization.class.php';
  15. require_once 'aiccResource.class.php';
  16. require_once 'aiccBlock.class.php';
  17. /**
  18. * Class
  19. * @package chamilo.learnpath
  20. */
  21. class aicc extends learnpath
  22. {
  23. public $config = array();
  24. public $config_basename = ''; // The configuration files might be multiple and might have
  25. // funny names. We need to keep the name of that file while we
  26. // install the content.
  27. public $config_files = array();
  28. public $config_exts = array(
  29. 'crs' => 0, // Course description file (mandatory)
  30. 'au' => 0, // Assignable Unit file (mandatory)
  31. 'des' => 0, // Descriptor file (mandatory)
  32. 'cst' => 0, // Course structure file (mandatory)
  33. 'ore' => 0, // Objectives relationshops file (optional)
  34. 'pre' => 0, // Prerequisites file (optional)
  35. 'cmp' => 0 // Completion Requirements file (optional)
  36. );
  37. public $aulist = array();
  38. public $au_order_list = array();
  39. public $au_order_list_new_id = array();
  40. public $deslist = array();
  41. public $cstlist = array();
  42. public $orelist = array();
  43. public $subdir = ''; // Path between the scorm/ directory and the config files e.g. maritime_nav/maritime_nav. This is the path that will be used in the lp_path when importing a package.
  44. public $zipname = ''; // Keeps the zipfile safe for the object's life so that we can use it if there is no title available.
  45. public $lastzipnameindex = 0; // Keeps an index of the number of uses of the zipname so far.
  46. public $config_encoding = 'ISO-8859-1';
  47. public $debug = 0;
  48. /**
  49. * Class constructor. Based on the parent constructor.
  50. * @param string Course code
  51. * @param integer Learnpath ID in DB
  52. * @param integer User ID
  53. */
  54. public function __construct($course_code = null, $resource_id = null, $user_id = null)
  55. {
  56. if ($this->debug > 0) {
  57. error_log('In aicc::aicc()', 0);
  58. }
  59. if (!empty($course_code) && !empty($resource_id) && !empty($user_id)) {
  60. parent::__construct($course_code, $resource_id, $user_id);
  61. } else {
  62. //do nothing but still build the aicc object
  63. }
  64. }
  65. /**
  66. * Opens a resource
  67. * @param integer Database ID of the resource
  68. */
  69. public function open($id)
  70. {
  71. if ($this->debug > 0) {
  72. error_log('In aicc::open()', 0);
  73. }
  74. // Redefine parent method.
  75. }
  76. /**
  77. * Parses a set of AICC config files and puts everything into the $config array
  78. * @param string Path to the config files dir on the system. If not defined, uses the base path of the course's scorm dir
  79. * @return array Structured array representing the config files' contents
  80. */
  81. function parse_config_files($dir = '')
  82. {
  83. if ($this->debug > 0) {
  84. error_log('New LP - In aicc::parse_config_files('.$dir.')', 0);
  85. }
  86. if (empty($dir)) {
  87. // Get the path of the AICC config files dir.
  88. $dir = $this->subdir;
  89. }
  90. if (is_dir($dir) && is_readable($dir)) {
  91. // Now go through all the config files one by one and parse everything into AICC objects.
  92. // The basename for the config files is stored in $this->config_basename.
  93. // Parse the Course Description File (.crs) - ini-type.
  94. $crs_file = $dir.'/'.$this->config_files['crs'];
  95. $crs_params = $this->parse_ini_file_quotes_safe($crs_file);
  96. //echo '<pre>crs:'.print_r($crs_params, true).'</pre>';
  97. if ($this->debug > 1) {
  98. error_log('New LP - In aicc::parse_config_files() - '.$crs_file.' has been parsed', 0);
  99. }
  100. // CRS distribute crs params into the aicc object.
  101. if (!empty($crs_params['course']['course_creator'])) {
  102. $this->course_creator = Database::escape_string($crs_params['course']['course_creator']);
  103. }
  104. if (!empty($crs_params['course']['course_id'])) {
  105. $this->course_id = Database::escape_string($crs_params['course']['course_id']);
  106. }
  107. if (!empty($crs_params['course']['course_system'])) {
  108. $this->course_system = $crs_params['course']['course_system'];
  109. }
  110. if (!empty($crs_params['course']['course_title'])) {
  111. $this->course_title = Database::escape_string($crs_params['course']['course_title']);
  112. }
  113. if (!empty($crs_params['course']['course_level'])) {
  114. $this->course_level = $crs_params['course']['course_level'];
  115. }
  116. if (!empty($crs_params['course']['max_fields_cst'])) {
  117. $this->course_max_fields_cst = $crs_params['course']['max_fields_cst'];
  118. }
  119. if (!empty($crs_params['course']['max_fields_ort'])) {
  120. $this->course_max_fields_ort = $crs_params['course']['max_fields_ort'];
  121. }
  122. if (!empty($crs_params['course']['total_aus'])) {
  123. $this->course_total_aus = $crs_params['course']['total_aus'];
  124. }
  125. if (!empty($crs_params['course']['total_blocks'])) {
  126. $this->course_total_blocks = $crs_params['course']['total_blocks'];
  127. }
  128. if (!empty($crs_params['course']['total_objectives'])) {
  129. $this->course_total_objectives = $crs_params['course']['total_objectives'];
  130. }
  131. if (!empty($crs_params['course']['total_complex_objectives'])) {
  132. $this->course_total_complex_objectives = $crs_params['course']['total_complex_objectives'];
  133. }
  134. if (!empty($crs_params['course']['version'])) {
  135. $this->course_version = $crs_params['course']['version'];
  136. }
  137. if (!empty($crs_params['course_description'])) {
  138. $this->course_description = Database::escape_string($crs_params['course_description']);
  139. }
  140. // Parse the Descriptor File (.des) - csv-type.
  141. $des_file = $dir.'/'.$this->config_files['des'];
  142. $des_params = $this->parse_csv_file($des_file);
  143. //echo '<pre>des:'.print_r($des_params, true).'</pre>';
  144. if ($this->debug > 1) {
  145. error_log('New LP - In aicc::parse_config_files() - '.$des_file.' has been parsed', 0);
  146. }
  147. // Distribute des params into the aicc object.
  148. foreach ($des_params as $des) {
  149. // One AU in AICC is equivalent to one SCO in SCORM (scormItem class).
  150. $oDes = new aiccResource('config', $des);
  151. $this->deslist[$oDes->identifier] = $oDes;
  152. }
  153. // Parse the Assignable Unit File (.au) - csv-type.
  154. $au_file = $dir.'/'.$this->config_files['au'];
  155. $au_params = $this->parse_csv_file($au_file);
  156. //echo '<pre>au:'.print_r($au_params, true).'</pre>';
  157. if ($this->debug > 1) {
  158. error_log('New LP - In aicc::parse_config_files() - '.$au_file.' has been parsed', 0);
  159. }
  160. // Distribute au params into the aicc object.
  161. foreach ($au_params as $au) {
  162. $oAu = new aiccItem('config', $au);
  163. $this->aulist[$oAu->identifier] = $oAu;
  164. $this->au_order_list[] = $oAu->identifier;
  165. }
  166. // Parse the Course Structure File (.cst) - csv-type.
  167. $cst_file = $dir.'/'.$this->config_files['cst'];
  168. $cst_params = $this->parse_csv_file($cst_file, ',', '"', true);
  169. //echo '<pre>cst:'.print_r($cst_params, true).'</pre>';
  170. if ($this->debug > 1) {
  171. error_log('New LP - In aicc::parse_config_files() - '.$cst_file.' has been parsed', 0);
  172. }
  173. // Distribute cst params into the aicc object.
  174. foreach ($cst_params as $cst) {
  175. $oCst = new aiccBlock('config', $cst);
  176. $this->cstlist[$oCst->identifier] = $oCst;
  177. }
  178. // Parse the Objectives Relationships File (.ore) - csv-type - if exists.
  179. // TODO: Implement these objectives. For now they're just parsed.
  180. if (!empty($this->config_files['ore'])) {
  181. $ore_file = $dir.'/'.$this->config_files['ore'];
  182. $ore_params = $this->parse_csv_file($ore_file, ',', '"', true);
  183. //echo '<pre>ore:'.print_r($ore_params,true).'</pre>';
  184. if ($this->debug > 1) {
  185. error_log('New LP - In aicc::parse_config_files() - '.$ore_file.' has been parsed', 0);
  186. }
  187. // Distribute ore params into the aicc object.
  188. foreach ($ore_params as $ore) {
  189. $oOre = new aiccObjective('config', $ore);
  190. $this->orelist[$oOre->identifier] = $oOre;
  191. }
  192. }
  193. // Parse the Prerequisites File (.pre) - csv-type - if exists.
  194. if (!empty($this->config_files['pre'])) {
  195. $pre_file = $dir.'/'.$this->config_files['pre'];
  196. $pre_params = $this->parse_csv_file($pre_file);
  197. //echo '<pre>pre:'.print_r($pre_params, true).'</pre>';
  198. if ($this->debug > 1) {
  199. error_log('New LP - In aicc::parse_config_files() - '.$pre_file.' has been parsed', 0);
  200. }
  201. // Distribute pre params into the aicc object.
  202. foreach ($pre_params as $pre) {
  203. // Place a constraint on the corresponding block or AU.
  204. if (in_array(api_strtolower($pre['structure_element']), array_keys($this->cstlist))) {
  205. // If this references a block element:
  206. $this->cstlist[api_strtolower($pre['structure_element'])]->prereq_string = api_strtolower(
  207. $pre['prerequisite']
  208. );
  209. }
  210. if (in_array(api_strtolower($pre['structure_element']), array_keys($this->aulist))) {
  211. // If this references a block element:
  212. $this->aulist[api_strtolower($pre['structure_element'])]->prereq_string = api_strtolower(
  213. $pre['prerequisite']
  214. );
  215. }
  216. }
  217. }
  218. // Parse the Completion Requirements File (.cmp) - csv-type - if exists.
  219. // TODO: Implement this set of requirements (needs database changes).
  220. if (!empty($this->config_files['cmp'])) {
  221. $cmp_file = $dir.'/'.$this->config_files['cmp'];
  222. $cmp_params = $this->parse_csv_file($cmp_file);
  223. //echo '<pre>cmp:'.print_r($cmp_params, true).'</pre>';
  224. if ($this->debug > 1) {
  225. error_log('New LP - In aicc::parse_config_files() - '.$cmp_file.' has been parsed', 0);
  226. }
  227. // Distribute cmp params into the aicc object.
  228. foreach ($cmp_params as $cmp) {
  229. //$oCmp = new aiccCompletionRequirements('config', $cmp);
  230. //$this->cmplist[$oCmp->identifier] =& $oCmp;
  231. }
  232. }
  233. }
  234. return $this->config;
  235. }
  236. /**
  237. * Import the aicc object (as a result from the parse_config_files function) into the database structure
  238. * @param string Unique course code
  239. * @return bool Returns -1 on error
  240. */
  241. function import_aicc($course_code)
  242. {
  243. $course_id = api_get_course_int_id();
  244. if ($this->debug > 0) {
  245. error_log('New LP - In aicc::import_aicc('.$course_code.')', 0);
  246. }
  247. // Get table names.
  248. $new_lp = 'lp';
  249. $new_lp_item = 'lp_item';
  250. // The previous method wasn't safe to get the database name, so do it manually with the course_code.
  251. $sql = "SELECT * FROM ".Database::get_main_table(TABLE_MAIN_COURSE)." WHERE code='$course_code'";
  252. $res = Database::query($sql);
  253. if (Database::num_rows($res) < 1) {
  254. error_log('New LP - Database for '.$course_code.' not found '.__FILE__.' '.__LINE__, 0);
  255. return -1;
  256. }
  257. $row = Database::fetch_array($res);
  258. $new_lp = Database::get_course_table(TABLE_LP_MAIN);
  259. $new_lp_item = Database::get_course_table(TABLE_LP_ITEM);
  260. $get_max = "SELECT MAX(display_order) FROM $new_lp WHERE c_id = $course_id";
  261. $res_max = Database::query($get_max);
  262. if (Database::num_rows($res_max) < 1) {
  263. $dsp = 1;
  264. } else {
  265. $row = Database::fetch_array($res_max);
  266. $dsp = $row[0] + 1;
  267. }
  268. $this->config_encoding = "ISO-8859-1"; // TODO: We may apply detection for this value, see the function api_detect_encoding().
  269. $sql = "INSERT INTO $new_lp (c_id, lp_type, name, ref, description, path, force_commit, default_view_mod, default_encoding, js_lib, content_maker,display_order)".
  270. "VALUES ".
  271. "($course_id, 3, '".$this->course_title."', '".$this->course_id."','".$this->course_description."',".
  272. "'".$this->subdir."', 0, 'embedded', '".$this->config_encoding."',".
  273. "'aicc_api.php','".$this->course_creator."',$dsp)";
  274. if ($this->debug > 2) {
  275. error_log('New LP - In import_aicc(), inserting path: '.$sql, 0);
  276. }
  277. $res = Database::query($sql);
  278. $lp_id = Database::insert_id();
  279. $this->lp_id = $lp_id;
  280. api_item_property_update(
  281. api_get_course_info($course_code),
  282. TOOL_LEARNPATH,
  283. $this->lp_id,
  284. 'LearnpathAdded',
  285. api_get_user_id()
  286. );
  287. api_item_property_update(
  288. api_get_course_info($course_code),
  289. TOOL_LEARNPATH,
  290. $this->lp_id,
  291. 'visible',
  292. api_get_user_id()
  293. );
  294. $previous = 0;
  295. foreach ($this->aulist as $identifier => $dummy) {
  296. $oAu =& $this->aulist[$identifier];
  297. //echo "Item ".$oAu->identifier;
  298. $field_add = '';
  299. $value_add = '';
  300. if (!empty($oAu->masteryscore)) {
  301. $field_add = 'mastery_score, ';
  302. $value_add = $oAu->masteryscore.',';
  303. }
  304. $title = $oAu->identifier;
  305. if (is_object($this->deslist[$identifier])) {
  306. $title = $this->deslist[$identifier]->title;
  307. }
  308. $path = $oAu->path;
  309. //$max_score = $oAu->max_score // TODO: Check if special constraint exists for this item.
  310. //$min_score = $oAu->min_score // TODO: Check if special constraint exists for this item.
  311. $parent = 0; // TODO: Deal with the parent.
  312. $previous = 0;
  313. $prereq = $oAu->prereq_string;
  314. //$previous = (!empty($this->au_order_list_new_id[x]) ? $this->au_order_list_new_id[x] : 0); // TODO: Deal with the previous.
  315. $sql_item = "INSERT INTO $new_lp_item (c_id, lp_id,item_type,ref,title, path,min_score,max_score, $field_add parent_item_id,previous_item_id,next_item_id, prerequisite,display_order) ".
  316. "VALUES ".
  317. "($course_id, $lp_id, 'au','".$oAu->identifier."','".$title."',".
  318. "'$path',0,100, $value_add".
  319. "$parent, $previous, 0, ".
  320. "'$prereq', 0".
  321. ")";
  322. $res_item = Database::query($sql_item);
  323. if ($this->debug > 1) {
  324. error_log('New LP - In aicc::import_aicc() - inserting item : '.$sql_item.' : '.Database::error(), 0);
  325. }
  326. $item_id = Database::insert_id();
  327. // Now update previous item to change next_item_id.
  328. if ($previous != 0) {
  329. $upd = "UPDATE $new_lp_item SET next_item_id = $item_id WHERE c_id = $course_id AND id = $previous";
  330. $upd_res = Database::query($upd);
  331. // Update the previous item id.
  332. }
  333. $previous = $item_id;
  334. }
  335. }
  336. /**
  337. * Intermediate to import_package only to allow import from local zip files
  338. * @param string Path to the zip file, from the dokeos sys root
  339. * @param string Current path (optional)
  340. * @return string Absolute path to the AICC description files or empty string on error
  341. */
  342. function import_local_package($file_path, $current_dir = '')
  343. {
  344. // TODO: Prepare info as given by the $_FILES[''] vector.
  345. $file_info = array();
  346. $file_info['tmp_name'] = $file_path;
  347. $file_info['name'] = basename($file_path);
  348. // Call the normal import_package function.
  349. return $this->import_package($file_info, $current_dir);
  350. }
  351. /**
  352. * Imports a zip file (presumably AICC) into the Chamilo structure
  353. * @param string Zip file info as given by $_FILES['userFile']
  354. * @return string Absolute path to the AICC config files directory or empty string on error
  355. */
  356. function import_package($zip_file_info, $current_dir = '')
  357. {
  358. if ($this->debug > 0) {
  359. error_log('In aicc::import_package('.print_r($zip_file_info, true).',"'.$current_dir.'") method', 0);
  360. }
  361. //ini_set('error_log', 'E_ALL');
  362. $maxFilledSpace = 1000000000;
  363. $zip_file_path = $zip_file_info['tmp_name'];
  364. $zip_file_name = $zip_file_info['name'];
  365. if ($this->debug > 0) {
  366. error_log(
  367. 'New LP - aicc::import_package() - Zip file path = '.$zip_file_path.', zip file name = '.$zip_file_name,
  368. 0
  369. );
  370. }
  371. $course_rel_dir = api_get_course_path().'/scorm'; // Scorm dir web path starting from /courses
  372. $course_sys_dir = api_get_path(SYS_COURSE_PATH).$course_rel_dir; // The absolute system path of this course.
  373. $current_dir = api_replace_dangerous_char(trim($current_dir), 'strict'); // Current dir we are in, inside scorm/
  374. if ($this->debug > 0) {
  375. error_log('New LP - aicc::import_package() - Current_dir = '.$current_dir, 0);
  376. }
  377. //$uploaded_filename = $_FILES['userFile']['name'];
  378. // Get the name of the zip file without the extension.
  379. if ($this->debug > 0) {
  380. error_log('New LP - aicc::import_package() - Received zip file name: '.$zip_file_path, 0);
  381. }
  382. $file_info = pathinfo($zip_file_name);
  383. $filename = $file_info['basename'];
  384. $extension = $file_info['extension'];
  385. $file_base_name = str_replace('.'.$extension, '', $filename); // Filename without its extension.
  386. $this->zipname = $file_base_name; // Save for later in case we don't have a title.
  387. if ($this->debug > 0) {
  388. error_log('New LP - aicc::import_package() - Base file name is : '.$file_base_name, 0);
  389. }
  390. $new_dir = api_replace_dangerous_char(trim($file_base_name), 'strict');
  391. $this->subdir = $new_dir;
  392. if ($this->debug > 0) {
  393. error_log('New LP - aicc::import_package() - Subdir is first set to : '.$this->subdir, 0);
  394. }
  395. /*
  396. if (check_name_exist($course_sys_dir.$current_dir.'/'.$new_dir)) {
  397. $dialogBox = get_lang('FileExists');
  398. $stopping_error = true;
  399. }
  400. */
  401. $zipFile = new PclZip($zip_file_path);
  402. // Check the zip content (real size and file extension).
  403. $zipContentArray = $zipFile->listContent();
  404. $package_type = ''; // The type of the package. Should be 'aicc' after the next few lines.
  405. $package = ''; // The basename of the config files (if 'courses.crs' => 'courses').
  406. $at_root = false; // Check if the config files are at zip root.
  407. $config_dir = ''; // The directory in which the config files are. May remain empty.
  408. $files_found = array();
  409. $subdir_isset = false;
  410. // The following loop should be stopped as soon as we found the right config files (.crs, .au, .des and .cst).
  411. foreach ($zipContentArray as $thisContent) {
  412. if (preg_match('~.(php.*|phtml)$~i', $thisContent['filename'])) {
  413. // If a php file is found, do not authorize (security risk).
  414. if ($this->debug > 1) {
  415. error_log(
  416. 'New LP - aicc::import_package() - Found unauthorized file: '.$thisContent['filename'],
  417. 0
  418. );
  419. }
  420. return api_failure::set_failure('php_file_in_zip_file');
  421. } elseif (preg_match('?.*/aicc/$?', $thisContent['filename'])) {
  422. // If a directory named 'aicc' is found, package type = aicc, but continue,
  423. // because we need to find the right AICC files;
  424. if ($this->debug > 1) {
  425. error_log('New LP - aicc::import_package() - Found aicc directory: '.$thisContent['filename'], 0);
  426. }
  427. $package_type = 'aicc';
  428. } else {
  429. // else, look for one of the files we're searching for (something.crs case insensitive).
  430. $res = array();
  431. if (preg_match('?^(.*)\.(crs|au|des|cst|ore|pre|cmp)$?i', $thisContent['filename'], $res)) {
  432. if ($this->debug > 1) {
  433. error_log(
  434. 'New LP - aicc::import_package() - Found AICC config file: '.$thisContent['filename'].'. Now splitting: '.$res[1].' and '.$res[2],
  435. 0
  436. );
  437. }
  438. if ($thisContent['filename'] == basename($thisContent['filename'])) {
  439. if ($this->debug > 2) {
  440. error_log(
  441. 'New LP - aicc::import_package() - '.$thisContent['filename'].' is at root level',
  442. 0
  443. );
  444. }
  445. $at_root = true;
  446. if (!is_array($files_found[$res[1]])) {
  447. $files_found[$res[1]] = $this->config_exts; // Initialise list of expected extensions (defined in class definition).
  448. }
  449. $files_found[$res[1]][api_strtolower($res[2])] = $thisContent['filename'];
  450. $subdir_isset = true;
  451. } else {
  452. if (!$subdir_isset) {
  453. if (preg_match('?^.*/aicc$?i', dirname($thisContent['filename']))) {
  454. //echo "Cutting subdir<br/>";
  455. $this->subdir .= '/'.substr(dirname($thisContent['filename']), 0, -5);
  456. } else {
  457. //echo "Not cutting subdir<br/>";
  458. $this->subdir .= '/'.dirname($thisContent['filename']);
  459. }
  460. $subdir_isset = true;
  461. }
  462. if ($this->debug > 2) {
  463. error_log(
  464. 'New LP - aicc::import_package() - '.$thisContent['filename'].' is not at root level - recording subdir '.$this->subdir,
  465. 0
  466. );
  467. }
  468. $config_dir = dirname($thisContent['filename']); // Just the relative directory inside scorm/
  469. if (!is_array($files_found[basename($res[1])])) {
  470. $files_found[basename($res[1])] = $this->config_exts;
  471. }
  472. $files_found[basename($res[1])][api_strtolower($res[2])] = basename($thisContent['filename']);
  473. }
  474. $package_type = 'aicc';
  475. } else {
  476. if ($this->debug > 3) {
  477. error_log(
  478. 'New LP - aicc::import_package() - File '.$thisContent['filename'].' didnt match any check',
  479. 0
  480. );
  481. }
  482. }
  483. }
  484. $realFileSize += $thisContent['size'];
  485. }
  486. if ($this->debug > 2) {
  487. error_log('New LP - aicc::import_package() - $files_found: '.print_r($files_found, true), 0);
  488. }
  489. if ($this->debug > 1) {
  490. error_log('New LP - aicc::import_package() - Package type is now '.$package_type, 0);
  491. }
  492. $mandatory = false;
  493. foreach ($files_found as $file_name => $file_exts) {
  494. $temp = (
  495. !empty($files_found[$file_name]['crs'])
  496. AND !empty($files_found[$file_name]['au'])
  497. AND !empty($files_found[$file_name]['des'])
  498. AND !empty($files_found[$file_name]['cst'])
  499. );
  500. if ($temp) {
  501. if ($this->debug > 1) {
  502. error_log('New LP - aicc::import_package() - Found all config files for '.$file_name, 0);
  503. }
  504. $mandatory = true;
  505. $package = $file_name;
  506. // Store base config file name for reuse in parse_config_files().
  507. $this->config_basename = $file_name;
  508. // Store filenames for reuse in parse_config_files().
  509. $this->config_files = $files_found[$file_name];
  510. // Get out, we only want one config files set.
  511. break;
  512. }
  513. }
  514. if ($package_type == '' || !$mandatory) // && defined('CHECK_FOR_AICC') && CHECK_FOR_AICC)
  515. {
  516. return api_failure::set_failure('not_aicc_content');
  517. }
  518. if (!FileManager::enough_size($realFileSize, $course_sys_dir, $maxFilledSpace)) {
  519. return api_failure::set_failure('not_enough_space');
  520. }
  521. // It happens on Linux that $new_dir sometimes doesn't start with '/'
  522. if ($new_dir[0] != '/') {
  523. $new_dir = '/'.$new_dir;
  524. }
  525. // Cut trailing slash.
  526. if ($new_dir[strlen($new_dir) - 1] == '/') {
  527. $new_dir = substr($new_dir, 0, -1);
  528. }
  529. /* Uncompressing phase */
  530. /*
  531. We need to process each individual file in the zip archive to
  532. - add it to the database
  533. - parse & change relative html links
  534. - make sure the filenames are secure (filter funny characters or php extensions)
  535. */
  536. if (is_dir($course_sys_dir.$new_dir) OR @mkdir(
  537. $course_sys_dir.$new_dir,
  538. api_get_permissions_for_new_directories()
  539. )
  540. ) {
  541. // PHP method - slower...
  542. if ($this->debug >= 1) {
  543. error_log('New LP - Changing dir to '.$course_sys_dir.$new_dir, 0);
  544. }
  545. $saved_dir = getcwd();
  546. chdir($course_sys_dir.$new_dir);
  547. $unzippingState = $zipFile->extract();
  548. for ($j = 0; $j < count($unzippingState); $j++) {
  549. $state = $unzippingState[$j];
  550. // TODO: Fix relative links in html files (?)
  551. $extension = strrchr($state["stored_filename"], '.');
  552. //if ($this->debug > 1) { error_log('New LP - found extension '.$extension.' in '.$state['stored_filename'], 0); }
  553. }
  554. if (!empty($new_dir)) {
  555. $new_dir = $new_dir.'/';
  556. }
  557. // Rename files, for example with \\ in it.
  558. if ($dir = @opendir($course_sys_dir.$new_dir)) {
  559. if ($this->debug == 1) {
  560. error_log('New LP - Opened dir '.$course_sys_dir.$new_dir, 0);
  561. }
  562. while ($file = readdir($dir)) {
  563. if ($file != '.' && $file != '..') {
  564. $filetype = 'file';
  565. if (is_dir($course_sys_dir.$new_dir.$file)) {
  566. $filetype = 'folder';
  567. }
  568. // TODO: RENAMING FILES CAN BE VERY DANGEROUS AICC-WISE, avoid that as much as possible!
  569. //$safe_file = replace_dangerous_char($file, 'strict');
  570. $find_str = array('\\', '.php', '.phtml');
  571. $repl_str = array('/', '.txt', '.txt');
  572. $safe_file = str_replace($find_str, $repl_str, $file);
  573. if ($safe_file != $file) {
  574. //@rename($course_sys_dir.$new_dir, $course_sys_dir.'/'.$safe_file);
  575. $mydir = dirname($course_sys_dir.$new_dir.$safe_file);
  576. if (!is_dir($mydir)) {
  577. $mysubdirs = split('/', $mydir);
  578. $mybasedir = '/';
  579. foreach ($mysubdirs as $mysubdir) {
  580. if (!empty($mysubdir)) {
  581. $mybasedir = $mybasedir.$mysubdir.'/';
  582. if (!is_dir($mybasedir)) {
  583. @mkdir($mybasedir, api_get_permissions_for_new_directories());
  584. if ($this->debug == 1) {
  585. error_log('New LP - Dir '.$mybasedir.' doesnt exist. Creating.', 0);
  586. }
  587. }
  588. }
  589. }
  590. }
  591. @rename($course_sys_dir.$new_dir.$file, $course_sys_dir.$new_dir.$safe_file);
  592. if ($this->debug == 1) {
  593. error_log(
  594. 'New LP - Renaming '.$course_sys_dir.$new_dir.$file.' to '.$course_sys_dir.$new_dir.$safe_file,
  595. 0
  596. );
  597. }
  598. }
  599. }
  600. }
  601. closedir($dir);
  602. chdir($saved_dir);
  603. }
  604. } else {
  605. return '';
  606. }
  607. return $course_sys_dir.$new_dir.$config_dir;
  608. }
  609. /**
  610. * Sets the proximity setting in the database
  611. * @param string Proximity setting
  612. */
  613. function set_proximity($proxy = '')
  614. {
  615. $course_id = api_get_course_int_id();
  616. if ($this->debug > 0) {
  617. error_log('In aicc::set_proximity('.$proxy.') method', 0);
  618. }
  619. $lp = $this->get_id();
  620. if ($lp != 0) {
  621. $tbl_lp = Database::get_course_table(TABLE_LP_MAIN);
  622. $sql = "UPDATE $tbl_lp SET content_local = '$proxy' WHERE c_id = ".$course_id." id = ".$lp;
  623. $res = Database::query($sql);
  624. return $res;
  625. } else {
  626. return false;
  627. }
  628. }
  629. /**
  630. * Sets the theme setting in the database
  631. * @param string Theme setting
  632. */
  633. function set_theme($theme = '')
  634. {
  635. $course_id = api_get_course_int_id();
  636. if ($this->debug > 0) {
  637. error_log('In aicc::set_theme('.$theme.') method', 0);
  638. }
  639. $lp = $this->get_id();
  640. if ($lp != 0) {
  641. $tbl_lp = Database::get_course_table(TABLE_LP_MAIN);
  642. $sql = "UPDATE $tbl_lp SET theme = '$theme' WHERE c_id = ".$course_id." id = ".$lp;
  643. $res = Database::query($sql);
  644. return $res;
  645. } else {
  646. return false;
  647. }
  648. }
  649. /**
  650. * Sets the image LP in the database
  651. * @param string Theme setting
  652. */
  653. function set_preview_image($preview_image = '')
  654. {
  655. $course_id = api_get_course_int_id();
  656. if ($this->debug > 0) {
  657. error_log('In aicc::set_preview_image('.$preview_image.') method', 0);
  658. }
  659. $lp = $this->get_id();
  660. if ($lp != 0) {
  661. $tbl_lp = Database::get_course_table(TABLE_LP_MAIN);
  662. $sql = "UPDATE $tbl_lp SET preview_image = '$preview_image' WHERE c_id = ".$course_id." id = ".$lp;
  663. $res = Database::query($sql);
  664. return $res;
  665. } else {
  666. return false;
  667. }
  668. }
  669. /**
  670. * Sets the Author LP in the database
  671. * @param string Theme setting
  672. */
  673. function set_author($author = '')
  674. {
  675. $course_id = api_get_course_int_id();
  676. if ($this->debug > 0) {
  677. error_log('In aicc::set_author('.$author.') method', 0);
  678. }
  679. $lp = $this->get_id();
  680. if ($lp != 0) {
  681. $tbl_lp = Database::get_course_table(TABLE_LP_MAIN);
  682. $sql = "UPDATE $tbl_lp SET author = '$author' WHERE c_id = ".$course_id." id = ".$lp;
  683. $res = Database::query($sql);
  684. return $res;
  685. } else {
  686. return false;
  687. }
  688. }
  689. /**
  690. * Sets the content maker setting in the database
  691. * @param string Proximity setting
  692. */
  693. function set_maker($maker = '')
  694. {
  695. $course_id = api_get_course_int_id();
  696. if ($this->debug > 0) {
  697. error_log('In aicc::set_maker method('.$maker.')', 0);
  698. }
  699. $lp = $this->get_id();
  700. if ($lp != 0) {
  701. $tbl_lp = Database::get_course_table(TABLE_LP_MAIN);
  702. $sql = "UPDATE $tbl_lp SET content_maker = '$maker' WHERE c_id = ".$course_id." id = ".$lp;
  703. $res = Database::query($sql);
  704. return $res;
  705. } else {
  706. return false;
  707. }
  708. }
  709. /**
  710. * Exports the current AICC object's files as a zip. Excerpts taken from learnpath_functions.inc.php::exportpath()
  711. * @param integer Learnpath ID (optional, taken from object context if not defined)
  712. */
  713. function export_zip($lp_id = null)
  714. {
  715. if ($this->debug > 0) {
  716. error_log('In aicc::export_zip method('.$lp_id.')', 0);
  717. }
  718. if (empty($lp_id)) {
  719. if (!is_object($this)) {
  720. return false;
  721. } else {
  722. $id = $this->get_id();
  723. if (empty($id)) {
  724. return false;
  725. } else {
  726. $lp_id = $this->get_id();
  727. }
  728. }
  729. }
  730. //error_log('New LP - in export_zip()', 0);
  731. // Zip everything that is in the corresponding scorm dir.
  732. // Write the zip file somewhere (might be too big to return).
  733. require_once 'learnpath_functions.inc.php';
  734. $course_id = api_get_course_int_id();
  735. $tbl_lp = Database::get_course_table(TABLE_LP_MAIN);
  736. $_course = api_get_course_info(api_get_course_id());
  737. $sql = "SELECT * FROM $tbl_lp WHERE c_id = ".$course_id." id=".$lp_id;
  738. $result = Database::query($sql);
  739. $row = Database::fetch_array($result);
  740. $LPname = $row['path'];
  741. $list = split('/', $LPname);
  742. $LPnamesafe = $list[0];
  743. //$zipfoldername = '/tmp';
  744. //$zipfoldername = '../../courses/'.$_course['directory'].'/temp/'.$LPnamesafe;
  745. $zipfoldername = api_get_path(SYS_COURSE_PATH).$_course['directory'].'/temp/'.$LPnamesafe;
  746. $scormfoldername = api_get_path(SYS_COURSE_PATH).$_course['directory'].'/scorm/'.$LPnamesafe;
  747. $zipfilename = $zipfoldername.'/'.$LPnamesafe.'.zip';
  748. // Get a temporary dir for creating the zip file.
  749. //error_log('New LP - cleaning dir '.$zipfoldername, 0);
  750. deldir($zipfoldername); //make sure the temp dir is cleared
  751. mkdir($zipfoldername, api_get_permissions_for_new_directories());
  752. //error_log('New LP - made dir '.$zipfoldername, 0);
  753. // Create zipfile of given directory.
  754. $zip_folder = new PclZip($zipfilename);
  755. $zip_folder->create($scormfoldername.'/', PCLZIP_OPT_REMOVE_PATH, $scormfoldername.'/');
  756. //$zipfilename = '/var/www/dokeos-comp/courses/TEST2/scorm/example_document.html';
  757. //this file sending implies removing the default mime-type from php.ini
  758. //DocumentManager :: file_send_for_download($zipfilename, true, $LPnamesafe.".zip");
  759. DocumentManager :: file_send_for_download($zipfilename, true);
  760. // Delete the temporary zip file and directory in fileManager.lib.php
  761. FileManager::my_delete($zipfilename);
  762. FileManager::my_delete($zipfoldername);
  763. return true;
  764. }
  765. /**
  766. * Gets a resource's path if available, otherwise return empty string
  767. * @param string Resource ID as used in resource array
  768. * @return string The resource's path as declared in config file course.crs
  769. */
  770. function get_res_path($id)
  771. {
  772. if ($this->debug > 0) {
  773. error_log('In aicc::get_res_path('.$id.') method', 0);
  774. }
  775. $path = '';
  776. if (isset($this->resources[$id])) {
  777. $oRes =& $this->resources[$id];
  778. $path = @$oRes->get_path();
  779. }
  780. return $path;
  781. }
  782. /**
  783. * Gets a resource's type if available, otherwise return empty string
  784. * @param string Resource ID as used in resource array
  785. * @return string The resource's type as declared in the assignable unit (.au) file
  786. */
  787. function get_res_type($id)
  788. {
  789. if ($this->debug > 0) {
  790. error_log('In aicc::get_res_type('.$id.') method', 0);
  791. }
  792. $type = '';
  793. if (isset($this->resources[$id])) {
  794. $oRes =& $this->resources[$id];
  795. $temptype = $oRes->get_scorm_type();
  796. if (!empty($temptype)) {
  797. $type = $temptype;
  798. }
  799. }
  800. return $type;
  801. }
  802. /**
  803. * Gets the default organisation's title
  804. * @return string The organization's title
  805. */
  806. function get_title()
  807. {
  808. if ($this->debug > 0) {
  809. error_log('In aicc::get_title() method', 0);
  810. }
  811. $title = '';
  812. if (isset($this->config['organizations']['default'])) {
  813. $title = $this->organizations[$this->config['organizations']['default']]->get_name();
  814. } elseif (count($this->organizations) == 1) {
  815. // This will only get one title but so we don't need to know the index.
  816. foreach ($this->organizations as $id => $value) {
  817. $title = $this->organizations[$id]->get_name();
  818. break;
  819. }
  820. }
  821. return $title;
  822. }
  823. /**
  824. * TODO: Implement this function to restore items data from a set of AICC config files,
  825. * updating the existing table... This will prove very useful in case initial data
  826. * from config files were not imported well enough.
  827. */
  828. function reimport_aicc()
  829. {
  830. if ($this->debug > 0) {
  831. error_log('In aicc::reimport_aicc() method', 0);
  832. }
  833. //query current items list
  834. //get the identifiers
  835. //parse the config files
  836. //match both
  837. //update DB accordingly
  838. return true;
  839. }
  840. /**
  841. * Static function to parse AICC ini files.
  842. * Based on work by sinedeo at gmail dot com published on php.net (parse_ini_file()).
  843. * @param string File path
  844. * @return array Structured array
  845. */
  846. function parse_ini_file_quotes_safe($f)
  847. {
  848. $null = '';
  849. $r = $null;
  850. $sec = $null;
  851. $f = @file_get_contents($f);
  852. $f = api_convert_encoding($f, api_get_system_encoding(), $this->config_encoding);
  853. $f = preg_split('/\r?\n/', $f);
  854. for ($i = 0; $i < @count($f); $i++) {
  855. $newsec = 0;
  856. $w = @trim($f[$i]);
  857. if ($w) {
  858. if ((!$r) or ($sec)) {
  859. if ((@substr($w, 0, 1) == '[') and (@substr($w, -1, 1)) == ']') {
  860. $sec = @substr($w, 1, @strlen($w) - 2);
  861. $newsec = 1;
  862. }
  863. }
  864. if (!$newsec) {
  865. $w = @explode('=', $w);
  866. $k = @trim($w[0]);
  867. unset($w[0]);
  868. $v = @trim(@implode('=', $w));
  869. if ((@substr($v, 0, 1) == "\"") and (@substr($v, -1, 1) == "\"")) {
  870. $v = @substr($v, 1, @strlen($v) - 2);
  871. }
  872. if ($sec) {
  873. if (api_strtolower($sec) == 'course_description') { // A special case.
  874. $r[api_strtolower($sec)] = $k;
  875. } else {
  876. $r[api_strtolower($sec)][api_strtolower($k)] = $v;
  877. }
  878. } else {
  879. $r[api_strtolower($k)] = $v;
  880. }
  881. }
  882. }
  883. }
  884. return $r;
  885. }
  886. /**
  887. * Static function to parse AICC ini strings.
  888. * Based on work by sinedeo at gmail dot com published on php.net (parse_ini_file()).
  889. * @param string INI File string
  890. * @param array List of names of sections that should be considered as containing only hard string data (no variables), provided in lower case
  891. * @return array Structured array
  892. */
  893. function parse_ini_string_quotes_safe($s, $pure_strings = array())
  894. {
  895. $null = '';
  896. $r = $null;
  897. $sec = $null;
  898. $s = api_convert_encoding($s, api_get_system_encoding(), $this->config_encoding);
  899. //$f = split("\r\n", $s);
  900. $f = preg_split('/\r?\n/', $s);
  901. for ($i = 0; $i < @count($f); $i++) {
  902. $newsec = 0;
  903. $w = @trim($f[$i]);
  904. if ($w) {
  905. if ((!$r) or ($sec)) {
  906. if ((@substr($w, 0, 1) == '[') and (@substr($w, -1, 1)) == ']') {
  907. $sec = @substr($w, 1, @strlen($w) - 2);
  908. $pure_data = 0;
  909. if (in_array(api_strtolower($sec), $pure_strings)) {
  910. // This section can only be considered as pure string data (until the next section).
  911. $pure_data = 1;
  912. $r[api_strtolower($sec)] = '';
  913. }
  914. $newsec = 1;
  915. }
  916. }
  917. if (!$newsec) {
  918. $w = @explode('=', $w);
  919. $k = @trim($w[0]);
  920. unset($w[0]);
  921. $v = @trim(@implode('=', $w));
  922. if ((@substr($v, 0, 1) == "\"") and (@substr($v, -1, 1) == "\"")) {
  923. $v = @substr($v, 1, @strlen($v) - 2);
  924. }
  925. if ($sec) {
  926. if ($pure_data) {
  927. $r[api_strtolower($sec)] .= $f[$i];
  928. } else {
  929. if (api_strtolower($sec) == 'course_description') { // A special case.
  930. $r[api_strtolower($sec)] = $k;
  931. } else {
  932. $r[api_strtolower($sec)][api_strtolower($k)] = $v;
  933. }
  934. }
  935. } else {
  936. $r[api_strtolower($k)] = $v;
  937. }
  938. }
  939. }
  940. }
  941. return $r;
  942. }
  943. /**
  944. * Static function that parses CSV files into simple arrays, based on a function
  945. * by spam at cyber-space dot nl published on php.net (fgetcsv()).
  946. * @param string Filepath
  947. * @param string CSV delimiter
  948. * @param string CSV enclosure
  949. * @param boolean Might one field name happen more than once on the same line? (then split by comma in the values)
  950. * @return array Simple structured array
  951. */
  952. function parse_csv_file($f, $delim = ',', $enclosure = '"', $multiples = false)
  953. {
  954. $data = @file_get_contents($f);
  955. $data = api_convert_encoding($data, api_get_system_encoding(), $this->config_encoding);
  956. $enclosed = false;
  957. $fldcount = 0;
  958. $linecount = 0;
  959. $fldval = '';
  960. for ($i = 0; $i < strlen($data); $i++) {
  961. $chr = $data{$i};
  962. switch ($chr) {
  963. case $enclosure:
  964. if ($enclosed && $data{$i + 1} == $enclosure) {
  965. $fldval .= $chr;
  966. ++$i; // Skip the next character.
  967. } else {
  968. $enclosed = !$enclosed;
  969. }
  970. break;
  971. case $delim:
  972. if (!$enclosed) {
  973. $ret_array[$linecount][$fldcount++] = $fldval;
  974. $fldval = '';
  975. } else {
  976. $fldval .= $chr;
  977. }
  978. break;
  979. case "\r":
  980. if (!$enclosed && $data{$i + 1} == "\n") {
  981. continue;
  982. }
  983. case "\n":
  984. if (!$enclosed) {
  985. $ret_array[$linecount++][$fldcount] = $fldval;
  986. $fldcount = 0;
  987. $fldval = '';
  988. } else {
  989. $fldval .= $chr;
  990. }
  991. break;
  992. case "\\r":
  993. if (!$enclosed && $data{$i + 1} == "\\n") {
  994. continue;
  995. }
  996. case "\\n":
  997. if (!$enclosed) {
  998. $ret_array[$linecount++][$fldcount] = $fldval;
  999. $fldcount = 0;
  1000. $fldval = '';
  1001. } else {
  1002. $fldval .= $chr;
  1003. }
  1004. break;
  1005. default:
  1006. $fldval .= $chr;
  1007. }
  1008. }
  1009. if ($fldval) {
  1010. $ret_array[$linecount][$fldcount] = $fldval;
  1011. }
  1012. // Transform the array to use the first line as titles.
  1013. $titles = array();
  1014. $ret_ret_array = array();
  1015. foreach ($ret_array as $line_idx => $line) {
  1016. if ($line_idx == 0) {
  1017. $titles = $line;
  1018. } else {
  1019. $ret_ret_array[$line_idx] = array();
  1020. foreach ($line as $idx => $val) {
  1021. if ($multiples && !empty($ret_ret_array[$line_idx][api_strtolower($titles[$idx])])) {
  1022. $ret_ret_array[$line_idx][api_strtolower($titles[$idx])] .= ','.$val;
  1023. } else {
  1024. $ret_ret_array[$line_idx][api_strtolower($titles[$idx])] = $val;
  1025. }
  1026. }
  1027. }
  1028. }
  1029. return $ret_ret_array;
  1030. }
  1031. }