From e1ab51a2834756ac1ecbe313ce07d71562c0ac37 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?C=C3=A9line=20Perv=C3=A8s?= Date: Fri, 28 Jun 2024 18:16:28 +0200 Subject: [PATCH 1/4] add new developped moosh commands that maybe usefull to others --- .../Command/Generic/Plugin/PluginHideShow.php | 289 ++++++++++++++++++ .../Command/Generic/Request/RequestSelect.php | 43 +++ Moosh/Command/Moodle39/Cache/CacheSet.php | 64 ++++ ...tegoryMoveCoursesFromCategoryToAnother.php | 62 ++++ .../Command/Moodle39/Cohort/CohortDelete.php | 50 +++ .../Moodle39/Context/ContextFreeze.php | 74 +++++ .../Course/CourseEnrolChangeStatus.php | 83 +++++ .../Moodle39/Dashboard/DashboardResetAll.php | 33 ++ Moosh/Command/Moodle39/Task/TaskSchedule.php | 93 ++++++ .../Moodle39/Webservice/WebserviceInstall.php | 128 ++++++++ .../Moodle41/Cache/CacheStoreClear.php | 81 +++++ 11 files changed, 1000 insertions(+) create mode 100755 Moosh/Command/Generic/Plugin/PluginHideShow.php create mode 100755 Moosh/Command/Generic/Request/RequestSelect.php create mode 100644 Moosh/Command/Moodle39/Cache/CacheSet.php create mode 100644 Moosh/Command/Moodle39/Category/CategoryMoveCoursesFromCategoryToAnother.php create mode 100755 Moosh/Command/Moodle39/Cohort/CohortDelete.php create mode 100644 Moosh/Command/Moodle39/Context/ContextFreeze.php create mode 100755 Moosh/Command/Moodle39/Course/CourseEnrolChangeStatus.php create mode 100644 Moosh/Command/Moodle39/Dashboard/DashboardResetAll.php create mode 100644 Moosh/Command/Moodle39/Task/TaskSchedule.php create mode 100644 Moosh/Command/Moodle39/Webservice/WebserviceInstall.php create mode 100755 Moosh/Command/Moodle41/Cache/CacheStoreClear.php diff --git a/Moosh/Command/Generic/Plugin/PluginHideShow.php b/Moosh/Command/Generic/Plugin/PluginHideShow.php new file mode 100755 index 00000000..e57a5ee9 --- /dev/null +++ b/Moosh/Command/Generic/Plugin/PluginHideShow.php @@ -0,0 +1,289 @@ + + * @copyright Université de Strasbourg unistra.fr + * @license http://www.gnu.org/copyleft/gpl.html GNU GPL v3 or later + */ + +namespace Moosh\Command\Generic\Plugin; +use Moosh\MooshCommand; + +class PluginHideShow extends MooshCommand +{ + + public function __construct() + { + parent::__construct('hideshow', 'plugin'); + $this->addArgument('plugintype'); + $this->addArgument('pluginname'); + $this->addArgument('show'); + } + + public function execute() + { + global $CFG,$DB; + + if(!is_numeric($this->arguments[2])){ + echo "hideshow must be an integer\n"; + die; + } + $show=(int)$this->arguments[2]; + $plugintype = $this->arguments[0]; + $pluginname = $this->arguments[1]; + switch($plugintype){ + case 'block': + self::hideshow_values_check($pluginname,array(0,1)); + if (!$block = $DB->get_record('block', array('name'=>$pluginname))) { + echo "$plugintype $pluginname not exists"; + break; + } + $DB->set_field('block', 'visible', $show, array('id'=>$block->id)); + add_to_config_log('block_visibility', $block->visible, ''.$show, $block->name); + \core_plugin_manager::reset_caches(); + break; + case 'mod': + self::hideshow_values_check($show,array(0,1)); + if (!$module = $DB->get_record("modules", array("name"=>$pluginname))) { + echo "$plugintype $pluginname not exists"; + break; + } + $DB->set_field("modules", "visible", $show, array("id"=>$module->id)); // Hide main module + if($show==0){ + // Remember the visibility status in visibleold. + // And hide... + $sql = "UPDATE {course_modules} + SET visibleold=visible, visible=0 + WHERE module=?"; + $DB->execute($sql, array($module->id)); + // Increment course.cacherev for courses where we just made something invisible. + // This will force cache rebuilding on the next request. + increment_revision_number('course', 'cacherev', + "id IN (SELECT DISTINCT course + FROM {course_modules} + WHERE visibleold=1 AND module=?)", + array($module->id)); + \core_plugin_manager::reset_caches(); + }else{ + $DB->set_field('course_modules', 'visible', '1', array('visibleold'=>1, 'module'=>$module->id)); // Get the previous saved visible state for the course module. + // Increment course.cacherev for courses where we just made something visible. + // This will force cache rebuilding on the next request. + increment_revision_number('course', 'cacherev', + "id IN (SELECT DISTINCT course + FROM {course_modules} + WHERE visible=1 AND module=?)", + array($module->id)); + \core_plugin_manager::reset_caches(); + } + break; + case 'assignfeedback': + case 'assignsubmission': + self::hideshow_values_check($show,array(0,1)); + //check pluginname + if(count((array)get_config($plugintype.'_'.$pluginname))==0){ + cli_error(get_string('hideshow_plugin_cli_pluginnotexists','tool_cmdlinetools',array('name'=>$pluginname, 'type'=> $plugintype))); + } + set_config('disabled', $show==0?1:0, $plugintype . '_' . $pluginname); + break; + case 'qtype': + require_once($CFG->dirroot.'/question/engine/bank.php'); + self::hideshow_values_check($show,array(0,1)); + //check qtype + if(!array_key_exists ($pluginname, \question_bank::get_all_qtypes())){ + echo "qtype $pluginname does not exist"; + } + if($show==0){ + set_config($pluginname . '_disabled',1, 'question'); + }else{ + $qtype = \question_bank::get_qtype($pluginname,true); + if (!$qtype->menu_name()) { + echo "cannot enable qtype $pluginname"; + break; + } + unset_config($pluginname . '_disabled', 'question'); + } + break; + case 'qbehaviour': + self::hideshow_values_check($show,array(0,1)); + require_once($CFG->dirroot.'/question/engine/lib.php'); + //check qbehaviour + if(!array_key_exists ($pluginname, core_component::get_plugin_list('qbehaviour'))){ + echo "qbehaviour $pluginname does not exist"; + break; + } + if(!question_engine::is_behaviour_archetypal($pluginname)){ + echo "can't enable/disable qbehaviour $pluginname"; + break; + } + $config = get_config('question'); + if (!empty($config->disabledbehaviours)) { + $disabledbehaviours = explode(',', $config->disabledbehaviours); + } else { + $disabledbehaviours = array(); + } + $disabledbehaviours_index = array_search($pluginname,$disabledbehaviours); + if($disabledbehaviours_index!==false && $show==1){ + unset($disabledbehaviours[$disabledbehaviours_index]); + set_config('disabledbehaviours', implode(',', $disabledbehaviours), 'question'); + }else if ($disabledbehaviours_index === false && $show ==0){ + $disabledbehaviours[] = $pluginname; + set_config('disabledbehaviours', implode(',', $disabledbehaviours), 'question'); + } + \core_plugin_manager::reset_caches(); + break; + case 'enrol': + self::hideshow_values_check($show,array(0,1)); + $syscontext = \context_system::instance(); + $enabled = enrol_get_plugins(true); + $all = enrol_get_plugins(false); + if(!array_key_exists($pluginname,$all)){ + echo "enrol method $pluginname not exists"; + break; + } + if($show==0 && array_key_exists($pluginname,$enabled)){ + unset($enabled[$pluginname]); + set_config('enrol_plugins_enabled', implode(',', array_keys($enabled))); + \core_plugin_manager::reset_caches(); + $syscontext->mark_dirty(); // resets all enrol caches + }else if($show == 1 && !array_key_exists($pluginname, $enabled)){ + $enabled = array_keys($enabled); + $enabled[] = $pluginname; + set_config('enrol_plugins_enabled', implode(',', $enabled)); + \core_plugin_manager::reset_caches(); + $syscontext->mark_dirty(); // resets all enrol caches + } + break; + case 'filter': + require_once($CFG->libdir.'/filterlib.php'); + self::hideshow_values_check($show,array(1,-1,-9999)); + //check if filter exists + if(!array_key_exists($pluginname,filter_get_all_installed())){ + echo "filter $pluginname does not exist"; + break; + } + filter_set_global_state('filter/'.$pluginname, $show); + if ($show == TEXTFILTER_DISABLED) { + filter_set_applies_to_strings('filter/'.$pluginname, false); + } + reset_text_filters_cache(); + break; + case 'editor': + self::hideshow_values_check($show,array(0,1)); + $editors=editors_get_available(); + if(!array_key_exists($pluginname,$editors)){ + echo "editor $pluginname doesn't exists"; + break; + } + $active_editors = explode(',', $CFG->texteditors); + $active_editors_index = array_search($pluginname,$active_editors); + if($show==0 && $active_editors_index !== false){ + unset($active_editors[$active_editors_index]); + set_config('texteditors', implode(',', $active_editors)); + + }else if($show == 1 && $active_editors_index === false){ + $active_editors[]=$pluginname; + set_config('texteditors', implode(',', $active_editors)); + } + break; + case 'auth': + self::hideshow_values_check($show, array(0,1)); + if(!exists_auth_plugin($pluginname)){ + echo "auth plugin $pluginname doesn't exists"; + break; + } + get_enabled_auth_plugins(true); // fix the list of enabled auths + if (empty($CFG->auth)) { + $authsenabled = array(); + } else { + $authsenabled = explode(',', $CFG->auth); + } + if($show==1){ + if (!in_array($pluginname, $authsenabled)) { + $authsenabled[] = $pluginname; + $authsenabled = array_unique($authsenabled); + set_config('auth', implode(',', $authsenabled)); + } + \core\session\manager::gc(); // Remove stale sessions. + \core_plugin_manager::reset_caches(); + }else{ + $key = array_search($pluginname, $authsenabled); + if ($key !== false) { + unset($authsenabled[$key]); + set_config('auth', implode(',', $authsenabled)); + } + + if ($pluginname == $CFG->registerauth) { + set_config('registerauth', ''); + } + \core\session\manager::gc(); // Remove stale sessions. + \core_plugin_manager::reset_caches(); + } + break; + case 'license': + self::hideshow_values_check($show, array(0,1)); + require_once($CFG->libdir.'/licenselib.php'); + if(license_manager::get_license_by_shortname($pluginname)==null){ + echo "license plugin $pluginname doesn't exists"; + break; + } + if($pluginname == $CFG->sitedefaultlicense){ + echo "can\'t enable/disable license $pluginname"; + break; + } + if($show==1){ + license_manager::enable($pluginname); + }else{ + license_manager::disable($pluginname); + } + break; + case 'repository': + echo 'not implemented because require repository setting form'; + break; + case 'courseformat': + self::hideshow_values_check($show, array(0,1)); + require_once($CFG->libdir.'/classes/plugin_manager.php'); + $formatplugins = \core_plugin_manager::instance()->get_plugins_of_type('format'); + if (!isset($formatplugins[$pluginname])) { + echo "cours eformat $pluginname doesn't exits"; + break; + } + if (get_config('moodlecourse', 'format') === $pluginname){ + echo "course foramt $pluginname can't be enabled/disabled"; + break; + } + if($show == 0){ + set_config('disabled', 1, 'format_'. $pluginname); + \core_plugin_manager::reset_caches(); + }else{ + unset_config('disabled', 'format_'. $pluginname); + \core_plugin_manager::reset_caches(); + } + break; + case 'availability': + if($show==0){ + set_config('disabled', 1, 'availability_' . $pluginname); + }else{ + unset_config('disabled', 'availability_' . $pluginname); + } + \core_plugin_manager::reset_caches(); + break; + default: + echo "bad plugin type $plugintype or not taken in charge"; + break; + } + } + private static function hideshow_values_check($hideshow,$possiblevalues){ + if(!in_array($hideshow, $possiblevalues)){ + cli_error(get_string('hideshow_plugin_cli_parametervalues','tool_cmdlinetools',implode(',', $possiblevalues))); + } + } + + protected function getArgumentsHelp() { + $help = parent::getArgumentsHelp(); + $help .= "\n\n"; + $help .= "This command enable to hide show a plugin"; + $help .= "\n\nplugintype can be block, mod, assignfeedback, assignsubmission, qtype, qbehaviour, enrol, filter, editor, auth, license, repository, courseformat or avaibility"; + return $help; + } +} diff --git a/Moosh/Command/Generic/Request/RequestSelect.php b/Moosh/Command/Generic/Request/RequestSelect.php new file mode 100755 index 00000000..005f8ffe --- /dev/null +++ b/Moosh/Command/Generic/Request/RequestSelect.php @@ -0,0 +1,43 @@ + + * @copyright Université de Strasbourg unistra.fr + * @license http://www.gnu.org/copyleft/gpl.html GNU GPL v3 or later + */ + +namespace Moosh\Command\Generic\Request; + +use Moosh\MooshCommand; + +class RequestSelect extends MooshCommand { + public function __construct() { + parent::__construct('select', 'request'); + $this->addArgument('select_query'); + $this->minArguments = 1; + $this->maxArguments = 1; + } + + protected function getArgumentsHelp() { + $help = parent::getArgumentsHelp(); + $help .= "\n\n"; + $help .= "This command enable to make a select query and return resultset into a csv format"; + return $help; + } + + + public function execute() { + global $CFG, $DB; + $selectquery = trim($this->arguments[0]); + cli_writeln("$selectquery"); + $results = $DB->get_records_sql($selectquery); + $output = ''; + foreach($results as $index=>$result) { + $output.=implode(';',(array) $result)."\n"; + } + echo $output; + } +} + + diff --git a/Moosh/Command/Moodle39/Cache/CacheSet.php b/Moosh/Command/Moodle39/Cache/CacheSet.php new file mode 100644 index 00000000..4025aeec --- /dev/null +++ b/Moosh/Command/Moodle39/Cache/CacheSet.php @@ -0,0 +1,64 @@ + + * @license http://www.gnu.org/copyleft/gpl.html GNU GPL v3 or later + */ + +namespace Moosh\Command\Moodle39\Cache; +use Moosh\MooshCommand; + +class CacheSet extends MooshCommand +{ + public function __construct() { + parent::__construct('set', 'cache'); + $this->addArgument('definition'); + $this->addArgument('mapping'); + } + + public function execute() + { + global $CFG; + require_once($CFG->dirroot.'/cache/locallib.php'); + $var_definition=$this->arguments[0]; + $var_mappings=$this->arguments[1]; + if(!isset($var_definition) || !isset($var_mappings) ){ + cli_error('Defintion and mampings are required'); + } + $factory = \cache_factory::instance(); + list($component, $area) = explode('/', $var_definition, 2); + $config = \cache_config::instance(); + $writer = \cache_config_writer::instance(); + $writer->update_definitions(); + $definition_check = $writer->get_definition_by_id($var_definition); + if (!$definition_check) { + cli_error("$var_definition cache definition not exists"); + } + $definition = $factory->create_definition($component, $area); + $possiblestores = $config->get_stores($definition->get_mode(), $definition->get_requirements_bin()); + + $var_mappings = explode(',',$var_mappings); + $mappings = array(); + foreach ($var_mappings as $index => $var_mapping){ + // Check mapping is available. + if (array_key_exists($var_mapping, $possiblestores)){ + $mappings[$index]= $var_mapping; + }else if(!empty($var_mapping)){ + cli_error("Bad store instance name mapping $var_mapping : does not exists or not usable for this cache mode"); + } + } + $writer->set_definition_mappings($var_definition, $mappings); + return true; + } + + protected function getArgumentsHelp() { + $help = parent::getArgumentsHelp(); + $help .= "\n\n"; + $help .= "This command enable to attribute cache to specific cache store"; + $help .= "\ne.g moosh -n cache-set core/calendar_subscriptions redisstore"; + return $help; + } +} \ No newline at end of file diff --git a/Moosh/Command/Moodle39/Category/CategoryMoveCoursesFromCategoryToAnother.php b/Moosh/Command/Moodle39/Category/CategoryMoveCoursesFromCategoryToAnother.php new file mode 100644 index 00000000..6db5dcad --- /dev/null +++ b/Moosh/Command/Moodle39/Category/CategoryMoveCoursesFromCategoryToAnother.php @@ -0,0 +1,62 @@ + + * @license http://www.gnu.org/copyleft/gpl.html GNU GPL v3 or later + */ + +namespace Moosh\Command\Moodle39\Category; +use Moosh\MooshCommand; + +class CategoryMoveCoursesFromCategoryToAnother extends MooshCommand { + + public function __construct() { + parent::__construct('move-courses-from-category-to-another', 'category'); + $this->addArgument('sourcecategory'); + $this->addArgument('destinationcategory'); + $this->minArguments = 2; + } + + public function execute() + { + global $DB, $CFG; + require_once($CFG->dirroot.'/course/classes/category.php'); + require_once($CFG->dirroot.'/course/lib.php'); + // Check that category ids exists. + try { + $sourcecategory = \core_course_category::get($this->arguments[0]); + } catch (moodle_exception $me){ + cli_error('source category does not exists'); + } + try { + $destinationcategory = \core_course_category::get($this->arguments[1]); + } catch(moodle_exception $me){ + cli_error('destination category does not exists'); + } + $courses = $DB->get_records('course', array('category'=> $sourcecategory->id)); + if(count($courses)==0){ + cli_writeln('no courses in source category'); + } + foreach( $courses as $coursedata){ + // change category + $coursedata->category = $destinationcategory->id; + $msgobject = new \stdClass(); + $msgobject->courseid = $coursedata->id; + $msgobject->course = $coursedata->shortname; + $msgobject->sourcecategory = $sourcecategory->name; + $msgobject->sourcecategoryid = $sourcecategory->id; + $msgobject->destinationcategory = $destinationcategory->name; + $msgobject->destinationcategoryid = $destinationcategory->id; + update_course($coursedata); + cli_writeln("course $msgobject->course ($msgobject->courseid) moved from category $msgobject->sourcecategory (id=$msgobject->sourcecategoryid) to category $msgobject->destinationcategory (id=$msgobject->destinationcategoryid)"); + } + } + protected function getArgumentsHelp() { + $help = parent::getArgumentsHelp(); + $help .= "\n\n"; + $help .= "This command enable to move all courses included into a source cateogry into a destination category"; + return $help; + } +} \ No newline at end of file diff --git a/Moosh/Command/Moodle39/Cohort/CohortDelete.php b/Moosh/Command/Moodle39/Cohort/CohortDelete.php new file mode 100755 index 00000000..1b775061 --- /dev/null +++ b/Moosh/Command/Moodle39/Cohort/CohortDelete.php @@ -0,0 +1,50 @@ + + * @license http://www.gnu.org/copyleft/gpl.html GNU GPL v3 or later + */ + +namespace Moosh\Command\Moodle39\Cohort; +use Moosh\MooshCommand; +use context_coursecat; + +class CohortDelete extends MooshCommand +{ + public function __construct() + { + parent::__construct('delete', 'cohort'); + + $this->addArgument('ids'); + $this->maxArguments = 255; + } + + public function execute() + { + global $CFG, $DB; + + require_once $CFG->dirroot . '/cohort/lib.php'; + + foreach ($this->arguments as $argument) { + $ids = explode(',', $argument); + foreach ($ids as $id) { + $cohort = $DB->get_record('cohort',array('id'=>$id)); + if (!$cohort) { + echo "Cohort $id does not exists\n"; + continue; + } + cohort_delete_cohort($cohort); + cli_writeln("cohort $id deleted"); + } + } + } + protected function getArgumentsHelp() { + $help = parent::getArgumentsHelp(); + $help .= "\n\n"; + $help .= "Delete cohort"; + $help .= "\n'cohort ids separated by , (max 255 characters)'"; + return $help; + } +} diff --git a/Moosh/Command/Moodle39/Context/ContextFreeze.php b/Moosh/Command/Moodle39/Context/ContextFreeze.php new file mode 100644 index 00000000..b6f9136f --- /dev/null +++ b/Moosh/Command/Moodle39/Context/ContextFreeze.php @@ -0,0 +1,74 @@ + + * @license http://www.gnu.org/copyleft/gpl.html GNU GPL v3 or later + */ + +namespace Moosh\Command\Moodle39\Context; + +use Moosh\MooshCommand; +use context_course; +use course_enrolment_manager; + +class ContextFreeze extends MooshCommand { + public function __construct() { + parent::__construct('freeze', 'context'); + $this->addArgument('instanceid'); + $this->addArgument('contextlevel'); + $this->addArgument('lock'); + } + public function execute(){ + global $CFG; + require_once("{$CFG->libdir}/accesslib.php"); + require_once("{$CFG->libdir}/adminlib.php"); + $instanceid = $this->arguments[0]; + $contextlevel = $this->arguments[1]; + $lock = $this->arguments[2]; + $context = null; + switch($contextlevel){ + case CONTEXT_SYSTEM : + $context = \context_system::instance(); + break; + case CONTEXT_COURSE : + $context = \context_course::instance($instanceid); + break; + case CONTEXT_COURSECAT : + $context = \context_coursecat::instance($instanceid); + break; + case CONTEXT_MODULE : + $context = \context_module::instance($instanceid); + break; + case CONTEXT_BLOCK : + $context = \context_block::instance($instanceid); + break; + case CONTEXT_USER : + $context = \context_user::instance($instanceid); + break; + default: + cli_error("bad context level $contextlevel"); + } + if($context){ + $context->set_locked($lock); + if($lock){ + cli_writeln("context locked"); + } else { + cli_writeln("context unlocked"); + } + } else { + cli_error("context not found"); + } + } + + protected function getArgumentsHelp() { + $help = parent::getArgumentsHelp(); + $help .= "\n\n"; + $help .= "Freeze a context"; + $help .= "\ncontext are moodle context levels integer"; + $help .= "\nlock is 1 or 0"; + + return $help; + } +} \ No newline at end of file diff --git a/Moosh/Command/Moodle39/Course/CourseEnrolChangeStatus.php b/Moosh/Command/Moodle39/Course/CourseEnrolChangeStatus.php new file mode 100755 index 00000000..78f74067 --- /dev/null +++ b/Moosh/Command/Moodle39/Course/CourseEnrolChangeStatus.php @@ -0,0 +1,83 @@ + + * @license http://www.gnu.org/copyleft/gpl.html GNU GPL v3 or later + */ + +namespace Moosh\Command\Moodle39\Course; +use Moosh\MooshCommand; + +class CourseEnrolChangeStatus extends MooshCommand +{ + public function __construct() + { + parent::__construct('enrol-change-status', 'course'); + + $this->addOption('i|instanceid:', 'enrolment instance id', 0); + $this->addOption('s|status', 'status to be applied 0 for disable, 1 for enable',0); + $this->addArgument('courseid'); + $this->maxArguments = 255; + } + + function choice_value_check_and_prompt($promptmsg, $values){ + $var = trim(cli_input($promptmsg)); + if (!is_numeric($var) && ( in_array($var, $values))) { + cli_writeln('Entered value must be an int in ('.implode(',', $values).')'); + return self::choice_value_check_and_prompt(); + } + return (int)$var; + } + + public function execute() + { + global $CFG, $DB; + + require_once $CFG->dirroot . '/course/lib.php'; + $instanceid = $this->expandedOptions['instanceid']; + $status = $this->expandedOptions['status']; + $courseid = $this->arguments[0]; + if($courseid == 0) { + cli_error('you must enter a valid courseid'); + } + $course = $DB->get_record('course', array('id' => $courseid)); + if (!$course) { + cli_error("course $courseid not found"); + } + $instances = enrol_get_instances($course->id, false); + if( $instanceid == 0) { + // Enter in interactive mode + cli_writeln('Interactive mode'); + $options = array(); + foreach ($instances as $instance) { + $options[] = $instance->id; + cli_writeln("$instance->id : $instance->name - $instance->enrol - " + .($instance->status == ENROL_INSTANCE_ENABLED ? "enabled" : "disabled")); + } + $instanceid = self::choice_value_check_and_prompt( + "please choose what enrol method you want to enable/disable by typeing is instance id", + $options + ); + $status = self::choice_value_check_and_prompt( + "enable ".ENROL_INSTANCE_ENABLED." or disable ".ENROL_INSTANCE_DISABLED, + array(ENROL_INSTANCE_ENABLED,ENROL_INSTANCE_DISABLED)); + //$progress = new \progress_trace_buffer(new \text_progress_trace(), false); + $plugin = enrol_get_plugin($instances[$instanceid]->enrol); + $plugin->update_status($instances[$instanceid],$status); + cli_writeln("instance $instanceid as be passed to ".($status == ENROL_INSTANCE_ENABLED? "enabled" : "disabled" )); + } else { + // non interactive mode change status by id + $plugin = enrol_get_plugin($instances[$instanceid]->enrol); + $plugin->update_status($instances[$instanceid],$status); + cli_writeln("instance $instanceid as be passed to ".($status == ENROL_INSTANCE_ENABLED? "enabled" : "disabled" )); + } + } + protected function getArgumentsHelp() { + $help = parent::getArgumentsHelp(); + $help .= "\n\n"; + $help .= "Change enrol status"; + + return $help; + } +} diff --git a/Moosh/Command/Moodle39/Dashboard/DashboardResetAll.php b/Moosh/Command/Moodle39/Dashboard/DashboardResetAll.php new file mode 100644 index 00000000..febf3ec3 --- /dev/null +++ b/Moosh/Command/Moodle39/Dashboard/DashboardResetAll.php @@ -0,0 +1,33 @@ + + * @license http://www.gnu.org/copyleft/gpl.html GNU GPL v3 or later + */ + +namespace Moosh\Command\Moodle39\Dashboard; +use Moosh\MooshCommand; + +class DashboardResetAll extends MooshCommand +{ + public function __construct() { + parent::__construct('reset-all', 'dashboard'); + } + public function execute() + { + global $CFG; + require_once($CFG->dirroot.'/my/lib.php'); + my_reset_page_for_all_users(MY_PAGE_PRIVATE, 'my-index'); + } + + protected function getArgumentsHelp() { + $help = parent::getArgumentsHelp(); + $help .= "\n\n"; + $help .= "Reset all dashboards"; + + return $help; + } +} \ No newline at end of file diff --git a/Moosh/Command/Moodle39/Task/TaskSchedule.php b/Moosh/Command/Moodle39/Task/TaskSchedule.php new file mode 100644 index 00000000..c5d52830 --- /dev/null +++ b/Moosh/Command/Moodle39/Task/TaskSchedule.php @@ -0,0 +1,93 @@ + + * @license http://www.gnu.org/copyleft/gpl.html GNU GPL v3 or later + */ + +namespace Moosh\Command\Moodle39\Task; +use Moosh\MooshCommand; + +class TaskSchedule extends MooshCommand +{ + + public function __construct() { + parent::__construct('schedule', 'task'); + $this->addOption('M|minute:', 'minute'); + $this->addOption('H|hour:', 'hour'); + $this->addOption('d|day:', 'day'); + $this->addOption('m|month:', 'month'); + $this->addOption('w|dayofweek:', 'Day of week'); + $this->addOption('x|disabled:', 'Disbaled'); + $this->addOption('r|resettodefaults:', 'Reset to defaults', 0); + $this->addArgument('taskname'); + $this->minArguments = 0; + $this->maxArguments = 1; + + } + + public function execute() + { + if(count($this->arguments) == 0){ + $tasks = \core\task\manager::get_all_scheduled_tasks(); + $tasknames = array(); + foreach($tasks as $currenttask){ + $tasknames[] = get_class($currenttask); + } + cli_writeln('Available task names are :'); + cli_writeln(implode(PHP_EOL, $tasknames)); + die; + } + $taskname = $this->arguments[0]; + $task = \core\task\manager::get_scheduled_task($taskname); + if (!$task) { + cli_error("task $taskname not exists"); + } + if (!empty($this->expandedOptions['resettodefaults'])) { + $defaulttask = \core\task\manager::get_default_scheduled_task($taskname); + $task->set_minute($defaulttask->get_minute()); + $task->set_hour($defaulttask->get_hour()); + $task->set_month($defaulttask->get_month()); + $task->set_day_of_week($defaulttask->get_day_of_week()); + $task->set_day($defaulttask->get_day()); + $task->set_disabled($defaulttask->get_disabled()); + $task->set_customised(false); + } else { + if(!empty($this->expandedOptions['minute'])){ + $task->set_minute($this->expandedOptions['minute']); + } + if(!empty($this->expandedOptions['hour'])) { + $task->set_hour($this->expandedOptions['hour']); + } + if(!empty($this->expandedOptions['month'])) { + $task->set_month($this->expandedOptions['month']); + } + if(!empty($this->expandedOptions['dayofweek'])) { + $task->set_day_of_week($this->expandedOptions['dayofweek']); + } + if(!empty($this->expandedOptions['day'])) { + $task->set_day($this->expandedOptions['day']); + } + if(!empty($this->expandedOptions['disabled'])) { + $task->set_disabled($this->expandedOptions['disabled']); + } + $task->set_customised(true); + } + try { + \core\task\manager::configure_scheduled_task($task); + cli_writeln("Task configured"); + } catch (Exception $e) { + cli_writeln("error while configuring task $taskname"); + cli_error($e->getMessage()); + } + } + protected function getArgumentsHelp() { + $help = parent::getArgumentsHelp(); + $help .= "\n\n"; + $help .= "Schedule Moodle task"; + + return $help; + } +} \ No newline at end of file diff --git a/Moosh/Command/Moodle39/Webservice/WebserviceInstall.php b/Moosh/Command/Moodle39/Webservice/WebserviceInstall.php new file mode 100644 index 00000000..bebe7efa --- /dev/null +++ b/Moosh/Command/Moodle39/Webservice/WebserviceInstall.php @@ -0,0 +1,128 @@ + + * @license http://www.gnu.org/copyleft/gpl.html GNU GPL v3 or later + */ + +namespace Moosh\Command\Moodle39\Webservice; +use Moosh\MooshCommand; + +class WebserviceInstall extends MooshCommand +{ + public function __construct() { + parent::__construct('install', 'webservice'); + $this->addArgument('servicename'); + $this->addArgument('capabilities'); + $this->addOption('m|mail:', 'mail'); + $this->addOption('u|username:', 'user name'); + $this->addOption('r|rolename:', 'role name'); + $this->addOption('i|iprestriction:',' ip restriction', ''); + $this->addOption('v|validuntil:', 'valid until', 0); + } + + public function execute() + { + global $DB, $CFG; + require_once($CFG->dirroot.'/user/lib.php'); + require_once($CFG->dirroot.'/webservice/lib.php'); + $servicename = $this->arguments[0]; + $service = $DB->get_record('external_services', array('shortname' => $servicename)); + if(!$service) { + echo "service $servicename does not exist".PHP_EOL; + return; + } + $capabilities = explode(',', $this->arguments[1]); + $rolename = $this->expandedOptions['rolename'] ? $this->expandedOptions['rolename'] : 'role_'.$servicename; + $username= $this->expandedOptions['username'] ? $this->expandedOptions['username'] : 'user_'.$servicename; + $iprestriction = $this->expandedOptions['iprestriction']; + $validuntil = $this->expandedOptions['validuntil']; + // Create user. + $email = null; + if ($this->expandedOptions['mail']) { + $email = $this->expandedOptions['mail']; + } else { + $noreplyuser = \core_user::get_noreply_user(); + $email= $servicename.'_'.$noreplyuser->email; + } + // Check if user exits. + $wsuserid = 0; + $userobject = $DB->get_record('user', array('username' => $username)); + if($userobject) { + $wsuserid = $userobject->id; + echo "user $username already exists so will use it".PHP_EOL; + } else { + $user = new \stdClass(); + $user->username = $username; + $user->firstname = $username; + $user->lastname = $username; + $user->email = $email; + $user->confirmed=1; + $user->policyagreed=1; + $user->mnethostid = $CFG->mnet_localhost_id; + $wsuserid = user_create_user($user); + } + // Create Role + $systemcontext = \context_system::instance(); + $rolerecord = $DB->get_record('role', array('shortname' => $rolename)); + $wsroleid = 0; + if ($rolerecord) { + $wsroleid = $rolerecord->id; + echo "role $rolename already exists, we'll use it".PHP_EOL; + } else { + $wsroleid = create_role( + $rolename, + $rolename, + $rolename + ); + } + // Assign necessary capabilities + foreach($capabilities as $capability){ + $capability = trim($capability); + $capaobject = $DB->get_record('capabilities', array('name' => $capability)); + if(!$capability){ + echo "capability $capability not exists".PHP_EOL; + continue; + } + assign_capability($capability, CAP_ALLOW, + $wsroleid, $systemcontext->id, true); + } + // Allow role assignment on system. + set_role_contextlevels($wsroleid, array(10 => 10)); + // Assign role to user. + role_assign($wsroleid, $wsuserid, $systemcontext->id); + // Assign user to webservice. + $webservicemanager = new \webservice(); + $serviceuser = new \stdClass(); + $serviceuser->externalserviceid = $service->id; + $serviceuser->userid = $wsuserid; + $serviceuser->iprestriction = $iprestriction; + + $webservicemanager->add_ws_authorised_user($serviceuser); + // Trigger events. + $params = array( + 'objectid' => $serviceuser->externalserviceid, + 'relateduserid' => $serviceuser->userid + ); + $event = \core\event\webservice_service_user_added::create($params); + $event->trigger(); + // Take admin role + // Generate Token. + $token = external_generate_token(EXTERNAL_TOKEN_PERMANENT, $service->id, $wsuserid, $systemcontext->id, $validuntil, $iprestriction); + echo "Webservice $servicename installed : user $username and role $rolename".PHP_EOL; + echo "Token generated $token"; + } + protected function getArgumentsHelp() { + $help = parent::getArgumentsHelp(); + $help .= "\n\n"; + $help .= "Install a webservice "; + $help .= "\ncapabilities are seperated by a commas.\n\n"; + $help .= "\nif user is not defined, user_servicename will be created"; + $help .= "\nif role is not defined, role_servicename will be created"; + + return $help; + } +} \ No newline at end of file diff --git a/Moosh/Command/Moodle41/Cache/CacheStoreClear.php b/Moosh/Command/Moodle41/Cache/CacheStoreClear.php new file mode 100755 index 00000000..f8e7e3fc --- /dev/null +++ b/Moosh/Command/Moodle41/Cache/CacheStoreClear.php @@ -0,0 +1,81 @@ + + * @license http://www.gnu.org/copyleft/gpl.html GNU GPL v3 or later + */ + +namespace Moosh\Command\Moodle41\Cache; +use Moosh\MooshCommand; + +class CacheStoreClear extends MooshCommand +{ + public function __construct() + { + parent::__construct('store-clear', 'cache'); + $this->addOption('d|definition:', 'cache definition, command will only purge this cache definition'); + $this->addArgument('storename'); + $this->minArguments = 1; + } + + public function execute() + { + global $CFG; + require_once($CFG->dirroot.'/cache/classes/factory.php'); + require_once($CFG->dirroot.'/cache/classes/helper.php'); + $instance = \cache_config::instance(); + $factory = \cache_factory::instance(); + $config = $factory->create_config_instance(); + $stores = $instance->get_all_stores(); + $this->expandOptionsManually($this->arguments); + $options = $this->expandedOptions; + $storename = $this->arguments[0]; + if (!array_key_exists($storename, $stores)) { + cli_error("Store with name $storename does not exists."); + } + $storedetails= $stores[$storename]; + $class = $storedetails['class']; + $storeinstance = new $class($storedetails['name'], $storedetails['configuration']); + if (!$storeinstance->are_requirements_met()){ + cli_error("Store $storename does not meet requirements."); + } + if(!$storeinstance->is_ready()) { + cli_error("Store $storename si not ready."); + } + $definitionid = $options['definition']; + if (!empty($options['definition'])) { + $definitions = $config->get_definitions(); + if (!array_key_exists($definitionid, $definitions)) { + cli_error("Cache definition $definitionid not found in the $storename cache store."); + } + $definition = $definitions[$definitionid]; + $definition = \cache_definition::load($definitionid, $definition); + $definitioninstance = clone($storeinstance); + $definitioninstance->initialise($definition); + if (!$definitioninstance->purge()) { + cli_error("For $storename cache store , Cache definition $definitionid purge return false."); + } else { + cli_writeln("Cache definition $definitionid in $storename cache store sucessfully purged."); + } + unset($definitioninstance); + } else { + // Purge all cache definitions. + if (\cache_helper::purge_store($storename,$config)) { + cli_writeln("Store $storename purged."); + } else { + cli_error("Store $storename purge returned false."); + } + } + } + + protected function getArgumentsHelp() { + $help = parent::getArgumentsHelp(); + $help .= "\n\n"; + $help .= "Clear a specific cache"; + $help .= "\nIt is possible to only clean a cache definition"; + + return $help; + } +} From 0749ea27615684045da9b9cdbbb5e84083257133 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?C=C3=A9line=20Perv=C3=A8s?= Date: Fri, 28 Jun 2024 18:26:14 +0200 Subject: [PATCH 2/4] correction hemp --- Moosh/Command/Moodle39/Cache/CacheSet.php | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/Moosh/Command/Moodle39/Cache/CacheSet.php b/Moosh/Command/Moodle39/Cache/CacheSet.php index 4025aeec..994d0538 100644 --- a/Moosh/Command/Moodle39/Cache/CacheSet.php +++ b/Moosh/Command/Moodle39/Cache/CacheSet.php @@ -57,7 +57,7 @@ public function execute() protected function getArgumentsHelp() { $help = parent::getArgumentsHelp(); $help .= "\n\n"; - $help .= "This command enable to attribute cache to specific cache store"; + $help .= "This command enable to attribute cache to specific cache definition"; $help .= "\ne.g moosh -n cache-set core/calendar_subscriptions redisstore"; return $help; } From fc01b9688c3f2e3bc14088b0fa8a42e95e6c08ad Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?C=C3=A9line=20Perv=C3=A8s?= Date: Mon, 8 Jul 2024 16:24:54 +0200 Subject: [PATCH 3/4] correct min argument for task-schedule --- Moosh/Command/Moodle39/Task/TaskSchedule.php | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/Moosh/Command/Moodle39/Task/TaskSchedule.php b/Moosh/Command/Moodle39/Task/TaskSchedule.php index c5d52830..2490704c 100644 --- a/Moosh/Command/Moodle39/Task/TaskSchedule.php +++ b/Moosh/Command/Moodle39/Task/TaskSchedule.php @@ -20,10 +20,10 @@ public function __construct() { $this->addOption('d|day:', 'day'); $this->addOption('m|month:', 'month'); $this->addOption('w|dayofweek:', 'Day of week'); - $this->addOption('x|disabled:', 'Disbaled'); + $this->addOption('x|disabled:', 'Disabled'); $this->addOption('r|resettodefaults:', 'Reset to defaults', 0); $this->addArgument('taskname'); - $this->minArguments = 0; + $this->minArguments = 1; $this->maxArguments = 1; } From c147af2f6cb13323f409f5f9b3fb4ba5d5a9dc3c Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?C=C3=A9line=20Perv=C3=A8s?= Date: Mon, 8 Jul 2024 16:29:41 +0200 Subject: [PATCH 4/4] complete index documentation for new commands previously pushed --- www/commands/index.md | 190 ++++++++++++++++++++++++++++++++++++++++++ 1 file changed, 190 insertions(+) diff --git a/www/commands/index.md b/www/commands/index.md index a2e7184d..f7b80012 100755 --- a/www/commands/index.md +++ b/www/commands/index.md @@ -413,6 +413,35 @@ Example 3: Set Application to "store name", Session to "Tests" and Request to "d moosh cache-edit-mappings -a "store name" -s Tests -r default_request +cache-set +------------------- + +Enable to attribute cache to specific cache definition + +Requires cache definition and cache store + +Example 1: Set "redisstore" to "core/calendar"_subscriptions definition + + moosh cache-set core/calendar_subscriptions redisstore + +cache-store-clear +----------------- + +Clear a specific cache store or only a definition for the given cache store + +Requires cache store name + +Option: +* --definition, -d : cache definition name + +Example 1 : Clear redisstore cache store + + moosh cache-store-clear --defintion=calendar_subscriptions redisstore + +Example 2 : Clear cache definition core/calendar_subscriptions for store named redisstore + + moosh cache-store-clear --definition=core/calendar_subscriptions redisstore + category-config-set ------------------- @@ -505,6 +534,17 @@ Example 2: Make the category with id 3 a top-level category moosh category-move 3 0 +category-move-courses-from-category-to-another +---------------------------------------------- + +Move all courses included into a source category into a destination category + +Requires Source course category id and Destination course category id + +Example 1: Move all courses included into category with id 1 to category with id 2 + + moosh category-move-courses-from-category-to-another 1 2 + category-resortcourses ---------------------- @@ -561,6 +601,22 @@ Example 2: Create cohort "my cohort18" with id "cohort18" under category id 2, w moosh cohort-create -d "Long description" -i cohort18 -c 2 "my cohort18" +cohort-delete +------------ + +Delete one or more cohorts + +Requires cohort ids to delete separated by , + +Example 1: Delete cohort of id 42 + + moosh cohort-cohort-delete 42 + +Example 2: Delete cohorts of ids 42 and 2012 + + moosh cohort-cohort-delete 42,2012 + + cohort-enrol ------------ @@ -677,6 +733,22 @@ Example 2: Set URL to logo for Sky High theme. moosh config-set logo http://example.com/logo.png theme_sky_high +context-freeze +-------------- +Freeze or unfreeze a given context + +Requires instance id, context level and lock + +Lock is 1 if locked, 0 if unlock + +Example 1 : Lock course context of id 20 + + moosh context-freeze 20 50 1 + +Example 1 : Unlock course context of id 20 + + moosh context-freeze 20 50 0 + context-rebuild --------------- @@ -796,6 +868,23 @@ Example 1: Enroll user with firstname test42 and lastname user42 into the course moosh course-enrolbyname -r editingteacher -f test42 -l user42 -c T12345 +course-enrol-change-status +------------------ + +Requires course id + +Options : +* --instanceid=?, --i=? : enrolment instance id, if not command enter in interactive mode and show all available enrolment instance for the given course, you'll have to choose status from prompt +* --status=0 or 1, --s=0 or 1 : status of enrolment instance 0 (default value) -> enabled, 1 -> disabled + +Example1 : change course enrolment instance status in interactive mode + + moosh course-enrol-change-status 2 + +Example1 : change course enrolment instance status to disable for instance 42 + + moosh course-enrol-change-status --i=42 --s=1 2 + course-enableselfenrol ---------------------- @@ -975,6 +1064,14 @@ Example 1: Unenrol users with id 7, 9, 12 and 16 from course with id 2. moosh course-unenrol 2 7 9 12 16 +dashboard-reset-all +---------- +Reset all users dashboard + +Example 1: Reset all users dashboard + + moosh dashboard-reset-all + data-stats ---------- @@ -1743,6 +1840,22 @@ Example: moosh php-eval 'var_dump(get_object_vars($CFG))' +plugin-hideshow +--------------- +Hide or show a plugin in all site context + +Requires plugin type, plugin name and show option. + +Show option is 0 if hide and 1 if show. + +This will work for the following plugin types: + +block, mod, assignfeedback, assignsubmission, qtype, qbehaviour, enrol, filter, editor, auth, license, repository, courseformat or avaibility + +Example 1 : Hide chat module plugin for entire site + + moosh plugin-hideshow mod chat 0 + plugin-download --------------- @@ -1901,6 +2014,28 @@ Example 2: Create the report for the last week. Could be used in a cronjob. start=$(date --date="7 days ago" +"%Y-%m-%d");finish=$(date +"%Y-%m-%d");moosh report-concurrency --from $start --to $finish +request-select +------- + +Run any custom SQL Select against bootstrapped Moodle instance DB and return resultset with csv format with ; separator. + +Requires a select query as argument. + +Usefull to create a csv file to import datas in moodle + +Usefull also to retrieve an value to reuse it later in a script composed of moosh commands + +Example 1: Select username, firstname, lastname and email + + moosh request-select "select username,firstname,lastname, email from {user}" + +Output: + + egrieg;Edward,Grieg,egrieg@example.com + + hpurcell;Henry;Purcell,hpurcell@example.com + + restore-settings ---------------- @@ -2100,6 +2235,27 @@ Example 2: show all tasks and their status (locked/unlocked) moosh task-lock-check -a +task-schedule +------------- + +Schedule Moodle task + +Requires task name with namespace + +Options: +* -M, --minute :minute +* -H, --hour : hour +* -d, --day : day +* -m, --month : month +* -w, --dayofweek : Day of week +* -x, --disabled : Disabled +* -r, --resettodefaults : Reset to defaults + + +Example 1 : Schedule cleanup_task to launch every day at 01:01 + + moosh -d=* -H=1 -m=1 "\tool_messageinbound\task\cleanup_task" + theme-info ---------- @@ -2362,3 +2518,37 @@ Calls Example: Get list of all courses enroled for a user moosh webservice-call --token 4ac42118db3ee8d4b1ae78f2c1232afd --params userid=3 core_enrol_get_users_courses + +webservice-install +------------------ + +Install a Moodle Webservice + +Requires service name and capabilities +* service name is the services["shortname'] defined in db/services.php file of a plugin +* Capabilities are separated by commas + +Options: +* -m, --mail : mail, by default noreply user mail +* -u, --username : username + * by default 'user_ will be used + * if not exists user is created +* -r, --rolename : role shortname + * by default role_ will used + * if not exists role will be created +* i, --iprestriction : ip restriction + * empty by default so all ip authorized +-v, --validuntil : valid until + * empty so valid forever + +Output: +* user user_servicename is created +* role role_servicename are created with definied capabilities +* user has role role_servicename on system +* webservice is instanciate with its function for user user_servicename +* token is genrated and output + +Example 1 : Install a service named wsservicename with capabilities + + moosh webservice-install wsservicename 'plugintype/pluginname:capability1,plugintype/pluginname:capability2' +