Task #1765 - LP tool, cleaning files (1).

skala
Ivan Tcholakov 15 years ago
parent f4adb7c366
commit 643a575db9
  1. 367
      main/newscorm/aicc.class.php
  2. 21
      main/newscorm/aiccBlock.class.php
  3. 31
      main/newscorm/aiccItem.class.php
  4. 20
      main/newscorm/aiccObjective.class.php
  5. 19
      main/newscorm/aiccResource.class.php
  6. 52
      main/newscorm/aicc_api.php
  7. 47
      main/newscorm/aicc_hacp.php
  8. 6
      main/newscorm/js/api_wrapper.js

@ -1,11 +1,13 @@
<?php
/* For licensing terms, see /license.txt */
/**
* Defines the AICC class, which is meant to contain the aicc items (nuclear elements)
* @package chamilo.learnpath
* @author Yannick Warnier <ywarnier@beeznest.org>
* @license GNU/GPL
*/
require_once 'aiccItem.class.php';
//require_once 'aiccMetadata.class.php';
//require_once 'aiccOrganization.class.php';
@ -15,7 +17,7 @@ require_once 'aiccBlock.class.php';
class aicc extends learnpath {
public $config = array();
public $config_basename = ''; //the configuration files might be multiple and might have
public $config_basename = ''; // The configuration files might be multiple and might have
// funny names. We need to keep the name of that file while we
// install the content.
public $config_files = array();
@ -35,11 +37,12 @@ class aicc extends learnpath {
public $cstlist = array();
public $orelist = array();
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
public $zipname = ''; //keeps the zipfile safe for the object's life so that we can use it if no title avail
public $lastzipnameindex = 0; //keeps an index of the number of uses of the zipname so far
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.
public $zipname = ''; // Keeps the zipfile safe for the object's life so that we can use it if there is no title available.
public $lastzipnameindex = 0; // Keeps an index of the number of uses of the zipname so far.
public $config_encoding = 'ISO-8859-1';
public $debug = 0;
/**
* Class constructor. Based on the parent constructor.
* @param string Course code
@ -48,48 +51,45 @@ class aicc extends learnpath {
*/
public function __construct($course_code = null, $resource_id = null, $user_id = null) {
if ($this->debug > 0) { error_log('In aicc::aicc()', 0); }
if(!empty($course_code) and !empty($resource_id) and !empty($user_id))
{
if (!empty($course_code) && !empty($resource_id) && !empty($user_id)) {
parent::__construct($course_code, $resource_id, $user_id);
} else {
//do nothing but still build the aicc object
}
}
/**
* Opens a resource
* @param integer Database ID of the resource
*/
public function open($id)
{
public function open($id) {
if ($this->debug > 0) { error_log('In aicc::open()', 0); }
// redefine parent method
// Redefine parent method.
}
/**
* Parses a set of AICC config files and puts everything into the $config array
* @param string Path to the config files dir on the system. If not defined, uses the base path of the course's scorm dir
* @return array Structured array representing the config files' contents
*/
function parse_config_files($dir='')
{
function parse_config_files($dir = '') {
if ($this->debug > 0) {error_log('New LP - In aicc::parse_config_files('.$dir.')', 0); }
if (empty($dir)) {
//get the path of the AICC config files dir
// Get the path of the AICC config files dir.
$dir = $this->subdir;
}
if(is_dir($dir) and is_readable($dir))
{
// Now go through all the config files one by one and parse everything into
// AICC objects.
if (is_dir($dir) && is_readable($dir)) {
// Now go through all the config files one by one and parse everything into AICC objects.
// The basename for the config files is stored in $this->config_basename
// The basename for the config files is stored in $this->config_basename.
// Parse the Course Description File (.crs) - ini-type
// Parse the Course Description File (.crs) - ini-type.
$crs_file = $dir.'/'.$this->config_files['crs'];
$crs_params = $this->parse_ini_file_quotes_safe($crs_file);
//echo '<pre>crs:'.print_r($crs_params, true).'</pre>';
if ($this->debug > 1) { error_log('New LP - In aicc::parse_config_files() - '.$crs_file.' has been parsed', 0); }
//CRS distribute crs params into the aicc object
// CRS distribute crs params into the aicc object.
if (!empty($crs_params['course']['course_creator'])) {
$this->course_creator = Database::escape_string($crs_params['course']['course_creator']);
}
@ -130,83 +130,83 @@ class aicc extends learnpath {
$this->course_description = Database::escape_string($crs_params['course_description']);
}
// Parse the Descriptor File (.des) - csv-type
// Parse the Descriptor File (.des) - csv-type.
$des_file = $dir.'/'.$this->config_files['des'];
$des_params = $this->parse_csv_file($des_file);
//echo '<pre>des:'.print_r($des_params, true).'</pre>';
if ($this->debug > 1) { error_log('New LP - In aicc::parse_config_files() - '.$des_file.' has been parsed', 0); }
//distribute des params into the aicc object
// Distribute des params into the aicc object.
foreach ($des_params as $des){
//one AU in AICC is equivalent to one SCO in SCORM (scormItem class)
// One AU in AICC is equivalent to one SCO in SCORM (scormItem class).
$oDes = new aiccResource('config', $des);
$this->deslist[$oDes->identifier] = $oDes;
}
// Parse the Assignable Unit File (.au) - csv-type
// Parse the Assignable Unit File (.au) - csv-type.
$au_file = $dir.'/'.$this->config_files['au'];
$au_params = $this->parse_csv_file($au_file);
//echo '<pre>au:'.print_r($au_params, true).'</pre>';
if ($this->debug > 1) { error_log('New LP - In aicc::parse_config_files() - '.$au_file.' has been parsed', 0); }
//distribute au params into the aicc object
// Distribute au params into the aicc object.
foreach ($au_params as $au) {
$oAu = new aiccItem('config', $au);
$this->aulist[$oAu->identifier] = $oAu;
$this->au_order_list[] = $oAu->identifier;
}
// Parse the Course Structure File (.cst) - csv-type
// Parse the Course Structure File (.cst) - csv-type.
$cst_file = $dir.'/'.$this->config_files['cst'];
$cst_params = $this->parse_csv_file($cst_file, ',', '"', true);
//echo '<pre>cst:'.print_r($cst_params, true).'</pre>';
if ($this->debug > 1) { error_log('New LP - In aicc::parse_config_files() - '.$cst_file.' has been parsed', 0); }
//distribute cst params into the aicc object
// Distribute cst params into the aicc object.
foreach ($cst_params as $cst) {
$oCst = new aiccBlock('config', $cst);
$this->cstlist[$oCst->identifier] = $oCst;
}
// Parse the Objectives Relationships File (.ore) - csv-type - if exists
//TODO @TODO implement these objectives. For now they're just parsed
// Parse the Objectives Relationships File (.ore) - csv-type - if exists.
// TODO: Implement these objectives. For now they're just parsed.
if (!empty($this->config_files['ore'])) {
$ore_file = $dir.'/'.$this->config_files['ore'];
$ore_params = $this->parse_csv_file($ore_file, ',', '"', true);
//echo '<pre>ore:'.print_r($ore_params,true).'</pre>';
if ($this->debug > 1) { error_log('New LP - In aicc::parse_config_files() - '.$ore_file.' has been parsed', 0); }
//distribute ore params into the aicc object
// Distribute ore params into the aicc object.
foreach ($ore_params as $ore) {
$oOre = new aiccObjective('config', $ore);
$this->orelist[$oOre->identifier] = $oOre;
}
}
// Parse the Prerequisites File (.pre) - csv-type - if exists
// Parse the Prerequisites File (.pre) - csv-type - if exists.
if (!empty($this->config_files['pre'])) {
$pre_file = $dir.'/'.$this->config_files['pre'];
$pre_params = $this->parse_csv_file($pre_file);
//echo '<pre>pre:'.print_r($pre_params, true).'</pre>';
if ($this->debug > 1) { error_log('New LP - In aicc::parse_config_files() - '.$pre_file.' has been parsed', 0); }
//distribute pre params into the aicc object
// Distribute pre params into the aicc object.
foreach ($pre_params as $pre) {
//place a constraint on the corresponding block or AU
// Place a constraint on the corresponding block or AU.
if (in_array(strtolower($pre['structure_element']), array_keys($this->cstlist))) {
//if this references a block element
// If this references a block element:
$this->cstlist[strtolower($pre['structure_element'])]->prereq_string = strtolower($pre['prerequisite']);
}
if (in_array(strtolower($pre['structure_element']), array_keys($this->aulist))) {
//if this references a block element
// If this references a block element:
$this->aulist[strtolower($pre['structure_element'])]->prereq_string = strtolower($pre['prerequisite']);
}
}
}
// Parse the Completion Requirements File (.cmp) - csv-type - if exists
//TODO @TODO implement this set of requirements (needs database changes)
// Parse the Completion Requirements File (.cmp) - csv-type - if exists.
// TODO: Implement this set of requirements (needs database changes).
if (!empty($this->config_files['cmp'])) {
$cmp_file = $dir.'/'.$this->config_files['cmp'];
$cmp_params = $this->parse_csv_file($cmp_file);
//echo '<pre>cmp:'.print_r($cmp_params, true).'</pre>';
if ($this->debug > 1) { error_log('New LP - In aicc::parse_config_files() - '.$cmp_file.' has been parsed', 0); }
//distribute cmp params into the aicc object
// Distribute cmp params into the aicc object.
foreach ($cmp_params as $cmp) {
//$oCmp = new aiccCompletionRequirements('config', $cmp);
//$this->cmplist[$oCmp->identifier] =& $oCmp;
@ -215,6 +215,7 @@ class aicc extends learnpath {
}
return $this->config;
}
/**
* Import the aicc object (as a result from the parse_config_files function) into the database structure
* @param string Unique course code
@ -222,11 +223,11 @@ class aicc extends learnpath {
*/
function import_aicc($course_code) {
if ($this->debug > 0) { error_log('New LP - In aicc::import_aicc('.$course_code.')', 0); }
//get table names
// Get table names.
$new_lp = 'lp';
$new_lp_item = 'lp_item';
//The previous method wasn't safe to get the database name, so do it manually with the course_code
// The previous method wasn't safe to get the database name, so do it manually with the course_code.
$sql = "SELECT * FROM ".Database::get_main_table(TABLE_MAIN_COURSE)." WHERE code='$course_code'";
$res = Database::query($sql);
if (Database::num_rows($res) < 1) { error_log('New LP - Database for '.$course_code.' not found '.__FILE__.' '.__LINE__, 0); return -1; }
@ -262,8 +263,7 @@ class aicc extends learnpath {
api_item_property_update(api_get_course_info($course_code), TOOL_LEARNPATH, $this->lp_id, 'visible', api_get_user_id());
$previous = 0;
foreach($this->aulist as $identifier => $dummy)
{
foreach ($this->aulist as $identifier => $dummy) {
$oAu =& $this->aulist[$identifier];
//echo "Item ".$oAu->identifier;
$field_add = '';
@ -277,12 +277,12 @@ class aicc extends learnpath {
$title = $this->deslist[$identifier]->title;
}
$path = $oAu->path;
//$max_score = $oAu->max_score //TODO check if special constraint exists for this item
//$min_score = $oAu->min_score //TODO check if special constraint exists for this item
$parent = 0; //TODO deal with parent
//$max_score = $oAu->max_score // TODO: Check if special constraint exists for this item.
//$min_score = $oAu->min_score // TODO: Check if special constraint exists for this item.
$parent = 0; // TODO: Deal with the parent.
$previous = 0;
$prereq = $oAu->prereq_string;
//$previous = (!empty($this->au_order_list_new_id[x])?$this->au_order_list_new_id[x]:0); //TODO deal with previous
//$previous = (!empty($this->au_order_list_new_id[x]) ? $this->au_order_list_new_id[x] : 0); // TODO: Deal with the previous.
$sql_item = "INSERT INTO $new_lp_item " .
"(lp_id,item_type,ref,title," .
"path,min_score,max_score, $field_add" .
@ -297,37 +297,37 @@ class aicc extends learnpath {
$res_item = Database::query($sql_item);
if ($this->debug > 1) { error_log('New LP - In aicc::import_aicc() - inserting item : '.$sql_item.' : '.Database::error(), 0); }
$item_id = Database::insert_id();
//now update previous item to change next_item_id
// Now update previous item to change next_item_id.
if ($previous != 0) {
$upd = "UPDATE $new_lp_item SET next_item_id = $item_id WHERE id = $previous";
$upd_res = Database::query($upd);
//update previous item id
// Update the previous item id.
}
$previous = $item_id;
}
}
/**
* Intermediate to import_package only to allow import from local zip files
* @param string Path to the zip file, from the dokeos sys root
* @param string Current path (optional)
* @return string Absolute path to the AICC description files or empty string on error
*/
function import_local_package($file_path,$current_dir='')
{
//todo prepare info as given by the $_FILES[''] vector
function import_local_package($file_path, $current_dir = '') {
// TODO: Prepare info as given by the $_FILES[''] vector.
$file_info = array();
$file_info['tmp_name'] = $file_path;
$file_info['name'] = basename($file_path);
//call the normal import_package function
// Call the normal import_package function.
return $this->import_package($file_info,$current_dir);
}
/**
* Imports a zip file (presumably AICC) into the Dokeos structure
* @param string Zip file info as given by $_FILES['userFile']
* @return string Absolute path to the AICC config files directory or empty string on error
*/
function import_package($zip_file_info,$current_dir = '')
{
function import_package($zip_file_info, $current_dir = '') {
if ($this->debug > 0) { error_log('In aicc::import_package('.print_r($zip_file_info, true).',"'.$current_dir.'") method', 0); }
//ini_set('error_log', 'E_ALL');
$maxFilledSpace = 1000000000;
@ -335,19 +335,19 @@ class aicc extends learnpath {
$zip_file_name = $zip_file_info['name'];
if ($this->debug > 0) { error_log('New LP - aicc::import_package() - Zip file path = '.$zip_file_path.', zip file name = '.$zip_file_name, 0); }
$course_rel_dir = api_get_course_path().'/scorm'; //scorm dir web path starting from /courses
$course_sys_dir = api_get_path(SYS_COURSE_PATH).$course_rel_dir; //absolute system path for this course
$current_dir = replace_dangerous_char(trim($current_dir),'strict'); //current dir we are in, inside scorm/
$course_rel_dir = api_get_course_path().'/scorm'; // Scorm dir web path starting from /courses
$course_sys_dir = api_get_path(SYS_COURSE_PATH).$course_rel_dir; // The absolute system path of this course.
$current_dir = replace_dangerous_char(trim($current_dir),'strict'); // Current dir we are in, inside scorm/
if ($this->debug > 0) { error_log('New LP - aicc::import_package() - Current_dir = '.$current_dir, 0); }
//$uploaded_filename = $_FILES['userFile']['name'];
//get name of the zip file without the extension
// Get the name of the zip file without the extension.
if ($this->debug > 0) { error_log('New LP - aicc::import_package() - Received zip file name: '.$zip_file_path, 0); }
$file_info = pathinfo($zip_file_name);
$filename = $file_info['basename'];
$extension = $file_info['extension'];
$file_base_name = str_replace('.'.$extension,'',$filename); //filename without its extension
$this->zipname = $file_base_name; //save for later in case we don't have a title
$file_base_name = str_replace('.'.$extension, '', $filename); // Filename without its extension.
$this->zipname = $file_base_name; // Save for later in case we don't have a title.
if ($this->debug > 0) { error_log('New LP - aicc::import_package() - Base file name is : '.$file_base_name, 0); }
$new_dir = replace_dangerous_char(trim($file_base_name),'strict');
@ -355,48 +355,44 @@ class aicc extends learnpath {
if($this->debug > 0) { error_log('New LP - aicc::import_package() - Subdir is first set to : '.$this->subdir, 0); }
/*
if( check_name_exist($course_sys_dir.$current_dir."/".$new_dir) )
{
if (check_name_exist($course_sys_dir.$current_dir.'/'.$new_dir)) {
$dialogBox = get_lang('FileExists');
$stopping_error = true;
}
*/
$zipFile = new pclZip($zip_file_path);
// Check the zip content (real size and file extension)
// Check the zip content (real size and file extension).
$zipContentArray = $zipFile->listContent();
$package_type=''; //the type of the package. Should be 'aicc' after the next few lines
$package = ''; //the basename of the config files (if 'courses.crs' => 'courses')
$at_root = false; //check if the config files are at zip root
$config_dir = ''; //the directory in which the config files are. May remain empty
$package_type = ''; // The type of the package. Should be 'aicc' after the next few lines.
$package = ''; // The basename of the config files (if 'courses.crs' => 'courses').
$at_root = false; // Check if the config files are at zip root.
$config_dir = ''; // The directory in which the config files are. May remain empty.
$files_found = array();
$subdir_isset = false;
//the following loop should be stopped as soon as we found the right config files (.crs, .au, .des and .cst)
foreach($zipContentArray as $thisContent)
{
if ( preg_match('~.(php.*|phtml)$~i', $thisContent['filename']) )
{
//if a php file is found, do not authorize (security risk)
// The following loop should be stopped as soon as we found the right config files (.crs, .au, .des and .cst).
foreach ($zipContentArray as $thisContent) {
if (preg_match('~.(php.*|phtml)$~i', $thisContent['filename'])) {
// If a php file is found, do not authorize (security risk).
if ($this->debug > 1) {error_log('New LP - aicc::import_package() - Found unauthorized file: '.$thisContent['filename'], 0); }
return api_failure::set_failure('php_file_in_zip_file');
} elseif (preg_match('?.*/aicc/$?', $thisContent['filename'])) {
//if a directory named 'aicc' is found, package type = aicc, but continue
//because we need to find the right AICC files
// If a directory named 'aicc' is found, package type = aicc, but continue,
// because we need to find the right AICC files;
if ($this->debug > 1) { error_log('New LP - aicc::import_package() - Found aicc directory: '.$thisContent['filename'], 0); }
$package_type = 'aicc';
} else {
//else, look for one of the files we're searching for (something.crs case insensitive)
// else, look for one of the files we're searching for (something.crs case insensitive).
$res = array();
if(preg_match('?^(.*)\.(crs|au|des|cst|ore|pre|cmp)$?i',$thisContent['filename'],$res))
{
if (preg_match('?^(.*)\.(crs|au|des|cst|ore|pre|cmp)$?i', $thisContent['filename'], $res)) {
if ($this->debug > 1) { error_log('New LP - aicc::import_package() - Found AICC config file: '.$thisContent['filename'].'. Now splitting: '.$res[1].' and '.$res[2], 0); }
if ($thisContent['filename'] == basename($thisContent['filename'])) {
if ($this->debug > 2) { error_log('New LP - aicc::import_package() - '.$thisContent['filename'].' is at root level', 0); }
$at_root = true;
if (!is_array($files_found[$res[1]])) {
$files_found[$res[1]] = $this->config_exts; //initialise list of expected extensions (defined in class definition)
$files_found[$res[1]] = $this->config_exts; // Initialise list of expected extensions (defined in class definition).
}
$files_found[$res[1]][strtolower($res[2])] = $thisContent['filename'];
$subdir_isset = true;
@ -412,7 +408,7 @@ class aicc extends learnpath {
$subdir_isset = true;
}
if ($this->debug > 2) { error_log('New LP - aicc::import_package() - '.$thisContent['filename'].' is not at root level - recording subdir '.$this->subdir, 0); }
$config_dir = dirname($thisContent['filename']); //just the relative directory inside scorm/
$config_dir = dirname($thisContent['filename']); // Just the relative directory inside scorm/
if (!is_array($files_found[basename($res[1])])) {
$files_found[basename($res[1])] = $this->config_exts;
}
@ -439,11 +435,11 @@ class aicc extends learnpath {
if ($this->debug > 1) { error_log('New LP - aicc::import_package() - Found all config files for '.$file_name, 0); }
$mandatory = true;
$package = $file_name;
//store base config file name for reuse in parse_config_files()
// Store base config file name for reuse in parse_config_files().
$this->config_basename = $file_name;
//store filenames for reuse in parse_config_files()
// Store filenames for reuse in parse_config_files().
$this->config_files = $files_found[$file_name];
//get out, we only want one config files set
// Get out, we only want one config files set.
break;
}
}
@ -454,67 +450,55 @@ class aicc extends learnpath {
return api_failure::set_failure('not_aicc_content');
}
if (! enough_size($realFileSize, $course_sys_dir, $maxFilledSpace) )
{
if (!enough_size($realFileSize, $course_sys_dir, $maxFilledSpace)) {
return api_failure::set_failure('not_enough_space');
}
// it happens on Linux that $new_dir sometimes doesn't start with '/'
if($new_dir[0] != '/')
{
// It happens on Linux that $new_dir sometimes doesn't start with '/'
if ($new_dir[0] != '/') {
$new_dir = '/'.$new_dir;
}
//cut trailing slash
if($new_dir[strlen($new_dir)-1] == '/')
{
// Cut trailing slash.
if ($new_dir[strlen($new_dir) - 1] == '/') {
$new_dir = substr($new_dir, 0, -1);
}
/*
--------------------------------------
Uncompressing phase
--------------------------------------
*/
/* Uncompressing phase */
/*
We need to process each individual file in the zip archive to
- add it to the database
- parse & change relative html links
- make sure the filenames are secure (filter funny characters or php extensions)
*/
if(is_dir($course_sys_dir.$new_dir) OR @mkdir($course_sys_dir.$new_dir, api_get_permissions_for_new_directories()))
{
if (is_dir($course_sys_dir.$new_dir) OR @mkdir($course_sys_dir.$new_dir, api_get_permissions_for_new_directories())) {
// PHP method - slower...
if ($this->debug >= 1) { error_log('New LP - Changing dir to '.$course_sys_dir.$new_dir, 0); }
$saved_dir = getcwd();
chdir($course_sys_dir.$new_dir);
$unzippingState = $zipFile->extract();
for($j=0;$j<count($unzippingState);$j++)
{
for ($j = 0; $j < count($unzippingState); $j++) {
$state = $unzippingState[$j];
//TODO fix relative links in html files (?)
$extension = strrchr($state["stored_filename"], ".");
// TODO: Fix relative links in html files (?)
$extension = strrchr($state["stored_filename"], '.');
//if ($this->debug > 1) { error_log('New LP - found extension '.$extension.' in '.$state['stored_filename'], 0); }
}
if(!empty($new_dir))
{
if (!empty($new_dir)) {
$new_dir = $new_dir.'/';
}
//rename files, for example with \\ in it
if($dir=@opendir($course_sys_dir.$new_dir))
{
// Rename files, for example with \\ in it.
if ($dir = @opendir($course_sys_dir.$new_dir)) {
if ($this->debug == 1) { error_log('New LP - Opened dir '.$course_sys_dir.$new_dir, 0); }
while($file=readdir($dir))
{
if($file != '.' && $file != '..')
{
$filetype="file";
while ($file = readdir($dir)) {
if ($file != '.' && $file != '..') {
$filetype = 'file';
if(is_dir($course_sys_dir.$new_dir.$file)) $filetype="folder";
if (is_dir($course_sys_dir.$new_dir.$file)) $filetype = 'folder';
//TODO RENAMING FILES CAN BE VERY DANGEROUS AICC-WISE, avoid that as much as possible!
// TODO: RENAMING FILES CAN BE VERY DANGEROUS AICC-WISE, avoid that as much as possible!
//$safe_file = replace_dangerous_char($file, 'strict');
$find_str = array('\\','.php','.phtml');
$repl_str = array('/', '.txt','.txt');
@ -551,6 +535,7 @@ class aicc extends learnpath {
}
return $course_sys_dir.$new_dir.$config_dir;
}
/**
* Sets the proximity setting in the database
* @param string Proximity setting
@ -619,8 +604,6 @@ class aicc extends learnpath {
}
}
/**
* Sets the content maker setting in the database
* @param string Proximity setting
@ -637,6 +620,7 @@ class aicc extends learnpath {
return false;
}
}
/**
* Exports the current AICC object's files as a zip. Excerpts taken from learnpath_functions.inc.php::exportpath()
* @param integer Learnpath ID (optional, taken from object context if not defined)
@ -644,23 +628,20 @@ class aicc extends learnpath {
function export_zip($lp_id = null) {
if ($this->debug > 0) { error_log('In aicc::export_zip method('.$lp_id.')', 0); }
if (empty($lp_id)) {
if(!is_object($this))
{
if (!is_object($this)) {
return false;
}
else{
} else {
$id = $this->get_id();
if (empty($id)) {
return false;
}
else{
} else {
$lp_id = $this->get_id();
}
}
}
//error_log('New LP - in export_zip()', 0);
//zip everything that is in the corresponding scorm dir
//write the zip file somewhere (might be too big to return)
// Zip everything that is in the corresponding scorm dir.
// Write the zip file somewhere (might be too big to return).
require_once api_get_path(LIBRARY_PATH).'fileUpload.lib.php';
require_once api_get_path(LIBRARY_PATH).'fileManage.lib.php';
require_once api_get_path(LIBRARY_PATH).'document.lib.php';
@ -681,14 +662,14 @@ class aicc extends learnpath {
$scormfoldername = api_get_path(SYS_COURSE_PATH).$_course['directory'].'/scorm/'.$LPnamesafe;
$zipfilename = $zipfoldername.'/'.$LPnamesafe.'.zip';
// Get a temporary dir for creating the zip file
// Get a temporary dir for creating the zip file.
//error_log('New LP - cleaning dir '.$zipfoldername, 0);
deldir($zipfoldername); //make sure the temp dir is cleared
mkdir($zipfoldername, api_get_permissions_for_new_directories());
//error_log('New LP - made dir '.$zipfoldername, 0);
//create zipfile of given directory
// Create zipfile of given directory.
$zip_folder = new PclZip($zipfilename);
$zip_folder->create($scormfoldername.'/', PCLZIP_OPT_REMOVE_PATH, $scormfoldername.'/');
@ -701,9 +682,9 @@ class aicc extends learnpath {
my_delete($zipfilename);
my_delete($zipfoldername);
return true;
}
/**
* Gets a resource's path if available, otherwise return empty string
* @param string Resource ID as used in resource array
@ -718,6 +699,7 @@ class aicc extends learnpath {
}
return $path;
}
/**
* Gets a resource's type if available, otherwise return empty string
* @param string Resource ID as used in resource array
@ -735,6 +717,7 @@ class aicc extends learnpath {
}
return $type;
}
/**
* Gets the default organisation's title
* @return string The organization's title
@ -745,7 +728,7 @@ class aicc extends learnpath {
if (isset($this->config['organizations']['default'])) {
$title = $this->organizations[$this->config['organizations']['default']]->get_name();
} elseif (count($this->organizations) == 1) {
//this will only get one title but so we don't need to know the index
// This will only get one title but so we don't need to know the index.
foreach ($this->organizations as $id => $value) {
$title = $this->organizations[$id]->get_name();
break;
@ -753,10 +736,11 @@ class aicc extends learnpath {
}
return $title;
}
/**
* //TODO @TODO implement this function to restore items data from a set of AICC config files,
* TODO: Implement this function to restore items data from a set of AICC config files,
* updating the existing table... This will prove very useful in case initial data
* from config files were not imported well enough
* from config files were not imported well enough.
*/
function reimport_aicc() {
if ($this->debug > 0) { error_log('In aicc::reimport_aicc() method', 0); }
@ -767,51 +751,43 @@ class aicc extends learnpath {
//update DB accordingly
return true;
}
/**
* Static function to parse AICC ini files.
* Based on work by sinedeo at gmail dot com published on php.net (parse_ini_file())
* Based on work by sinedeo at gmail dot com published on php.net (parse_ini_file()).
* @param string File path
* @return array Structured array
*/
function parse_ini_file_quotes_safe($f)
{
$null = "";
function parse_ini_file_quotes_safe($f) {
$null = '';
$r = $null;
$sec = $null;
$f = @file($f);
for ($i=0;$i<@count($f);$i++)
{
for ($i = 0; $i < @count($f); $i++) {
$newsec = 0;
$w = @trim($f[$i]);
if ($w)
{
if ((!$r) or ($sec))
{
if ((@substr($w,0,1)=="[") and (@substr($w,-1,1))=="]")
{
if ($w) {
if ((!$r) or ($sec)) {
if ((@substr($w, 0, 1) == '[') and (@substr($w, -1, 1)) == ']') {
$sec = @substr($w, 1, @strlen($w) - 2);
$newsec = 1;
}
}
if (!$newsec)
{
$w=@explode("=",$w);
if (!$newsec) {
$w = @explode('=', $w);
$k = @trim($w[0]);
unset($w[0]);
$v=@trim(@implode("=",$w));
if ((@substr($v,0,1)=="\"") and (@substr($v,-1,1)=="\""))
{
$v = @trim(@implode('=', $w));
if ((@substr($v, 0, 1) == "\"") and (@substr($v, -1, 1) == "\"")) {
$v = @substr($v, 1, @strlen($v) - 2);
}
if ($sec)
{
if(strtolower($sec)=='course_description'){//special case
if ($sec) {
if (strtolower($sec) == 'course_description') { //A special case.
$r[strtolower($sec)] = $k;
} else {
$r[strtolower($sec)][strtolower($k)] = $v;
}
} else
{
} else {
$r[strtolower($k)] = $v;
}
}
@ -819,62 +795,54 @@ class aicc extends learnpath {
}
return $r;
}
/**
* Static function to parse AICC ini strings.
* Based on work by sinedeo at gmail dot com published on php.net (parse_ini_file())
* Based on work by sinedeo at gmail dot com published on php.net (parse_ini_file()).
* @param string INI File string
* @param array List of names of sections that should be considered as containing only hard string data (no variables), provided in lower case
* @return array Structured array
*/
function parse_ini_string_quotes_safe($s,$pure_strings=array())
{
$null = "";
function parse_ini_string_quotes_safe($s, $pure_strings = array()) {
$null = '';
$r = $null;
$sec = $null;
$f = split("\r\n", $s);
for ($i=0;$i<@count($f);$i++)
{
for ($i = 0; $i < @count($f); $i++) {
$newsec = 0;
$w = @trim($f[$i]);
if ($w)
{
if ((!$r) or ($sec))
{
if ((@substr($w,0,1)=="[") and (@substr($w,-1,1))=="]")
{
if ($w) {
if ((!$r) or ($sec)) {
if ((@substr($w, 0, 1) == '[') and (@substr($w, -1, 1)) == ']') {
$sec = @substr($w, 1, @strlen($w) - 2);
$pure_data = 0;
if (in_array(strtolower($sec), $pure_strings)) {
//this section can only be considered as pure string data (until the next section)
// This section can only be considered as pure string data (until the next section).
$pure_data = 1;
$r[strtolower($sec)] = '';
}
$newsec = 1;
}
}
if (!$newsec)
{
$w=@explode("=",$w);
if (!$newsec) {
$w = @explode('=', $w);
$k = @trim($w[0]);
unset($w[0]);
$v=@trim(@implode("=",$w));
if ((@substr($v,0,1)=="\"") and (@substr($v,-1,1)=="\""))
{
$v = @trim(@implode('=', $w));
if ((@substr($v, 0, 1) == "\"") and (@substr($v, -1, 1) == "\"")) {
$v = @substr($v, 1, @strlen($v) - 2);
}
if ($sec)
{
if ($sec) {
if ($pure_data) {
$r[strtolower($sec)] .= $f[$i];
} else {
if(strtolower($sec)=='course_description'){//special case
if (strtolower($sec) == 'course_description') { // A special case.
$r[strtolower($sec)] = $k;
} else {
$r[strtolower($sec)][strtolower($k)] = $v;
}
}
} else
{
} else {
$r[strtolower($k)] = $v;
}
}
@ -882,69 +850,59 @@ class aicc extends learnpath {
}
return $r;
}
/**
* Static function that parses CSV files into simple arrays, based on a function
* by spam at cyber-space dot nl published on php.net (fgetcsv())
* by spam at cyber-space dot nl published on php.net (fgetcsv()).
* @param string Filepath
* @param string CSV delimiter
* @param string CSV enclosure
* @param boolean Might one field name happen more than once on the same line? (then split by comma in the values)
* @return array Simple structured array
*/
function parse_csv_file($f,$delim=',',$enclosure='"',$multiples=false)
{
function parse_csv_file($f, $delim = ',', $enclosure = '"', $multiples = false) {
$data = file_get_contents($f);
$enclosed = false;
$fldcount = 0;
$linecount = 0;
$fldval = '';
for($i=0;$i<strlen($data);$i++)
{
for ($i = 0; $i < strlen($data); $i++) {
$chr = $data{$i};
switch($chr)
{
switch ($chr) {
case $enclosure:
if($enclosed&&$data{$i+1}==$enclosure)
{
if ($enclosed && $data{$i+1} == $enclosure) {
$fldval .= $chr;
++$i; //skip next char
}
else
++$i; // Skip the next character.
} else
$enclosed = !$enclosed;
break;
case $delim:
if(!$enclosed)
{
if (!$enclosed) {
$ret_array[$linecount][$fldcount++] = $fldval;
$fldval = '';
}
else
} else
$fldval .= $chr;
break;
case "\r":
if (!$enclosed&&$data{$i+1} == "\n")
continue;
case "\n":
if(!$enclosed)
{
if (!$enclosed) {
$ret_array[$linecount++][$fldcount] = $fldval;
$fldcount = 0;
$fldval = '';
}
else
} else
$fldval .= $chr;
break;
case "\\r":
if (!$enclosed&&$data{$i+1} == "\\n")
continue;
case "\\n":
if(!$enclosed)
{
if (!$enclosed) {
$ret_array[$linecount++][$fldcount] = $fldval;
$fldcount = 0;
$fldval = '';
}
else
} else
$fldval .= $chr;
break;
default:
@ -954,7 +912,7 @@ class aicc extends learnpath {
if ($fldval) {
$ret_array[$linecount][$fldcount] = $fldval;
}
//transform the array to use the first line as titles
// Transform the array to use the first line as titles.
$titles = array();
$ret_ret_array = array();
foreach ($ret_array as $line_idx => $line) {
@ -962,10 +920,9 @@ class aicc extends learnpath {
$titles = $line;
} else {
$ret_ret_array[$line_idx] = array();
foreach($line as $idx=>$val)
{
foreach ($line as $idx => $val) {
if ($multiples && !empty($ret_ret_array[$line_idx][strtolower($titles[$idx])])) {
$ret_ret_array[$line_idx][strtolower($titles[$idx])].=",".$val;
$ret_ret_array[$line_idx][strtolower($titles[$idx])] .= ','.$val;
} else {
$ret_ret_array[$line_idx][strtolower($titles[$idx])] = $val;
}
@ -974,6 +931,4 @@ class aicc extends learnpath {
}
return $ret_ret_array;
}
}
?>

@ -1,14 +1,15 @@
<?php
/* For licensing terms, see /license.txt */
/**
* Container for the aiccResource class that deals with elemens from AICC Course Structure file
* @package chamilo.learnpath
* @author Yannick Warnier <ywarnier@beeznest.org>
* @license GNU/GPL
*/
/**
* Class defining the Block elements in an AICC Course Structure file
*
* Class defining the Block elements in an AICC Course Structure file.
*/
require_once 'learnpathItem.class.php';
class aiccBlock extends learnpathItem {
@ -22,19 +23,15 @@ class aiccBlock extends learnpathItem{
* @param mixed Depending on the type given, DB id for the lp_item or parameters array
*/
function aiccBlock($type = 'config', $params) {
if(isset($params))
{
if (isset($params)) {
switch ($type) {
case 'db':
//TODO implement this way of object creation
//TODO: Implement this way of object creation.
return false;
case 'config': //do the same as the default
case 'config': // Do the same as the default.
default:
foreach($params as $a => $value)
{
switch($a)
{
foreach ($params as $a => $value) {
switch ($a) {
case 'system_id':
$this->identifier = strtolower($value);
break;
@ -51,10 +48,8 @@ class aiccBlock extends learnpathItem{
}
}
return true;
}
}
return false;
}
}
?>

@ -1,11 +1,13 @@
<?php
/* For licensing terms, see /license.txt */
/**
* Container for the aiccItem class that deals with AICC Assignable Units (AUs)
* @package chamilo.learnpath
* @author Yannick Warnier <ywarnier@beeznest.org>
* @license GNU/GPL
*/
/**
* This class handles the elements from an AICC Descriptor file.
*/
@ -16,7 +18,7 @@ class aiccItem extends learnpathItem{
public $parameters = ''; // AICC AU's web_launch
public $title = ''; // no AICC equivalent
public $sub_items = array(); // AICC elements (des)
//public $prerequisites = ''; - defined in learnpathItem.class.php
//public $prerequisites = ''; // defined in learnpathItem.class.php
//public $max_score = ''; // defined in learnpathItem
//public $path = ''; // defined in learnpathItem
public $maxtimeallowed = '00:00:00'; // AICC AU's max_time_allowed
@ -35,21 +37,18 @@ class aiccItem extends learnpathItem{
* @param mixed Depending on the type given, DB id for the lp_item or parameters array
*/
public function aiccItem($type = 'config', $params) {
if(isset($params))
{
if (isset($params)) {
switch ($type) {
case 'db':
parent::__construct($params,api_get_user_id());
$this->aicc_contact = false;
//TODO implement this way of metadata object creation
//TODO: Implement this way of metadata object creation.
return false;
case 'config': //do the same as the default
case 'config': // Do the same as the default.
default:
//if($first_item->type == XML_ELEMENT_NODE) this is already check prior to the call to this function
foreach($params as $a => $value)
{
switch($a)
{
foreach ($params as $a => $value) {
switch ($a) {
case 'system_id':
$this->identifier = Database::escape_string(strtolower($value));
break;
@ -90,6 +89,7 @@ class aiccItem extends learnpathItem{
}
return false;
}
/**
* Builds a flat list with the current item and calls itself recursively on all children
* @param array Reference to the array to complete with the current item
@ -97,8 +97,7 @@ class aiccItem extends learnpathItem{
* @param integer Optional relative order of the item at this level
* @param integer Optional level. If not given, assumes it's level 0
*/
function get_flat_list(&$list,&$abs_order,$rel_order=1,$level=0)
{
function get_flat_list(&$list, &$abs_order, $rel_order = 1, $level = 0) {
$list[] = array(
'au_type' => $this->au_type,
'command_line' => $this->command_line,
@ -114,25 +113,23 @@ class aiccItem extends learnpathItem{
);
$abs_order++;
$i = 1;
foreach($this->sub_items as $id => $dummy)
{
foreach ($this->sub_items as $id => $dummy) {
$oSubitem =& $this->sub_items[$id];
$oSubitem->get_flat_list($list, $abs_order, $i, $level + 1);
$i++;
}
}
/**
* Save function. Uses the parent save function and adds a layer for AICC.
* @param boolean Save from URL params (1) or from object attributes (0)
*/
function save($from_outside=true, $prereqs_complete=false)
{
function save($from_outside = true, $prereqs_complete = false) {
parent::save($from_outside, $prereqs_complete = false);
//under certain conditions, the scorm_contact should not be set, because no scorm signal was sent
// Under certain conditions, the scorm_contact should not be set, because no scorm signal was sent.
$this->aicc_contact = true;
if (!$this->aicc_contact) {
//error_log('New LP - was expecting SCORM message but none received', 0);
}
}
}
?>

@ -1,14 +1,15 @@
<?php
/* For licensing terms, see /license.txt */
/**
* Container for the aiccResource class that deals with elemens from AICC Objectives file
* @package chamilo.learnpath
* @author Yannick Warnier <ywarnier@beeznest.org>
* @license GNU/GPL
*/
/**
* Class defining the Block elements in an AICC Course Structure file
*
* Class defining the Block elements in an AICC Course Structure file.
*/
require_once 'learnpathItem.class.php';
class aiccObjective extends learnpathItem {
@ -23,18 +24,15 @@ class aiccObjective extends learnpathItem{
*/
function aiccObjective($type = 'config', $params) {
if(isset($params))
{
if (isset($params)) {
switch ($type) {
case 'db':
//TODO implement this way of object creation
// TODO: Implement this way of object creation.
return false;
case 'config': //do the same as the default
case 'config': // Do the same as the default.
default:
foreach($params as $a => $value)
{
switch($a)
{
foreach ($params as $a => $value) {
switch ($a) {
case 'system_id':
$this->identifier = strtolower($value);
break;
@ -51,10 +49,8 @@ class aiccObjective extends learnpathItem{
}
}
return true;
}
}
return false;
}
}
?>

@ -1,13 +1,15 @@
<?php
/* For licensing terms, see /license.txt */
/**
* Container for the aiccResource class that deals with elemens from AICC Descriptor file
* @package chamilo.learnpath
* @author Yannick Warnier <ywarnier@beeznest.org>
* @license GNU/GPL
*/
/**
* Class defining the elements from an AICC Descriptor file
* Class defining the elements from an AICC Descriptor file.
*/
class aiccResource {
public $identifier = '';
@ -23,18 +25,15 @@ class aiccResource {
*/
public function aiccResource($type = 'config', $params) {
if(isset($params))
{
if (isset($params)) {
switch ($type) {
case 'db':
//TODO implement this way of object creation
// TODO: Implement this way of object creation.
return false;
case 'config': //do the same as the default
case 'config': // Do the same as the default.
default:
foreach($params as $a => $value)
{
switch($a)
{
foreach ($params as $a => $value) {
switch ($a) {
case 'system_id':
$this->identifier = strtolower($value);
break;
@ -49,10 +48,8 @@ class aiccResource {
}
}
return true;
}
}
return false;
}
}
?>

@ -1,5 +1,6 @@
<?php
/* For licensing terms, see /license.txt */
/**
* API event handler functions for AICC / CMIv4 in API communication mode
*
@ -10,32 +11,31 @@
* @package chamilo.learnpath
* @license GNU/GPL
*/
/**
* This script is divided into three sections.
* The first section (below) is the initialisation part.
* The second section is the AICC object part
* The third section defines the event handlers for Dokeos' internal messaging
* The third section defines the event handlers for Chamilo's internal messaging
* and frames refresh
*
* This script implements the API messaging for AICC. The HACP messaging is
* made by another set of scripts.
*/
/*
* INIT SECTION
*/
//flag to allow for anonymous user - needs to be set before global.inc.php
/* INIT SECTION */
// Flag to allow for anonymous user - needs to be set before global.inc.php.
$use_anonymous = true;
//Load common libraries using a compatibility script to bridge between 1.6 and 1.8
// Load common libraries using a compatibility script to bridge between 1.6 and 1.8.
require_once 'back_compat.inc.php';
//Load learning path libraries so we can use the objects to define the initial values
//of the API
// Load learning path libraries so we can use the objects to define the initial values of the API.
require_once 'learnpath.class.php';
require_once 'learnpathItem.class.php';
require_once 'aicc.class.php';
// Is this needed? This is probabaly done in the header file
// Is this needed? This is probabaly done in the header file.
// $_user = $_SESSION['_user'];
$file = $_SESSION['file'];
$oLP = unserialize($_SESSION['lpobject']);
@ -46,11 +46,8 @@ if(!is_object($oItem)){
}
$autocomplete_when_80pct = 0;
/*
==============================================================================
JavaScript Functions
==============================================================================
*/
/* JavaScript Functions */
?>var scorm_logs=<?php echo (empty($oLP->scorm_debug) ? '0' : '3'); ?>; //debug log level for SCORM. 0 = none, 1=light, 2=a lot, 3=all - displays logs in log frame
var lms_logs=0; //debug log level for LMS actions. 0=none, 1=light, 2=a lot, 3=all
//logit_lms('scormfunctions.php included',0);
@ -66,7 +63,7 @@ function APIobject() {
this.LMSGetDiagnostic=LMSGetDiagnostic;
}
//it is not sure that the scos use the above declarations
// It is not sure that the scos use the above declarations.
API = new APIobject(); //for scorm 1.2
@ -87,7 +84,7 @@ var G_LastError = G_NoError ;
var commit = false ;
//Strictly scorm variables
// Strictly SCORM variables.
var score=<?php echo $oItem->get_score();?>;
var max=<?php echo $oItem->get_max();?>;
var min=<?php echo $oItem->get_min();?>;
@ -97,7 +94,7 @@ var suspend_data = '<?php echo $oItem->get_suspend_data();?>';
var lesson_location = '<?php echo $oItem->get_lesson_location();?>';
var total_time = '<?php echo $oItem->get_scorm_time('js');?>';
//Dokeos internal variables
// Chamilo internal variables.
var saved_lesson_status = 'not attempted';
var lms_lp_id = <?php echo $oLP->get_id();?>;
var lms_item_id = <?php echo $oItem->get_id();?>;
@ -116,7 +113,7 @@ var lms_previous_item = '<?php echo $oLP->get_previous_item_id();?>';
var lms_lp_type = '<?php echo $oLP->get_type();?>';
var lms_item_type = '<?php echo $oItem->get_type();?>';
//Backup for old values
// Backup for old values.
var old_score = 0;
var old_max = 0;
var old_min = 0;
@ -293,7 +290,7 @@ function savedata(origin) { //origin can be 'commit', 'finish' or 'terminate'
url="http://<?php
$self = api_get_self();
$url = $_SERVER['HTTP_HOST'].$self;
$url=substr($url,0,-14);//14 is the length of this file's name (/scorm_api.php)
$url = substr($url, 0, -14); // 14 is the length of this file's name (/scorm_api.php).
echo $url;
?>/lp_controller.php?cidReq=<?php echo api_get_course_id();?>&action=save&lp_id=<?php echo $oLP->get_id();?>&" + param + "";
logit_lms('saving data (status='+lesson_status+')',1);
@ -331,17 +328,16 @@ function LMSGetDiagnostic(errCode){
return(API.LMSGetLastError());
}
<?php
//--------------------------------------------------------------------//
/**
* Dokeos-specific code that deals with event handling and inter-frames
* Chamilo-specific code that deals with event handling and inter-frames
* messaging/refreshing.
* Note that from now on, the Dokeos JS code in this library will act as
* Note that from now on, the Chamilo JS code in this library will act as
* a controller, of the MVC pattern, and receive all requests for frame
* updates, then redispatch to any frame concerned.
*/
?>
/**
* Defining the AJAX-object class to be made available from other frames
* Defining the AJAX-object class to be made available from other frames.
*/
function XAJAXobject() {
this.xajax_switch_item_details=xajax_switch_item_details;
@ -389,8 +385,8 @@ function addListeners(){
//assign event handlers to objects
if(lms_lp_type==1 || lms_item_type=='asset'){
logit_lms('Dokeos LP or asset',2);
//if this path is a Dokeos learnpath, then start manual save
//when something is loaded in there
// If this path is a Chamilo learnpath, then start manual save
// when something is loaded in there.
var myelem = document.getElementById('content_id');
if(!myelem){logit_lms("Impossible to find content_id element in document",2);}
addEvent(myelem,'unload',dokeos_save_asset,false);
@ -433,12 +429,12 @@ function load_item(item_id,url){
return false;
}
/**
* Save a Dokeos learnpath item's time and mark as completed upon
* Save a Chamilo learnpath item's time and mark as completed upon
* leaving it
*/
function dokeos_save_asset(){
//var linkparams = 'id='+lms_item_id+'&score='+score+'&max='+max+'&min='+min+'&lesson_status='+lesson_status+'&time='+session_time+'&suspend_data='+suspend_data;
//var url = "<?php echo api_get_path(WEB_CODE_PATH).'newscorm/lp_controller.php' ?>?action=save&" + linkparams + "";
//var url = "<?php echo api_get_path(WEB_CODE_PATH).'newscorm/lp_controller.php'; ?>?action=save&" + linkparams + "";
logit_lms('dokeos_save_asset: '+url,0);
//frames["message_name"].src = url;
xajax_save_item(lms_lp_id, lms_user_id, lms_view_id, lms_item_id, score, max, min, lesson_status, session_time, suspend_data, lesson_location);
@ -598,7 +594,7 @@ function update_message_frame(msg_msg)
* current item, (2) refresh all the values inside the SCORM API object, (3) open the
* new item into the content_id frame, (4) refresh the table of contents, (5) refresh
* the progress bar (completion), (6) refresh the message frame
* @param integer Dokeos ID for the current item
* @param integer Chamilo ID for the current item
* @param string This parameter can be a string specifying the next
* item (like 'next', 'previous', 'first' or 'last') or the id to the next item
*/

@ -1,5 +1,6 @@
<?php // $Id: $
<?php
/* For licensing terms, see /license.txt */
/**
* API event handler functions for AICC / CMIv4 in HACP communication mode
*
@ -10,6 +11,7 @@
* @package chamilo.learnpath
* @license GNU/GPL
*/
/**
* This script is divided into three sections.
* The first section (below) is the initialisation part.
@ -29,32 +31,28 @@
* Only suspend_data and core.lesson_location should be sent updated to a late GetParam
* request. All other params should be as when the AU was launched.
*/
/*
==============================================================================
INIT SECTION
==============================================================================
*/
/* INIT SECTION */
$debug = 0;
//flag to allow for anonymous user - needs to be set before global.inc.php
// Flag to allow for anonymous user - needs to be set before global.inc.php.
$use_anonymous = true;
//Use session ID as provided by the request
if(!empty($_REQUEST['aicc_sid']))
{
// Use session ID as provided by the request.
if (!empty($_REQUEST['aicc_sid'])) {
session_id($_REQUEST['aicc_sid']);
if ($debug > 1) { error_log('New LP - '.__FILE__.','.__LINE__.' - reusing session ID '.$_REQUEST['aicc_sid'], 0); }
}
//Load common libraries using a compatibility script to bridge between 1.6 and 1.8
//Load common libraries using a compatibility script to bridge between 1.6 and 1.8.
require_once 'back_compat.inc.php';
if ($debug > 2) { error_log('New LP - '.__FILE__.','.__LINE__.' - Current session ID: '.session_id(), 0); }
//Load learning path libraries so we can use the objects to define the initial values
//of the API
//Load learning path libraries so we can use the objects to define the initial values of the API.
require_once 'learnpath.class.php';
require_once 'learnpathItem.class.php';
require_once 'aicc.class.php';
// Is this needed? This is probabaly done in the header file
// Is this needed? This is probabaly done in the header file.
//$_user = $_SESSION['_user'];
$file = $_SESSION['file'];
$oLP = unserialize($_SESSION['lpobject']);
@ -86,14 +84,12 @@ $error_code = 0;
$error_text = '';
$aicc_data = '';
$result = '';
//GET REQUEST
if(!empty($_REQUEST['command']))
{
// Get REQUEST
if (!empty($_REQUEST['command'])) {
//error_log('In '.__FILE__.', '.__LINE__.' - request is '.$_REQUEST['command'], 0);
switch(strtolower($_REQUEST['command']))
{
switch (strtolower($_REQUEST['command'])) {
case 'getparam':
//request for all available data to be printed out in the answer
// Request for all available data to be printed out in the answer.
if (!empty($_REQUEST['version'])) {
$hacp_version = learnpath::escape_string($_REQUEST['version']);
}
@ -134,11 +130,9 @@ if(!empty($_REQUEST['command']))
$hacp_version = '';
$hacp_session_id = '';
$hacp_aicc_data = '';
foreach($_REQUEST as $name => $value)
{
foreach ($_REQUEST as $name => $value) {
//escape the value as described in the AICC documentation p170
switch(strtolower($name))
{
switch (strtolower($name)) {
case 'version':
$hacp_version = $value;
break;
@ -154,7 +148,7 @@ if(!empty($_REQUEST['command']))
}
}
//error_log('In '.__FILE__.', '.__LINE__.' - aicc data is '.$hacp_aicc_data, 0);
//treat the incoming request:
// Treat the incoming request:
$msg_array = aicc::parse_ini_string_quotes_safe($hacp_aicc_data, array('core_lesson', 'core_vendor'));
//error_log('Message is now in this form: '.print_r($msg_array, true), 0);
foreach ($msg_array as $key => $dummy) {
@ -238,7 +232,6 @@ if(!empty($_REQUEST['command']))
}
}
$_SESSION['lpobject'] = serialize($oLP);
//content type must be text/plain
// Content type must be text/plain.
header('Content-type: text/plain');
echo $result;
?>

@ -1,5 +1,5 @@
/**
* Wrapper to the SCORM API provided by Dokeos
* Wrapper to the SCORM API provided by Chamilo
* The complete set of functions and variables are in this file to avoid unnecessary file
* accesses.
* Only event triggers and answer data are inserted into the final document.
@ -9,7 +9,7 @@
/**
* Initialisation of the SCORM API section.
* Find the SCO functions (startTimer, computeTime, etc in the second section)
* Find the Dokeos-proper functions (checkAnswers, etc in the third section)
* Find the Chamilo-proper functions (checkAnswers, etc in the third section)
*/
var _debug = false;
var findAPITries = 0;
@ -413,7 +413,7 @@ function unloadPage(status)
}
}
/**
* Third section - depending on Dokeos - check answers and set score
* Third section - depending on Chamilo - check answers and set score
*/
var questions = new Array();
var questions_answers = new Array();

Loading…
Cancel
Save