MDL-22543, IMSCP, check if files exist before migrating to moodle 2
[moodle.git] / mod / imscp / db / upgradelib.php
1 <?php
3 // This file is part of Moodle - http://moodle.org/
4 //
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.
9 //
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.
14 //
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/>.
18 /**
19  * IMS CP module upgrade related helper functions
20  *
21  * @package   mod-imscp
22  * @copyright 2009 Petr Skoda (http://skodak.org)
23  * @license   http://www.gnu.org/copyleft/gpl.html GNU GPL v3 or later
24  */
26 /**
27  * Migrate imscp module data from 1.9 resource_old table to new imscp table
28  * @return void
29  */
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
38         return;
39     }
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
46         return;
47     }
49     if (!$candidates = $DB->get_recordset('resource_old', array('type'=>'ims', 'migrated'=>0))) {
50         return;
51     }
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;
63         $imscp->revision     = 1;
64         $imscp->keepold      = 1;
65         $imscp->timemodified = time();
67         if (!$imscp = resource_migrate_to_module('imscp', $candidate, $imscp)) {
68             continue;
69         }
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',
81                                      'itemid'    => 1,
82                                      'filepath'  => '/',
83                                      'filename'  => $package);
84                 $fs->create_file_from_pathname($file_record, $fullpath);
85             }
86         }
88         // migrate extracted package data
89         $files = imsc_migrate_get_old_files($root, '');
90         if (empty($files)) {
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");
93             continue;
94         }
96         $file_record = array('contextid'=>$context->id, 'filearea'=>'imscp_content', 'itemid'=>1);
97         $error = false;
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);
104             try {
105                 $fs->create_file_from_pathname($file_record, $fullpath);
106             } catch (Exception $e) {
107                 //continue on error, we can not recover anyway
108                 $error = true;
109                 echo $OUTPUT->notification("IMSCP: failed migrating file: $relname");
110             }
111         }
112         unset($files);
114         // parse manifest
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)) {
121             fulldelete($root);
122         }
123     }
125     $candidates->close();
127     // clear all course modinfo caches
128     rebuild_course_cache(0, true);
131 /**
132  * Private function returning all extracted IMS content package file
133  */
134 function imsc_migrate_get_old_files($path, $relative) {
135     global $OUTPUT;
136     $result = array();
137     if (!file_exists($path)) {
138         echo $OUTPUT->notification("File path doesn't exist: $path <br/> Please fix it manually.");
139         return array();
140     }
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
145             continue;
146         }
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));
153         }
154         unset($item);
155     }
156     unset($items);
157     return $result;