PageRenderTime 52ms CodeModel.GetById 24ms RepoModel.GetById 0ms app.codeStats 0ms

/mod/quiz/backup/moodle2/restore_quiz_stepslib.php

https://bitbucket.org/kudutest/moodlegit
PHP | 359 lines | 232 code | 74 blank | 53 comment | 28 complexity | 40c15abacfe659f62630c43a482ea170 MD5 | raw file
  1. <?php
  2. // This file is part of Moodle - http://moodle.org/
  3. //
  4. // Moodle is free software: you can redistribute it and/or modify
  5. // it under the terms of the GNU General Public License as published by
  6. // the Free Software Foundation, either version 3 of the License, or
  7. // (at your option) any later version.
  8. //
  9. // Moodle is distributed in the hope that it will be useful,
  10. // but WITHOUT ANY WARRANTY; without even the implied warranty of
  11. // MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
  12. // GNU General Public License for more details.
  13. //
  14. // You should have received a copy of the GNU General Public License
  15. // along with Moodle. If not, see <http://www.gnu.org/licenses/>.
  16. /**
  17. * @package moodlecore
  18. * @subpackage backup-moodle2
  19. * @copyright 2010 onwards Eloy Lafuente (stronk7) {@link http://stronk7.com}
  20. * @license http://www.gnu.org/copyleft/gpl.html GNU GPL v3 or later
  21. */
  22. defined('MOODLE_INTERNAL') || die();
  23. /**
  24. * Structure step to restore one quiz activity
  25. *
  26. * @copyright 2010 onwards Eloy Lafuente (stronk7) {@link http://stronk7.com}
  27. * @license http://www.gnu.org/copyleft/gpl.html GNU GPL v3 or later
  28. */
  29. class restore_quiz_activity_structure_step extends restore_questions_activity_structure_step {
  30. protected function define_structure() {
  31. $paths = array();
  32. $userinfo = $this->get_setting_value('userinfo');
  33. $quiz = new restore_path_element('quiz', '/activity/quiz');
  34. $paths[] = $quiz;
  35. // A chance for access subplugings to set up their quiz data.
  36. $this->add_subplugin_structure('quizaccess', $quiz);
  37. $paths[] = new restore_path_element('quiz_question_instance',
  38. '/activity/quiz/question_instances/question_instance');
  39. $paths[] = new restore_path_element('quiz_feedback', '/activity/quiz/feedbacks/feedback');
  40. $paths[] = new restore_path_element('quiz_override', '/activity/quiz/overrides/override');
  41. if ($userinfo) {
  42. $paths[] = new restore_path_element('quiz_grade', '/activity/quiz/grades/grade');
  43. if ($this->task->get_old_moduleversion() > 2011010100) {
  44. // Restoring from a version 2.1 dev or later.
  45. // Process the new-style attempt data.
  46. $quizattempt = new restore_path_element('quiz_attempt',
  47. '/activity/quiz/attempts/attempt');
  48. $paths[] = $quizattempt;
  49. // Add states and sessions.
  50. $this->add_question_usages($quizattempt, $paths);
  51. // A chance for access subplugings to set up their attempt data.
  52. $this->add_subplugin_structure('quizaccess', $quizattempt);
  53. } else {
  54. // Restoring from a version 2.0.x+ or earlier.
  55. // Upgrade the legacy attempt data.
  56. $quizattempt = new restore_path_element('quiz_attempt_legacy',
  57. '/activity/quiz/attempts/attempt',
  58. true);
  59. $paths[] = $quizattempt;
  60. $this->add_legacy_question_attempt_data($quizattempt, $paths);
  61. }
  62. }
  63. // Return the paths wrapped into standard activity structure.
  64. return $this->prepare_activity_structure($paths);
  65. }
  66. protected function process_quiz($data) {
  67. global $CFG, $DB;
  68. $data = (object)$data;
  69. $oldid = $data->id;
  70. $data->course = $this->get_courseid();
  71. $data->timeopen = $this->apply_date_offset($data->timeopen);
  72. $data->timeclose = $this->apply_date_offset($data->timeclose);
  73. $data->timecreated = $this->apply_date_offset($data->timecreated);
  74. $data->timemodified = $this->apply_date_offset($data->timemodified);
  75. // Needed by {@link process_quiz_attempt_legacy}.
  76. $this->oldquizlayout = $data->questions;
  77. $data->questions = $this->questions_recode_layout($data->questions);
  78. // The setting quiz->attempts can come both in data->attempts and
  79. // data->attempts_number, handle both. MDL-26229.
  80. if (isset($data->attempts_number)) {
  81. $data->attempts = $data->attempts_number;
  82. unset($data->attempts_number);
  83. }
  84. // The old optionflags and penaltyscheme from 2.0 need to be mapped to
  85. // the new preferredbehaviour. See MDL-20636.
  86. if (!isset($data->preferredbehaviour)) {
  87. if (empty($data->optionflags)) {
  88. $data->preferredbehaviour = 'deferredfeedback';
  89. } else if (empty($data->penaltyscheme)) {
  90. $data->preferredbehaviour = 'adaptivenopenalty';
  91. } else {
  92. $data->preferredbehaviour = 'adaptive';
  93. }
  94. unset($data->optionflags);
  95. unset($data->penaltyscheme);
  96. }
  97. // The old review column from 2.0 need to be split into the seven new
  98. // review columns. See MDL-20636.
  99. if (isset($data->review)) {
  100. require_once($CFG->dirroot . '/mod/quiz/locallib.php');
  101. if (!defined('QUIZ_OLD_IMMEDIATELY')) {
  102. define('QUIZ_OLD_IMMEDIATELY', 0x3c003f);
  103. define('QUIZ_OLD_OPEN', 0x3c00fc0);
  104. define('QUIZ_OLD_CLOSED', 0x3c03f000);
  105. define('QUIZ_OLD_RESPONSES', 1*0x1041);
  106. define('QUIZ_OLD_SCORES', 2*0x1041);
  107. define('QUIZ_OLD_FEEDBACK', 4*0x1041);
  108. define('QUIZ_OLD_ANSWERS', 8*0x1041);
  109. define('QUIZ_OLD_SOLUTIONS', 16*0x1041);
  110. define('QUIZ_OLD_GENERALFEEDBACK', 32*0x1041);
  111. define('QUIZ_OLD_OVERALLFEEDBACK', 1*0x4440000);
  112. }
  113. $oldreview = $data->review;
  114. $data->reviewattempt =
  115. mod_quiz_display_options::DURING |
  116. ($oldreview & QUIZ_OLD_IMMEDIATELY & QUIZ_OLD_RESPONSES ?
  117. mod_quiz_display_options::IMMEDIATELY_AFTER : 0) |
  118. ($oldreview & QUIZ_OLD_OPEN & QUIZ_OLD_RESPONSES ?
  119. mod_quiz_display_options::LATER_WHILE_OPEN : 0) |
  120. ($oldreview & QUIZ_OLD_CLOSED & QUIZ_OLD_RESPONSES ?
  121. mod_quiz_display_options::AFTER_CLOSE : 0);
  122. $data->reviewcorrectness =
  123. mod_quiz_display_options::DURING |
  124. ($oldreview & QUIZ_OLD_IMMEDIATELY & QUIZ_OLD_SCORES ?
  125. mod_quiz_display_options::IMMEDIATELY_AFTER : 0) |
  126. ($oldreview & QUIZ_OLD_OPEN & QUIZ_OLD_SCORES ?
  127. mod_quiz_display_options::LATER_WHILE_OPEN : 0) |
  128. ($oldreview & QUIZ_OLD_CLOSED & QUIZ_OLD_SCORES ?
  129. mod_quiz_display_options::AFTER_CLOSE : 0);
  130. $data->reviewmarks =
  131. mod_quiz_display_options::DURING |
  132. ($oldreview & QUIZ_OLD_IMMEDIATELY & QUIZ_OLD_SCORES ?
  133. mod_quiz_display_options::IMMEDIATELY_AFTER : 0) |
  134. ($oldreview & QUIZ_OLD_OPEN & QUIZ_OLD_SCORES ?
  135. mod_quiz_display_options::LATER_WHILE_OPEN : 0) |
  136. ($oldreview & QUIZ_OLD_CLOSED & QUIZ_OLD_SCORES ?
  137. mod_quiz_display_options::AFTER_CLOSE : 0);
  138. $data->reviewspecificfeedback =
  139. ($oldreview & QUIZ_OLD_IMMEDIATELY & QUIZ_OLD_FEEDBACK ?
  140. mod_quiz_display_options::DURING : 0) |
  141. ($oldreview & QUIZ_OLD_IMMEDIATELY & QUIZ_OLD_FEEDBACK ?
  142. mod_quiz_display_options::IMMEDIATELY_AFTER : 0) |
  143. ($oldreview & QUIZ_OLD_OPEN & QUIZ_OLD_FEEDBACK ?
  144. mod_quiz_display_options::LATER_WHILE_OPEN : 0) |
  145. ($oldreview & QUIZ_OLD_CLOSED & QUIZ_OLD_FEEDBACK ?
  146. mod_quiz_display_options::AFTER_CLOSE : 0);
  147. $data->reviewgeneralfeedback =
  148. ($oldreview & QUIZ_OLD_IMMEDIATELY & QUIZ_OLD_GENERALFEEDBACK ?
  149. mod_quiz_display_options::DURING : 0) |
  150. ($oldreview & QUIZ_OLD_IMMEDIATELY & QUIZ_OLD_GENERALFEEDBACK ?
  151. mod_quiz_display_options::IMMEDIATELY_AFTER : 0) |
  152. ($oldreview & QUIZ_OLD_OPEN & QUIZ_OLD_GENERALFEEDBACK ?
  153. mod_quiz_display_options::LATER_WHILE_OPEN : 0) |
  154. ($oldreview & QUIZ_OLD_CLOSED & QUIZ_OLD_GENERALFEEDBACK ?
  155. mod_quiz_display_options::AFTER_CLOSE : 0);
  156. $data->reviewrightanswer =
  157. ($oldreview & QUIZ_OLD_IMMEDIATELY & QUIZ_OLD_ANSWERS ?
  158. mod_quiz_display_options::DURING : 0) |
  159. ($oldreview & QUIZ_OLD_IMMEDIATELY & QUIZ_OLD_ANSWERS ?
  160. mod_quiz_display_options::IMMEDIATELY_AFTER : 0) |
  161. ($oldreview & QUIZ_OLD_OPEN & QUIZ_OLD_ANSWERS ?
  162. mod_quiz_display_options::LATER_WHILE_OPEN : 0) |
  163. ($oldreview & QUIZ_OLD_CLOSED & QUIZ_OLD_ANSWERS ?
  164. mod_quiz_display_options::AFTER_CLOSE : 0);
  165. $data->reviewoverallfeedback =
  166. 0 |
  167. ($oldreview & QUIZ_OLD_IMMEDIATELY & QUIZ_OLD_OVERALLFEEDBACK ?
  168. mod_quiz_display_options::IMMEDIATELY_AFTER : 0) |
  169. ($oldreview & QUIZ_OLD_OPEN & QUIZ_OLD_OVERALLFEEDBACK ?
  170. mod_quiz_display_options::LATER_WHILE_OPEN : 0) |
  171. ($oldreview & QUIZ_OLD_CLOSED & QUIZ_OLD_OVERALLFEEDBACK ?
  172. mod_quiz_display_options::AFTER_CLOSE : 0);
  173. }
  174. // The old popup column from from <= 2.1 need to be mapped to
  175. // the new browsersecurity. See MDL-29627.
  176. if (!isset($data->browsersecurity)) {
  177. if (empty($data->popup)) {
  178. $data->browsersecurity = '-';
  179. } else if ($data->popup == 1) {
  180. $data->browsersecurity = 'securewindow';
  181. } else if ($data->popup == 2) {
  182. $data->browsersecurity = 'safebrowser';
  183. } else {
  184. $data->preferredbehaviour = '-';
  185. }
  186. unset($data->popup);
  187. }
  188. // Insert the quiz record.
  189. $newitemid = $DB->insert_record('quiz', $data);
  190. // Immediately after inserting "activity" record, call this.
  191. $this->apply_activity_instance($newitemid);
  192. }
  193. protected function process_quiz_question_instance($data) {
  194. global $DB;
  195. $data = (object)$data;
  196. $oldid = $data->id;
  197. $data->quiz = $this->get_new_parentid('quiz');
  198. $data->question = $this->get_mappingid('question', $data->question);
  199. $DB->insert_record('quiz_question_instances', $data);
  200. }
  201. protected function process_quiz_feedback($data) {
  202. global $DB;
  203. $data = (object)$data;
  204. $oldid = $data->id;
  205. $data->quizid = $this->get_new_parentid('quiz');
  206. $newitemid = $DB->insert_record('quiz_feedback', $data);
  207. $this->set_mapping('quiz_feedback', $oldid, $newitemid, true); // Has related files.
  208. }
  209. protected function process_quiz_override($data) {
  210. global $DB;
  211. $data = (object)$data;
  212. $oldid = $data->id;
  213. // Based on userinfo, we'll restore user overides or no.
  214. $userinfo = $this->get_setting_value('userinfo');
  215. // Skip user overrides if we are not restoring userinfo.
  216. if (!$userinfo && !is_null($data->userid)) {
  217. return;
  218. }
  219. $data->quiz = $this->get_new_parentid('quiz');
  220. $data->userid = $this->get_mappingid('user', $data->userid);
  221. $data->groupid = $this->get_mappingid('group', $data->groupid);
  222. $data->timeopen = $this->apply_date_offset($data->timeopen);
  223. $data->timeclose = $this->apply_date_offset($data->timeclose);
  224. $newitemid = $DB->insert_record('quiz_overrides', $data);
  225. // Add mapping, restore of logs needs it.
  226. $this->set_mapping('quiz_override', $oldid, $newitemid);
  227. }
  228. protected function process_quiz_grade($data) {
  229. global $DB;
  230. $data = (object)$data;
  231. $oldid = $data->id;
  232. $data->quiz = $this->get_new_parentid('quiz');
  233. $data->userid = $this->get_mappingid('user', $data->userid);
  234. $data->grade = $data->gradeval;
  235. $data->timemodified = $this->apply_date_offset($data->timemodified);
  236. $DB->insert_record('quiz_grades', $data);
  237. }
  238. protected function process_quiz_attempt($data) {
  239. $data = (object)$data;
  240. $data->quiz = $this->get_new_parentid('quiz');
  241. $data->attempt = $data->attemptnum;
  242. $data->userid = $this->get_mappingid('user', $data->userid);
  243. $data->timestart = $this->apply_date_offset($data->timestart);
  244. $data->timefinish = $this->apply_date_offset($data->timefinish);
  245. $data->timemodified = $this->apply_date_offset($data->timemodified);
  246. if (!empty($data->timecheckstate)) {
  247. $data->timecheckstate = $this->apply_date_offset($data->timecheckstate);
  248. } else {
  249. $data->timecheckstate = 0;
  250. }
  251. // Deals with up-grading pre-2.3 back-ups to 2.3+.
  252. if (!isset($data->state)) {
  253. if ($data->timefinish > 0) {
  254. $data->state = 'finished';
  255. } else {
  256. $data->state = 'inprogress';
  257. }
  258. }
  259. // The data is actually inserted into the database later in inform_new_usage_id.
  260. $this->currentquizattempt = clone($data);
  261. }
  262. protected function process_quiz_attempt_legacy($data) {
  263. global $DB;
  264. $this->process_quiz_attempt($data);
  265. $quiz = $DB->get_record('quiz', array('id' => $this->get_new_parentid('quiz')));
  266. $quiz->oldquestions = $this->oldquizlayout;
  267. $this->process_legacy_quiz_attempt_data($data, $quiz);
  268. }
  269. protected function inform_new_usage_id($newusageid) {
  270. global $DB;
  271. $data = $this->currentquizattempt;
  272. $oldid = $data->id;
  273. $data->uniqueid = $newusageid;
  274. $newitemid = $DB->insert_record('quiz_attempts', $data);
  275. // Save quiz_attempt->id mapping, because logs use it.
  276. $this->set_mapping('quiz_attempt', $oldid, $newitemid, false);
  277. }
  278. protected function after_execute() {
  279. parent::after_execute();
  280. // Add quiz related files, no need to match by itemname (just internally handled context).
  281. $this->add_related_files('mod_quiz', 'intro', null);
  282. // Add feedback related files, matching by itemname = 'quiz_feedback'.
  283. $this->add_related_files('mod_quiz', 'feedback', 'quiz_feedback');
  284. }
  285. }