3 // This file is part of Moodle - http://moodle.org/
5 // Moodle is free software: you can redistribute it and/or modify
6 // it under the terms of the GNU General Public License as published by
7 // the Free Software Foundation, either version 3 of the License, or
8 // (at your option) any later version.
10 // Moodle is distributed in the hope that it will be useful,
11 // but WITHOUT ANY WARRANTY; without even the implied warranty of
12 // MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
13 // GNU General Public License for more details.
15 // You should have received a copy of the GNU General Public License
16 // along with Moodle. If not, see <http://www.gnu.org/licenses/>.
19 * IMS CP module upgrade related helper functions
22 * @copyright 2009 Petr Skoda (http://skodak.org)
23 * @license http://www.gnu.org/copyleft/gpl.html GNU GPL v3 or later
27 * Migrate imscp module data from 1.9 resource_old table to new imscp table
30 function imscp_20_migrate() {
31 global $CFG, $DB, $OUTPUT;
32 require_once("$CFG->libdir/filelib.php");
33 require_once("$CFG->dirroot/course/lib.php");
34 require_once("$CFG->dirroot/mod/imscp/locallib.php");
36 if (!file_exists("$CFG->dirroot/mod/resource/db/upgradelib.php")) {
37 // bad luck, somebody deleted resource module
41 require_once("$CFG->dirroot/mod/resource/db/upgradelib.php");
43 // create resource_old table and copy resource table there if needed
44 if (!resource_20_prepare_migration()) {
45 // no modules or fresh install
49 if (!$candidates = $DB->get_recordset('resource_old', array('type'=>'ims', 'migrated'=>0))) {
53 $fs = get_file_storage();
55 foreach ($candidates as $candidate) {
56 upgrade_set_timeout(60);
58 $imscp = new object();
59 $imscp->course = $candidate->course;
60 $imscp->name = $candidate->name;
61 $imscp->intro = $candidate->intro;
62 $imscp->introformat = $candidate->introformat;
65 $imscp->timemodified = time();
67 if (!$imscp = resource_migrate_to_module('imscp', $candidate, $imscp)) {
71 $context = get_context_instance(CONTEXT_MODULE, $candidate->cmid);
72 $root = "$CFG->dataroot/$candidate->course/$CFG->moddata/resource/$candidate->oldid";
74 // migrate package backup file
75 if ($candidate->reference) {
76 $package = basename($candidate->reference);
77 $fullpath = $root.'/'.$package;
78 if (file_exists($fullpath)) {
79 $file_record = array('contextid' => $context->id,
80 'filearea' => 'imscp_backup',
83 'filename' => $package);
84 $fs->create_file_from_pathname($file_record, $fullpath);
88 // migrate extracted package data
89 $files = imsc_migrate_get_old_files($root, '');
91 // if ims package doesn't exist, continue loop
92 echo $OUTPUT->notification("IMS package data cannot be found, failed migrating activity: \"$candidate->name\", please fix it manually");
96 $file_record = array('contextid'=>$context->id, 'filearea'=>'imscp_content', 'itemid'=>1);
98 foreach ($files as $relname=>$fullpath) {
99 $parts = explode('/', $relname);
100 $file_record['filename'] = array_pop($parts);
101 $parts[] = ''; // keep trailing slash
102 $file_record['filepath'] = implode('/', $parts);
105 $fs->create_file_from_pathname($file_record, $fullpath);
106 } catch (Exception $e) {
107 //continue on error, we can not recover anyway
109 echo $OUTPUT->notification("IMSCP: failed migrating file: $relname");
115 $structure = imscp_parse_structure($imscp, $context);
116 $imscp->structure = is_array($structure) ? serialize($structure) : null;
117 $DB->update_record('imscp', $imscp);
119 // remove old moddata dir only if no error and manifest ok
120 if (!$error and is_array($structure)) {
125 $candidates->close();
127 // clear all course modinfo caches
128 rebuild_course_cache(0, true);
132 * Private function returning all extracted IMS content package file
134 function imsc_migrate_get_old_files($path, $relative) {
137 if (!file_exists($path)) {
138 echo $OUTPUT->notification("File path doesn't exist: $path <br/> Please fix it manually.");
141 $items = new DirectoryIterator($path);
142 foreach ($items as $item) {
143 if ($item->isDot() or $item->isLink()) {
144 // symbolik links could create infinite loops or cause unintended file migration, sorry
147 $pathname = $item->getPathname();
148 $relname = $relative.'/'.$item->getFilename();
149 if ($item->isFile()) {
150 $result[$relname] = $pathname;
151 } else if ($item->isDir()) {
152 $result = array_merge($result, imsc_migrate_get_old_files($pathname, $relname));