You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
489 lines
18 KiB
489 lines
18 KiB
<?php
|
|
// This file is part of Moodle - http://moodle.org/
|
|
//
|
|
// Moodle is free software: you can redistribute it and/or modify
|
|
// it under the terms of the GNU General Public License as published by
|
|
// the Free Software Foundation, either version 3 of the License, or
|
|
// (at your option) any later version.
|
|
//
|
|
// Moodle is distributed in the hope that it will be useful,
|
|
// but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
// GNU General Public License for more details.
|
|
//
|
|
// You should have received a copy of the GNU General Public License
|
|
// along with Moodle. If not, see <http://www.gnu.org/licenses/>.
|
|
|
|
/**
|
|
* This file contains the class to import a competency framework.
|
|
*
|
|
* @package tool_lpimportcsv
|
|
* @copyright 2015 Damyon Wiese
|
|
* @license http://www.gnu.org/copyleft/gpl.html GNU GPL v3 or later
|
|
*/
|
|
|
|
namespace tool_lpimportcsv;
|
|
|
|
defined('MOODLE_INTERNAL') || die('Direct access to this script is forbidden.');
|
|
|
|
use core_competency\api;
|
|
use grade_scale;
|
|
use stdClass;
|
|
use context_system;
|
|
use csv_import_reader;
|
|
|
|
/**
|
|
* This file contains the class to import a competency framework.
|
|
*
|
|
* @package tool_lpimportcsv
|
|
* @copyright 2015 Damyon Wiese
|
|
* @license http://www.gnu.org/copyleft/gpl.html GNU GPL v3 or later
|
|
*/
|
|
class framework_importer {
|
|
|
|
/** @var string $error The errors message from reading the xml */
|
|
protected $error = '';
|
|
|
|
/** @var array $flat The flat competencies tree */
|
|
protected $flat = array();
|
|
/** @var array $framework The framework info */
|
|
protected $framework = array();
|
|
protected $mappings = array();
|
|
protected $importid = 0;
|
|
protected $importer = null;
|
|
protected $foundheaders = array();
|
|
protected $scalecache = array();
|
|
/** @var bool $useprogressbar Control whether importing should use progress bars or not. */
|
|
protected $useprogressbar = false;
|
|
/** @var \core\progress\display_if_slow|null $progress The progress bar instance. */
|
|
protected $progress = null;
|
|
|
|
/**
|
|
* Store an error message for display later
|
|
* @param string $msg
|
|
*/
|
|
public function fail($msg) {
|
|
$this->error = $msg;
|
|
return false;
|
|
}
|
|
|
|
/**
|
|
* Get the CSV import id
|
|
* @return string The import id.
|
|
*/
|
|
public function get_importid() {
|
|
return $this->importid;
|
|
}
|
|
|
|
/**
|
|
* Get the list of headers required for import.
|
|
* @return array The headers (lang strings)
|
|
*/
|
|
public static function list_required_headers() {
|
|
return array(
|
|
get_string('parentidnumber', 'tool_lpimportcsv'),
|
|
get_string('idnumber', 'tool_lpimportcsv'),
|
|
get_string('shortname', 'tool_lpimportcsv'),
|
|
get_string('description', 'tool_lpimportcsv'),
|
|
get_string('descriptionformat', 'tool_lpimportcsv'),
|
|
get_string('scalevalues', 'tool_lpimportcsv'),
|
|
get_string('scaleconfiguration', 'tool_lpimportcsv'),
|
|
get_string('ruletype', 'tool_lpimportcsv'),
|
|
get_string('ruleoutcome', 'tool_lpimportcsv'),
|
|
get_string('ruleconfig', 'tool_lpimportcsv'),
|
|
get_string('relatedidnumbers', 'tool_lpimportcsv'),
|
|
get_string('exportid', 'tool_lpimportcsv'),
|
|
get_string('isframework', 'tool_lpimportcsv'),
|
|
get_string('taxonomy', 'tool_lpimportcsv'),
|
|
);
|
|
}
|
|
|
|
/**
|
|
* Get the list of headers found in the import.
|
|
* @return array The found headers (names from import)
|
|
*/
|
|
public function list_found_headers() {
|
|
return $this->foundheaders;
|
|
}
|
|
|
|
/**
|
|
* Read the data from the mapping form.
|
|
* @param array The mapping data.
|
|
*/
|
|
protected function read_mapping_data($data) {
|
|
if ($data) {
|
|
return array(
|
|
'parentidnumber' => $data->header0,
|
|
'idnumber' => $data->header1,
|
|
'shortname' => $data->header2,
|
|
'description' => $data->header3,
|
|
'descriptionformat' => $data->header4,
|
|
'scalevalues' => $data->header5,
|
|
'scaleconfiguration' => $data->header6,
|
|
'ruletype' => $data->header7,
|
|
'ruleoutcome' => $data->header8,
|
|
'ruleconfig' => $data->header9,
|
|
'relatedidnumbers' => $data->header10,
|
|
'exportid' => $data->header11,
|
|
'isframework' => $data->header12,
|
|
'taxonomies' => $data->header13
|
|
);
|
|
} else {
|
|
return array(
|
|
'parentidnumber' => 0,
|
|
'idnumber' => 1,
|
|
'shortname' => 2,
|
|
'description' => 3,
|
|
'descriptionformat' => 4,
|
|
'scalevalues' => 5,
|
|
'scaleconfiguration' => 6,
|
|
'ruletype' => 7,
|
|
'ruleoutcome' => 8,
|
|
'ruleconfig' => 9,
|
|
'relatedidnumbers' => 10,
|
|
'exportid' => 11,
|
|
'isframework' => 12,
|
|
'taxonomies' => 13
|
|
);
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Get the a column from the imported data.
|
|
* @param array The imported raw row
|
|
* @param index The column index we want
|
|
* @return string The column data.
|
|
*/
|
|
protected function get_column_data($row, $index) {
|
|
if ($index < 0) {
|
|
return '';
|
|
}
|
|
return isset($row[$index]) ? $row[$index] : '';
|
|
}
|
|
|
|
/**
|
|
* Constructor - parses the raw text for sanity.
|
|
* @param string $text The raw csv text.
|
|
* @param string $encoding The encoding of the csv file.
|
|
* @param string delimiter The specified delimiter for the file.
|
|
* @param string importid The id of the csv import.
|
|
* @param array mappingdata The mapping data from the import form.
|
|
* @param bool $useprogressbar Whether progress bar should be displayed, to avoid html output on CLI.
|
|
*/
|
|
public function __construct($text = null, $encoding = null, $delimiter = null, $importid = 0, $mappingdata = null,
|
|
$useprogressbar = false) {
|
|
|
|
global $CFG;
|
|
|
|
// The format of our records is:
|
|
// Parent ID number, ID number, Shortname, Description, Description format, Scale values, Scale configuration,
|
|
// Rule type, Rule outcome, Rule config, Is framework, Taxonomy.
|
|
|
|
// The idnumber is concatenated with the category names.
|
|
require_once($CFG->libdir . '/csvlib.class.php');
|
|
|
|
$type = 'competency_framework';
|
|
|
|
if (!$importid) {
|
|
if ($text === null) {
|
|
return;
|
|
}
|
|
$this->importid = csv_import_reader::get_new_iid($type);
|
|
|
|
$this->importer = new csv_import_reader($this->importid, $type);
|
|
|
|
if (!$this->importer->load_csv_content($text, $encoding, $delimiter)) {
|
|
$this->fail(get_string('invalidimportfile', 'tool_lpimportcsv'));
|
|
$this->importer->cleanup();
|
|
return;
|
|
}
|
|
|
|
} else {
|
|
$this->importid = $importid;
|
|
|
|
$this->importer = new csv_import_reader($this->importid, $type);
|
|
}
|
|
|
|
if (!$this->importer->init()) {
|
|
$this->fail(get_string('invalidimportfile', 'tool_lpimportcsv'));
|
|
$this->importer->cleanup();
|
|
return;
|
|
}
|
|
|
|
$this->foundheaders = $this->importer->get_columns();
|
|
$this->useprogressbar = $useprogressbar;
|
|
$domainid = 1;
|
|
|
|
$flat = array();
|
|
$framework = null;
|
|
|
|
while ($row = $this->importer->next()) {
|
|
$mapping = $this->read_mapping_data($mappingdata);
|
|
|
|
$parentidnumber = $this->get_column_data($row, $mapping['parentidnumber']);
|
|
$idnumber = $this->get_column_data($row, $mapping['idnumber']);
|
|
$shortname = $this->get_column_data($row, $mapping['shortname']);
|
|
$description = $this->get_column_data($row, $mapping['description']);
|
|
$descriptionformat = $this->get_column_data($row, $mapping['descriptionformat']);
|
|
$scalevalues = $this->get_column_data($row, $mapping['scalevalues']);
|
|
$scaleconfiguration = $this->get_column_data($row, $mapping['scaleconfiguration']);
|
|
$ruletype = $this->get_column_data($row, $mapping['ruletype']);
|
|
$ruleoutcome = $this->get_column_data($row, $mapping['ruleoutcome']);
|
|
$ruleconfig = $this->get_column_data($row, $mapping['ruleconfig']);
|
|
$relatedidnumbers = $this->get_column_data($row, $mapping['relatedidnumbers']);
|
|
$exportid = $this->get_column_data($row, $mapping['exportid']);
|
|
$isframework = $this->get_column_data($row, $mapping['isframework']);
|
|
$taxonomies = $this->get_column_data($row, $mapping['taxonomies']);
|
|
|
|
if ($isframework) {
|
|
$framework = new stdClass();
|
|
$framework->idnumber = shorten_text(clean_param($idnumber, PARAM_TEXT), 100);
|
|
$framework->shortname = shorten_text(clean_param($shortname, PARAM_TEXT), 100);
|
|
$framework->description = clean_param($description, PARAM_RAW);
|
|
$framework->descriptionformat = clean_param($descriptionformat, PARAM_INT);
|
|
$framework->scalevalues = $scalevalues;
|
|
$framework->scaleconfiguration = $scaleconfiguration;
|
|
$framework->taxonomies = $taxonomies;
|
|
$framework->children = array();
|
|
} else {
|
|
$competency = new stdClass();
|
|
$competency->parentidnumber = clean_param($parentidnumber, PARAM_TEXT);
|
|
$competency->idnumber = shorten_text(clean_param($idnumber, PARAM_TEXT), 100);
|
|
$competency->shortname = shorten_text(clean_param($shortname, PARAM_TEXT), 100);
|
|
$competency->description = clean_param($description, PARAM_RAW);
|
|
$competency->descriptionformat = clean_param($descriptionformat, PARAM_INT);
|
|
$competency->ruletype = $ruletype;
|
|
$competency->ruleoutcome = clean_param($ruleoutcome, PARAM_INT);
|
|
$competency->ruleconfig = $ruleconfig;
|
|
$competency->relatedidnumbers = $relatedidnumbers;
|
|
$competency->exportid = $exportid;
|
|
$competency->scalevalues = $scalevalues;
|
|
$competency->scaleconfiguration = $scaleconfiguration;
|
|
$competency->children = array();
|
|
$flat[$idnumber] = $competency;
|
|
}
|
|
}
|
|
$this->flat = $flat;
|
|
$this->framework = $framework;
|
|
|
|
$this->importer->close();
|
|
if ($this->framework == null) {
|
|
$this->fail(get_string('invalidimportfile', 'tool_lpimportcsv'));
|
|
return;
|
|
} else {
|
|
// We are calling from browser, display progress bar.
|
|
if ($this->useprogressbar === true) {
|
|
$this->progress = new \core\progress\display_if_slow(get_string('processingfile', 'tool_lpimportcsv'));
|
|
} else {
|
|
// Avoid html output on CLI scripts.
|
|
$this->progress = new \core\progress\none();
|
|
}
|
|
$this->progress->start_progress('', count($this->flat));
|
|
// Build a tree from this flat list.
|
|
raise_memory_limit(MEMORY_EXTRA);
|
|
$this->add_children($this->framework, '');
|
|
$this->progress->end_progress();
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Add a competency to the parent with the specified idnumber.
|
|
*
|
|
* @param competency $node (pass by reference)
|
|
* @param string $parentidnumber Add this competency to the parent with this idnumber.
|
|
*/
|
|
public function add_children(& $node, $parentidnumber) {
|
|
foreach ($this->flat as $competency) {
|
|
if ($competency->parentidnumber == $parentidnumber) {
|
|
$this->progress->increment_progress();
|
|
$node->children[] = $competency;
|
|
$this->add_children($competency, $competency->idnumber);
|
|
}
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Get parse errors.
|
|
* @return array of errors from parsing the xml.
|
|
*/
|
|
public function get_error() {
|
|
return $this->error;
|
|
}
|
|
|
|
/**
|
|
* Recursive function to add a competency with all it's children.
|
|
*
|
|
* @param stdClass $record Raw data for the new competency
|
|
* @param competency $parent
|
|
* @param competency_framework $framework
|
|
*/
|
|
public function create_competency($record, $parent, $framework) {
|
|
$competency = new stdClass();
|
|
$competency->competencyframeworkid = $framework->get('id');
|
|
$competency->shortname = $record->shortname;
|
|
if (!empty($record->description)) {
|
|
$competency->description = $record->description;
|
|
$competency->descriptionformat = $record->descriptionformat;
|
|
}
|
|
if ($record->scalevalues) {
|
|
$competency->scaleid = $this->get_scale_id($record->scalevalues, $competency->shortname);
|
|
$competency->scaleconfiguration = $this->get_scale_configuration($competency->scaleid, $record->scaleconfiguration);
|
|
}
|
|
if ($parent) {
|
|
$competency->parentid = $parent->get('id');
|
|
} else {
|
|
$competency->parentid = 0;
|
|
}
|
|
$competency->idnumber = $record->idnumber;
|
|
|
|
if (!empty($competency->idnumber) && !empty($competency->shortname)) {
|
|
$comp = api::create_competency($competency);
|
|
if ($record->exportid) {
|
|
$this->mappings[$record->exportid] = $comp;
|
|
}
|
|
$record->createdcomp = $comp;
|
|
foreach ($record->children as $child) {
|
|
$this->create_competency($child, $comp, $framework);
|
|
}
|
|
|
|
return $comp;
|
|
}
|
|
return false;
|
|
}
|
|
|
|
/**
|
|
* Recreate the scale config to point to a new scaleid.
|
|
* @param int $scaleid
|
|
* @param string $config json encoded scale data.
|
|
*/
|
|
public function get_scale_configuration($scaleid, $config) {
|
|
$asarray = json_decode($config);
|
|
$asarray[0]->scaleid = $scaleid;
|
|
return json_encode($asarray);
|
|
}
|
|
|
|
/**
|
|
* Search for a global scale that matches this set of scalevalues.
|
|
* If one is not found it will be created.
|
|
* @param array $scalevalues
|
|
* @param string $competencyname (Used to create a new scale if required)
|
|
* @return int The id of the scale
|
|
*/
|
|
public function get_scale_id($scalevalues, $competencyname) {
|
|
global $CFG, $USER;
|
|
|
|
require_once($CFG->libdir . '/gradelib.php');
|
|
|
|
if (empty($this->scalecache)) {
|
|
$allscales = grade_scale::fetch_all_global();
|
|
foreach ($allscales as $scale) {
|
|
$scale->load_items();
|
|
$this->scalecache[$scale->compact_items()] = $scale;
|
|
}
|
|
}
|
|
$matchingscale = false;
|
|
if (isset($this->scalecache[$scalevalues])) {
|
|
$matchingscale = $this->scalecache[$scalevalues];
|
|
}
|
|
if (!$matchingscale) {
|
|
// Create it.
|
|
$newscale = new grade_scale();
|
|
$newscale->name = get_string('competencyscale', 'tool_lpimportcsv', $competencyname);
|
|
$newscale->courseid = 0;
|
|
$newscale->userid = $USER->id;
|
|
$newscale->scale = $scalevalues;
|
|
$newscale->description = get_string('competencyscaledescription', 'tool_lpimportcsv');
|
|
$newscale->insert();
|
|
$this->scalecache[$scalevalues] = $newscale;
|
|
return $newscale->id;
|
|
}
|
|
return $matchingscale->id;
|
|
}
|
|
|
|
/**
|
|
* Walk through the idnumbers in the relatedidnumbers col and set the relations.
|
|
* @param stdClass $record
|
|
*/
|
|
protected function set_related($record) {
|
|
$comp = $record->createdcomp;
|
|
if ($record->relatedidnumbers) {
|
|
$allidnumbers = explode(',', $record->relatedidnumbers);
|
|
foreach ($allidnumbers as $rawidnumber) {
|
|
$idnumber = str_replace('%2C', ',', $rawidnumber);
|
|
|
|
if (isset($this->flat[$idnumber])) {
|
|
$relatedcomp = $this->flat[$idnumber]->createdcomp;
|
|
api::add_related_competency($comp->get('id'), $relatedcomp->get('id'));
|
|
}
|
|
}
|
|
}
|
|
foreach ($record->children as $child) {
|
|
$this->set_related($child);
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Create any completion rule attached to this competency.
|
|
* @param stdClass $record
|
|
*/
|
|
protected function set_rules($record) {
|
|
$comp = $record->createdcomp;
|
|
if ($record->ruletype) {
|
|
$class = $record->ruletype;
|
|
if (class_exists($class)) {
|
|
$oldruleconfig = $record->ruleconfig;
|
|
if ($oldruleconfig == "null") {
|
|
$oldruleconfig = null;
|
|
}
|
|
$newruleconfig = $class::migrate_config($oldruleconfig, $this->mappings);
|
|
$comp->set('ruleconfig', $newruleconfig);
|
|
$comp->set('ruletype', $class);
|
|
$comp->set('ruleoutcome', $record->ruleoutcome);
|
|
$comp->update();
|
|
}
|
|
}
|
|
foreach ($record->children as $child) {
|
|
$this->set_rules($child);
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Do the job.
|
|
* @return competency_framework
|
|
*/
|
|
public function import() {
|
|
$record = clone $this->framework;
|
|
unset($record->children);
|
|
|
|
$record->scaleid = $this->get_scale_id($record->scalevalues, $record->shortname);
|
|
$record->scaleconfiguration = $this->get_scale_configuration($record->scaleid, $record->scaleconfiguration);
|
|
unset($record->scalevalues);
|
|
$record->contextid = context_system::instance()->id;
|
|
|
|
$framework = api::create_framework($record);
|
|
if ($this->useprogressbar === true) {
|
|
$this->progress = new \core\progress\display_if_slow(get_string('importingfile', 'tool_lpimportcsv'));
|
|
} else {
|
|
$this->progress = new \core\progress\none();
|
|
}
|
|
|
|
$this->progress->start_progress('', (count($this->framework->children) * 2));
|
|
raise_memory_limit(MEMORY_EXTRA);
|
|
// Now all the children.
|
|
foreach ($this->framework->children as $comp) {
|
|
$this->progress->increment_progress();
|
|
$this->create_competency($comp, null, $framework);
|
|
}
|
|
|
|
// Now create the rules.
|
|
foreach ($this->framework->children as $record) {
|
|
$this->progress->increment_progress();
|
|
$this->set_rules($record);
|
|
$this->set_related($record);
|
|
}
|
|
$this->progress->end_progress();
|
|
|
|
$this->importer->cleanup();
|
|
return $framework;
|
|
}
|
|
}
|
|
|