. /** * General database importer class * * @package core_dtl * @copyright 2008 Andrei Bautu * @license http://www.gnu.org/copyleft/gpl.html GNU GPL v3 or later */ defined('MOODLE_INTERNAL') || die(); /** * Base class for database import operations. This class implements * basic callbacks for import operations and defines the @see import_database * method as a common method for all importers. In general, subclasses will * override import_database and call other methods in appropriate moments. * Between a single pair of calls to @see begin_database_import and * @see finish_database_import, multiple non-overlapping pairs of calls may * be made to @see begin_table_import and @see finish_database_import for * different tables. * Between one pair of calls to @see begin_table_import and * @see finish_database_import multiple calls may be made to * @see import_table_data for the same table. * This class can be used directly, if the standard control flow (defined above) * is respected. */ class database_importer { /** @var moodle_database Connection to the target database (a @see moodle_database object). */ protected $mdb; /** @var database_manager Database manager of the target database (a @see database_manager object). */ protected $manager; /** @var xmldb_structure Target database schema in XMLDB format (a @see xmldb_structure object). */ protected $schema; /** * Boolean flag - whether or not to check that XML database schema matches * the RDBMS database schema before importing (used by * @see begin_database_import). * @var bool */ protected $check_schema; /** @var string How to use transactions. */ protected $transactionmode = 'allinone'; /** @var moodle_transaction Transaction object */ protected $transaction; /** * Object constructor. * * @param moodle_database $mdb Connection to the target database (a * @see moodle_database object). Use null to use the current $DB connection. * @param boolean $check_schema - whether or not to check that XML database * schema matches the RDBMS database schema before importing (inside * @see begin_database_import). */ public function __construct(moodle_database $mdb, $check_schema=true) { $this->mdb = $mdb; $this->manager = $mdb->get_manager(); $this->schema = $this->manager->get_install_xml_schema(); $this->check_schema = $check_schema; } /** * How to use transactions during the import. * @param string $mode 'pertable', 'allinone' or 'none'. */ public function set_transaction_mode($mode) { if (!in_array($mode, array('pertable', 'allinone', 'none'))) { throw new coding_exception('Unknown transaction mode', $mode); } $this->transactionmode = $mode; } /** * Callback function. Should be called only once database per import * operation, before any database changes are made. It will check the database * schema if @see check_schema is true * * @throws dbtransfer_exception if any checking (e.g. database schema, Moodle * version) fails * * @param float $version the version of the system which generated the data * @param string $timestamp the timestamp of the data (in ISO 8601) format. * @return void */ public function begin_database_import($version, $timestamp) { global $CFG; if (!$this->mdb->get_tables()) { // No tables present yet, time to create all tables. $this->manager->install_from_xmldb_structure($this->schema); } if (round($version, 2) !== round($CFG->version, 2)) { // version might be in decimal format too $a = (object)array('schemaver'=>$version, 'currentver'=>$CFG->version); throw new dbtransfer_exception('importversionmismatchexception', $a); } $options = array('changedcolumns' => false); // Column types may be fixed by transfer. if ($this->check_schema and $errors = $this->manager->check_database_schema($this->schema, $options)) { $details = ''; foreach ($errors as $table=>$items) { $details .= '