Skip to content

Commit

Permalink
Use old namespaces
Browse files Browse the repository at this point in the history
  • Loading branch information
crazyserver committed Oct 23, 2023
1 parent 96378fb commit c9e59a2
Show file tree
Hide file tree
Showing 5 changed files with 313 additions and 4 deletions.
6 changes: 5 additions & 1 deletion structure/get_all_ws_structures.php
Original file line number Diff line number Diff line change
Expand Up @@ -34,7 +34,11 @@

require($moodlepath . '/config.php');
require($CFG->dirroot . '/webservice/lib.php');
require_once('ws_to_ts_functions.php');
if (file_exists("$CFG->libdir/external/classes/external_api.php")) {
require_once('ws_to_ts_functions.php');
} else {
require_once('ws_to_ts_functions_40.php');
}

$structures = get_all_ws_structures();

Expand Down
6 changes: 5 additions & 1 deletion structure/get_ws_changes.php
Original file line number Diff line number Diff line change
Expand Up @@ -36,7 +36,11 @@
define('CLI_SCRIPT', true);
define('CACHE_DISABLE_ALL', true);
define('SERIALIZED', true);
require_once('ws_to_ts_functions.php');
if (file_exists("$CFG->libdir/external/classes/external_api.php")) {
require_once('ws_to_ts_functions.php');
} else {
require_once('ws_to_ts_functions_40.php');
}

$versions = array('master', '310', '39', '38', '37', '36', '35', '34', '33', '32', '31');

Expand Down
6 changes: 5 additions & 1 deletion structure/get_ws_structure.php
Original file line number Diff line number Diff line change
Expand Up @@ -45,7 +45,11 @@

require($moodlepath . '/config.php');
require($CFG->dirroot . '/webservice/lib.php');
require_once('ws_to_ts_functions.php');
if (file_exists("$CFG->libdir/external/classes/external_api.php")) {
require_once('ws_to_ts_functions.php');
} else {
require_once('ws_to_ts_functions_40.php');
}

$structure = get_ws_structure($wsname, $useparams);

Expand Down
1 change: 0 additions & 1 deletion structure/ws_to_ts_functions.php
Original file line number Diff line number Diff line change
Expand Up @@ -19,7 +19,6 @@
*/

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;
Expand Down
298 changes: 298 additions & 0 deletions structure/ws_to_ts_functions_40.php
Original file line number Diff line number Diff line change
@@ -0,0 +1,298 @@
<?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/>.

/**
* Helper functions for converting a Moodle WS structure to a TS type.
*/


require_once("$CFG->libdir/externallib.php");

use external_api;
use external_function_parameters;
use external_value;
use external_warnings;
use external_files;
use external_single_structure;
use 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);
}
}

0 comments on commit c9e59a2

Please sign in to comment.