2009-05-25 17:44:05 +00:00
|
|
|
<?php
|
|
|
|
|
|
|
|
// This file is part of Moodle - http://moodle.org/
|
|
|
|
//
|
|
|
|
// Moodle is free software: you can redistribute it and/or modify
|
|
|
|
// it under the terms of the GNU General Public License as published by
|
|
|
|
// the Free Software Foundation, either version 3 of the License, or
|
|
|
|
// (at your option) any later version.
|
|
|
|
//
|
|
|
|
// Moodle is distributed in the hope that it will be useful,
|
|
|
|
// but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
|
|
// GNU General Public License for more details.
|
|
|
|
//
|
|
|
|
// You should have received a copy of the GNU General Public License
|
|
|
|
// along with Moodle. If not, see <http://www.gnu.org/licenses/>.
|
2009-01-29 21:17:52 +00:00
|
|
|
|
|
|
|
/**
|
2009-05-26 09:52:38 +00:00
|
|
|
* Various upgrade/install related functions and classes.
|
2009-01-29 21:17:52 +00:00
|
|
|
*
|
2009-05-25 17:44:05 +00:00
|
|
|
* @package moodlecore
|
|
|
|
* @subpackage upgrade
|
|
|
|
* @copyright 1999 onwards Martin Dougiamas (http://dougiamas.com)
|
|
|
|
* @license http://www.gnu.org/copyleft/gpl.html GNU GPL v3 or later
|
2009-01-29 21:17:52 +00:00
|
|
|
*/
|
|
|
|
|
2009-05-26 05:18:52 +00:00
|
|
|
/** UPGRADE_LOG_NORMAL = 0 */
|
2009-01-29 21:17:52 +00:00
|
|
|
define('UPGRADE_LOG_NORMAL', 0);
|
2009-05-26 05:18:52 +00:00
|
|
|
/** UPGRADE_LOG_NOTICE = 1 */
|
2009-01-29 21:17:52 +00:00
|
|
|
define('UPGRADE_LOG_NOTICE', 1);
|
2009-05-26 05:18:52 +00:00
|
|
|
/** UPGRADE_LOG_ERROR = 2 */
|
2009-01-31 20:07:32 +00:00
|
|
|
define('UPGRADE_LOG_ERROR', 2);
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Exception indicating unknown error during upgrade.
|
2009-05-26 05:18:52 +00:00
|
|
|
*
|
|
|
|
* @package moodlecore
|
|
|
|
* @subpackage upgrade
|
|
|
|
* @copyright 1999 onwards Martin Dougiamas (http://dougiamas.com)
|
|
|
|
* @license http://www.gnu.org/copyleft/gpl.html GNU GPL v3 or later
|
2009-01-31 20:07:32 +00:00
|
|
|
*/
|
|
|
|
class upgrade_exception extends moodle_exception {
|
2010-03-31 07:41:31 +00:00
|
|
|
function __construct($plugin, $version, $debuginfo=NULL) {
|
2009-01-31 20:07:32 +00:00
|
|
|
global $CFG;
|
|
|
|
$a = (object)array('plugin'=>$plugin, 'version'=>$version);
|
2010-03-31 07:41:31 +00:00
|
|
|
parent::__construct('upgradeerror', 'admin', "$CFG->wwwroot/$CFG->admin/index.php", $a, $debuginfo);
|
2009-01-31 20:07:32 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Exception indicating downgrade error during upgrade.
|
2009-05-26 05:18:52 +00:00
|
|
|
*
|
|
|
|
* @package moodlecore
|
|
|
|
* @subpackage upgrade
|
|
|
|
* @copyright 1999 onwards Martin Dougiamas (http://dougiamas.com)
|
|
|
|
* @license http://www.gnu.org/copyleft/gpl.html GNU GPL v3 or later
|
2009-01-31 20:07:32 +00:00
|
|
|
*/
|
|
|
|
class downgrade_exception extends moodle_exception {
|
|
|
|
function __construct($plugin, $oldversion, $newversion) {
|
|
|
|
global $CFG;
|
|
|
|
$plugin = is_null($plugin) ? 'moodle' : $plugin;
|
|
|
|
$a = (object)array('plugin'=>$plugin, 'oldversion'=>$oldversion, 'newversion'=>$newversion);
|
|
|
|
parent::__construct('cannotdowngrade', 'debug', "$CFG->wwwroot/$CFG->admin/index.php", $a);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2009-05-26 05:18:52 +00:00
|
|
|
/**
|
|
|
|
* @package moodlecore
|
|
|
|
* @subpackage upgrade
|
|
|
|
* @copyright 1999 onwards Martin Dougiamas (http://dougiamas.com)
|
|
|
|
* @license http://www.gnu.org/copyleft/gpl.html GNU GPL v3 or later
|
|
|
|
*/
|
2009-01-31 20:07:32 +00:00
|
|
|
class upgrade_requires_exception extends moodle_exception {
|
|
|
|
function __construct($plugin, $pluginversion, $currentmoodle, $requiremoodle) {
|
|
|
|
global $CFG;
|
|
|
|
$a = new object();
|
|
|
|
$a->pluginname = $plugin;
|
|
|
|
$a->pluginversion = $pluginversion;
|
|
|
|
$a->currentmoodle = $currentmoodle;
|
2009-06-19 14:25:56 +00:00
|
|
|
$a->requiremoodle = $requiremoodle;
|
2009-01-31 20:07:32 +00:00
|
|
|
parent::__construct('pluginrequirementsnotmet', 'error', "$CFG->wwwroot/$CFG->admin/index.php", $a);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2009-05-26 05:18:52 +00:00
|
|
|
/**
|
|
|
|
* @package moodlecore
|
|
|
|
* @subpackage upgrade
|
|
|
|
* @copyright 1999 onwards Martin Dougiamas (http://dougiamas.com)
|
|
|
|
* @license http://www.gnu.org/copyleft/gpl.html GNU GPL v3 or later
|
|
|
|
*/
|
2009-01-31 20:07:32 +00:00
|
|
|
class plugin_defective_exception extends moodle_exception {
|
|
|
|
function __construct($plugin, $details) {
|
|
|
|
global $CFG;
|
|
|
|
parent::__construct('detectedbrokenplugin', 'error', "$CFG->wwwroot/$CFG->admin/index.php", $plugin, $details);
|
|
|
|
}
|
|
|
|
}
|
2009-01-29 21:17:52 +00:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Insert or update log display entry. Entry may already exist.
|
|
|
|
* $module, $action must be unique
|
|
|
|
*
|
2009-05-26 05:18:52 +00:00
|
|
|
* @global object
|
2009-01-29 21:17:52 +00:00
|
|
|
* @param string $module
|
|
|
|
* @param string $action
|
|
|
|
* @param string $mtable
|
|
|
|
* @param string $field
|
|
|
|
* @return void
|
|
|
|
*
|
|
|
|
*/
|
|
|
|
function update_log_display_entry($module, $action, $mtable, $field) {
|
|
|
|
global $DB;
|
|
|
|
|
|
|
|
if ($type = $DB->get_record('log_display', array('module'=>$module, 'action'=>$action))) {
|
|
|
|
$type->mtable = $mtable;
|
|
|
|
$type->field = $field;
|
|
|
|
$DB->update_record('log_display', $type);
|
|
|
|
|
|
|
|
} else {
|
|
|
|
$type = new object();
|
|
|
|
$type->module = $module;
|
|
|
|
$type->action = $action;
|
|
|
|
$type->mtable = $mtable;
|
|
|
|
$type->field = $field;
|
|
|
|
|
|
|
|
$DB->insert_record('log_display', $type, false);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Upgrade savepoint, marks end of each upgrade block.
|
|
|
|
* It stores new main version, resets upgrade timeout
|
|
|
|
* and abort upgrade if user cancels page loading.
|
|
|
|
*
|
|
|
|
* Please do not make large upgrade blocks with lots of operations,
|
|
|
|
* for example when adding tables keep only one table operation per block.
|
|
|
|
*
|
2009-05-26 05:18:52 +00:00
|
|
|
* @global object
|
2009-01-29 21:17:52 +00:00
|
|
|
* @param bool $result false if upgrade step failed, true if completed
|
|
|
|
* @param string or float $version main version
|
|
|
|
* @param bool $allowabort allow user to abort script execution here
|
|
|
|
* @return void
|
|
|
|
*/
|
|
|
|
function upgrade_main_savepoint($result, $version, $allowabort=true) {
|
|
|
|
global $CFG;
|
|
|
|
|
2009-01-31 20:07:32 +00:00
|
|
|
if (!$result) {
|
2009-06-26 09:02:05 +00:00
|
|
|
throw new upgrade_exception(null, $version);
|
2009-01-31 20:07:32 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
if ($CFG->version >= $version) {
|
|
|
|
// something really wrong is going on in main upgrade script
|
|
|
|
throw new downgrade_exception(null, $CFG->version, $version);
|
2009-01-29 21:17:52 +00:00
|
|
|
}
|
|
|
|
|
2009-01-31 20:07:32 +00:00
|
|
|
set_config('version', $version);
|
|
|
|
upgrade_log(UPGRADE_LOG_NORMAL, null, 'Upgrade savepoint reached');
|
|
|
|
|
2009-01-29 21:17:52 +00:00
|
|
|
// reset upgrade timeout to default
|
|
|
|
upgrade_set_timeout();
|
|
|
|
|
|
|
|
// this is a safe place to stop upgrades if user aborts page loading
|
|
|
|
if ($allowabort and connection_aborted()) {
|
|
|
|
die;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Module upgrade savepoint, marks end of module upgrade blocks
|
|
|
|
* It stores module version, resets upgrade timeout
|
|
|
|
* and abort upgrade if user cancels page loading.
|
|
|
|
*
|
2009-05-26 05:18:52 +00:00
|
|
|
* @global object
|
2009-01-29 21:17:52 +00:00
|
|
|
* @param bool $result false if upgrade step failed, true if completed
|
|
|
|
* @param string or float $version main version
|
|
|
|
* @param string $modname name of module
|
|
|
|
* @param bool $allowabort allow user to abort script execution here
|
|
|
|
* @return void
|
|
|
|
*/
|
|
|
|
function upgrade_mod_savepoint($result, $version, $modname, $allowabort=true) {
|
|
|
|
global $DB;
|
|
|
|
|
2009-01-31 20:07:32 +00:00
|
|
|
if (!$result) {
|
2009-06-26 09:02:05 +00:00
|
|
|
throw new upgrade_exception("mod_$modname", $version);
|
2009-01-31 20:07:32 +00:00
|
|
|
}
|
|
|
|
|
2009-01-29 21:17:52 +00:00
|
|
|
if (!$module = $DB->get_record('modules', array('name'=>$modname))) {
|
|
|
|
print_error('modulenotexist', 'debug', '', $modname);
|
|
|
|
}
|
|
|
|
|
2009-01-31 20:07:32 +00:00
|
|
|
if ($module->version >= $version) {
|
|
|
|
// something really wrong is going on in upgrade script
|
2009-06-26 09:02:05 +00:00
|
|
|
throw new downgrade_exception("mod_$modname", $module->version, $version);
|
2009-01-29 21:17:52 +00:00
|
|
|
}
|
2009-01-31 20:07:32 +00:00
|
|
|
$module->version = $version;
|
|
|
|
$DB->update_record('modules', $module);
|
2009-06-26 09:02:05 +00:00
|
|
|
upgrade_log(UPGRADE_LOG_NORMAL, "mod_$modname", 'Upgrade savepoint reached');
|
2009-01-29 21:17:52 +00:00
|
|
|
|
|
|
|
// reset upgrade timeout to default
|
|
|
|
upgrade_set_timeout();
|
|
|
|
|
|
|
|
// this is a safe place to stop upgrades if user aborts page loading
|
|
|
|
if ($allowabort and connection_aborted()) {
|
|
|
|
die;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Blocks upgrade savepoint, marks end of blocks upgrade blocks
|
|
|
|
* It stores block version, resets upgrade timeout
|
|
|
|
* and abort upgrade if user cancels page loading.
|
|
|
|
*
|
2009-05-26 05:18:52 +00:00
|
|
|
* @global object
|
2009-01-29 21:17:52 +00:00
|
|
|
* @param bool $result false if upgrade step failed, true if completed
|
|
|
|
* @param string or float $version main version
|
|
|
|
* @param string $blockname name of block
|
|
|
|
* @param bool $allowabort allow user to abort script execution here
|
|
|
|
* @return void
|
|
|
|
*/
|
2009-01-31 20:07:32 +00:00
|
|
|
function upgrade_block_savepoint($result, $version, $blockname, $allowabort=true) {
|
2009-01-29 21:17:52 +00:00
|
|
|
global $DB;
|
|
|
|
|
2009-01-31 20:07:32 +00:00
|
|
|
if (!$result) {
|
2009-06-26 09:02:05 +00:00
|
|
|
throw new upgrade_exception("block_$blockname", $version);
|
2009-01-31 20:07:32 +00:00
|
|
|
}
|
|
|
|
|
2009-01-29 21:17:52 +00:00
|
|
|
if (!$block = $DB->get_record('block', array('name'=>$blockname))) {
|
|
|
|
print_error('blocknotexist', 'debug', '', $blockname);
|
|
|
|
}
|
|
|
|
|
2009-01-31 20:07:32 +00:00
|
|
|
if ($block->version >= $version) {
|
|
|
|
// something really wrong is going on in upgrade script
|
2009-06-26 09:02:05 +00:00
|
|
|
throw new downgrade_exception("block_$blockname", $block->version, $version);
|
2009-01-29 21:17:52 +00:00
|
|
|
}
|
2009-01-31 20:07:32 +00:00
|
|
|
$block->version = $version;
|
|
|
|
$DB->update_record('block', $block);
|
2009-06-26 09:02:05 +00:00
|
|
|
upgrade_log(UPGRADE_LOG_NORMAL, "block_$blockname", 'Upgrade savepoint reached');
|
2009-01-29 21:17:52 +00:00
|
|
|
|
|
|
|
// reset upgrade timeout to default
|
|
|
|
upgrade_set_timeout();
|
|
|
|
|
|
|
|
// this is a safe place to stop upgrades if user aborts page loading
|
|
|
|
if ($allowabort and connection_aborted()) {
|
|
|
|
die;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Plugins upgrade savepoint, marks end of blocks upgrade blocks
|
|
|
|
* It stores plugin version, resets upgrade timeout
|
|
|
|
* and abort upgrade if user cancels page loading.
|
|
|
|
*
|
|
|
|
* @param bool $result false if upgrade step failed, true if completed
|
|
|
|
* @param string or float $version main version
|
|
|
|
* @param string $type name of plugin
|
|
|
|
* @param string $dir location of plugin
|
|
|
|
* @param bool $allowabort allow user to abort script execution here
|
|
|
|
* @return void
|
|
|
|
*/
|
2009-06-19 14:25:56 +00:00
|
|
|
function upgrade_plugin_savepoint($result, $version, $type, $plugin, $allowabort=true) {
|
|
|
|
$component = $type.'_'.$plugin;
|
|
|
|
|
2009-01-31 20:07:32 +00:00
|
|
|
if (!$result) {
|
2009-06-19 14:25:56 +00:00
|
|
|
throw new upgrade_exception($component, $version);
|
2009-01-29 21:17:52 +00:00
|
|
|
}
|
|
|
|
|
2009-06-19 14:25:56 +00:00
|
|
|
$installedversion = get_config($component, 'version');
|
2009-01-31 20:07:32 +00:00
|
|
|
if ($installedversion >= $version) {
|
|
|
|
// Something really wrong is going on in the upgrade script
|
|
|
|
throw new downgrade_exception($component, $installedversion, $version);
|
|
|
|
}
|
2009-06-19 14:25:56 +00:00
|
|
|
set_config('version', $version, $component);
|
2009-01-31 20:07:32 +00:00
|
|
|
upgrade_log(UPGRADE_LOG_NORMAL, $component, 'Upgrade savepoint reached');
|
|
|
|
|
2009-01-29 21:17:52 +00:00
|
|
|
// Reset upgrade timeout to default
|
|
|
|
upgrade_set_timeout();
|
|
|
|
|
|
|
|
// This is a safe place to stop upgrades if user aborts page loading
|
|
|
|
if ($allowabort and connection_aborted()) {
|
|
|
|
die;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Upgrade plugins
|
|
|
|
* @param string $type The type of plugins that should be updated (e.g. 'enrol', 'qtype')
|
2009-06-19 14:25:56 +00:00
|
|
|
* return void
|
2009-01-29 21:17:52 +00:00
|
|
|
*/
|
2009-06-19 14:25:56 +00:00
|
|
|
function upgrade_plugins($type, $startcallback, $endcallback, $verbose) {
|
2009-01-29 21:17:52 +00:00
|
|
|
global $CFG, $DB;
|
|
|
|
|
|
|
|
/// special cases
|
|
|
|
if ($type === 'mod') {
|
2009-05-26 19:02:11 +00:00
|
|
|
return upgrade_plugins_modules($startcallback, $endcallback, $verbose);
|
2009-01-31 20:07:32 +00:00
|
|
|
} else if ($type === 'block') {
|
2009-05-26 19:02:11 +00:00
|
|
|
return upgrade_plugins_blocks($startcallback, $endcallback, $verbose);
|
2009-01-29 21:17:52 +00:00
|
|
|
}
|
|
|
|
|
2009-06-19 14:25:56 +00:00
|
|
|
$plugs = get_plugin_list($type);
|
2009-01-29 21:17:52 +00:00
|
|
|
|
2009-06-19 14:25:56 +00:00
|
|
|
foreach ($plugs as $plug=>$fullplug) {
|
|
|
|
$component = $type.'_'.$plug; // standardised plugin name
|
2009-01-29 21:17:52 +00:00
|
|
|
|
2009-01-31 20:07:32 +00:00
|
|
|
if (!is_readable($fullplug.'/version.php')) {
|
|
|
|
continue;
|
2009-01-29 21:17:52 +00:00
|
|
|
}
|
|
|
|
|
2009-01-31 20:07:32 +00:00
|
|
|
$plugin = new object();
|
|
|
|
require($fullplug.'/version.php'); // defines $plugin with version etc
|
2009-01-29 21:17:52 +00:00
|
|
|
|
2009-01-31 20:07:32 +00:00
|
|
|
if (empty($plugin->version)) {
|
|
|
|
throw new plugin_defective_exception($component, 'Missing version value in version.php');
|
2009-01-29 21:17:52 +00:00
|
|
|
}
|
|
|
|
|
2009-06-19 14:25:56 +00:00
|
|
|
$plugin->name = $plug;
|
|
|
|
$plugin->fullname = $component;
|
2009-01-31 20:07:32 +00:00
|
|
|
|
|
|
|
|
2009-01-29 21:17:52 +00:00
|
|
|
if (!empty($plugin->requires)) {
|
|
|
|
if ($plugin->requires > $CFG->version) {
|
2009-01-31 20:07:32 +00:00
|
|
|
throw new upgrade_requires_exception($component, $plugin->version, $CFG->version, $plugin->requires);
|
2009-01-29 21:17:52 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2009-08-06 19:59:16 +00:00
|
|
|
// try to recover from interrupted install.php if needed
|
|
|
|
if (file_exists($fullplug.'/db/install.php')) {
|
|
|
|
if (get_config($plugin->fullname, 'installrunning')) {
|
|
|
|
require_once($fullplug.'/db/install.php');
|
|
|
|
$recover_install_function = 'xmldb_'.$plugin->fullname.'_install_recovery';
|
|
|
|
if (function_exists($recover_install_function)) {
|
|
|
|
$startcallback($component, true, $verbose);
|
|
|
|
$recover_install_function();
|
|
|
|
unset_config('installrunning', 'block_'.$plugin->fullname);
|
|
|
|
update_capabilities($component);
|
2009-09-13 21:26:51 +00:00
|
|
|
external_update_descriptions($component);
|
2009-08-06 19:59:16 +00:00
|
|
|
events_update_definition($component);
|
|
|
|
message_update_providers($component);
|
mnet MDL-21261 large overhaul. This commit changes:
- The way that mnet rpc functions are registered. Plugins must now
create db/mnet.php which is an array, similar to services.php. This
*replaces* the old mnet_publishes() functions. version.php must be
bumped to trigger this.
- More information about each rpc-available function is stored in the
database, including the class it belongs to, the file it is found in,
and whether or not it is static. Methods that are not static must be
in a class with a constructor that takes no arguments (this can easily
be achieved with a small wrapper if necessary)
- The xmlrpc dispatcher has been rewritten to remove all the
dependencies on hardcoded information about auth,mnet,portfolio and
repository, and just use the information in the database.
- The old hardcoded hidden mnet/testclient.php has been moved to the
Admin menu under "Development" and rewritten.
- The xmlrpc introspection method profiling is now using php and zend
reflection - which is a lot nicer than the old way, which was using a
php-based php parser. This fixes some inconsistent handling of
methods without arguments that were advertising their return value as
the only method parameter. While this is a *fix*, it breaks BC
slightly - the old 1.9 broken mnet/testclient.php will now not work
properly with 2.0
- Dangerous mode is still supported, but old mod/*/rpclib.php is
now unsupported, due to the fact that any plugin can export mnet
functions with db/mnet.php. This is a slight BC break.
Still TODO:
- TEST TEST TEST
- Document the two small BC breaks in release notes
- Document the contract for db/mnet.php
2010-01-28 05:25:50 +00:00
|
|
|
upgrade_plugin_mnet_functions($component);
|
2009-08-06 19:59:16 +00:00
|
|
|
$endcallback($component, true, $verbose);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2009-01-29 21:17:52 +00:00
|
|
|
|
2009-08-06 19:59:16 +00:00
|
|
|
$installedversion = get_config($plugin->fullname, 'version');
|
2009-01-31 20:07:32 +00:00
|
|
|
if (empty($installedversion)) { // new installation
|
2009-05-26 19:02:11 +00:00
|
|
|
$startcallback($component, true, $verbose);
|
2009-01-29 21:17:52 +00:00
|
|
|
|
2009-01-31 20:07:32 +00:00
|
|
|
/// Install tables if defined
|
|
|
|
if (file_exists($fullplug.'/db/install.xml')) {
|
|
|
|
$DB->get_manager()->install_from_xmldb_file($fullplug.'/db/install.xml');
|
2009-01-29 21:17:52 +00:00
|
|
|
}
|
2009-08-06 19:59:16 +00:00
|
|
|
|
|
|
|
/// store version
|
|
|
|
upgrade_plugin_savepoint(true, $plugin->version, $type, $plug, false);
|
|
|
|
|
2009-01-31 20:07:32 +00:00
|
|
|
/// execute post install file
|
|
|
|
if (file_exists($fullplug.'/db/install.php')) {
|
|
|
|
require_once($fullplug.'/db/install.php');
|
2009-08-06 19:59:16 +00:00
|
|
|
set_config('installrunning', 1, 'block_'.$plugin->fullname);
|
2009-01-31 20:07:32 +00:00
|
|
|
$post_install_function = 'xmldb_'.$plugin->fullname.'_install';;
|
|
|
|
$post_install_function();
|
2009-08-06 19:59:16 +00:00
|
|
|
unset_config('installrunning', 'block_'.$plugin->fullname);
|
2009-01-31 20:07:32 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/// Install various components
|
|
|
|
update_capabilities($component);
|
2009-09-13 21:26:51 +00:00
|
|
|
external_update_descriptions($component);
|
2009-01-31 20:07:32 +00:00
|
|
|
events_update_definition($component);
|
|
|
|
message_update_providers($component);
|
mnet MDL-21261 large overhaul. This commit changes:
- The way that mnet rpc functions are registered. Plugins must now
create db/mnet.php which is an array, similar to services.php. This
*replaces* the old mnet_publishes() functions. version.php must be
bumped to trigger this.
- More information about each rpc-available function is stored in the
database, including the class it belongs to, the file it is found in,
and whether or not it is static. Methods that are not static must be
in a class with a constructor that takes no arguments (this can easily
be achieved with a small wrapper if necessary)
- The xmlrpc dispatcher has been rewritten to remove all the
dependencies on hardcoded information about auth,mnet,portfolio and
repository, and just use the information in the database.
- The old hardcoded hidden mnet/testclient.php has been moved to the
Admin menu under "Development" and rewritten.
- The xmlrpc introspection method profiling is now using php and zend
reflection - which is a lot nicer than the old way, which was using a
php-based php parser. This fixes some inconsistent handling of
methods without arguments that were advertising their return value as
the only method parameter. While this is a *fix*, it breaks BC
slightly - the old 1.9 broken mnet/testclient.php will now not work
properly with 2.0
- Dangerous mode is still supported, but old mod/*/rpclib.php is
now unsupported, due to the fact that any plugin can export mnet
functions with db/mnet.php. This is a slight BC break.
Still TODO:
- TEST TEST TEST
- Document the two small BC breaks in release notes
- Document the contract for db/mnet.php
2010-01-28 05:25:50 +00:00
|
|
|
upgrade_plugin_mnet_functions($component);
|
2009-01-31 20:07:32 +00:00
|
|
|
|
2010-07-13 15:09:44 +00:00
|
|
|
purge_all_caches();
|
2009-05-26 19:02:11 +00:00
|
|
|
$endcallback($component, true, $verbose);
|
2009-01-31 20:07:32 +00:00
|
|
|
|
|
|
|
} else if ($installedversion < $plugin->version) { // upgrade
|
|
|
|
/// Run the upgrade function for the plugin.
|
2009-05-26 19:02:11 +00:00
|
|
|
$startcallback($component, false, $verbose);
|
2009-01-31 20:07:32 +00:00
|
|
|
|
|
|
|
if (is_readable($fullplug.'/db/upgrade.php')) {
|
|
|
|
require_once($fullplug.'/db/upgrade.php'); // defines upgrading function
|
|
|
|
|
|
|
|
$newupgrade_function = 'xmldb_'.$plugin->fullname.'_upgrade';
|
|
|
|
$result = $newupgrade_function($installedversion);
|
|
|
|
} else {
|
|
|
|
$result = true;
|
|
|
|
}
|
|
|
|
|
|
|
|
$installedversion = get_config($plugin->fullname, 'version');
|
|
|
|
if ($installedversion < $plugin->version) {
|
|
|
|
// store version if not already there
|
|
|
|
upgrade_plugin_savepoint($result, $plugin->version, $type, $plug, false);
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Upgrade various components
|
|
|
|
update_capabilities($component);
|
2009-09-13 21:26:51 +00:00
|
|
|
external_update_descriptions($component);
|
2009-01-31 20:07:32 +00:00
|
|
|
events_update_definition($component);
|
|
|
|
message_update_providers($component);
|
mnet MDL-21261 large overhaul. This commit changes:
- The way that mnet rpc functions are registered. Plugins must now
create db/mnet.php which is an array, similar to services.php. This
*replaces* the old mnet_publishes() functions. version.php must be
bumped to trigger this.
- More information about each rpc-available function is stored in the
database, including the class it belongs to, the file it is found in,
and whether or not it is static. Methods that are not static must be
in a class with a constructor that takes no arguments (this can easily
be achieved with a small wrapper if necessary)
- The xmlrpc dispatcher has been rewritten to remove all the
dependencies on hardcoded information about auth,mnet,portfolio and
repository, and just use the information in the database.
- The old hardcoded hidden mnet/testclient.php has been moved to the
Admin menu under "Development" and rewritten.
- The xmlrpc introspection method profiling is now using php and zend
reflection - which is a lot nicer than the old way, which was using a
php-based php parser. This fixes some inconsistent handling of
methods without arguments that were advertising their return value as
the only method parameter. While this is a *fix*, it breaks BC
slightly - the old 1.9 broken mnet/testclient.php will now not work
properly with 2.0
- Dangerous mode is still supported, but old mod/*/rpclib.php is
now unsupported, due to the fact that any plugin can export mnet
functions with db/mnet.php. This is a slight BC break.
Still TODO:
- TEST TEST TEST
- Document the two small BC breaks in release notes
- Document the contract for db/mnet.php
2010-01-28 05:25:50 +00:00
|
|
|
upgrade_plugin_mnet_functions($component);
|
2009-01-31 20:07:32 +00:00
|
|
|
|
2010-07-13 15:09:44 +00:00
|
|
|
purge_all_caches();
|
2009-05-26 19:02:11 +00:00
|
|
|
$endcallback($component, false, $verbose);
|
2009-01-31 20:07:32 +00:00
|
|
|
|
|
|
|
} else if ($installedversion > $plugin->version) {
|
|
|
|
throw new downgrade_exception($component, $installedversion, $plugin->version);
|
2009-01-29 21:17:52 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Find and check all modules and load them up or upgrade them if necessary
|
2009-05-26 05:18:52 +00:00
|
|
|
*
|
|
|
|
* @global object
|
|
|
|
* @global object
|
2009-01-29 21:17:52 +00:00
|
|
|
*/
|
2009-05-26 19:02:11 +00:00
|
|
|
function upgrade_plugins_modules($startcallback, $endcallback, $verbose) {
|
2009-01-29 21:17:52 +00:00
|
|
|
global $CFG, $DB;
|
|
|
|
|
2009-06-19 14:25:56 +00:00
|
|
|
$mods = get_plugin_list('mod');
|
2009-01-29 21:17:52 +00:00
|
|
|
|
2009-06-19 14:25:56 +00:00
|
|
|
foreach ($mods as $mod=>$fullmod) {
|
2009-01-29 21:17:52 +00:00
|
|
|
|
|
|
|
if ($mod == 'NEWMODULE') { // Someone has unzipped the template, ignore it
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
2009-06-19 14:25:56 +00:00
|
|
|
$component = 'mod_'.$mod;
|
2009-01-29 21:17:52 +00:00
|
|
|
|
2009-01-31 20:07:32 +00:00
|
|
|
if (!is_readable($fullmod.'/version.php')) {
|
|
|
|
throw new plugin_defective_exception($component, 'Missing version.php');
|
2009-01-29 21:17:52 +00:00
|
|
|
}
|
|
|
|
|
2009-01-31 20:07:32 +00:00
|
|
|
$module = new object();
|
|
|
|
require($fullmod .'/version.php'); // defines $module with version etc
|
2009-01-29 21:17:52 +00:00
|
|
|
|
2009-01-31 20:07:32 +00:00
|
|
|
if (empty($module->version)) {
|
2009-05-12 08:12:48 +00:00
|
|
|
if (isset($module->version)) {
|
|
|
|
// Version is empty but is set - it means its value is 0 or ''. Let us skip such module.
|
2010-05-22 20:24:28 +00:00
|
|
|
// This is intended for developers so they can work on the early stages of the module.
|
2009-05-12 08:12:48 +00:00
|
|
|
continue;
|
|
|
|
}
|
2009-01-31 20:07:32 +00:00
|
|
|
throw new plugin_defective_exception($component, 'Missing version value in version.php');
|
2009-01-29 21:17:52 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
if (!empty($module->requires)) {
|
|
|
|
if ($module->requires > $CFG->version) {
|
2009-01-31 20:07:32 +00:00
|
|
|
throw new upgrade_requires_exception($component, $module->version, $CFG->version, $module->requires);
|
2009-01-29 21:17:52 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
$module->name = $mod; // The name MUST match the directory
|
|
|
|
|
2009-01-31 20:07:32 +00:00
|
|
|
$currmodule = $DB->get_record('modules', array('name'=>$module->name));
|
2009-01-29 21:17:52 +00:00
|
|
|
|
2009-08-06 19:59:16 +00:00
|
|
|
if (file_exists($fullmod.'/db/install.php')) {
|
|
|
|
if (get_config($module->name, 'installrunning')) {
|
|
|
|
require_once($fullmod.'/db/install.php');
|
|
|
|
$recover_install_function = 'xmldb_'.$module->name.'_install_recovery';
|
|
|
|
if (function_exists($recover_install_function)) {
|
|
|
|
$startcallback($component, true, $verbose);
|
|
|
|
$recover_install_function();
|
|
|
|
unset_config('installrunning', $module->name);
|
|
|
|
// Install various components too
|
|
|
|
update_capabilities($component);
|
2009-09-13 21:26:51 +00:00
|
|
|
external_update_descriptions($component);
|
2009-08-06 19:59:16 +00:00
|
|
|
events_update_definition($component);
|
|
|
|
message_update_providers($component);
|
mnet MDL-21261 large overhaul. This commit changes:
- The way that mnet rpc functions are registered. Plugins must now
create db/mnet.php which is an array, similar to services.php. This
*replaces* the old mnet_publishes() functions. version.php must be
bumped to trigger this.
- More information about each rpc-available function is stored in the
database, including the class it belongs to, the file it is found in,
and whether or not it is static. Methods that are not static must be
in a class with a constructor that takes no arguments (this can easily
be achieved with a small wrapper if necessary)
- The xmlrpc dispatcher has been rewritten to remove all the
dependencies on hardcoded information about auth,mnet,portfolio and
repository, and just use the information in the database.
- The old hardcoded hidden mnet/testclient.php has been moved to the
Admin menu under "Development" and rewritten.
- The xmlrpc introspection method profiling is now using php and zend
reflection - which is a lot nicer than the old way, which was using a
php-based php parser. This fixes some inconsistent handling of
methods without arguments that were advertising their return value as
the only method parameter. While this is a *fix*, it breaks BC
slightly - the old 1.9 broken mnet/testclient.php will now not work
properly with 2.0
- Dangerous mode is still supported, but old mod/*/rpclib.php is
now unsupported, due to the fact that any plugin can export mnet
functions with db/mnet.php. This is a slight BC break.
Still TODO:
- TEST TEST TEST
- Document the two small BC breaks in release notes
- Document the contract for db/mnet.php
2010-01-28 05:25:50 +00:00
|
|
|
upgrade_plugin_mnet_functions($component);
|
2009-08-06 19:59:16 +00:00
|
|
|
$endcallback($component, true, $verbose);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2009-01-31 20:07:32 +00:00
|
|
|
if (empty($currmodule->version)) {
|
2009-05-26 19:02:11 +00:00
|
|
|
$startcallback($component, true, $verbose);
|
2009-01-29 21:17:52 +00:00
|
|
|
|
2009-01-31 20:07:32 +00:00
|
|
|
/// Execute install.xml (XMLDB) - must be present in all modules
|
|
|
|
$DB->get_manager()->install_from_xmldb_file($fullmod.'/db/install.xml');
|
2009-01-29 21:17:52 +00:00
|
|
|
|
2009-07-03 13:35:36 +00:00
|
|
|
/// Add record into modules table - may be needed in install.php already
|
|
|
|
$module->id = $DB->insert_record('modules', $module);
|
|
|
|
|
2009-01-29 21:17:52 +00:00
|
|
|
/// Post installation hook - optional
|
|
|
|
if (file_exists("$fullmod/db/install.php")) {
|
|
|
|
require_once("$fullmod/db/install.php");
|
2009-08-06 19:59:16 +00:00
|
|
|
// Set installation running flag, we need to recover after exception or error
|
|
|
|
set_config('installrunning', 1, $module->name);
|
2009-01-29 21:17:52 +00:00
|
|
|
$post_install_function = 'xmldb_'.$module->name.'_install';;
|
|
|
|
$post_install_function();
|
2009-08-06 19:59:16 +00:00
|
|
|
unset_config('installrunning', $module->name);
|
2009-01-29 21:17:52 +00:00
|
|
|
}
|
|
|
|
|
2009-01-31 20:07:32 +00:00
|
|
|
/// Install various components
|
|
|
|
update_capabilities($component);
|
2009-09-13 21:26:51 +00:00
|
|
|
external_update_descriptions($component);
|
2009-01-31 20:07:32 +00:00
|
|
|
events_update_definition($component);
|
|
|
|
message_update_providers($component);
|
mnet MDL-21261 large overhaul. This commit changes:
- The way that mnet rpc functions are registered. Plugins must now
create db/mnet.php which is an array, similar to services.php. This
*replaces* the old mnet_publishes() functions. version.php must be
bumped to trigger this.
- More information about each rpc-available function is stored in the
database, including the class it belongs to, the file it is found in,
and whether or not it is static. Methods that are not static must be
in a class with a constructor that takes no arguments (this can easily
be achieved with a small wrapper if necessary)
- The xmlrpc dispatcher has been rewritten to remove all the
dependencies on hardcoded information about auth,mnet,portfolio and
repository, and just use the information in the database.
- The old hardcoded hidden mnet/testclient.php has been moved to the
Admin menu under "Development" and rewritten.
- The xmlrpc introspection method profiling is now using php and zend
reflection - which is a lot nicer than the old way, which was using a
php-based php parser. This fixes some inconsistent handling of
methods without arguments that were advertising their return value as
the only method parameter. While this is a *fix*, it breaks BC
slightly - the old 1.9 broken mnet/testclient.php will now not work
properly with 2.0
- Dangerous mode is still supported, but old mod/*/rpclib.php is
now unsupported, due to the fact that any plugin can export mnet
functions with db/mnet.php. This is a slight BC break.
Still TODO:
- TEST TEST TEST
- Document the two small BC breaks in release notes
- Document the contract for db/mnet.php
2010-01-28 05:25:50 +00:00
|
|
|
upgrade_plugin_mnet_functions($component);
|
2009-01-31 20:07:32 +00:00
|
|
|
|
2010-07-13 15:09:44 +00:00
|
|
|
purge_all_caches();
|
2009-05-26 19:02:11 +00:00
|
|
|
$endcallback($component, true, $verbose);
|
2009-01-29 21:17:52 +00:00
|
|
|
|
2009-01-31 20:07:32 +00:00
|
|
|
} else if ($currmodule->version < $module->version) {
|
|
|
|
/// If versions say that we need to upgrade but no upgrade files are available, notify and continue
|
2009-05-26 19:02:11 +00:00
|
|
|
$startcallback($component, false, $verbose);
|
2009-01-31 20:07:32 +00:00
|
|
|
|
|
|
|
if (is_readable($fullmod.'/db/upgrade.php')) {
|
|
|
|
require_once($fullmod.'/db/upgrade.php'); // defines new upgrading function
|
|
|
|
$newupgrade_function = 'xmldb_'.$module->name.'_upgrade';
|
|
|
|
$result = $newupgrade_function($currmodule->version, $module);
|
|
|
|
} else {
|
|
|
|
$result = true;
|
|
|
|
}
|
|
|
|
|
|
|
|
$currmodule = $DB->get_record('modules', array('name'=>$module->name));
|
|
|
|
if ($currmodule->version < $module->version) {
|
|
|
|
// store version if not already there
|
|
|
|
upgrade_mod_savepoint($result, $module->version, $mod, false);
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Upgrade various components
|
|
|
|
update_capabilities($component);
|
2009-09-13 21:26:51 +00:00
|
|
|
external_update_descriptions($component);
|
2009-01-31 20:07:32 +00:00
|
|
|
events_update_definition($component);
|
|
|
|
message_update_providers($component);
|
mnet MDL-21261 large overhaul. This commit changes:
- The way that mnet rpc functions are registered. Plugins must now
create db/mnet.php which is an array, similar to services.php. This
*replaces* the old mnet_publishes() functions. version.php must be
bumped to trigger this.
- More information about each rpc-available function is stored in the
database, including the class it belongs to, the file it is found in,
and whether or not it is static. Methods that are not static must be
in a class with a constructor that takes no arguments (this can easily
be achieved with a small wrapper if necessary)
- The xmlrpc dispatcher has been rewritten to remove all the
dependencies on hardcoded information about auth,mnet,portfolio and
repository, and just use the information in the database.
- The old hardcoded hidden mnet/testclient.php has been moved to the
Admin menu under "Development" and rewritten.
- The xmlrpc introspection method profiling is now using php and zend
reflection - which is a lot nicer than the old way, which was using a
php-based php parser. This fixes some inconsistent handling of
methods without arguments that were advertising their return value as
the only method parameter. While this is a *fix*, it breaks BC
slightly - the old 1.9 broken mnet/testclient.php will now not work
properly with 2.0
- Dangerous mode is still supported, but old mod/*/rpclib.php is
now unsupported, due to the fact that any plugin can export mnet
functions with db/mnet.php. This is a slight BC break.
Still TODO:
- TEST TEST TEST
- Document the two small BC breaks in release notes
- Document the contract for db/mnet.php
2010-01-28 05:25:50 +00:00
|
|
|
upgrade_plugin_mnet_functions($component);
|
2009-01-31 20:07:32 +00:00
|
|
|
|
2010-07-13 15:09:44 +00:00
|
|
|
purge_all_caches();
|
2009-01-31 20:07:32 +00:00
|
|
|
|
2009-05-26 19:02:11 +00:00
|
|
|
$endcallback($component, false, $verbose);
|
2009-01-31 20:07:32 +00:00
|
|
|
|
|
|
|
} else if ($currmodule->version > $module->version) {
|
|
|
|
throw new downgrade_exception($component, $currmodule->version, $module->version);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2009-01-29 21:17:52 +00:00
|
|
|
|
|
|
|
|
2009-01-31 20:07:32 +00:00
|
|
|
/**
|
|
|
|
* This function finds all available blocks and install them
|
|
|
|
* into blocks table or do all the upgrade process if newer.
|
2009-05-26 05:18:52 +00:00
|
|
|
*
|
|
|
|
* @global object
|
|
|
|
* @global object
|
2009-01-31 20:07:32 +00:00
|
|
|
*/
|
2009-05-26 19:02:11 +00:00
|
|
|
function upgrade_plugins_blocks($startcallback, $endcallback, $verbose) {
|
2009-01-31 20:07:32 +00:00
|
|
|
global $CFG, $DB;
|
|
|
|
|
|
|
|
require_once($CFG->dirroot.'/blocks/moodleblock.class.php');
|
|
|
|
|
|
|
|
$blocktitles = array(); // we do not want duplicate titles
|
|
|
|
|
|
|
|
//Is this a first install
|
|
|
|
$first_install = null;
|
|
|
|
|
2009-06-19 14:25:56 +00:00
|
|
|
$blocks = get_plugin_list('block');
|
2009-01-31 20:07:32 +00:00
|
|
|
|
2009-06-19 14:25:56 +00:00
|
|
|
foreach ($blocks as $blockname=>$fullblock) {
|
2009-01-31 20:07:32 +00:00
|
|
|
|
|
|
|
if (is_null($first_install)) {
|
2010-04-06 19:52:45 +00:00
|
|
|
$first_install = ($DB->count_records('block_instances') == 0);
|
2009-01-31 20:07:32 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
if ($blockname == 'NEWBLOCK') { // Someone has unzipped the template, ignore it
|
|
|
|
continue;
|
2009-01-29 21:17:52 +00:00
|
|
|
}
|
|
|
|
|
2009-06-19 14:25:56 +00:00
|
|
|
$component = 'block_'.$blockname;
|
2009-01-29 21:17:52 +00:00
|
|
|
|
2010-07-04 12:18:22 +00:00
|
|
|
if (!is_readable($fullblock.'/version.php')) {
|
|
|
|
throw new plugin_defective_exception('block/'.$blockname, 'Missing version.php file.');
|
|
|
|
}
|
|
|
|
$plugin = new object();
|
|
|
|
$plugin->version = NULL;
|
|
|
|
$plugin->cron = 0;
|
|
|
|
include($fullblock.'/version.php');
|
|
|
|
$block = $plugin;
|
|
|
|
|
2009-01-31 20:07:32 +00:00
|
|
|
if (!is_readable($fullblock.'/block_'.$blockname.'.php')) {
|
|
|
|
throw new plugin_defective_exception('block/'.$blockname, 'Missing main block class file.');
|
2009-01-29 21:17:52 +00:00
|
|
|
}
|
2010-07-04 12:18:22 +00:00
|
|
|
include_once($fullblock.'/block_'.$blockname.'.php');
|
2009-01-29 21:17:52 +00:00
|
|
|
|
2009-01-31 20:07:32 +00:00
|
|
|
$classname = 'block_'.$blockname;
|
|
|
|
|
|
|
|
if (!class_exists($classname)) {
|
|
|
|
throw new plugin_defective_exception($component, 'Can not load main class.');
|
|
|
|
}
|
|
|
|
|
2010-05-22 20:24:28 +00:00
|
|
|
$blockobj = new $classname; // This is what we'll be testing
|
2009-01-31 20:07:32 +00:00
|
|
|
$blocktitle = $blockobj->get_title();
|
|
|
|
|
|
|
|
// OK, it's as we all hoped. For further tests, the object will do them itself.
|
|
|
|
if (!$blockobj->_self_test()) {
|
|
|
|
throw new plugin_defective_exception($component, 'Self test failed.');
|
|
|
|
}
|
|
|
|
|
|
|
|
$block->name = $blockname; // The name MUST match the directory
|
|
|
|
$block->multiple = $blockobj->instance_allow_multiple() ? 1 : 0;
|
|
|
|
|
|
|
|
if (empty($block->version)) {
|
|
|
|
throw new plugin_defective_exception($component, 'Missing block version.');
|
|
|
|
}
|
|
|
|
|
|
|
|
$currblock = $DB->get_record('block', array('name'=>$block->name));
|
|
|
|
|
2009-08-06 19:59:16 +00:00
|
|
|
if (file_exists($fullblock.'/db/install.php')) {
|
|
|
|
if (get_config('block_'.$blockname, 'installrunning')) {
|
|
|
|
require_once($fullblock.'/db/install.php');
|
|
|
|
$recover_install_function = 'xmldb_block_'.$blockname.'_install_recovery';
|
|
|
|
if (function_exists($recover_install_function)) {
|
|
|
|
$startcallback($component, true, $verbose);
|
|
|
|
$recover_install_function();
|
|
|
|
unset_config('installrunning', 'block_'.$blockname);
|
|
|
|
// Install various components
|
|
|
|
update_capabilities($component);
|
2009-09-13 21:26:51 +00:00
|
|
|
external_update_descriptions($component);
|
2009-08-06 19:59:16 +00:00
|
|
|
events_update_definition($component);
|
|
|
|
message_update_providers($component);
|
mnet MDL-21261 large overhaul. This commit changes:
- The way that mnet rpc functions are registered. Plugins must now
create db/mnet.php which is an array, similar to services.php. This
*replaces* the old mnet_publishes() functions. version.php must be
bumped to trigger this.
- More information about each rpc-available function is stored in the
database, including the class it belongs to, the file it is found in,
and whether or not it is static. Methods that are not static must be
in a class with a constructor that takes no arguments (this can easily
be achieved with a small wrapper if necessary)
- The xmlrpc dispatcher has been rewritten to remove all the
dependencies on hardcoded information about auth,mnet,portfolio and
repository, and just use the information in the database.
- The old hardcoded hidden mnet/testclient.php has been moved to the
Admin menu under "Development" and rewritten.
- The xmlrpc introspection method profiling is now using php and zend
reflection - which is a lot nicer than the old way, which was using a
php-based php parser. This fixes some inconsistent handling of
methods without arguments that were advertising their return value as
the only method parameter. While this is a *fix*, it breaks BC
slightly - the old 1.9 broken mnet/testclient.php will now not work
properly with 2.0
- Dangerous mode is still supported, but old mod/*/rpclib.php is
now unsupported, due to the fact that any plugin can export mnet
functions with db/mnet.php. This is a slight BC break.
Still TODO:
- TEST TEST TEST
- Document the two small BC breaks in release notes
- Document the contract for db/mnet.php
2010-01-28 05:25:50 +00:00
|
|
|
upgrade_plugin_mnet_functions($component);
|
2009-08-06 19:59:16 +00:00
|
|
|
$endcallback($component, true, $verbose);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2009-01-31 20:07:32 +00:00
|
|
|
if (empty($currblock->version)) { // block not installed yet, so install it
|
|
|
|
// If it allows multiples, start with it enabled
|
|
|
|
|
|
|
|
$conflictblock = array_search($blocktitle, $blocktitles);
|
|
|
|
if ($conflictblock !== false) {
|
|
|
|
// Duplicate block titles are not allowed, they confuse people
|
|
|
|
// AND PHP's associative arrays ;)
|
|
|
|
throw new plugin_defective_exception($component, get_string('blocknameconflict', '', (object)array('name'=>$block->name, 'conflict'=>$conflictblock)));
|
|
|
|
}
|
2009-05-26 19:02:11 +00:00
|
|
|
$startcallback($component, true, $verbose);
|
2009-01-31 20:07:32 +00:00
|
|
|
|
|
|
|
if (file_exists($fullblock.'/db/install.xml')) {
|
|
|
|
$DB->get_manager()->install_from_xmldb_file($fullblock.'/db/install.xml');
|
|
|
|
}
|
|
|
|
$block->id = $DB->insert_record('block', $block);
|
|
|
|
|
|
|
|
if (file_exists($fullblock.'/db/install.php')) {
|
|
|
|
require_once($fullblock.'/db/install.php');
|
2009-08-06 19:59:16 +00:00
|
|
|
// Set installation running flag, we need to recover after exception or error
|
|
|
|
set_config('installrunning', 1, 'block_'.$blockname);
|
2009-01-31 20:07:32 +00:00
|
|
|
$post_install_function = 'xmldb_block_'.$blockname.'_install';;
|
|
|
|
$post_install_function();
|
2009-08-06 19:59:16 +00:00
|
|
|
unset_config('installrunning', 'block_'.$blockname);
|
2009-01-31 20:07:32 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
$blocktitles[$block->name] = $blocktitle;
|
|
|
|
|
|
|
|
// Install various components
|
|
|
|
update_capabilities($component);
|
2009-09-13 21:26:51 +00:00
|
|
|
external_update_descriptions($component);
|
2009-01-31 20:07:32 +00:00
|
|
|
events_update_definition($component);
|
|
|
|
message_update_providers($component);
|
mnet MDL-21261 large overhaul. This commit changes:
- The way that mnet rpc functions are registered. Plugins must now
create db/mnet.php which is an array, similar to services.php. This
*replaces* the old mnet_publishes() functions. version.php must be
bumped to trigger this.
- More information about each rpc-available function is stored in the
database, including the class it belongs to, the file it is found in,
and whether or not it is static. Methods that are not static must be
in a class with a constructor that takes no arguments (this can easily
be achieved with a small wrapper if necessary)
- The xmlrpc dispatcher has been rewritten to remove all the
dependencies on hardcoded information about auth,mnet,portfolio and
repository, and just use the information in the database.
- The old hardcoded hidden mnet/testclient.php has been moved to the
Admin menu under "Development" and rewritten.
- The xmlrpc introspection method profiling is now using php and zend
reflection - which is a lot nicer than the old way, which was using a
php-based php parser. This fixes some inconsistent handling of
methods without arguments that were advertising their return value as
the only method parameter. While this is a *fix*, it breaks BC
slightly - the old 1.9 broken mnet/testclient.php will now not work
properly with 2.0
- Dangerous mode is still supported, but old mod/*/rpclib.php is
now unsupported, due to the fact that any plugin can export mnet
functions with db/mnet.php. This is a slight BC break.
Still TODO:
- TEST TEST TEST
- Document the two small BC breaks in release notes
- Document the contract for db/mnet.php
2010-01-28 05:25:50 +00:00
|
|
|
upgrade_plugin_mnet_functions($component);
|
2009-01-31 20:07:32 +00:00
|
|
|
|
2010-07-13 15:09:44 +00:00
|
|
|
purge_all_caches();
|
2009-05-26 19:02:11 +00:00
|
|
|
$endcallback($component, true, $verbose);
|
2009-01-31 20:07:32 +00:00
|
|
|
|
|
|
|
} else if ($currblock->version < $block->version) {
|
2009-05-26 19:02:11 +00:00
|
|
|
$startcallback($component, false, $verbose);
|
2009-01-31 20:07:32 +00:00
|
|
|
|
|
|
|
if (is_readable($fullblock.'/db/upgrade.php')) {
|
|
|
|
require_once($fullblock.'/db/upgrade.php'); // defines new upgrading function
|
|
|
|
$newupgrade_function = 'xmldb_block_'.$blockname.'_upgrade';
|
|
|
|
$result = $newupgrade_function($currblock->version, $block);
|
|
|
|
} else {
|
|
|
|
$result = true;
|
|
|
|
}
|
|
|
|
|
|
|
|
$currblock = $DB->get_record('block', array('name'=>$block->name));
|
|
|
|
if ($currblock->version < $block->version) {
|
|
|
|
// store version if not already there
|
|
|
|
upgrade_block_savepoint($result, $block->version, $block->name, false);
|
|
|
|
}
|
|
|
|
|
|
|
|
if ($currblock->cron != $block->cron) {
|
|
|
|
// update cron flag if needed
|
|
|
|
$currblock->cron = $block->cron;
|
|
|
|
$DB->update_record('block', $currblock);
|
|
|
|
}
|
|
|
|
|
2010-05-22 20:24:28 +00:00
|
|
|
// Upgrade various components
|
2009-01-31 20:07:32 +00:00
|
|
|
update_capabilities($component);
|
2009-09-13 21:26:51 +00:00
|
|
|
external_update_descriptions($component);
|
2009-06-19 14:25:56 +00:00
|
|
|
events_update_definition($component);
|
2009-01-31 20:07:32 +00:00
|
|
|
message_update_providers($component);
|
mnet MDL-21261 large overhaul. This commit changes:
- The way that mnet rpc functions are registered. Plugins must now
create db/mnet.php which is an array, similar to services.php. This
*replaces* the old mnet_publishes() functions. version.php must be
bumped to trigger this.
- More information about each rpc-available function is stored in the
database, including the class it belongs to, the file it is found in,
and whether or not it is static. Methods that are not static must be
in a class with a constructor that takes no arguments (this can easily
be achieved with a small wrapper if necessary)
- The xmlrpc dispatcher has been rewritten to remove all the
dependencies on hardcoded information about auth,mnet,portfolio and
repository, and just use the information in the database.
- The old hardcoded hidden mnet/testclient.php has been moved to the
Admin menu under "Development" and rewritten.
- The xmlrpc introspection method profiling is now using php and zend
reflection - which is a lot nicer than the old way, which was using a
php-based php parser. This fixes some inconsistent handling of
methods without arguments that were advertising their return value as
the only method parameter. While this is a *fix*, it breaks BC
slightly - the old 1.9 broken mnet/testclient.php will now not work
properly with 2.0
- Dangerous mode is still supported, but old mod/*/rpclib.php is
now unsupported, due to the fact that any plugin can export mnet
functions with db/mnet.php. This is a slight BC break.
Still TODO:
- TEST TEST TEST
- Document the two small BC breaks in release notes
- Document the contract for db/mnet.php
2010-01-28 05:25:50 +00:00
|
|
|
upgrade_plugin_mnet_functions($component);
|
2009-01-31 20:07:32 +00:00
|
|
|
|
2010-07-13 15:09:44 +00:00
|
|
|
purge_all_caches();
|
2009-05-26 19:02:11 +00:00
|
|
|
$endcallback($component, false, $verbose);
|
2009-01-31 20:07:32 +00:00
|
|
|
|
|
|
|
} else if ($currblock->version > $block->version) {
|
|
|
|
throw new downgrade_exception($component, $currblock->version, $block->version);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// Finally, if we are in the first_install of BLOCKS setup frontpage and admin page blocks
|
|
|
|
if ($first_install) {
|
|
|
|
//Iterate over each course - there should be only site course here now
|
|
|
|
if ($courses = $DB->get_records('course')) {
|
|
|
|
foreach ($courses as $course) {
|
2009-05-07 07:05:22 +00:00
|
|
|
blocks_add_default_course_blocks($course);
|
2009-01-29 21:17:52 +00:00
|
|
|
}
|
|
|
|
}
|
2009-01-31 20:07:32 +00:00
|
|
|
|
2009-05-07 07:05:22 +00:00
|
|
|
blocks_add_default_system_blocks();
|
2009-01-31 20:07:32 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2009-09-13 21:26:51 +00:00
|
|
|
/**
|
|
|
|
* Web service discovery function used during install and upgrade.
|
|
|
|
* @param string $component name of component (moodle, mod_assignment, etc.)
|
|
|
|
* @return void
|
|
|
|
*/
|
|
|
|
function external_update_descriptions($component) {
|
|
|
|
global $DB;
|
|
|
|
|
|
|
|
$defpath = get_component_directory($component).'/db/services.php';
|
|
|
|
|
|
|
|
if (!file_exists($defpath)) {
|
|
|
|
external_delete_descriptions($component);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
// load new info
|
|
|
|
$functions = array();
|
|
|
|
$services = array();
|
|
|
|
include($defpath);
|
|
|
|
|
|
|
|
// update all function fist
|
|
|
|
$dbfunctions = $DB->get_records('external_functions', array('component'=>$component));
|
|
|
|
foreach ($dbfunctions as $dbfunction) {
|
|
|
|
if (empty($functions[$dbfunction->name])) {
|
|
|
|
$DB->delete_records('external_functions', array('id'=>$dbfunction->id));
|
|
|
|
// do not delete functions from external_services_functions, beacuse
|
|
|
|
// we want to notify admins when functions used in custom services disappear
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
$function = $functions[$dbfunction->name];
|
|
|
|
unset($functions[$dbfunction->name]);
|
|
|
|
$function['classpath'] = empty($function['classpath']) ? null : $function['classpath'];
|
|
|
|
|
|
|
|
$update = false;
|
|
|
|
if ($dbfunction->classname != $function['classname']) {
|
|
|
|
$dbfunction->classname = $function['classname'];
|
|
|
|
$update = true;
|
|
|
|
}
|
|
|
|
if ($dbfunction->methodname != $function['methodname']) {
|
|
|
|
$dbfunction->methodname = $function['methodname'];
|
|
|
|
$update = true;
|
|
|
|
}
|
|
|
|
if ($dbfunction->classpath != $function['classpath']) {
|
|
|
|
$dbfunction->classpath = $function['classpath'];
|
|
|
|
$update = true;
|
|
|
|
}
|
|
|
|
if ($update) {
|
|
|
|
$DB->update_record('external_functions', $dbfunction);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
foreach ($functions as $fname => $function) {
|
|
|
|
$dbfunction = new object();
|
|
|
|
$dbfunction->name = $fname;
|
|
|
|
$dbfunction->classname = $function['classname'];
|
|
|
|
$dbfunction->methodname = $function['methodname'];
|
|
|
|
$dbfunction->classpath = empty($function['classpath']) ? null : $function['classpath'];
|
|
|
|
$dbfunction->component = $component;
|
|
|
|
$dbfunction->id = $DB->insert_record('external_functions', $dbfunction);
|
|
|
|
}
|
|
|
|
unset($functions);
|
|
|
|
|
|
|
|
// now deal with services
|
|
|
|
$dbservices = $DB->get_records('external_services', array('component'=>$component));
|
|
|
|
foreach ($dbservices as $dbservice) {
|
|
|
|
if (empty($services[$dbservice->name])) {
|
|
|
|
$DB->delete_records('external_services_functions', array('externalserviceid'=>$dbservice->id));
|
|
|
|
$DB->delete_records('external_services_users', array('externalserviceid'=>$dbservice->id));
|
|
|
|
$DB->delete_records('external_services', array('id'=>$dbservice->id));
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
$service = $services[$dbservice->name];
|
|
|
|
unset($services[$dbservice->name]);
|
|
|
|
$service['enabled'] = empty($service['enabled']) ? 0 : $service['enabled'];
|
|
|
|
$service['requiredcapability'] = empty($service['requiredcapability']) ? null : $service['requiredcapability'];
|
|
|
|
$service['restrictedusers'] = !isset($service['restrictedusers']) ? 1 : $service['restrictedusers'];
|
|
|
|
|
|
|
|
$update = false;
|
|
|
|
if ($dbservice->enabled != $service['enabled']) {
|
|
|
|
$dbservice->enabled = $service['enabled'];
|
|
|
|
$update = true;
|
|
|
|
}
|
|
|
|
if ($dbservice->requiredcapability != $service['requiredcapability']) {
|
|
|
|
$dbservice->requiredcapability = $service['requiredcapability'];
|
|
|
|
$update = true;
|
|
|
|
}
|
|
|
|
if ($dbservice->restrictedusers != $service['restrictedusers']) {
|
|
|
|
$dbservice->restrictedusers = $service['restrictedusers'];
|
|
|
|
$update = true;
|
|
|
|
}
|
|
|
|
if ($update) {
|
|
|
|
$DB->update_record('external_services', $dbservice);
|
|
|
|
}
|
|
|
|
|
|
|
|
$functions = $DB->get_records('external_services_functions', array('externalserviceid'=>$dbservice->id));
|
|
|
|
foreach ($functions as $function) {
|
|
|
|
$key = array_search($function->functionname, $service['functions']);
|
|
|
|
if ($key === false) {
|
|
|
|
$DB->delete_records('external_services_functions', array('id'=>$function->id));
|
|
|
|
} else {
|
|
|
|
unset($service['functions'][$key]);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
foreach ($service['functions'] as $fname) {
|
|
|
|
$newf = new object();
|
|
|
|
$newf->externalserviceid = $dbservice->id;
|
|
|
|
$newf->functionname = $fname;
|
|
|
|
$DB->insert_record('external_services_functions', $newf);
|
|
|
|
}
|
|
|
|
unset($functions);
|
|
|
|
}
|
|
|
|
foreach ($services as $name => $service) {
|
|
|
|
$dbservice = new object();
|
|
|
|
$dbservice->name = $name;
|
|
|
|
$dbservice->enabled = empty($service['enabled']) ? 0 : $service['enabled'];
|
|
|
|
$dbservice->requiredcapability = empty($service['requiredcapability']) ? null : $service['requiredcapability'];
|
|
|
|
$dbservice->restrictedusers = !isset($service['restrictedusers']) ? 1 : $service['restrictedusers'];
|
|
|
|
$dbservice->component = $component;
|
2009-11-08 22:12:36 +00:00
|
|
|
$dbservice->timecreated = time();
|
2009-09-13 21:26:51 +00:00
|
|
|
$dbservice->id = $DB->insert_record('external_services', $dbservice);
|
|
|
|
foreach ($service['functions'] as $fname) {
|
|
|
|
$newf = new object();
|
|
|
|
$newf->externalserviceid = $dbservice->id;
|
|
|
|
$newf->functionname = $fname;
|
|
|
|
$DB->insert_record('external_services_functions', $newf);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2010-05-22 20:24:28 +00:00
|
|
|
* Delete all service and external functions information defined in the specified component.
|
2009-09-13 21:26:51 +00:00
|
|
|
* @param string $component name of component (moodle, mod_assignment, etc.)
|
|
|
|
* @return void
|
|
|
|
*/
|
|
|
|
function external_delete_descriptions($component) {
|
|
|
|
global $DB;
|
|
|
|
|
|
|
|
$params = array($component);
|
|
|
|
|
|
|
|
$DB->delete_records_select('external_services_users', "externalserviceid IN (SELECT id FROM {external_services} WHERE component = ?)", $params);
|
|
|
|
$DB->delete_records_select('external_services_functions', "externalserviceid IN (SELECT id FROM {external_services} WHERE component = ?)", $params);
|
|
|
|
$DB->delete_records('external_services', array('component'=>$component));
|
|
|
|
$DB->delete_records('external_functions', array('component'=>$component));
|
|
|
|
}
|
|
|
|
|
2009-05-26 05:18:52 +00:00
|
|
|
/**
|
|
|
|
* upgrade logging functions
|
|
|
|
*/
|
2009-06-29 00:34:49 +00:00
|
|
|
function upgrade_handle_exception($ex, $plugin = null) {
|
2009-11-01 10:00:30 +00:00
|
|
|
// rollback everything, we need to log all upgrade problems
|
|
|
|
abort_all_db_transactions();
|
|
|
|
|
2009-10-31 13:52:39 +00:00
|
|
|
$info = get_exception_info($ex);
|
|
|
|
|
|
|
|
// First log upgrade error
|
|
|
|
upgrade_log(UPGRADE_LOG_ERROR, $plugin, 'Exception: ' . get_class($ex), $info->message, $info->backtrace);
|
|
|
|
|
|
|
|
// Always turn on debugging - admins need to know what is going on
|
|
|
|
$CFG->debug = DEBUG_DEVELOPER;
|
|
|
|
|
2009-06-29 00:34:49 +00:00
|
|
|
default_exception_handler($ex, true, $plugin);
|
2009-01-29 21:17:52 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Adds log entry into upgrade_log table
|
|
|
|
*
|
2009-05-26 05:18:52 +00:00
|
|
|
* @global object
|
|
|
|
* @global object
|
|
|
|
* @global object
|
2009-01-29 21:17:52 +00:00
|
|
|
* @param int $type UPGRADE_LOG_NORMAL, UPGRADE_LOG_NOTICE or UPGRADE_LOG_ERROR
|
|
|
|
* @param string $plugin plugin or null if main
|
|
|
|
* @param string $info short description text of log entry
|
|
|
|
* @param string $details long problem description
|
|
|
|
* @param string $backtrace string used for errors only
|
|
|
|
* @return void
|
|
|
|
*/
|
|
|
|
function upgrade_log($type, $plugin, $info, $details=null, $backtrace=null) {
|
|
|
|
global $DB, $USER, $CFG;
|
|
|
|
|
2009-01-31 20:07:32 +00:00
|
|
|
$plugin = ($plugin==='moodle') ? null : $plugin;
|
|
|
|
|
2009-06-26 09:06:16 +00:00
|
|
|
$backtrace = format_backtrace($backtrace, true);
|
2009-01-29 21:17:52 +00:00
|
|
|
|
|
|
|
$version = null;
|
|
|
|
|
|
|
|
//first try to find out current version number
|
2009-01-31 20:07:32 +00:00
|
|
|
if (empty($plugin) or $plugin === 'moodle') {
|
2009-01-29 21:17:52 +00:00
|
|
|
//main
|
|
|
|
$version = $CFG->version;
|
|
|
|
|
2009-01-31 20:07:32 +00:00
|
|
|
} else if ($plugin === 'local') {
|
|
|
|
//customisation
|
|
|
|
$version = $CFG->local_version;
|
|
|
|
|
|
|
|
} else if (strpos($plugin, 'mod/') === 0) {
|
2009-01-29 21:17:52 +00:00
|
|
|
try {
|
|
|
|
$modname = substr($plugin, strlen('mod/'));
|
|
|
|
$version = $DB->get_field('modules', 'version', array('name'=>$modname));
|
2009-01-31 20:07:32 +00:00
|
|
|
$version = ($version === false) ? null : $version;
|
2009-01-29 21:17:52 +00:00
|
|
|
} catch (Exception $ignored) {
|
|
|
|
}
|
|
|
|
|
2009-01-31 20:07:32 +00:00
|
|
|
} else if (strpos($plugin, 'block/') === 0) {
|
2009-01-29 21:17:52 +00:00
|
|
|
try {
|
2009-01-31 20:07:32 +00:00
|
|
|
$blockname = substr($plugin, strlen('block/'));
|
2009-01-29 21:17:52 +00:00
|
|
|
if ($block = $DB->get_record('block', array('name'=>$blockname))) {
|
|
|
|
$version = $block->version;
|
|
|
|
}
|
|
|
|
} catch (Exception $ignored) {
|
|
|
|
}
|
2009-01-31 20:07:32 +00:00
|
|
|
|
|
|
|
} else {
|
|
|
|
$pluginversion = get_config(str_replace('/', '_', $plugin), 'version');
|
|
|
|
if (!empty($pluginversion)) {
|
|
|
|
$version = $pluginversion;
|
|
|
|
}
|
2009-01-29 21:17:52 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
$log = new object();
|
|
|
|
$log->type = $type;
|
|
|
|
$log->plugin = $plugin;
|
|
|
|
$log->version = $version;
|
|
|
|
$log->info = $info;
|
|
|
|
$log->details = $details;
|
|
|
|
$log->backtrace = $backtrace;
|
|
|
|
$log->userid = $USER->id;
|
|
|
|
$log->timemodified = time();
|
|
|
|
try {
|
|
|
|
$DB->insert_record('upgrade_log', $log);
|
|
|
|
} catch (Exception $ignored) {
|
2009-01-31 20:07:32 +00:00
|
|
|
// possible during install or 2.0 upgrade
|
2009-01-29 21:17:52 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Marks start of upgrade, blocks any other access to site.
|
|
|
|
* The upgrade is finished at the end of script or after timeout.
|
2009-05-26 05:18:52 +00:00
|
|
|
*
|
|
|
|
* @global object
|
|
|
|
* @global object
|
|
|
|
* @global object
|
2009-01-29 21:17:52 +00:00
|
|
|
*/
|
|
|
|
function upgrade_started($preinstall=false) {
|
2009-09-03 06:59:25 +00:00
|
|
|
global $CFG, $DB, $PAGE, $OUTPUT;
|
2009-01-29 21:17:52 +00:00
|
|
|
|
|
|
|
static $started = false;
|
|
|
|
|
|
|
|
if ($preinstall) {
|
|
|
|
ignore_user_abort(true);
|
|
|
|
upgrade_setup_debug(true);
|
|
|
|
|
|
|
|
} else if ($started) {
|
|
|
|
upgrade_set_timeout(120);
|
|
|
|
|
|
|
|
} else {
|
2009-05-06 08:29:22 +00:00
|
|
|
if (!CLI_SCRIPT and !$PAGE->headerprinted) {
|
2009-01-29 21:17:52 +00:00
|
|
|
$strupgrade = get_string('upgradingversion', 'admin');
|
2009-12-16 18:00:58 +00:00
|
|
|
$PAGE->set_pagelayout('maintenance');
|
2010-01-20 18:07:49 +00:00
|
|
|
upgrade_init_javascript();
|
2009-09-03 06:59:25 +00:00
|
|
|
$PAGE->set_title($strupgrade.' - Moodle '.$CFG->target_release);
|
|
|
|
$PAGE->set_heading($strupgrade);
|
|
|
|
$PAGE->navbar->add($strupgrade);
|
|
|
|
$PAGE->set_cacheable(false);
|
|
|
|
echo $OUTPUT->header();
|
2009-01-29 21:17:52 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
ignore_user_abort(true);
|
|
|
|
register_shutdown_function('upgrade_finished_handler');
|
|
|
|
upgrade_setup_debug(true);
|
|
|
|
set_config('upgraderunning', time()+300);
|
|
|
|
$started = true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2010-05-22 20:24:28 +00:00
|
|
|
* Internal function - executed if upgrade interrupted.
|
2009-01-29 21:17:52 +00:00
|
|
|
*/
|
|
|
|
function upgrade_finished_handler() {
|
|
|
|
upgrade_finished();
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Indicates upgrade is finished.
|
|
|
|
*
|
|
|
|
* This function may be called repeatedly.
|
2009-05-26 05:18:52 +00:00
|
|
|
*
|
|
|
|
* @global object
|
|
|
|
* @global object
|
2009-01-29 21:17:52 +00:00
|
|
|
*/
|
|
|
|
function upgrade_finished($continueurl=null) {
|
2009-08-06 14:21:34 +00:00
|
|
|
global $CFG, $DB, $OUTPUT;
|
2009-01-29 21:17:52 +00:00
|
|
|
|
|
|
|
if (!empty($CFG->upgraderunning)) {
|
|
|
|
unset_config('upgraderunning');
|
|
|
|
upgrade_setup_debug(false);
|
|
|
|
ignore_user_abort(false);
|
|
|
|
if ($continueurl) {
|
2009-08-18 05:20:12 +00:00
|
|
|
echo $OUTPUT->continue_button($continueurl);
|
2009-08-06 14:21:34 +00:00
|
|
|
echo $OUTPUT->footer();
|
2009-01-29 21:17:52 +00:00
|
|
|
die;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2009-05-26 05:18:52 +00:00
|
|
|
/**
|
|
|
|
* @global object
|
|
|
|
* @global object
|
|
|
|
*/
|
2009-01-29 21:17:52 +00:00
|
|
|
function upgrade_setup_debug($starting) {
|
|
|
|
global $CFG, $DB;
|
|
|
|
|
|
|
|
static $originaldebug = null;
|
|
|
|
|
|
|
|
if ($starting) {
|
|
|
|
if ($originaldebug === null) {
|
|
|
|
$originaldebug = $DB->get_debug();
|
|
|
|
}
|
|
|
|
if (!empty($CFG->upgradeshowsql)) {
|
|
|
|
$DB->set_debug(true);
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
$DB->set_debug($originaldebug);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2009-05-26 05:18:52 +00:00
|
|
|
/**
|
|
|
|
* @global object
|
|
|
|
*/
|
2009-02-01 12:52:30 +00:00
|
|
|
function print_upgrade_reload($url) {
|
2009-07-02 11:09:15 +00:00
|
|
|
global $OUTPUT;
|
2009-02-01 12:52:30 +00:00
|
|
|
|
|
|
|
echo "<br />";
|
|
|
|
echo '<div class="continuebutton">';
|
2009-12-16 21:50:45 +00:00
|
|
|
echo '<a href="'.$url.'" title="'.get_string('reload').'" ><img src="'.$OUTPUT->pix_url('i/reload') . '" alt="" /> '.get_string('reload').'</a>';
|
2009-02-01 12:52:30 +00:00
|
|
|
echo '</div><br />';
|
|
|
|
}
|
|
|
|
|
2009-01-29 21:17:52 +00:00
|
|
|
function print_upgrade_separator() {
|
|
|
|
if (!CLI_SCRIPT) {
|
|
|
|
echo '<hr />';
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2009-01-31 20:07:32 +00:00
|
|
|
/**
|
|
|
|
* Default start upgrade callback
|
|
|
|
* @param string $plugin
|
2010-05-22 20:24:28 +00:00
|
|
|
* @param bool $installation true if installation, false means upgrade
|
2009-01-31 20:07:32 +00:00
|
|
|
*/
|
2009-05-26 19:02:11 +00:00
|
|
|
function print_upgrade_part_start($plugin, $installation, $verbose) {
|
2009-08-06 08:19:21 +00:00
|
|
|
global $OUTPUT;
|
2009-01-31 20:07:32 +00:00
|
|
|
if (empty($plugin) or $plugin == 'moodle') {
|
|
|
|
upgrade_started($installation); // does not store upgrade running flag yet
|
2009-05-26 19:02:11 +00:00
|
|
|
if ($verbose) {
|
2009-08-06 08:19:21 +00:00
|
|
|
echo $OUTPUT->heading(get_string('coresystem'));
|
2009-05-26 19:02:11 +00:00
|
|
|
}
|
2009-01-31 20:07:32 +00:00
|
|
|
} else {
|
|
|
|
upgrade_started();
|
2009-05-26 19:02:11 +00:00
|
|
|
if ($verbose) {
|
2009-08-06 08:19:21 +00:00
|
|
|
echo $OUTPUT->heading($plugin);
|
2009-05-26 19:02:11 +00:00
|
|
|
}
|
2009-01-31 20:07:32 +00:00
|
|
|
}
|
|
|
|
if ($installation) {
|
|
|
|
if (empty($plugin) or $plugin == 'moodle') {
|
|
|
|
// no need to log - log table not yet there ;-)
|
|
|
|
} else {
|
|
|
|
upgrade_log(UPGRADE_LOG_NORMAL, $plugin, 'Starting plugin installation');
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
if (empty($plugin) or $plugin == 'moodle') {
|
|
|
|
upgrade_log(UPGRADE_LOG_NORMAL, $plugin, 'Starting core upgrade');
|
|
|
|
} else {
|
|
|
|
upgrade_log(UPGRADE_LOG_NORMAL, $plugin, 'Starting plugin upgrade');
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Default end upgrade callback
|
|
|
|
* @param string $plugin
|
2010-05-22 20:24:28 +00:00
|
|
|
* @param bool $installation true if installation, false means upgrade
|
2009-01-31 20:07:32 +00:00
|
|
|
*/
|
2009-05-26 19:02:11 +00:00
|
|
|
function print_upgrade_part_end($plugin, $installation, $verbose) {
|
2009-08-18 05:20:12 +00:00
|
|
|
global $OUTPUT;
|
2009-01-31 20:07:32 +00:00
|
|
|
upgrade_started();
|
|
|
|
if ($installation) {
|
|
|
|
if (empty($plugin) or $plugin == 'moodle') {
|
|
|
|
upgrade_log(UPGRADE_LOG_NORMAL, $plugin, 'Core installed');
|
|
|
|
} else {
|
|
|
|
upgrade_log(UPGRADE_LOG_NORMAL, $plugin, 'Plugin installed');
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
if (empty($plugin) or $plugin == 'moodle') {
|
|
|
|
upgrade_log(UPGRADE_LOG_NORMAL, $plugin, 'Core upgraded');
|
|
|
|
} else {
|
|
|
|
upgrade_log(UPGRADE_LOG_NORMAL, $plugin, 'Plugin upgraded');
|
|
|
|
}
|
|
|
|
}
|
2009-05-26 19:02:11 +00:00
|
|
|
if ($verbose) {
|
2009-08-18 05:20:12 +00:00
|
|
|
echo $OUTPUT->notification(get_string('success'), 'notifysuccess');
|
2009-05-26 19:02:11 +00:00
|
|
|
print_upgrade_separator();
|
2009-05-25 17:44:05 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2009-05-26 05:18:52 +00:00
|
|
|
/**
|
2010-05-22 20:24:28 +00:00
|
|
|
* Sets up JS code required for all upgrade scripts.
|
2009-05-26 05:18:52 +00:00
|
|
|
* @global object
|
|
|
|
*/
|
2010-01-20 18:07:49 +00:00
|
|
|
function upgrade_init_javascript() {
|
2009-06-26 09:38:14 +00:00
|
|
|
global $PAGE;
|
2010-01-20 18:07:49 +00:00
|
|
|
// scroll to the end of each upgrade page so that ppl see either error or continue button,
|
|
|
|
// no need to scroll continuously any more, it is enough to jump to end once the footer is printed ;-)
|
|
|
|
$js = "window.scrollTo(0, 5000000);";
|
|
|
|
$PAGE->requires->js_init_code($js);
|
2009-01-29 21:17:52 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Try to upgrade the given language pack (or current language)
|
2010-05-26 00:29:33 +00:00
|
|
|
*
|
|
|
|
* @todo hardcoded Moodle version here - shall be provided by version.php or similar script
|
2009-01-29 21:17:52 +00:00
|
|
|
*/
|
|
|
|
function upgrade_language_pack($lang='') {
|
2009-08-06 08:19:21 +00:00
|
|
|
global $CFG, $OUTPUT;
|
2009-01-29 21:17:52 +00:00
|
|
|
|
2010-05-26 00:29:33 +00:00
|
|
|
get_string_manager()->reset_caches();
|
|
|
|
|
2009-01-29 21:17:52 +00:00
|
|
|
if (empty($lang)) {
|
|
|
|
$lang = current_language();
|
|
|
|
}
|
|
|
|
|
2010-04-10 07:24:56 +00:00
|
|
|
if ($lang == 'en') {
|
2009-01-29 21:17:52 +00:00
|
|
|
return true; // Nothing to do
|
|
|
|
}
|
|
|
|
|
2009-02-08 23:14:13 +00:00
|
|
|
upgrade_started(false);
|
2009-08-06 08:19:21 +00:00
|
|
|
echo $OUTPUT->heading(get_string('langimport', 'admin').': '.$lang);
|
2009-01-29 21:17:52 +00:00
|
|
|
|
|
|
|
@mkdir ($CFG->dataroot.'/temp/'); //make it in case it's a fresh install, it might not be there
|
|
|
|
@mkdir ($CFG->dataroot.'/lang/');
|
|
|
|
|
|
|
|
require_once($CFG->libdir.'/componentlib.class.php');
|
2010-05-26 00:29:33 +00:00
|
|
|
|
|
|
|
if ($cd = new component_installer('http://download.moodle.org', 'langpack/2.0', $lang.'.zip', 'languages.md5', 'lang')) {
|
2009-01-29 21:17:52 +00:00
|
|
|
$status = $cd->install(); //returns COMPONENT_(ERROR | UPTODATE | INSTALLED)
|
|
|
|
|
|
|
|
if ($status == COMPONENT_INSTALLED) {
|
2010-05-26 00:29:33 +00:00
|
|
|
remove_dir($CFG->dataroot.'/cache/languages');
|
2009-02-08 23:14:13 +00:00
|
|
|
if ($parentlang = get_parent_language($lang)) {
|
2010-05-26 00:29:33 +00:00
|
|
|
if ($cd = new component_installer('http://download.moodle.org', 'langpack/2.0', $parentlang.'.zip', 'languages.md5', 'lang')) {
|
2009-02-08 23:14:13 +00:00
|
|
|
$cd->install();
|
|
|
|
}
|
|
|
|
}
|
2009-08-18 05:20:12 +00:00
|
|
|
echo $OUTPUT->notification(get_string('success'), 'notifysuccess');
|
2009-01-29 21:17:52 +00:00
|
|
|
}
|
|
|
|
}
|
2010-05-26 00:29:33 +00:00
|
|
|
|
|
|
|
get_string_manager()->reset_caches();
|
|
|
|
|
2009-02-08 23:14:13 +00:00
|
|
|
print_upgrade_separator();
|
2009-01-29 21:17:52 +00:00
|
|
|
}
|
2009-05-26 09:52:38 +00:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Install core moodle tables and initialize
|
|
|
|
* @param float $version target version
|
|
|
|
* @param bool $verbose
|
|
|
|
* @return void, may throw exception
|
|
|
|
*/
|
|
|
|
function install_core($version, $verbose) {
|
|
|
|
global $CFG, $DB;
|
|
|
|
|
|
|
|
try {
|
2009-06-07 08:46:13 +00:00
|
|
|
set_time_limit(600);
|
2009-05-26 19:02:11 +00:00
|
|
|
print_upgrade_part_start('moodle', true, $verbose); // does not store upgrade running flag
|
2009-05-26 09:52:38 +00:00
|
|
|
|
|
|
|
$DB->get_manager()->install_from_xmldb_file("$CFG->libdir/db/install.xml");
|
|
|
|
upgrade_started(); // we want the flag to be stored in config table ;-)
|
|
|
|
|
|
|
|
// set all core default records and default settings
|
|
|
|
require_once("$CFG->libdir/db/install.php");
|
|
|
|
xmldb_main_install();
|
|
|
|
|
|
|
|
// store version
|
|
|
|
upgrade_main_savepoint(true, $version, false);
|
|
|
|
|
MDL-21782 reworked enrolment framework, the core infrastructure is in place, the basic plugins are all implemented; see the tracker issue for list of unfinished bits, expect more changes and improvements during the next week
AMOS START
MOV [sendcoursewelcomemessage,core_admin],[sendcoursewelcomemessage,enrol_self]
MOV [configsendcoursewelcomemessage,core_admin],[sendcoursewelcomemessage_desc,enrol_self]
MOV [enrolstartdate,core],[enrolstartdate,enrol_self]
MOV [enrolenddate,core],[enrolenddate,enrol_self]
CPY [welcometocourse,core],[welcometocourse,enrol_self]
CPY [welcometocoursetext,core],[welcometocoursetext,enrol_self]
MOV [notenrollable,core],[notenrollable,core_enrol]
MOV [enrolenddaterror,core],[enrolenddaterror,enrol_self]
MOV [enrolmentkeyhint,core],[passwordinvalidhint,enrol_self]
MOV [coursemanager,core_admin],[coursecontact,core_admin]
MOV [configcoursemanager,core_admin],[coursecontact_desc,core_admin]
MOV [enrolledincourserole,core],[enrolledincourserole,enrol_manual]
MOV [enrolme,core],[enrolme,core_enrol]
MOV [unenrol,core],[unenrol,core_enrol]
MOV [unenrolme,core],[unenrolme,core_enrol]
MOV [enrolmentnew,core],[enrolmentnew,core_enrol]
MOV [enrolmentnewuser,core],[enrolmentnewuser,core_enrol]
MOV [enrolments,core],[enrolments,core_enrol]
MOV [enrolperiod,core],[enrolperiod,core_enrol]
MOV [unenrolroleusers,core],[unenrolroleusers,core_enrol]
AMOS END
2010-06-21 15:30:49 +00:00
|
|
|
// Continue with the installation
|
2009-05-26 09:52:38 +00:00
|
|
|
events_update_definition('moodle');
|
|
|
|
message_update_providers('moodle');
|
|
|
|
|
MDL-21782 reworked enrolment framework, the core infrastructure is in place, the basic plugins are all implemented; see the tracker issue for list of unfinished bits, expect more changes and improvements during the next week
AMOS START
MOV [sendcoursewelcomemessage,core_admin],[sendcoursewelcomemessage,enrol_self]
MOV [configsendcoursewelcomemessage,core_admin],[sendcoursewelcomemessage_desc,enrol_self]
MOV [enrolstartdate,core],[enrolstartdate,enrol_self]
MOV [enrolenddate,core],[enrolenddate,enrol_self]
CPY [welcometocourse,core],[welcometocourse,enrol_self]
CPY [welcometocoursetext,core],[welcometocoursetext,enrol_self]
MOV [notenrollable,core],[notenrollable,core_enrol]
MOV [enrolenddaterror,core],[enrolenddaterror,enrol_self]
MOV [enrolmentkeyhint,core],[passwordinvalidhint,enrol_self]
MOV [coursemanager,core_admin],[coursecontact,core_admin]
MOV [configcoursemanager,core_admin],[coursecontact_desc,core_admin]
MOV [enrolledincourserole,core],[enrolledincourserole,enrol_manual]
MOV [enrolme,core],[enrolme,core_enrol]
MOV [unenrol,core],[unenrol,core_enrol]
MOV [unenrolme,core],[unenrolme,core_enrol]
MOV [enrolmentnew,core],[enrolmentnew,core_enrol]
MOV [enrolmentnewuser,core],[enrolmentnewuser,core_enrol]
MOV [enrolments,core],[enrolments,core_enrol]
MOV [enrolperiod,core],[enrolperiod,core_enrol]
MOV [unenrolroleusers,core],[unenrolroleusers,core_enrol]
AMOS END
2010-06-21 15:30:49 +00:00
|
|
|
// Write default settings unconditionally
|
2009-05-26 09:52:38 +00:00
|
|
|
admin_apply_default_settings(NULL, true);
|
|
|
|
|
2009-05-26 19:02:11 +00:00
|
|
|
print_upgrade_part_end(null, true, $verbose);
|
2009-05-26 09:52:38 +00:00
|
|
|
} catch (exception $ex) {
|
|
|
|
upgrade_handle_exception($ex);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Upgrade moodle core
|
|
|
|
* @param float $version target version
|
|
|
|
* @param bool $verbose
|
|
|
|
* @return void, may throw exception
|
|
|
|
*/
|
|
|
|
function upgrade_core($version, $verbose) {
|
|
|
|
global $CFG;
|
|
|
|
|
2009-05-26 17:44:25 +00:00
|
|
|
require_once($CFG->libdir.'/db/upgrade.php'); // Defines upgrades
|
|
|
|
|
2009-05-26 09:52:38 +00:00
|
|
|
try {
|
2010-05-31 05:34:16 +00:00
|
|
|
// Reset caches before any output
|
2010-07-13 15:09:44 +00:00
|
|
|
purge_all_caches();
|
2010-05-31 05:34:16 +00:00
|
|
|
|
2009-05-26 09:52:38 +00:00
|
|
|
// Upgrade current language pack if we can
|
|
|
|
if (empty($CFG->skiplangupgrade)) {
|
|
|
|
upgrade_language_pack(false);
|
|
|
|
}
|
|
|
|
|
2009-05-26 19:02:11 +00:00
|
|
|
print_upgrade_part_start('moodle', false, $verbose);
|
2009-05-26 09:52:38 +00:00
|
|
|
|
2009-06-19 14:25:56 +00:00
|
|
|
// one time special local migration pre 2.0 upgrade script
|
|
|
|
if ($version < 2007101600) {
|
|
|
|
$pre20upgradefile = "$CFG->dirrot/local/upgrade_pre20.php";
|
|
|
|
if (file_exists($pre20upgradefile)) {
|
|
|
|
set_time_limit(0);
|
|
|
|
require($pre20upgradefile);
|
|
|
|
// reset upgrade timeout to default
|
|
|
|
upgrade_set_timeout();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2009-05-26 09:52:38 +00:00
|
|
|
$result = xmldb_main_upgrade($CFG->version);
|
|
|
|
if ($version > $CFG->version) {
|
|
|
|
// store version if not already there
|
|
|
|
upgrade_main_savepoint($result, $version, false);
|
|
|
|
}
|
|
|
|
|
|
|
|
// perform all other component upgrade routines
|
|
|
|
update_capabilities('moodle');
|
2009-09-13 21:26:51 +00:00
|
|
|
external_update_descriptions('moodle');
|
2009-05-26 09:52:38 +00:00
|
|
|
events_update_definition('moodle');
|
|
|
|
message_update_providers('moodle');
|
|
|
|
|
2010-05-31 05:34:16 +00:00
|
|
|
// Reset caches again, just to be sure
|
2010-07-13 15:09:44 +00:00
|
|
|
purge_all_caches();
|
2010-07-06 17:14:36 +00:00
|
|
|
|
|
|
|
// Clean up contexts - more and more stuff depends on existence of paths and contexts
|
|
|
|
cleanup_contexts();
|
|
|
|
create_contexts();
|
|
|
|
build_context_path();
|
MDL-21782 reworked enrolment framework, the core infrastructure is in place, the basic plugins are all implemented; see the tracker issue for list of unfinished bits, expect more changes and improvements during the next week
AMOS START
MOV [sendcoursewelcomemessage,core_admin],[sendcoursewelcomemessage,enrol_self]
MOV [configsendcoursewelcomemessage,core_admin],[sendcoursewelcomemessage_desc,enrol_self]
MOV [enrolstartdate,core],[enrolstartdate,enrol_self]
MOV [enrolenddate,core],[enrolenddate,enrol_self]
CPY [welcometocourse,core],[welcometocourse,enrol_self]
CPY [welcometocoursetext,core],[welcometocoursetext,enrol_self]
MOV [notenrollable,core],[notenrollable,core_enrol]
MOV [enrolenddaterror,core],[enrolenddaterror,enrol_self]
MOV [enrolmentkeyhint,core],[passwordinvalidhint,enrol_self]
MOV [coursemanager,core_admin],[coursecontact,core_admin]
MOV [configcoursemanager,core_admin],[coursecontact_desc,core_admin]
MOV [enrolledincourserole,core],[enrolledincourserole,enrol_manual]
MOV [enrolme,core],[enrolme,core_enrol]
MOV [unenrol,core],[unenrol,core_enrol]
MOV [unenrolme,core],[unenrolme,core_enrol]
MOV [enrolmentnew,core],[enrolmentnew,core_enrol]
MOV [enrolmentnewuser,core],[enrolmentnewuser,core_enrol]
MOV [enrolments,core],[enrolments,core_enrol]
MOV [enrolperiod,core],[enrolperiod,core_enrol]
MOV [unenrolroleusers,core],[unenrolroleusers,core_enrol]
AMOS END
2010-06-21 15:30:49 +00:00
|
|
|
$syscontext = get_context_instance(CONTEXT_SYSTEM);
|
|
|
|
mark_context_dirty($syscontext->path);
|
2009-05-26 09:52:38 +00:00
|
|
|
|
2009-05-26 19:02:11 +00:00
|
|
|
print_upgrade_part_end('moodle', false, $verbose);
|
2009-05-26 09:52:38 +00:00
|
|
|
} catch (Exception $ex) {
|
|
|
|
upgrade_handle_exception($ex);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Upgrade/install other parts of moodle
|
|
|
|
* @param bool $verbose
|
|
|
|
* @return void, may throw exception
|
|
|
|
*/
|
|
|
|
function upgrade_noncore($verbose) {
|
|
|
|
global $CFG;
|
|
|
|
|
|
|
|
// upgrade all plugins types
|
|
|
|
try {
|
|
|
|
$plugintypes = get_plugin_types();
|
|
|
|
foreach ($plugintypes as $type=>$location) {
|
2009-06-19 14:25:56 +00:00
|
|
|
upgrade_plugins($type, 'print_upgrade_part_start', 'print_upgrade_part_end', $verbose);
|
2009-05-26 09:52:38 +00:00
|
|
|
}
|
|
|
|
} catch (Exception $ex) {
|
|
|
|
upgrade_handle_exception($ex);
|
|
|
|
}
|
|
|
|
}
|
2009-05-26 17:44:25 +00:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Checks if the main tables have been installed yet or not.
|
|
|
|
* @return bool
|
|
|
|
*/
|
|
|
|
function core_tables_exist() {
|
|
|
|
global $DB;
|
|
|
|
|
|
|
|
if (!$tables = $DB->get_tables() ) { // No tables yet at all.
|
|
|
|
return false;
|
2009-08-06 19:59:16 +00:00
|
|
|
|
2009-05-26 17:44:25 +00:00
|
|
|
} else { // Check for missing main tables
|
|
|
|
$mtables = array('config', 'course', 'groupings'); // some tables used in 1.9 and 2.0, preferable something from the start and end of install.xml
|
|
|
|
foreach ($mtables as $mtable) {
|
|
|
|
if (!in_array($mtable, $tables)) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return true;
|
2009-08-06 19:59:16 +00:00
|
|
|
}
|
2009-05-30 18:51:47 +00:00
|
|
|
}
|
2010-02-02 21:16:56 +00:00
|
|
|
|
|
|
|
/**
|
|
|
|
* upgrades the mnet rpc definitions for the given component.
|
|
|
|
* this method doesn't return status, an exception will be thrown in the case of an error
|
|
|
|
*
|
|
|
|
* @param string $component the plugin to upgrade, eg auth_mnet
|
|
|
|
*/
|
|
|
|
function upgrade_plugin_mnet_functions($component) {
|
|
|
|
global $DB, $CFG;
|
|
|
|
|
|
|
|
list($type, $plugin) = explode('_', $component);
|
|
|
|
$path = get_plugin_directory($type, $plugin);
|
|
|
|
|
|
|
|
if (file_exists($path . '/db/mnet.php')) {
|
|
|
|
require_once($path . '/db/mnet.php'); // $publishes comes from this file
|
|
|
|
}
|
|
|
|
if (empty($publishes)) {
|
|
|
|
$publishes = array(); // still need this to be able to disable stuff later
|
|
|
|
}
|
|
|
|
if (empty($subscribes)) {
|
|
|
|
$subscribes = array(); // still need this to be able to disable stuff later
|
|
|
|
}
|
|
|
|
|
|
|
|
static $servicecache = array();
|
|
|
|
|
|
|
|
// rekey an array based on the rpc method for easy lookups later
|
|
|
|
$publishmethodservices = array();
|
|
|
|
$subscribemethodservices = array();
|
|
|
|
foreach($publishes as $servicename => $service) {
|
|
|
|
if (is_array($service['methods'])) {
|
|
|
|
foreach($service['methods'] as $methodname) {
|
|
|
|
$service['servicename'] = $servicename;
|
|
|
|
$publishmethodservices[$methodname][] = $service;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Disable functions that don't exist (any more) in the source
|
|
|
|
// Should these be deleted? What about their permissions records?
|
|
|
|
foreach ($DB->get_records('mnet_rpc', array('pluginname'=>$plugin, 'plugintype'=>$type), 'functionname ASC ') as $rpc) {
|
|
|
|
if (!array_key_exists($rpc->functionname, $publishmethodservices) && $rpc->enabled) {
|
|
|
|
$DB->set_field('mnet_rpc', 'enabled', 0, array('id' => $rpc->id));
|
|
|
|
} else if (array_key_exists($rpc->functionname, $publishmethodservices) && !$rpc->enabled) {
|
|
|
|
$DB->set_field('mnet_rpc', 'enabled', 1, array('id' => $rpc->id));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// reflect all the services we're publishing and save them
|
|
|
|
require_once($CFG->dirroot . '/lib/zend/Zend/Server/Reflection.php');
|
|
|
|
static $cachedclasses = array(); // to store reflection information in
|
|
|
|
foreach ($publishes as $service => $data) {
|
|
|
|
$f = $data['filename'];
|
|
|
|
$c = $data['classname'];
|
|
|
|
foreach ($data['methods'] as $method) {
|
|
|
|
$dataobject = new stdclass;
|
|
|
|
$dataobject->plugintype = $type;
|
|
|
|
$dataobject->pluginname = $plugin;
|
|
|
|
$dataobject->enabled = 1;
|
|
|
|
$dataobject->classname = $c;
|
|
|
|
$dataobject->filename = $f;
|
|
|
|
|
|
|
|
if (is_string($method)) {
|
|
|
|
$dataobject->functionname = $method;
|
|
|
|
|
|
|
|
} else if (is_array($method)) { // wants to override file or class
|
|
|
|
$dataobject->functionname = $method['method'];
|
|
|
|
$dataobject->classname = $method['classname'];
|
|
|
|
$dataobject->filename = $method['filename'];
|
|
|
|
}
|
|
|
|
$dataobject->xmlrpcpath = $type.'/'.$plugin.'/'.$dataobject->filename.'/'.$method;
|
|
|
|
$dataobject->static = false;
|
|
|
|
|
|
|
|
require_once($path . '/' . $dataobject->filename);
|
|
|
|
$functionreflect = null; // slightly different ways to get this depending on whether it's a class method or a function
|
|
|
|
if (!empty($dataobject->classname)) {
|
|
|
|
if (!class_exists($dataobject->classname)) {
|
|
|
|
throw new moodle_exception('installnosuchmethod', 'mnet', '', (object)array('method' => $dataobject->functionname, 'class' => $dataobject->classname));
|
|
|
|
}
|
|
|
|
$key = $dataobject->filename . '|' . $dataobject->classname;
|
|
|
|
if (!array_key_exists($key, $cachedclasses)) { // look to see if we've already got a reflection object
|
|
|
|
try {
|
|
|
|
$cachedclasses[$key] = Zend_Server_Reflection::reflectClass($dataobject->classname);
|
|
|
|
} catch (Zend_Server_Reflection_Exception $e) { // catch these and rethrow them to something more helpful
|
|
|
|
throw new moodle_exception('installreflectionclasserror', 'mnet', '', (object)array('method' => $dataobject->functionname, 'class' => $dataobject->classname, 'error' => $e->getMessage()));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
$r =& $cachedclasses[$key];
|
|
|
|
if (!$r->hasMethod($dataobject->functionname)) {
|
|
|
|
throw new moodle_exception('installnosuchmethod', 'mnet', '', (object)array('method' => $dataobject->functionname, 'class' => $dataobject->classname));
|
|
|
|
}
|
|
|
|
// stupid workaround for zend not having a getMethod($name) function
|
|
|
|
$ms = $r->getMethods();
|
|
|
|
foreach ($ms as $m) {
|
|
|
|
if ($m->getName() == $dataobject->functionname) {
|
|
|
|
$functionreflect = $m;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
$dataobject->static = (int)$functionreflect->isStatic();
|
|
|
|
} else {
|
|
|
|
if (!function_exists($dataobject->functionname)) {
|
|
|
|
throw new moodle_exception('installnosuchfunction', 'mnet', '', (object)array('method' => $dataobject->functionname, 'file' => $dataobject->filename));
|
|
|
|
}
|
|
|
|
try {
|
|
|
|
$functionreflect = Zend_Server_Reflection::reflectFunction($dataobject->functionname);
|
|
|
|
} catch (Zend_Server_Reflection_Exception $e) { // catch these and rethrow them to something more helpful
|
|
|
|
throw new moodle_exception('installreflectionfunctionerror', 'mnet', '', (object)array('method' => $dataobject->functionname, '' => $dataobject->filename, 'error' => $e->getMessage()));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
$dataobject->profile = serialize(admin_mnet_method_profile($functionreflect));
|
|
|
|
$dataobject->help = $functionreflect->getDescription();
|
|
|
|
|
|
|
|
if ($record_exists = $DB->get_record('mnet_rpc', array('xmlrpcpath'=>$dataobject->xmlrpcpath))) {
|
|
|
|
$dataobject->id = $record_exists->id;
|
|
|
|
$dataobject->enabled = $record_exists->enabled;
|
|
|
|
$DB->update_record('mnet_rpc', $dataobject);
|
|
|
|
} else {
|
|
|
|
$dataobject->id = $DB->insert_record('mnet_rpc', $dataobject, true);
|
|
|
|
}
|
|
|
|
|
2010-02-18 04:55:34 +00:00
|
|
|
foreach ($publishmethodservices[$dataobject->functionname] as $service) {
|
|
|
|
if ($serviceobj = $DB->get_record('mnet_service', array('name'=>$service['servicename']))) {
|
|
|
|
$serviceobj->apiversion = $service['apiversion'];
|
|
|
|
$DB->update_record('mnet_service', $serviceobj);
|
|
|
|
} else {
|
|
|
|
$serviceobj = new stdClass();
|
|
|
|
$serviceobj->name = $service['servicename'];
|
|
|
|
$serviceobj->apiversion = $service['apiversion'];
|
|
|
|
$serviceobj->offer = 1;
|
|
|
|
$serviceobj->id = $DB->insert_record('mnet_service', $serviceobj);
|
|
|
|
}
|
|
|
|
$servicecache[$service['servicename']] = $serviceobj;
|
|
|
|
if (!$DB->record_exists('mnet_service2rpc', array('rpcid'=>$dataobject->id, 'serviceid'=>$serviceobj->id))) {
|
|
|
|
$obj = new stdClass();
|
|
|
|
$obj->rpcid = $dataobject->id;
|
|
|
|
$obj->serviceid = $serviceobj->id;
|
|
|
|
$DB->insert_record('mnet_service2rpc', $obj, true);
|
|
|
|
}
|
2010-02-02 21:16:56 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
// finished with methods we publish, now do subscribable methods
|
|
|
|
foreach($subscribes as $service => $methods) {
|
|
|
|
if (!array_key_exists($service, $servicecache)) {
|
|
|
|
if (!$serviceobj = $DB->get_record('mnet_service', array('name' => $service))) {
|
2010-03-30 19:51:34 +00:00
|
|
|
debugging("TODO: skipping unknown service $service - somebody needs to fix MDL-21993");
|
2010-02-02 21:16:56 +00:00
|
|
|
continue;
|
|
|
|
}
|
|
|
|
$servicecache[$service] = $serviceobj;
|
|
|
|
} else {
|
|
|
|
$serviceobj = $servicecache[$service];
|
|
|
|
}
|
|
|
|
foreach ($methods as $method => $xmlrpcpath) {
|
|
|
|
if (!$rpcid = $DB->get_field('mnet_remote_rpc', 'id', array('xmlrpcpath'=>$xmlrpcpath))) {
|
|
|
|
$remoterpc = (object)array(
|
|
|
|
'functionname' => $method,
|
|
|
|
'xmlrpcpath' => $xmlrpcpath,
|
|
|
|
'plugintype' => $type,
|
|
|
|
'pluginname' => $plugin,
|
|
|
|
'enabled' => 1,
|
|
|
|
);
|
|
|
|
$rpcid = $remoterpc->id = $DB->insert_record('mnet_remote_rpc', $remoterpc, true);
|
|
|
|
}
|
|
|
|
if (!$DB->record_exists('mnet_remote_service2rpc', array('rpcid'=>$rpcid, 'serviceid'=>$serviceobj->id))) {
|
|
|
|
$obj = new stdClass();
|
|
|
|
$obj->rpcid = $rpcid;
|
|
|
|
$obj->serviceid = $serviceobj->id;
|
|
|
|
$DB->insert_record('mnet_remote_service2rpc', $obj, true);
|
|
|
|
}
|
2010-02-18 04:55:34 +00:00
|
|
|
$subscribemethodservices[$method][] = $service;
|
2010-02-02 21:16:56 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
foreach ($DB->get_records('mnet_remote_rpc', array('pluginname'=>$plugin, 'plugintype'=>$type), 'functionname ASC ') as $rpc) {
|
|
|
|
if (!array_key_exists($rpc->functionname, $subscribemethodservices) && $rpc->enabled) {
|
|
|
|
$DB->set_field('mnet_remote_rpc', 'enabled', 0, array('id' => $rpc->id));
|
|
|
|
} else if (array_key_exists($rpc->functionname, $subscribemethodservices) && !$rpc->enabled) {
|
|
|
|
$DB->set_field('mnet_remote_rpc', 'enabled', 1, array('id' => $rpc->id));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Given some sort of Zend Reflection function/method object, return a profile array, ready to be serialized and stored
|
|
|
|
*
|
|
|
|
* @param Zend_Server_Reflection_Function_Abstract $function can be any subclass of this object type
|
|
|
|
*
|
|
|
|
* @return array
|
|
|
|
*/
|
|
|
|
function admin_mnet_method_profile(Zend_Server_Reflection_Function_Abstract $function) {
|
|
|
|
$proto = array_pop($function->getPrototypes());
|
|
|
|
$ret = $proto->getReturnValue();
|
|
|
|
$profile = array(
|
|
|
|
'parameters' => array(),
|
|
|
|
'return' => array(
|
|
|
|
'type' => $ret->getType(),
|
|
|
|
'description' => $ret->getDescription(),
|
|
|
|
),
|
|
|
|
);
|
|
|
|
foreach ($proto->getParameters() as $p) {
|
|
|
|
$profile['parameters'][] = array(
|
|
|
|
'name' => $p->getName(),
|
|
|
|
'type' => $p->getType(),
|
|
|
|
'description' => $p->getDescription(),
|
|
|
|
);
|
|
|
|
}
|
|
|
|
return $profile;
|
|
|
|
}
|