Skip to content

Commit 7359d18

Browse files
committed
Fixed an RCE vulnerability
1 parent c0a37e1 commit 7359d18

File tree

4 files changed

+52
-1
lines changed

4 files changed

+52
-1
lines changed

CHANGELOG.md

+2
Original file line numberDiff line numberDiff line change
@@ -6,8 +6,10 @@
66
- The `up`, `migrate/up`, and `migrate/all` commands now revert any project config changes created by migrations on failure.
77
- The `up`, `migrate/up`, and `migrate/all` commands now prompt to restore the backup created at the outset of the command, or recommend restoring a backup, on failure.
88
- Added `craft\console\controllers\BackupTrait::restore()`.
9+
- Added `craft\helpers\Component::cleanseConfig()`.
910
- Fixed a bug where Single entries weren’t getting preloaded for template macros, if the template body wasn‘t rendered. ([#13312](https://github.com/craftcms/cms/issues/13312))
1011
- Fixed a bug where asset folders could get dynamically created for elements with temporary slugs. ([#13311](https://github.com/craftcms/cms/issues/13311))
12+
- Fixed an RCE vulnerability.
1113

1214
## 4.4.14 - 2023-06-13
1315

src/controllers/ConditionsController.php

+2-1
Original file line numberDiff line numberDiff line change
@@ -11,6 +11,7 @@
1111
use craft\base\conditions\ConditionInterface;
1212
use craft\base\conditions\ConditionRuleInterface;
1313
use craft\helpers\ArrayHelper;
14+
use craft\helpers\Component;
1415
use craft\helpers\Json;
1516
use craft\web\Controller;
1617
use Illuminate\Support\Collection;
@@ -41,7 +42,7 @@ public function beforeAction($action): bool
4142
$this->requireCpRequest();
4243

4344
$baseConfig = Json::decodeIfJson($this->request->getBodyParam('config'));
44-
$config = $this->request->getBodyParam($baseConfig['name']);
45+
$config = Component::cleanseConfig($this->request->getBodyParam($baseConfig['name']));
4546
$newRuleType = ArrayHelper::remove($config, 'new-rule-type');
4647
$conditionsService = Craft::$app->getConditions();
4748
$this->_condition = $conditionsService->createCondition($config);

src/helpers/Component.php

+21
Original file line numberDiff line numberDiff line change
@@ -81,6 +81,27 @@ public static function validateComponentClass(string $class, ?string $instanceOf
8181
return true;
8282
}
8383

84+
/**
85+
* Cleanses a component config of any `on X` or `as X` keys.
86+
*
87+
* @param array $config
88+
* @return array
89+
* @since 4.4.15
90+
*/
91+
public static function cleanseConfig(array $config): array
92+
{
93+
foreach ($config as $key => $value) {
94+
if (is_string($key) && (str_starts_with($key, 'on ') || str_starts_with($key, 'as '))) {
95+
unset($config[$key]);
96+
continue;
97+
}
98+
if (is_array($value)) {
99+
$config[$key] = static::cleanseConfig($value);
100+
}
101+
}
102+
return $config;
103+
}
104+
84105
/**
85106
* Instantiates and populates a component, and ensures that it is an instance of a given interface.
86107
*

tests/unit/helpers/ComponentHelperTest.php

+27
Original file line numberDiff line numberDiff line change
@@ -115,6 +115,16 @@ public function testIconSvg(string $needle, ?string $icon, string $label): void
115115
self::assertStringContainsString($needle, Component::iconSvg($icon, $label));
116116
}
117117

118+
/**
119+
* @dataProvider cleanseConfigDataProvider
120+
* @param array $expected
121+
* @param array $config
122+
*/
123+
public function testCleanseConfig(array $expected, array $config)
124+
{
125+
self::assertSame($expected, Component::cleanseConfig($config));
126+
}
127+
118128
/**
119129
* @return array
120130
*/
@@ -304,4 +314,21 @@ public function iconSvgDataProvider(): array
304314
'aria-hidden' => ['aria-hidden="true"', '<svg width="100px" height="100px" viewBox="0 0 100 100" version="1.1" xmlns="http://www.w3.org/2000/svg" xmlns:xlink="http://www.w3.org/1999/xlink"></svg>', 'Default'],
305315
];
306316
}
317+
318+
/**
319+
* @return array
320+
*/
321+
public function cleanseConfigDataProvider(): array
322+
{
323+
return [
324+
[
325+
['f' => 'foo', 'b' => 'bar'],
326+
['f' => 'foo', 'b' => 'bar', 'as f' => 'f', 'on b' => 'b'],
327+
],
328+
[
329+
['nested' => ['f' => 'foo', 'b' => 'bar']],
330+
['nested' => ['f' => 'foo', 'b' => 'bar', 'as f' => 'f', 'on b' => 'b']],
331+
],
332+
];
333+
}
307334
}

0 commit comments

Comments
 (0)