1
0
mirror of https://github.com/moodle/moodle.git synced 2025-04-23 09:23:09 +02:00
This commit is contained in:
Shamim Rezaie 2023-04-05 20:16:04 +10:00 committed by Paul Holden
commit b54fbcf532
No known key found for this signature in database
GPG Key ID: A81A96D6045F6164
12 changed files with 184 additions and 31 deletions

@ -102,6 +102,9 @@ abstract class base {
/**
* Returns the filter clauses to be used with SQL where
*
* Ideally the field SQL should be included only once in the returned expression, however if that is unavoidable then
* use the {@see filter::get_field_sql_and_params} helper to ensure uniqueness of any parameters included within
*
* @param array $values
* @return array [$sql, [...$params]]
*/

@ -61,8 +61,7 @@ class category extends base {
public function get_sql_filter(array $values): array {
global $DB;
$fieldsql = $this->filter->get_field_sql();
$params = $this->filter->get_field_params();
[$fieldsql, $params] = $this->filter->get_field_sql_and_params();
$category = (int) ($values["{$this->name}_value"] ?? 0);
$subcategories = !empty($values["{$this->name}_subcategories"]);
@ -79,6 +78,11 @@ class category extends base {
// Sub-category matching on path of selected category.
if ($subcategories) {
// We need to re-use the original filter SQL here, while ensuring parameter uniqueness is preserved.
[$fieldsql, $params1] = $this->filter->get_field_sql_and_params(1);
$params = array_merge($params, $params1);
$paramcategorypath = database::generate_param_name();
$params[$paramcategorypath] = "%/{$category}/%";
$sql .= " OR {$fieldsql} IN (

@ -175,30 +175,31 @@ class date extends base {
switch ($operator) {
case self::DATE_NOT_EMPTY:
$sql = "{$fieldsql} IS NOT NULL AND {$fieldsql} <> 0";
$sql = "COALESCE({$fieldsql}, 0) <> 0";
break;
case self::DATE_EMPTY:
$sql = "{$fieldsql} IS NULL OR {$fieldsql} = 0";
$sql = "COALESCE({$fieldsql}, 0) = 0";
break;
case self::DATE_RANGE:
$clauses = [];
$sql = '';
$datefrom = (int)($values["{$this->name}_from"] ?? 0);
if ($datefrom > 0) {
$paramdatefrom = database::generate_param_name();
$clauses[] = "{$fieldsql} >= :{$paramdatefrom}";
$params[$paramdatefrom] = $datefrom;
}
$dateto = (int)($values["{$this->name}_to"] ?? 0);
if ($dateto > 0) {
$paramdateto = database::generate_param_name();
$clauses[] = "{$fieldsql} < :{$paramdateto}";
[$paramdatefrom, $paramdateto] = database::generate_param_names(2);
if ($datefrom > 0 && $dateto > 0) {
$sql = "{$fieldsql} BETWEEN :{$paramdatefrom} AND :{$paramdateto}";
$params[$paramdatefrom] = $datefrom;
$params[$paramdateto] = $dateto;
} else if ($datefrom > 0) {
$sql = "{$fieldsql} >= :{$paramdatefrom}";
$params[$paramdatefrom] = $datefrom;
} else if ($dateto > 0) {
$sql = "{$fieldsql} < :{$paramdateto}";
$params[$paramdateto] = $dateto;
}
$sql = implode(' AND ', $clauses);
break;
case self::DATE_BEFORE:
$param = database::generate_param_name();
@ -219,7 +220,7 @@ class date extends base {
// Generate parameters and SQL clause for the relative date comparison.
[$paramdatefrom, $paramdateto] = database::generate_param_names(2);
$sql = "{$fieldsql} >= :{$paramdatefrom} AND {$fieldsql} <= :{$paramdateto}";
$sql = "{$fieldsql} BETWEEN :{$paramdatefrom} AND :{$paramdateto}";
[
$params[$paramdatefrom],

@ -132,10 +132,10 @@ class number extends base {
case self::ANY_VALUE:
return ['', []];
case self::IS_NOT_EMPTY:
$res = "{$fieldsql} IS NOT NULL AND {$fieldsql} <> 0";
$res = "COALESCE({$fieldsql}, 0) <> 0";
break;
case self::IS_EMPTY:
$res = "{$fieldsql} IS NULL OR {$fieldsql} = 0";
$res = "COALESCE({$fieldsql}, 0) = 0";
break;
case self::LESS_THAN:
$res = "{$fieldsql} < :{$param}";
@ -158,7 +158,7 @@ class number extends base {
$params[$param] = $value1;
break;
case self::RANGE:
$res = "({$fieldsql} >= :{$param} AND {$fieldsql} <= :{$param2})";
$res = "{$fieldsql} BETWEEN :{$param} AND :{$param2}";
$params[$param] = $value1;
$params[$param2] = $value2;
break;

@ -101,6 +101,29 @@ class database {
return true;
}
/**
* Replace parameter names within given SQL expression, allowing caller to specify callback to handle their replacement
* primarily to ensure uniqueness when the expression is to be used as part of a larger query
*
* @param string $sql
* @param array $params
* @param callable $callback Method that takes a single string parameter, and returns another string
* @return string
*/
public static function sql_replace_parameter_names(string $sql, array $params, callable $callback): string {
foreach ($params as $param) {
// Pattern to look for param within the SQL.
$pattern = '/:(?<param>' . preg_quote($param) . ')\b/';
$sql = preg_replace_callback($pattern, function(array $matches) use ($callback): string {
return ':' . $callback($matches['param']);
}, $sql);
}
return $sql;
}
/**
* Generate SQL expression for sorting group concatenated fields
*

@ -367,12 +367,12 @@ final class column {
$fields = [];
foreach ($this->fields as $alias => $sql) {
// Ensure params within SQL are prefixed with column index.
foreach ($this->params as $name => $value) {
$sql = preg_replace_callback('/:(?<param>' . preg_quote($name, '\b/') . ')/', function(array $matches): string {
return ':' . $this->unique_param_name($matches['param']);
}, $sql);
}
// Ensure parameter names within SQL are prefixed with column index.
$params = array_keys($this->params);
$sql = database::sql_replace_parameter_names($sql, $params, function(string $param): string {
return $this->unique_param_name($param);
});
$fields[$alias] = [
'sql' => $sql,

@ -21,6 +21,7 @@ namespace core_reportbuilder\local\report;
use lang_string;
use moodle_exception;
use core_reportbuilder\local\filters\base;
use core_reportbuilder\local\helpers\database;
use core_reportbuilder\local\models\filter as filter_model;
/**
@ -212,6 +213,38 @@ final class filter {
return $this->fieldparams;
}
/**
* Retrieve SQL expression and parameters for the field
*
* @param int $index
* @return array [$sql, [...$params]]
*/
public function get_field_sql_and_params(int $index = 0): array {
$fieldsql = $this->get_field_sql();
$fieldparams = $this->get_field_params();
// Shortcut if there aren't any parameters.
if (empty($fieldparams)) {
return [$fieldsql, $fieldparams];
}
// Simple callback for replacement of parameter names within filter SQL.
$transform = function(string $param) use ($index): string {
return "{$param}_{$index}";
};
$paramnames = array_keys($fieldparams);
$sql = database::sql_replace_parameter_names($fieldsql, $paramnames, $transform);
$params = [];
foreach ($paramnames as $paramname) {
$paramnametransform = $transform($paramname);
$params[$paramnametransform] = $fieldparams[$paramname];
}
return [$sql, $params];
}
/**
* Set the SQL expression for the field that is being filtered. It will be passed to the filter class
*

@ -20,6 +20,7 @@ namespace core_reportbuilder\local\filters;
use advanced_testcase;
use lang_string;
use core_reportbuilder\local\helpers\database;
use core_reportbuilder\local\report\filter;
/**
@ -88,4 +89,37 @@ class category_test extends advanced_testcase {
$categories = $DB->get_fieldset_select('course_categories', 'name', $select, $params);
$this->assertEqualsCanonicalizing($expectedcategories, $categories);
}
/**
* Test getting filter SQL with parameters
*/
public function test_get_sql_filter_parameters(): void {
global $DB;
$this->resetAfterTest();
$category1 = $this->getDataGenerator()->create_category(['name' => 'One']);
$category2 = $this->getDataGenerator()->create_category(['name' => 'Two', 'parent' => $category1->id]);
$category3 = $this->getDataGenerator()->create_category(['name' => 'Three']);
// Rather convoluted filter SQL, but enough to demonstrate usage of a parameter that gets used twice in the query.
$paramzero = database::generate_param_name();
$filter = new filter(
category::class,
'test',
new lang_string('yes'),
'testentity',
"id + :{$paramzero}",
[$paramzero => 0]
);
// When including sub-categories, the filter SQL is included twice (for the category itself, plus to find descendents).
[$select, $params] = category::create($filter)->get_sql_filter([
$filter->get_unique_identifier() . '_value' => $category1->id,
$filter->get_unique_identifier() . '_subcategories' => true,
]);
$categories = $DB->get_fieldset_select('course_categories', 'id', $select, $params);
$this->assertEqualsCanonicalizing([$category1->id, $category2->id], $categories);
}
}

@ -139,4 +139,31 @@ class database_test extends advanced_testcase {
$record = $DB->get_record_sql($sql, $params);
$this->assertEquals($admin->id, $record->{$userfieldalias});
}
/**
* Test replacement of parameter names within SQL statements
*/
public function test_sql_replace_parameter_names(): void {
global $DB;
// Predefine parameter names, to ensure they don't overwrite each other.
[$param0, $param1, $param10] = ['rbparam0', 'rbparam1', 'rbparam10'];
$sql = "SELECT :{$param0} AS field0, :{$param1} AS field1, :{$param10} AS field10" . $DB->sql_null_from_clause();
$sql = database::sql_replace_parameter_names($sql, [$param0, $param1, $param10], static function(string $param): string {
return "prefix_{$param}";
});
$record = $DB->get_record_sql($sql, [
"prefix_{$param0}" => 'Zero',
"prefix_{$param1}" => 'One',
"prefix_{$param10}" => 'Ten',
]);
$this->assertEquals((object) [
'field0' => 'Zero',
'field1' => 'One',
'field10' => 'Ten',
], $record);
}
}

@ -173,19 +173,31 @@ class column_test extends advanced_testcase {
* Test adding params to field, and retrieving them
*/
public function test_add_field_with_params(): void {
$param = database::generate_param_name();
[$param0, $param1] = database::generate_param_names(2);
$column = $this->create_column('test')
->set_index(1)
->add_field(":{$param}", 'foo', [$param => 'bar']);
->add_field(":{$param0}", 'foo', [$param0 => 'foo'])
->add_field(":{$param1}", 'bar', [$param1 => 'bar']);
// Select will look like the following: "p<index>_rbparam<counter>", where index is the column index and counter is
// a static value of the report helper class.
$select = $column->get_fields();
preg_match('/:(?<paramname>p1_rbparam[\d]+) AS c1_foo/', $select[0], $matches);
$fields = $column->get_fields();
$this->assertCount(2, $fields);
preg_match('/:(?<paramname>p1_rbparam[\d]+) AS c1_foo/', $fields[0], $matches);
$this->assertArrayHasKey('paramname', $matches);
$this->assertEquals([$matches['paramname'] => 'bar'], $column->get_params());
$fieldparam0 = $matches['paramname'];
preg_match('/:(?<paramname>p1_rbparam[\d]+) AS c1_bar/', $fields[1], $matches);
$this->assertArrayHasKey('paramname', $matches);
$fieldparam1 = $matches['paramname'];
// Ensure column parameters have been renamed appropriately.
$this->assertEquals([
$fieldparam0 => 'foo',
$fieldparam1 => 'bar',
], $column->get_params());
}
/**

@ -108,6 +108,18 @@ class filter_test extends advanced_testcase {
$this->assertEquals(['foo' => 'bar'], $filter->get_field_params());
}
/**
* Test getting field SQL and params, while providing index for uniqueness
*/
public function test_get_field_sql_and_params(): void {
$filter = $this->create_filter('username', 'u.username = :username AND u.idnumber = :idnumber',
['username' => 'test', 'idnumber' => 'bar']);
[$sql, $params] = $filter->get_field_sql_and_params(1);
$this->assertEquals('u.username = :username_1 AND u.idnumber = :idnumber_1', $sql);
$this->assertEquals(['username_1' => 'test', 'idnumber_1' => 'bar'], $params);
}
/**
* Test adding single join
*/

@ -7,6 +7,10 @@ Information provided here is intended especially for developers.
* Column callbacks are now passed a fourth argument to indicate the aggregation type currently being applied, which allows
for columns to define how the aggregated data is displayed
* New methods `[add|get]_attributes` added to report base class, for including custom attributes in report container HTML
* New database helper method `sql_replace_parameter_names` to help ensure uniqueness of parameters within an expression (where
that expression can be used multiple times as part of a larger query)
* The local report filter class has a new `get_field_sql_and_params` method which should be used by filter types that re-use
the filter field SQL within their generated expression, to ensure SQL containing parameters works correctly
* The following attributes can be added to custom reports in order to control card view display (via the `add_attributes` method):
- `data-force-card` to force cards view
- `data-force-table` to force table view