diff --git a/composer.json b/composer.json index 7f33acac9b9..dd2d9504b52 100644 --- a/composer.json +++ b/composer.json @@ -27,7 +27,7 @@ "composer/installers": "^2.2", "guzzlehttp/guzzle": "^7.5.0", "guzzlehttp/psr7": "^2.4.0", - "embed/embed": "^4.4.4", + "embed/embed": "^4.4.7", "league/csv": "^9.8.0", "m1/env": "^2.2.0", "monolog/monolog": "^3.2.0", diff --git a/src/Control/HTTPRequest.php b/src/Control/HTTPRequest.php index 2ceb7e23941..db22f8d4165 100644 --- a/src/Control/HTTPRequest.php +++ b/src/Control/HTTPRequest.php @@ -579,7 +579,7 @@ public function match($pattern, $shiftOnSuccess = false) $shiftCount = sizeof($patternParts ?? []); $remaining = count($this->dirParts ?? []) - $i; for ($j = 1; $j <= $remaining; $j++) { - $arguments["$${j}"] = $this->dirParts[$j + $i - 1]; + $arguments['$' . $j] = $this->dirParts[$j + $i - 1]; } $patternParts = array_merge($patternParts, array_keys($arguments ?? [])); break; diff --git a/src/Dev/Constraint/SSListContains.php b/src/Dev/Constraint/SSListContains.php index bb3b0b9dd7e..e9880ee22b6 100644 --- a/src/Dev/Constraint/SSListContains.php +++ b/src/Dev/Constraint/SSListContains.php @@ -20,6 +20,8 @@ class SSListContains extends Constraint implements TestOnly */ protected $matches = []; + protected SSListExporter $exporter; + /** * Check if the list has left over items that don't match * diff --git a/src/Dev/Constraint/SSListContainsOnlyMatchingItems.php b/src/Dev/Constraint/SSListContainsOnlyMatchingItems.php index f093b3e0157..a08b20f53cb 100644 --- a/src/Dev/Constraint/SSListContainsOnlyMatchingItems.php +++ b/src/Dev/Constraint/SSListContainsOnlyMatchingItems.php @@ -19,6 +19,8 @@ class SSListContainsOnlyMatchingItems extends Constraint implements TestOnly */ private $match; + protected SSListExporter $exporter; + /** * @var ViewableDataContains */ diff --git a/src/Dev/FixtureBlueprint.php b/src/Dev/FixtureBlueprint.php index e0c79bec6d4..7652f6ddb2d 100644 --- a/src/Dev/FixtureBlueprint.php +++ b/src/Dev/FixtureBlueprint.php @@ -34,6 +34,8 @@ class FixtureBlueprint */ protected $class; + private FixtureFactory $factory; + /** * @var array */ @@ -70,6 +72,17 @@ public function __construct($name, $class = null, $defaults = []) $this->defaults = $defaults; } + public function getFactory(): FixtureFactory + { + return $this->factory; + } + + public function setFactory(FixtureFactory $factory): static + { + $this->factory = $factory; + return $this; + } + /** * @param string $identifier Unique identifier for this fixture type * @param array $data Map of property names to their values. diff --git a/src/Dev/SapphireTest.php b/src/Dev/SapphireTest.php index a31675502c9..65c63dab5e4 100644 --- a/src/Dev/SapphireTest.php +++ b/src/Dev/SapphireTest.php @@ -161,6 +161,8 @@ abstract class SapphireTest extends TestCase implements TestOnly */ protected static $tempDB = null; + protected FixtureFactory|bool $fixtureFactory; + /** * @return TempDatabase */ diff --git a/src/ORM/Connect/MySQLDatabase.php b/src/ORM/Connect/MySQLDatabase.php index ef7cd1601a5..b80e02054b3 100644 --- a/src/ORM/Connect/MySQLDatabase.php +++ b/src/ORM/Connect/MySQLDatabase.php @@ -65,6 +65,8 @@ class MySQLDatabase extends Database implements TransactionManager */ private $transactionManager = null; + private int $transactionNesting = 0; + /** * Default collation * diff --git a/src/ORM/Connect/MySQLStatement.php b/src/ORM/Connect/MySQLStatement.php index da687898885..81edf8c50fe 100644 --- a/src/ORM/Connect/MySQLStatement.php +++ b/src/ORM/Connect/MySQLStatement.php @@ -73,7 +73,6 @@ public function __construct($statement, $metadata) public function __destruct() { $this->statement->close(); - $this->currentRecord = false; } /** diff --git a/src/View/ViewableData.php b/src/View/ViewableData.php index 2440a5cacf0..9731747ee48 100644 --- a/src/View/ViewableData.php +++ b/src/View/ViewableData.php @@ -66,6 +66,11 @@ class ViewableData implements IteratorAggregate */ private static $casting_cache = []; + /** + * Acts as a PHP 8.2+ compliant replacement for dynamic properties + */ + private array $data = []; + // ----------------------------------------------------------------------------------------------------------------- /** @@ -191,7 +196,7 @@ public function getFailover() */ public function hasField($field) { - return property_exists($this, $field ?? ''); + return property_exists($this, $field) || isset($this->data[$field]); } /** @@ -202,7 +207,10 @@ public function hasField($field) */ public function getField($field) { - return $this->$field; + if (property_exists($this, $field)) { + return $this->$field; + } + return $this->data[$field]; } /** @@ -215,7 +223,13 @@ public function getField($field) public function setField($field, $value) { $this->objCacheClear(); - $this->$field = $value; + // prior to PHP 8.2 support ViewableData::setField() simply used `$this->field = $value;` + // so the following logic essentially mimics this behaviour, though without the use + // of now deprecated dynamic properties + if (property_exists($this, $field)) { + $this->$field = $value; + } + $this->data[$field] = $value; return $this; } @@ -509,14 +523,14 @@ public function obj($fieldName, $arguments = [], $cache = false, $cacheName = nu * A simple wrapper around {@link ViewableData::obj()} that automatically caches the result so it can be used again * without re-running the method. * - * @param string $field + * @param string $fieldName * @param array $arguments * @param string $identifier an optional custom cache identifier * @return Object|DBField */ - public function cachedCall($field, $arguments = [], $identifier = null) + public function cachedCall($fieldName, $arguments = [], $identifier = null) { - return $this->obj($field, $arguments, true, $identifier); + return $this->obj($fieldName, $arguments, true, $identifier); } /** diff --git a/src/View/ViewableData_Customised.php b/src/View/ViewableData_Customised.php index 097da478183..a8589bb51a8 100644 --- a/src/View/ViewableData_Customised.php +++ b/src/View/ViewableData_Customised.php @@ -59,19 +59,26 @@ public function hasMethod($method) return $this->customised->hasMethod($method) || $this->original->hasMethod($method); } - public function cachedCall($field, $arguments = null, $identifier = null) + public function cachedCall($fieldName, $arguments = null, $identifier = null) { - if ($this->customised->hasMethod($field) || $this->customised->hasField($field)) { - return $this->customised->cachedCall($field, $arguments, $identifier); + if ($this->customisedHas($fieldName)) { + return $this->customised->cachedCall($fieldName, $arguments, $identifier); } - return $this->original->cachedCall($field, $arguments, $identifier); + return $this->original->cachedCall($fieldName, $arguments, $identifier); } public function obj($fieldName, $arguments = null, $cache = false, $cacheName = null) { - if ($this->customised->hasField($fieldName) || $this->customised->hasMethod($fieldName)) { + if ($this->customisedHas($fieldName)) { return $this->customised->obj($fieldName, $arguments, $cache, $cacheName); } return $this->original->obj($fieldName, $arguments, $cache, $cacheName); } + + private function customisedHas(string $fieldName): bool + { + return property_exists($this->customised, $fieldName) || + $this->customised->hasField($fieldName) || + $this->customised->hasMethod($fieldName); + } } diff --git a/tests/php/Core/Injector/AopProxyServiceTest/AnotherService.php b/tests/php/Core/Injector/AopProxyServiceTest/AnotherService.php index b91ecb8b4ad..cf4d574ccc9 100644 --- a/tests/php/Core/Injector/AopProxyServiceTest/AnotherService.php +++ b/tests/php/Core/Injector/AopProxyServiceTest/AnotherService.php @@ -4,5 +4,7 @@ class AnotherService { + public $config_property; + public $filters = []; } diff --git a/tests/php/Core/Injector/AopProxyServiceTest/SampleService.php b/tests/php/Core/Injector/AopProxyServiceTest/SampleService.php index 7190bdf76f0..af17e049286 100644 --- a/tests/php/Core/Injector/AopProxyServiceTest/SampleService.php +++ b/tests/php/Core/Injector/AopProxyServiceTest/SampleService.php @@ -4,6 +4,7 @@ class SampleService { + public $auto; public $constructorVarOne; public $constructorVarTwo; diff --git a/tests/php/Core/Injector/InjectorTest/TestObject.php b/tests/php/Core/Injector/InjectorTest/TestObject.php index 075dcee0232..01aa7d6c2e4 100644 --- a/tests/php/Core/Injector/InjectorTest/TestObject.php +++ b/tests/php/Core/Injector/InjectorTest/TestObject.php @@ -6,6 +6,7 @@ class TestObject implements TestOnly { + public $auto; public $sampleService; diff --git a/tests/php/Dev/FixtureBlueprintTest.php b/tests/php/Dev/FixtureBlueprintTest.php index 6224d578301..b8fa977fadc 100644 --- a/tests/php/Dev/FixtureBlueprintTest.php +++ b/tests/php/Dev/FixtureBlueprintTest.php @@ -17,6 +17,8 @@ class FixtureBlueprintTest extends SapphireTest protected $usesDatabase = true; + private int $_called = 0; + protected static $extra_dataobjects = [ TestDataObject::class, DataObjectRelation::class, diff --git a/tests/php/Forms/GridField/GridFieldFilterHeaderTest.php b/tests/php/Forms/GridField/GridFieldFilterHeaderTest.php index 357c9d7657a..1081f90f399 100644 --- a/tests/php/Forms/GridField/GridFieldFilterHeaderTest.php +++ b/tests/php/Forms/GridField/GridFieldFilterHeaderTest.php @@ -138,18 +138,17 @@ public function testHandleActionReset() public function testGetSearchForm() { $searchForm = $this->component->getSearchForm($this->gridField); - $this->assertTrue($searchForm instanceof Form); - $this->assertEquals('Search__q', $searchForm->fields[0]->Name); - $this->assertEquals('Search__Name', $searchForm->fields[1]->Name); - $this->assertEquals('Search__City', $searchForm->fields[2]->Name); - $this->assertEquals('Search__Cheerleader__Hat__Colour', $searchForm->fields[3]->Name); + $fields = $searchForm->Fields()->toArray(); + $this->assertEquals('Search__q', $fields[0]->Name); + $this->assertEquals('Search__Name', $fields[1]->Name); + $this->assertEquals('Search__City', $fields[2]->Name); + $this->assertEquals('Search__Cheerleader__Hat__Colour', $fields[3]->Name); $this->assertEquals('TeamsSearchForm', $searchForm->Name); - $this->assertEquals('cms-search-form', $searchForm->extraClasses['cms-search-form']); - - foreach ($searchForm->fields as $field) { - $this->assertEquals('stacked', $field->extraClasses['stacked']); - $this->assertEquals('no-change-track', $field->extraClasses['no-change-track']); + $this->assertTrue($searchForm->hasExtraClass('cms-search-form')); + foreach ($fields as $field) { + $this->assertTrue($field->hasExtraClass('stacked')); + $this->assertTrue($field->hasExtraClass('no-change-track')); } } diff --git a/tests/php/ORM/DataObjectTest.php b/tests/php/ORM/DataObjectTest.php index cb6536f57ce..e4617bb7c3e 100644 --- a/tests/php/ORM/DataObjectTest.php +++ b/tests/php/ORM/DataObjectTest.php @@ -826,10 +826,6 @@ public function testHasOneAsField() $team1->Captain = $captain2; $team1->write(); $this->assertEquals($captain2->ID, $team1->Captain->ID); - - // Setter: Custom data (required by DataDifferencer) - $team1->Captain = DBField::create_field('HTMLFragment', '

No captain

'); - $this->assertEquals('

No captain

', $team1->Captain); } /** diff --git a/tests/php/View/ArrayDataTest/NonEmptyObject.php b/tests/php/View/ArrayDataTest/NonEmptyObject.php index fa25d7c5c27..da4cc57e2a0 100644 --- a/tests/php/View/ArrayDataTest/NonEmptyObject.php +++ b/tests/php/View/ArrayDataTest/NonEmptyObject.php @@ -6,8 +6,9 @@ class NonEmptyObject implements TestOnly { - - static $c = "Cucumber"; + public $a; + public $b; + public static $c = "Cucumber"; public function __construct() { diff --git a/tests/php/View/Embed/MockResponse.php b/tests/php/View/Embed/MockResponse.php index 2c71e179bdb..8744e49c75c 100644 --- a/tests/php/View/Embed/MockResponse.php +++ b/tests/php/View/Embed/MockResponse.php @@ -8,7 +8,7 @@ class MockResponse implements ResponseInterface { private EmbedUnitTest $unitTest; - private string $firstReponse; + private string $firstResponse; private string $secondResponse; public function __construct(EmbedUnitTest $unitTest, string $firstResponse, string $secondResponse) diff --git a/thirdparty/php-peg/Parser.php b/thirdparty/php-peg/Parser.php index 1ce89193e72..bcb57ce06da 100644 --- a/thirdparty/php-peg/Parser.php +++ b/thirdparty/php-peg/Parser.php @@ -9,6 +9,13 @@ * the bracket if a failed match + restore has moved the current position backwards - so we have to check that too. */ class ParserRegexp { + + public $parser; + public $rx; + public $matches; + public $match_pos; + public $check_pos; + function __construct( $parser, $rx ) { $this->parser = $parser ; $this->rx = $rx . 'Sx' ;