Search moodle.org's
Developer Documentation

See Release Notes
Long Term Support Release

  • Bug fixes for general core bugs in 4.1.x will end 13 November 2023 (12 months).
  • Bug fixes for security issues in 4.1.x will end 10 November 2025 (36 months).
  • PHP version: minimum PHP 7.4.0 Note: minimum PHP version has increased since Moodle 4.0. PHP 8.0.x is supported too.
/portfolio/ -> add.php (source)

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 . '&amp;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