From dd122138d39d43c22fa338d9c195f9d5e921435d Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Pau=20Ferrer=20Oca=C3=B1a?= Date: Tue, 31 Oct 2023 16:08:09 +0100 Subject: [PATCH 1/3] MOBILE-4368 scripts: Lang packs path should be different in both scripts --- scripts/create_langindex_functions.sh | 2 ++ scripts/lang_functions.sh | 1 - scripts/update_lang_functions.sh | 2 ++ 3 files changed, 4 insertions(+), 1 deletion(-) diff --git a/scripts/create_langindex_functions.sh b/scripts/create_langindex_functions.sh index bc6eb3c52..33f160842 100644 --- a/scripts/create_langindex_functions.sh +++ b/scripts/create_langindex_functions.sh @@ -5,6 +5,8 @@ SERVER_URL='https://packaging.moodle.org/' +LANGPACKS_PATH='/tmp/moodle-lang' + # Downloads a file and if it's a zip file, unzip it. function download_file { local url=$1 diff --git a/scripts/lang_functions.sh b/scripts/lang_functions.sh index 649f57c1a..6e794d66c 100644 --- a/scripts/lang_functions.sh +++ b/scripts/lang_functions.sh @@ -6,7 +6,6 @@ LANG_PATH='../src/assets/lang' SUFFIX='' # Determines suffix of the langpacks to be merged. Ie, using wp will include en.json and en_wp.json # (and the later will override the earlier). -LANGPACKS_PATH='/tmp/moodleapp-lang' # Get the version of the Moodle App to fetch latest languages. function get_app_version { diff --git a/scripts/update_lang_functions.sh b/scripts/update_lang_functions.sh index 1b9384297..a2605115e 100755 --- a/scripts/update_lang_functions.sh +++ b/scripts/update_lang_functions.sh @@ -8,6 +8,8 @@ APPMODULENAME='local_moodlemobileapp' TOTAL_STRINGS=0 LANGINDEX_STRINGS=0 +LANGPACKS_PATH='/tmp/moodleapp-lang' + function progressbar { let _progress=(${1}*100/100*100)/100 let _done=(${_progress}*4)/10 From 9099655f1ec50e693662fdd1752704909543d217 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Pau=20Ferrer=20Oca=C3=B1a?= Date: Tue, 31 Oct 2023 16:21:35 +0100 Subject: [PATCH 2/3] MOBILE-4368 icon: Decode HTML entities that may be included in icon url --- src/core/components/mod-icon/mod-icon.ts | 3 +++ 1 file changed, 3 insertions(+) diff --git a/src/core/components/mod-icon/mod-icon.ts b/src/core/components/mod-icon/mod-icon.ts index a51788f87..d35294ff7 100644 --- a/src/core/components/mod-icon/mod-icon.ts +++ b/src/core/components/mod-icon/mod-icon.ts @@ -17,6 +17,7 @@ import { Component, ElementRef, Input, OnChanges, OnInit, SimpleChange } from '@ import { CoreCourse } from '@features/course/services/course'; import { CoreCourseModuleDelegate } from '@features/course/services/module-delegate'; import { CoreSites } from '@services/sites'; +import { CoreTextUtils } from '@services/utils/text'; import { CoreUrlUtils } from '@services/utils/url'; const assetsPath = 'assets/img/'; @@ -141,6 +142,8 @@ export class CoreModIconComponent implements OnInit, OnChanges { return false; } + this.icon = CoreTextUtils.decodeHTMLEntities(this.icon); + // If it's an Moodle Theme icon, check if filtericon is set and use it. if (this.icon && CoreUrlUtils.isThemeImageUrl(this.icon)) { const iconParams = CoreUrlUtils.extractUrlParams(this.icon); From db9865ca284e07ea0fa04760764b02e0f50efade Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Pau=20Ferrer=20Oca=C3=B1a?= Date: Tue, 31 Oct 2023 16:28:07 +0100 Subject: [PATCH 3/3] MOBILE-4368 scripts: WS Structure Scripts have been moved away from here https://github.com/moodlehq/moodle-local_moodlemobileapp/tree/main/structure --- scripts/get_all_ws_structures.php | 48 ----- scripts/get_ws_changes.php | 105 ----------- scripts/get_ws_structure.php | 63 ------- scripts/ws_to_ts_functions.php | 294 ------------------------------ 4 files changed, 510 deletions(-) delete mode 100644 scripts/get_all_ws_structures.php delete mode 100644 scripts/get_ws_changes.php delete mode 100644 scripts/get_ws_structure.php delete mode 100644 scripts/ws_to_ts_functions.php diff --git a/scripts/get_all_ws_structures.php b/scripts/get_all_ws_structures.php deleted file mode 100644 index 8a07b8e0c..000000000 --- a/scripts/get_all_ws_structures.php +++ /dev/null @@ -1,48 +0,0 @@ -. - -/** - * Script for getting the PHP structure of a WS returns or params. - * - * The first parameter (required) is the path to the Moodle installation to use. - * The second parameter (required) is the name to the WS to convert. - * The third parameter (optional) is a number: 1 to convert the params structure, - * 0 to convert the returns structure. Defaults to 0. - */ - -if (!isset($argv[1])) { - echo "ERROR: Please pass the Moodle path as the first parameter.\n"; - die(); -} - -$moodlepath = $argv[1]; - -define('CLI_SCRIPT', true); - -require($moodlepath . '/config.php'); -require($CFG->dirroot . '/webservice/lib.php'); -require_once('ws_to_ts_functions.php'); - -$structures = get_all_ws_structures(); - -foreach ($structures as $wsname => $structure) { - - remove_default_closures($structure->parameters_desc); - print_ws_structure($wsname, $structure->parameters_desc, true); - - remove_default_closures($structure->returns_desc); - print_ws_structure($wsname, $structure->returns_desc, false); -} diff --git a/scripts/get_ws_changes.php b/scripts/get_ws_changes.php deleted file mode 100644 index 4e5794d81..000000000 --- a/scripts/get_ws_changes.php +++ /dev/null @@ -1,105 +0,0 @@ -. - -/** - * Script for detecting changes in a WS params or return data, version by version. - * - * The first parameter (required) is the path to the Moodle installation to use. - * The second parameter (required) is the name to the WS to convert. - * The third parameter (optional) is a number: 1 to convert the params structure, - * 0 to convert the returns structure. Defaults to 0. - */ - -if (!isset($argv[1])) { - echo "ERROR: Please pass the path to the folder containing the Moodle installations as the first parameter.\n"; - die(); -} - -if (!isset($argv[2])) { - echo "ERROR: Please pass the WS name as the second parameter.\n"; - die(); -} - -define('CLI_SCRIPT', true); -define('CACHE_DISABLE_ALL', true); -define('SERIALIZED', true); -require_once('ws_to_ts_functions.php'); - -$versions = array('master', '310', '39', '38', '37', '36', '35', '34', '33', '32', '31'); - -$moodlespath = $argv[1]; -$wsname = $argv[2]; -$useparams = (bool)(isset($argv[3]) && $argv[3]); -$pathseparator = '/'; - -// Get the path to the script. -$index = strrpos(__FILE__, $pathseparator); -if ($index === false) { - $pathseparator = '\\'; - $index = strrpos(__FILE__, $pathseparator); -} -$scriptfolder = substr(__FILE__, 0, $index); -$scriptpath = concatenate_paths($scriptfolder, 'get_ws_structure.php', $pathseparator); - -$previousstructure = null; -$previousversion = null; -$libsloaded = false; - -foreach ($versions as $version) { - $moodlepath = concatenate_paths($moodlespath, 'stable_' . $version . '/moodle', $pathseparator); - - if (!file_exists($moodlepath)) { - echo "Folder does not exist for version $version, skipping...\n"; - continue; - } - - if (!$libsloaded) { - $libsloaded = true; - - require($moodlepath . '/config.php'); - require($CFG->dirroot . '/webservice/lib.php'); - } - - // Get the structure in this Moodle version. - $structure = shell_exec("php $scriptpath $moodlepath $wsname " . ($useparams ? 'true' : '')); - - if (strpos($structure, 'ERROR:') === 0) { - echo "WS not found in version $version. Stop.\n"; - break; - } - - $structure = unserialize($structure); - - if ($previousstructure != null) { - echo "*** Check changes from version $version to $previousversion ***\n"; - - $messages = detect_ws_changes($previousstructure, $structure); - - if (count($messages) > 0) { - $haschanged = true; - - foreach($messages as $message) { - echo "$message\n"; - } - } else { - echo "No changes found.\n"; - } - echo "\n"; - } - - $previousstructure = $structure; - $previousversion = $version; -} diff --git a/scripts/get_ws_structure.php b/scripts/get_ws_structure.php deleted file mode 100644 index 961708cc5..000000000 --- a/scripts/get_ws_structure.php +++ /dev/null @@ -1,63 +0,0 @@ -. - -/** - * Script for getting the PHP structure of a WS returns or params. - * - * The first parameter (required) is the path to the Moodle installation to use. - * The second parameter (required) is the name to the WS to convert. - * The third parameter (optional) is a number: 1 to convert the params structure, - * 0 to convert the returns structure. Defaults to 0. - */ - -if (!isset($argv[1])) { - echo "ERROR: Please pass the Moodle path as the first parameter.\n"; - die(); -} - -if (!isset($argv[2])) { - echo "ERROR: Please pass the WS name as the second parameter.\n"; - die(); -} - -if (!defined('SERIALIZED')) { - define('SERIALIZED', false); -} - -$moodlepath = $argv[1]; -$wsname = $argv[2]; -$useparams = (bool)(isset($argv[3]) && $argv[3]); - -define('CLI_SCRIPT', true); - -require($moodlepath . '/config.php'); -require($CFG->dirroot . '/webservice/lib.php'); -require_once('ws_to_ts_functions.php'); - -$structure = get_ws_structure($wsname, $useparams); - -if ($structure === false) { - echo "ERROR: The WS wasn't found in this Moodle installation.\n"; - die(); -} - -remove_default_closures($structure); - -if (SERIALIZED) { - echo serialize($structure); -} else { - print_ws_structure($wsname, $structure, $useparams); -} diff --git a/scripts/ws_to_ts_functions.php b/scripts/ws_to_ts_functions.php deleted file mode 100644 index ef11eab0a..000000000 --- a/scripts/ws_to_ts_functions.php +++ /dev/null @@ -1,294 +0,0 @@ -. - -/** - * Helper functions for converting a Moodle WS structure to a TS type. - */ - -use core_external\external_api; -use core_external\external_function_parameters; -use core_external\external_value; -use core_external\external_warnings; -use core_external\external_files; -use core_external\external_single_structure; -use core_external\external_multiple_structure; - -/** - * Get the structure of a WS params or returns. - */ -function get_ws_structure($wsname, $useparams) { - global $DB; - - // get all the function descriptions - $function = $DB->get_record('external_functions', array('services' => 'moodle_mobile_app', 'name' => $wsname)); - if (!$function) { - return false; - } - - $functiondesc = external_api::external_function_info($function); - - if ($useparams) { - return $functiondesc->parameters_desc; - } else { - return $functiondesc->returns_desc; - } -} - -/** - * Return all WS structures. - */ -function get_all_ws_structures() { - global $DB; - - // get all the function descriptions - $functions = $DB->get_records('external_functions', array('services' => 'moodle_mobile_app'), 'name'); - $functiondescs = array(); - foreach ($functions as $function) { - $functiondescs[$function->name] = external_api::external_function_info($function); - } - - return $functiondescs; -} - -/** - * Fix a comment: make sure first letter is uppercase and add a dot at the end if needed. - */ -function fix_comment($desc) { - $desc = trim($desc); - $desc = ucfirst($desc); - - if (substr($desc, -1) !== '.') { - $desc .= '.'; - } - - $lines = explode("\n", $desc); - if (count($lines) > 1) { - $desc = array_shift($lines)."\n"; - - foreach ($lines as $line) { - $spaces = strlen($line) - strlen(ltrim($line)); - $desc .= str_repeat(' ', $spaces - 3) . '// '. ltrim($line)."\n"; - } - } - - return $desc; -} - -/** - * Get an inline comment based on a certain text. - */ -function get_inline_comment($desc) { - if (empty($desc)) { - return ''; - } - - return ' // ' . fix_comment($desc); -} - -/** - * Add the TS documentation of a certain element. - */ -function get_ts_doc($type, $desc, $indentation) { - if (empty($desc)) { - // If no key, it's probably in an array. We only document object properties. - return ''; - } - - return $indentation . "/**\n" . - $indentation . " * " . fix_comment($desc) . "\n" . - (!empty($type) ? ($indentation . " * @type {" . $type . "}\n") : '') . - $indentation . " */\n"; -} - -/** - * Specify a certain type, with or without a key. - */ -function convert_key_type($key, $type, $required, $indentation) { - if ($key) { - // It has a key, it's inside an object. - return $indentation . "$key" . ($required == VALUE_OPTIONAL || $required == VALUE_DEFAULT ? '?' : '') . ": $type"; - } else { - // No key, it's probably in an array. Just include the type. - return $type; - } -} - -/** - * Convert a certain element into a TS structure. - */ -function convert_to_ts($key, $value, $boolisnumber = false, $indentation = '', $arraydesc = '') { - if ($value instanceof external_value || $value instanceof external_warnings || $value instanceof external_files) { - // It's a basic field or a pre-defined type like warnings. - $type = 'string'; - - if ($value instanceof external_warnings) { - $type = 'CoreWSExternalWarning[]'; - } else if ($value instanceof external_files) { - $type = 'CoreWSExternalFile[]'; - } else if ($value->type == PARAM_BOOL && !$boolisnumber) { - $type = 'boolean'; - } else if (($value->type == PARAM_BOOL && $boolisnumber) || $value->type == PARAM_INT || $value->type == PARAM_FLOAT || - $value->type == PARAM_LOCALISEDFLOAT || $value->type == PARAM_PERMISSION || $value->type == PARAM_INTEGER || - $value->type == PARAM_NUMBER) { - $type = 'number'; - } - - return convert_key_type($key, $type, $value->required, $indentation); - - } else if ($value instanceof external_single_structure) { - // It's an object. - $result = convert_key_type($key, '{', $value->required, $indentation); - - if ($arraydesc) { - // It's an array of objects. Print the array description now. - $result .= get_inline_comment($arraydesc); - } - - $result .= "\n"; - - foreach ($value->keys as $key => $value) { - $result .= convert_to_ts($key, $value, $boolisnumber, $indentation . ' ') . ';'; - - if (!$value instanceof external_multiple_structure || !$value->content instanceof external_single_structure) { - // Add inline comments after the field, except for arrays of objects where it's added at the start. - $result .= get_inline_comment($value->desc); - } - - $result .= "\n"; - } - - $result .= "$indentation}"; - - return $result; - - } else if ($value instanceof external_multiple_structure) { - // It's an array. - $result = convert_key_type($key, '', $value->required, $indentation); - - $result .= convert_to_ts(null, $value->content, $boolisnumber, $indentation, $value->desc); - - $result .= "[]"; - - return $result; - } else if ($value == null) { - return "{}; // WARNING: Null structure found"; - } else { - return "{}; // WARNING: Unknown structure: $key " . get_class($value); - } -} - -/** - * Print structure ready to use. - */ -function print_ws_structure($name, $structure, $useparams) { - if ($useparams) { - $type = implode('', array_map('ucfirst', explode('_', $name))) . 'WSParams'; - $comment = "Params of $name WS."; - } else { - $type = implode('', array_map('ucfirst', explode('_', $name))) . 'WSResponse'; - $comment = "Data returned by $name WS."; - } - - echo " -/** - * $comment - */ -export type $type = ".convert_to_ts(null, $structure).";\n"; -} - -/** - * Concatenate two paths. - */ -function concatenate_paths($left, $right, $separator = '/') { - if (!is_string($left) || $left == '') { - return $right; - } else if (!is_string($right) || $right == '') { - return $left; - } - - $lastCharLeft = substr($left, -1); - $firstCharRight = $right[0]; - - if ($lastCharLeft === $separator && $firstCharRight === $separator) { - return $left . substr($right, 1); - } else if ($lastCharLeft !== $separator && $firstCharRight !== '/') { - return $left . '/' . $right; - } else { - return $left . $right; - } -} - -/** - * Detect changes between 2 WS structures. We only detect fields that have been added or modified, not removed fields. - */ -function detect_ws_changes($new, $old, $key = '', $path = '') { - $messages = []; - - if (gettype($new) != gettype($old)) { - // The type has changed. - $messages[] = "Property '$key' has changed type, from '" . gettype($old) . "' to '" . gettype($new) . - ($path != '' ? "' inside $path." : "'."); - - } else if ($new instanceof external_value && $new->type != $old->type) { - // The type has changed. - $messages[] = "Property '$key' has changed type, from '" . $old->type . "' to '" . $new->type . - ($path != '' ? "' inside $path." : "'."); - - } else if ($new instanceof external_warnings || $new instanceof external_files) { - // Ignore these types. - - } else if ($new instanceof external_single_structure) { - // Check each subproperty. - $newpath = ($path != '' ? "$path." : '') . $key; - - foreach ($new->keys as $subkey => $value) { - if (!isset($old->keys[$subkey])) { - // New property. - $messages[] = "New property '$subkey' found" . ($newpath != '' ? " inside '$newpath'." : '.'); - } else { - $messages = array_merge($messages, detect_ws_changes($value, $old->keys[$subkey], $subkey, $newpath)); - } - } - } else if ($new instanceof external_multiple_structure) { - // Recursive call with the content. - $messages = array_merge($messages, detect_ws_changes($new->content, $old->content, $key, $path)); - } - - return $messages; -} - -/** - * Remove all closures (anonymous functions) in the default values so the object can be serialized. - */ -function remove_default_closures($value) { - if ($value instanceof external_warnings || $value instanceof external_files) { - // Ignore these types. - - } else if ($value instanceof external_value) { - if ($value->default instanceof Closure) { - $value->default = null; - } - - } else if ($value instanceof external_single_structure) { - - foreach ($value->keys as $subvalue) { - remove_default_closures($subvalue); - } - - } else if ($value instanceof external_multiple_structure) { - remove_default_closures($value->content); - } -}