See Release Notes
Long Term Support Release
Differences Between: [Versions 310 and 401] [Versions 311 and 401] [Versions 39 and 401] [Versions 400 and 401]
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 /** 18 * This file is the main controller to do with the portfolio export wizard. 19 * 20 * @package core_portfolio 21 * @copyright 2008 Penny Leach <penny@catalyst.net.nz>, 22 * Martin Dougiamas <http://dougiamas.com> 23 * @license http://www.gnu.org/copyleft/gpl.html GNU GPL 24 */ 25 require_once(__DIR__ . '/../config.php'); 26 27 if (empty($CFG->enableportfolios)) { 28 throw new \moodle_exception('disabled', 'portfolio'); 29 } 30 31 require_once($CFG->libdir . '/portfoliolib.php'); 32 require_once($CFG->libdir . '/portfolio/exporter.php'); 33 require_once($CFG->libdir . '/portfolio/caller.php'); 34 require_once($CFG->libdir . '/portfolio/plugin.php'); 35 36 $dataid = optional_param('id', 0, PARAM_INT); // The ID of partially completed export, corresponds to a record in portfolio_tempdata. 37 $type = optional_param('type', null, PARAM_SAFEDIR); // If we're returning from an external system (postcontrol) for a single-export only plugin. 38 $cancel = optional_param('cancel', 0, PARAM_RAW); // User has cancelled the request. 39 $cancelsure = optional_param('cancelsure', 0, PARAM_BOOL); // Make sure they confirm first. 40 $logreturn = optional_param('logreturn', 0, PARAM_BOOL); // When cancelling, we can also come from the log page, rather than the caller. 41 $instanceid = optional_param('instance', 0, PARAM_INT); // The instance of configured portfolio plugin. 42 $courseid = optional_param('course', 0, PARAM_INT); // The courseid the data being exported belongs to (caller object should provide this later). 43 $stage = optional_param('stage', PORTFOLIO_STAGE_CONFIG, PARAM_INT); // Stage of the export we're at (stored in the exporter). 44 $postcontrol = optional_param('postcontrol', 0, PARAM_INT); // When returning from some bounce to an external system, this gets passed. 45 $callbackcomponent = optional_param('callbackcomponent', null, PARAM_PATH); // Callback component eg mod_forum - the component of the exporting content. 46 $callbackclass = optional_param('callbackclass', null, PARAM_ALPHAEXT); // Callback class eg forum_portfolio_caller - the class to handle the exporting content. 47 $callerformats = optional_param('callerformats', null, PARAM_TAGLIST); // Comma separated list of formats the specific place exporting content supports. 48 49 require_login(); // this is selectively called again with $course later when we know for sure which one we're in. 50 $PAGE->set_context(context_system::instance()); 51 $PAGE->set_url('/portfolio/add.php', array('id' => $dataid, 'sesskey' => sesskey())); 52 $PAGE->set_pagelayout('admin'); 53 $exporter = null; 54 55 if ($postcontrol && $type && !$dataid) { 56 // we're returning from an external system that can't construct dynamic return urls 57 // this is a special "one export of this type only per session" case 58 if (portfolio_static_function($type, 'allows_multiple_exports')) { 59 throw new portfolio_exception('multiplesingleresume', 'portfolio'); 60 } 61 62 if (!$dataid = portfolio_export_type_to_id($type, $USER->id)) { 63 throw new portfolio_exception('invalidtempid', 'portfolio'); 64 } 65 } else { 66 // we can't do this in the above case, because we're redirecting straight back from an external system 67 // this is not really ideal, but since we're in a "staged" wizard, the session key is checked in other stages. 68 require_sesskey(); // pretty much everything in this page is a write that could be hijacked, so just do this at the top here 69 } 70 71 // if we have a dataid, it means we're in the middle of an export, 72 // so rewaken it and continue. 73 if (!empty($dataid)) { 74 try { 75 $exporter = portfolio_exporter::rewaken_object($dataid); 76 } catch (portfolio_exception $e) { 77 // this can happen in some cases, a cancel request is sent when something is already broken 78 // so process it elegantly and move on. 79 if ($cancel) { 80 if ($logreturn) { 81 redirect($CFG->wwwroot . '/user/portfoliologs.php'); 82 } 83 redirect($CFG->wwwroot); 84 } else { 85 throw $e; 86 } 87 } 88 // we have to wake it up first before we can cancel it 89 // so temporary directories etc get cleaned up. 90 if ($cancel) { 91 if ($cancelsure) { 92 $exporter->cancel_request($logreturn); 93 } else { 94 portfolio_export_pagesetup($PAGE, $exporter->get('caller')); 95 $exporter->print_header(get_string('confirmcancel', 'portfolio')); 96 echo $OUTPUT->box_start(); 97 $yesbutton = new single_button(new moodle_url('/portfolio/add.php', array('id' => $dataid, 'cancel' => 1, 'cancelsure' => 1, 'logreturn' => $logreturn)), get_string('yes')); 98 if ($logreturn) { 99 $nobutton = new single_button(new moodle_url('/user/portfoliologs.php'), get_string('no')); 100 } else { 101 $nobutton = new single_button(new moodle_url('/portfolio/add.php', array('id' => $dataid)), get_string('no')); 102 } 103 echo $OUTPUT->confirm(get_string('confirmcancel', 'portfolio'), $yesbutton, $nobutton); 104 echo $OUTPUT->box_end(); 105 echo $OUTPUT->footer(); 106 exit; 107 } 108 } 109 // verify we still belong to the correct user and permissions are still ok 110 $exporter->verify_rewaken(); 111 // if we don't have an instanceid in the exporter 112 // it means we've just posted from the 'choose portfolio instance' page 113 // so process that and start up the portfolio plugin 114 if (!$exporter->get('instance')) { 115 if ($instanceid) { 116 try { 117 $instance = portfolio_instance($instanceid); 118 } catch (portfolio_exception $e) { 119 portfolio_export_rethrow_exception($exporter, $e); 120 } 121 // this technically shouldn't happen but make sure anyway 122 if ($broken = portfolio_instance_sanity_check($instance)) { 123 throw new portfolio_export_exception($exporter, $broken[$instance->get('id')], 'portfolio_' . $instance->get('plugin')); 124 } 125 // now we're all set up, ready to go 126 $instance->set('user', $USER); 127 $exporter->set('instance', $instance); 128 $exporter->save(); 129 } 130 } 131 132 portfolio_export_pagesetup($PAGE, $exporter->get('caller')); // this calls require_login($course) if it can.. 133 134 // completely new request, look to see what information we've been passed and set up the exporter object. 135 } else { 136 // you cannot get here with no information for us, we must at least have the caller. 137 if (empty($_GET) && empty($_POST)) { 138 portfolio_exporter::print_expired_export(); 139 } 140 // we'e just posted here for the first time and have might the instance already 141 if ($instanceid) { 142 // this can throw exceptions but there's no point catching and rethrowing here 143 // as the exporter isn't created yet. 144 $instance = portfolio_instance($instanceid); 145 if ($broken = portfolio_instance_sanity_check($instance)) { 146 throw new portfolio_exception($broken[$instance->get('id')], 'portfolio_' . $instance->get('plugin')); 147 } 148 $instance->set('user', $USER); 149 } else { 150 $instance = null; 151 } 152 153 // we must be passed this from the caller, we cannot start a new export 154 // without knowing information about what part of moodle we come from. 155 if (empty($callbackcomponent) || empty($callbackclass)) { 156 debugging('no callback file or class'); 157 portfolio_exporter::print_expired_export(); 158 } 159 160 // so each place in moodle can pass callback args here 161 // process the entire request looking for ca_* 162 // be as lenient as possible while still being secure 163 // so only accept certain parameter types. 164 $callbackargs = array(); 165 foreach (array_keys(array_merge($_GET, $_POST)) as $key) { 166 if (strpos($key, 'ca_') === 0) { 167 if (!$value = optional_param($key, false, PARAM_ALPHAEXT)) { 168 if (!$value = optional_param($key, false, PARAM_FLOAT)) { 169 $value = optional_param($key, false, PARAM_PATH); 170 } 171 } 172 // strip off ca_ for niceness 173 $callbackargs[substr($key, 3)] = $value; 174 } 175 } 176 177 // Ensure that we found a file we can use, if not throw an exception. 178 portfolio_include_callback_file($callbackcomponent, $callbackclass); 179 180 $caller = new $callbackclass($callbackargs); 181 $caller->set('user', $USER); 182 if ($formats = explode(',', $callerformats)) { 183 $caller->set_formats_from_button($formats); 184 } 185 $caller->load_data(); 186 // this must check capabilities and either throw an exception or return false. 187 if (!$caller->check_permissions()) { 188 throw new portfolio_caller_exception('nopermissions', 'portfolio', $caller->get_return_url()); 189 } 190 191 portfolio_export_pagesetup($PAGE, $caller); // this calls require_login($course) if it can.. 192 193 // finally! set up the exporter object with the portfolio instance, and caller information elements 194 $exporter = new portfolio_exporter($instance, $caller, $callbackcomponent); 195 196 // set the export-specific variables, and save. 197 $exporter->set('user', $USER); 198 $exporter->save(); 199 } 200 201 if (!$exporter->get('instance')) { 202 // we've just arrived but have no instance 203 // in this case the exporter object and the caller object have been set up above 204 // so just make a little form to select the portfolio plugin instance, 205 // which is the last thing to do before starting the export. 206 // 207 // first check to make sure there is actually a point 208 $options = portfolio_instance_select( 209 portfolio_instances(), 210 $exporter->get('caller')->supported_formats(), 211 get_class($exporter->get('caller')), 212 $exporter->get('caller')->get_mimetype(), 213 'instance', 214 true, 215 true 216 ); 217 if (empty($options)) { 218 throw new portfolio_export_exception($exporter, 'noavailableplugins', 'portfolio'); 219 } else if (count($options) == 1) { 220 // no point displaying a form, just redirect. 221 $optionskeys = array_keys($options); 222 $instance = array_shift($optionskeys); 223 redirect($CFG->wwwroot . '/portfolio/add.php?id= ' . $exporter->get('id') . '&instance=' . $instance . '&sesskey=' . sesskey()); 224 } 225 // be very selective about not including this unless we really need to 226 require_once($CFG->libdir . '/portfolio/forms.php'); 227 $mform = new portfolio_instance_select('', array('id' => $exporter->get('id'), 'caller' => $exporter->get('caller'), 'options' => $options)); 228 if ($mform->is_cancelled()) { 229 $exporter->cancel_request(); 230 } else if ($fromform = $mform->get_data()){ 231 redirect($CFG->wwwroot . '/portfolio/add.php?instance=' . $fromform->instance . '&id=' . $exporter->get('id')); 232 exit; 233 } 234 else { 235 $exporter->print_header(get_string('selectplugin', 'portfolio')); 236 echo $OUTPUT->box_start(); 237 $mform->display(); 238 echo $OUTPUT->box_end(); 239 echo $OUTPUT->footer(); 240 exit; 241 } 242 } 243 244 // if we haven't been passed &stage= grab it from the exporter. 245 if (!$stage) { 246 $stage = $exporter->get('stage'); 247 } 248 249 // for places returning control to pass (rather than PORTFOLIO_STAGE_PACKAGE 250 // which is unstable if they can't get to the constant (eg external system) 251 $alreadystolen = false; 252 if ($postcontrol) { // the magic request variable plugins must pass on returning here 253 try { 254 // allow it to read whatever gets sent back in the request 255 // this is useful for plugins that redirect away and back again 256 // adding a token to the end of the url, for example box.net 257 $exporter->instance()->post_control($stage, array_merge($_GET, $_POST)); 258 } catch (portfolio_plugin_exception $e) { 259 portfolio_export_rethrow_exception($exporter, $e); 260 } 261 $alreadystolen = true; // remember this so we don't get caught in a steal control loop! 262 } 263 264 // actually do the work now.. 265 $exporter->process_stage($stage, $alreadystolen); 266 267
title
Description
Body
title
Description
Body
title
Description
Body
title
Body