mirror of
https://github.com/silverstripe/silverstripe-framework
synced 2024-10-22 14:05:37 +02:00
Merge pull request #7437 from open-sausages/pulls/4.0/stateless-extensions
API Extensions are now stateless
This commit is contained in:
commit
566d7baa48
@ -578,6 +578,69 @@ In some places it may still be necessary to access the session object where no r
|
|||||||
In rare cases it is still possible to access the request of the current controller via
|
In rare cases it is still possible to access the request of the current controller via
|
||||||
`Controller::curr()->getRequest()` to gain access to the current session.
|
`Controller::curr()->getRequest()` to gain access to the current session.
|
||||||
|
|
||||||
|
#### Upgrade extensions to work as singletons
|
||||||
|
|
||||||
|
In SilverStripe 4.0 extensions are now all singletons, meaning that state stored as protected vars
|
||||||
|
within extensions are now shared across all object instances that use this extension.
|
||||||
|
|
||||||
|
As a result, you must take care that all protected vars are either refactored to be stored against
|
||||||
|
the owner object, or are keyed in a fashion distinct to the parent.
|
||||||
|
|
||||||
|
E.g.
|
||||||
|
|
||||||
|
```diff
|
||||||
|
class MyExtension extends Extension {
|
||||||
|
public function getContent() {
|
||||||
|
- if (!$this->contentCache) {
|
||||||
|
- $this->contentCache = $this->generateContent();
|
||||||
|
- }
|
||||||
|
- return $this->contentCache;
|
||||||
|
+ $contentCache = $this->owner->getField('contentCache');
|
||||||
|
+ if (!$contentCache) {
|
||||||
|
+ $contentCache = $this->generateContent();
|
||||||
|
+ $this->owner->setField('contentCache', $contentCache);
|
||||||
|
+ }
|
||||||
|
+ return $contentCache;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
In addition when using extensions with distinct constructor arguments it's advisable to
|
||||||
|
use `yml` to register those constructor arguments and use a service name or alias in
|
||||||
|
`private static $extensions`.
|
||||||
|
|
||||||
|
E.g. This is the service definition for the various versioned extension variants
|
||||||
|
|
||||||
|
```yaml
|
||||||
|
---
|
||||||
|
Name: versionedextension
|
||||||
|
---
|
||||||
|
SilverStripe\Core\Injector\Injector:
|
||||||
|
# Versioning only
|
||||||
|
SilverStripe\Versioned\Versioned.versioned:
|
||||||
|
class: SilverStripe\Versioned\Versioned
|
||||||
|
constructor:
|
||||||
|
mode: Versioned
|
||||||
|
# Staging and Versioning
|
||||||
|
SilverStripe\Versioned\Versioned.stagedversioned:
|
||||||
|
class: SilverStripe\Versioned\Versioned
|
||||||
|
constructor:
|
||||||
|
mode: StagedVersioned
|
||||||
|
# Default is alias for .stagedversioned
|
||||||
|
SilverStripe\Versioned\Versioned: '%$SilverStripe\Versioned\Versioned.stagedversioned'
|
||||||
|
```
|
||||||
|
|
||||||
|
Upgrade your extension references as below:
|
||||||
|
|
||||||
|
```diff
|
||||||
|
class MyClass extends DataObject {
|
||||||
|
private static $extensions = [
|
||||||
|
- Versioned::class . '(Versioned)',
|
||||||
|
+ Versioned::class . '.versioned',
|
||||||
|
];
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
#### Compatibility with the new front-end building tools
|
#### Compatibility with the new front-end building tools
|
||||||
|
|
||||||
If you are using Requirements from 3.x then your scripts should continue to work as they did before.
|
If you are using Requirements from 3.x then your scripts should continue to work as they did before.
|
||||||
@ -1297,8 +1360,12 @@ Rather than declaring the list of stages a model has, the constructor for `Versi
|
|||||||
parameter, which declares whether or not the model is versioned and has a draft and live stage, or alternatively
|
parameter, which declares whether or not the model is versioned and has a draft and live stage, or alternatively
|
||||||
if it only has versioning without staging.
|
if it only has versioning without staging.
|
||||||
|
|
||||||
|
Each form of this extension is registered under the appropriate service identifier, which you should use in your
|
||||||
|
model as below:
|
||||||
|
|
||||||
```php
|
```php
|
||||||
use SilverStripe\ORM\DataObject;
|
use SilverStripe\ORM\DataObject;
|
||||||
|
use SilverStripe\ORM\Versioning\Versioned;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* This model has staging and versioning. Stages will be "Stage" and "Live"
|
* This model has staging and versioning. Stages will be "Stage" and "Live"
|
||||||
@ -1306,7 +1373,7 @@ use SilverStripe\ORM\DataObject;
|
|||||||
class MyStagedModel extends SilverStripe\ORM\DataObject
|
class MyStagedModel extends SilverStripe\ORM\DataObject
|
||||||
{
|
{
|
||||||
private static $extensions = [
|
private static $extensions = [
|
||||||
"SilverStripe\\ORM\\Versioning\\Versioned('StagedVersioned')"
|
Versioned::class . '.stagedversioned',
|
||||||
];
|
];
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -1316,7 +1383,7 @@ class MyStagedModel extends SilverStripe\ORM\DataObject
|
|||||||
class MyVersionedModel extends DataObject
|
class MyVersionedModel extends DataObject
|
||||||
{
|
{
|
||||||
private static $extensions = [
|
private static $extensions = [
|
||||||
"SilverStripe\\ORM\\Versioning\\Versioned('Versioned')"
|
Versioned::class . '.versioned',
|
||||||
];
|
];
|
||||||
}
|
}
|
||||||
```
|
```
|
||||||
@ -1860,6 +1927,8 @@ that it belongs to, e.g. `themes/mytheme/templates/MyVendor/Foobar/Model/MyModel
|
|||||||
* `RequestFilter` has been deprecated in favour of
|
* `RequestFilter` has been deprecated in favour of
|
||||||
[HTTPMiddleware](/developer_guides/controllers/middlewares). Also the legacy RequestFilter
|
[HTTPMiddleware](/developer_guides/controllers/middlewares). Also the legacy RequestFilter
|
||||||
API has changed: $session and $dataModel variables removed from preRequest / postRequest.
|
API has changed: $session and $dataModel variables removed from preRequest / postRequest.
|
||||||
|
* `Extension` instances are now singletons and no longer are constructed once per extended object.
|
||||||
|
See the 'Upgrade extensions to work as singletons' section on this page for more information.
|
||||||
|
|
||||||
#### <a name="overview-general-removed"></a>General and Core Removed API
|
#### <a name="overview-general-removed"></a>General and Core Removed API
|
||||||
|
|
||||||
@ -1945,6 +2014,8 @@ that it belongs to, e.g. `themes/mytheme/templates/MyVendor/Foobar/Model/MyModel
|
|||||||
Accordingly, `hasService()` has been renamed to `has()`, and `get()` will throw
|
Accordingly, `hasService()` has been renamed to `has()`, and `get()` will throw
|
||||||
`SilverStripe\Core\Injector\InjectorNotFoundException` when the service can't be found.
|
`SilverStripe\Core\Injector\InjectorNotFoundException` when the service can't be found.
|
||||||
* Removed `CustomMethods::createMethod()`. Use closures instead.
|
* Removed `CustomMethods::createMethod()`. Use closures instead.
|
||||||
|
* Removed `Extension::$ownerBaseClass` property. You should use $this->owner->baseClass() instead.
|
||||||
|
The second argument of `Extension::setOwner()` has also been removed.
|
||||||
|
|
||||||
#### <a name="overview-general-deprecated"></a>General and Core Deprecated API
|
#### <a name="overview-general-deprecated"></a>General and Core Deprecated API
|
||||||
|
|
||||||
|
@ -441,6 +441,10 @@ class ClassInfo
|
|||||||
} elseif (is_array($token) && $token[0] === T_NS_SEPARATOR) {
|
} elseif (is_array($token) && $token[0] === T_NS_SEPARATOR) {
|
||||||
$class .= $token[1];
|
$class .= $token[1];
|
||||||
$hadNamespace = true;
|
$hadNamespace = true;
|
||||||
|
} elseif ($token === '.') {
|
||||||
|
// Treat service name separator as NS separator
|
||||||
|
$class .= '.';
|
||||||
|
$hadNamespace = true;
|
||||||
} elseif ($hadNamespace && is_array($token) && $token[0] === T_STRING) {
|
} elseif ($hadNamespace && is_array($token) && $token[0] === T_STRING) {
|
||||||
$class .= $token[1];
|
$class .= $token[1];
|
||||||
$hadNamespace = false;
|
$hadNamespace = false;
|
||||||
@ -454,7 +458,11 @@ class ClassInfo
|
|||||||
$result = stripcslashes(substr($argString, 1, -1));
|
$result = stripcslashes(substr($argString, 1, -1));
|
||||||
break;
|
break;
|
||||||
case "'":
|
case "'":
|
||||||
$result = str_replace(array("\\\\", "\\'"), array("\\", "'"), substr($argString, 1, -1));
|
$result = str_replace(
|
||||||
|
["\\\\", "\\'"],
|
||||||
|
["\\", "'"],
|
||||||
|
substr($argString, 1, -1)
|
||||||
|
);
|
||||||
break;
|
break;
|
||||||
default:
|
default:
|
||||||
throw new Exception("Bad T_CONSTANT_ENCAPSED_STRING arg $argString");
|
throw new Exception("Bad T_CONSTANT_ENCAPSED_STRING arg $argString");
|
||||||
|
@ -64,6 +64,8 @@ class ExtensionMiddleware implements Middleware
|
|||||||
$extensions = $extensionSourceConfig['extensions'];
|
$extensions = $extensionSourceConfig['extensions'];
|
||||||
foreach ($extensions as $extension) {
|
foreach ($extensions as $extension) {
|
||||||
list($extensionClass, $extensionArgs) = ClassInfo::parse_class_spec($extension);
|
list($extensionClass, $extensionArgs) = ClassInfo::parse_class_spec($extension);
|
||||||
|
// Strip service name specifier
|
||||||
|
$extensionClass = strtok($extensionClass, '.');
|
||||||
if (!class_exists($extensionClass)) {
|
if (!class_exists($extensionClass)) {
|
||||||
throw new InvalidArgumentException("$class references nonexistent $extensionClass in 'extensions'");
|
throw new InvalidArgumentException("$class references nonexistent $extensionClass in 'extensions'");
|
||||||
}
|
}
|
||||||
|
@ -67,27 +67,30 @@ trait CustomMethods
|
|||||||
$this->{$property}[$index] :
|
$this->{$property}[$index] :
|
||||||
$this->{$property};
|
$this->{$property};
|
||||||
|
|
||||||
if ($obj) {
|
if (!$obj) {
|
||||||
if (!empty($config['callSetOwnerFirst'])) {
|
|
||||||
/** @var Extension $obj */
|
|
||||||
$obj->setOwner($this);
|
|
||||||
}
|
|
||||||
$retVal = call_user_func_array(array($obj, $method), $arguments);
|
|
||||||
if (!empty($config['callSetOwnerFirst'])) {
|
|
||||||
/** @var Extension $obj */
|
|
||||||
$obj->clearOwner();
|
|
||||||
}
|
|
||||||
return $retVal;
|
|
||||||
}
|
|
||||||
|
|
||||||
throw new BadMethodCallException(
|
throw new BadMethodCallException(
|
||||||
"Object->__call(): {$class} cannot pass control to {$property}({$index})."
|
"Object->__call(): {$class} cannot pass control to {$property}({$index})."
|
||||||
. ' Perhaps this object was mistakenly destroyed?'
|
. ' Perhaps this object was mistakenly destroyed?'
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Call without setOwner
|
||||||
|
if (empty($config['callSetOwnerFirst'])) {
|
||||||
|
return $obj->$method(...$arguments);
|
||||||
|
}
|
||||||
|
|
||||||
|
/** @var Extension $obj */
|
||||||
|
try {
|
||||||
|
$obj->setOwner($this);
|
||||||
|
return $obj->$method(...$arguments);
|
||||||
|
} finally {
|
||||||
|
$obj->clearOwner();
|
||||||
|
}
|
||||||
|
}
|
||||||
case isset($config['wrap']): {
|
case isset($config['wrap']): {
|
||||||
array_unshift($arguments, $config['method']);
|
array_unshift($arguments, $config['method']);
|
||||||
return call_user_func_array(array($this, $config['wrap']), $arguments);
|
$wrapped = $config['wrap'];
|
||||||
|
return $this->$wrapped(...$arguments);
|
||||||
}
|
}
|
||||||
case isset($config['function']): {
|
case isset($config['function']): {
|
||||||
return $config['function']($this, $arguments);
|
return $config['function']($this, $arguments);
|
||||||
@ -192,12 +195,15 @@ trait CustomMethods
|
|||||||
{
|
{
|
||||||
if (method_exists($extension, 'allMethodNames')) {
|
if (method_exists($extension, 'allMethodNames')) {
|
||||||
if ($extension instanceof Extension) {
|
if ($extension instanceof Extension) {
|
||||||
|
try {
|
||||||
$extension->setOwner($this);
|
$extension->setOwner($this);
|
||||||
}
|
|
||||||
$methods = $extension->allMethodNames(true);
|
$methods = $extension->allMethodNames(true);
|
||||||
if ($extension instanceof Extension) {
|
} finally {
|
||||||
$extension->clearOwner();
|
$extension->clearOwner();
|
||||||
}
|
}
|
||||||
|
} else {
|
||||||
|
$methods = $extension->allMethodNames(true);
|
||||||
|
}
|
||||||
} else {
|
} else {
|
||||||
$class = get_class($extension);
|
$class = get_class($extension);
|
||||||
if (!isset(self::$built_in_methods[$class])) {
|
if (!isset(self::$built_in_methods[$class])) {
|
||||||
|
@ -139,8 +139,8 @@ trait Extensible
|
|||||||
$this->addCallbackMethod($method, function ($inst, $args) use ($method, $extensionClass) {
|
$this->addCallbackMethod($method, function ($inst, $args) use ($method, $extensionClass) {
|
||||||
/** @var Extensible $inst */
|
/** @var Extensible $inst */
|
||||||
$extension = $inst->getExtensionInstance($extensionClass);
|
$extension = $inst->getExtensionInstance($extensionClass);
|
||||||
$extension->setOwner($inst);
|
|
||||||
try {
|
try {
|
||||||
|
$extension->setOwner($inst);
|
||||||
return call_user_func_array([$extension, $method], $args);
|
return call_user_func_array([$extension, $method], $args);
|
||||||
} finally {
|
} finally {
|
||||||
$extension->clearOwner();
|
$extension->clearOwner();
|
||||||
@ -343,6 +343,8 @@ trait Extensible
|
|||||||
|
|
||||||
foreach ($extensions as $extension) {
|
foreach ($extensions as $extension) {
|
||||||
list($extensionClass, $extensionArgs) = ClassInfo::parse_class_spec($extension);
|
list($extensionClass, $extensionArgs) = ClassInfo::parse_class_spec($extension);
|
||||||
|
// Strip service name specifier
|
||||||
|
$extensionClass = strtok($extensionClass, '.');
|
||||||
$sources[] = $extensionClass;
|
$sources[] = $extensionClass;
|
||||||
|
|
||||||
if (!class_exists($extensionClass)) {
|
if (!class_exists($extensionClass)) {
|
||||||
@ -465,8 +467,8 @@ trait Extensible
|
|||||||
|
|
||||||
foreach ($this->getExtensionInstances() as $instance) {
|
foreach ($this->getExtensionInstances() as $instance) {
|
||||||
if (method_exists($instance, $method)) {
|
if (method_exists($instance, $method)) {
|
||||||
$instance->setOwner($this);
|
|
||||||
try {
|
try {
|
||||||
|
$instance->setOwner($this);
|
||||||
$value = $instance->$method($a1, $a2, $a3, $a4, $a5, $a6, $a7);
|
$value = $instance->$method($a1, $a2, $a3, $a4, $a5, $a6, $a7);
|
||||||
} finally {
|
} finally {
|
||||||
$instance->clearOwner();
|
$instance->clearOwner();
|
||||||
@ -551,8 +553,7 @@ trait Extensible
|
|||||||
|
|
||||||
if ($extensions) {
|
if ($extensions) {
|
||||||
foreach ($extensions as $extension) {
|
foreach ($extensions as $extension) {
|
||||||
$instance = Injector::inst()->create($extension);
|
$instance = Injector::inst()->get($extension);
|
||||||
$instance->setOwner(null, $class);
|
|
||||||
$this->extension_instances[get_class($instance)] = $instance;
|
$this->extension_instances[get_class($instance)] = $instance;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -30,15 +30,7 @@ abstract class Extension
|
|||||||
protected $owner;
|
protected $owner;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* The base class that this extension was applied to; $this->owner must be one of these
|
* Stack of all parent owners, not including current owner
|
||||||
*
|
|
||||||
* @var DataObject
|
|
||||||
*/
|
|
||||||
protected $ownerBaseClass;
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Ownership stack for recursive methods.
|
|
||||||
* Last item is current owner.
|
|
||||||
*
|
*
|
||||||
* @var array
|
* @var array
|
||||||
*/
|
*/
|
||||||
@ -63,24 +55,29 @@ abstract class Extension
|
|||||||
/**
|
/**
|
||||||
* Set the owner of this extension.
|
* Set the owner of this extension.
|
||||||
*
|
*
|
||||||
* @param object $owner The owner object,
|
* @param object $owner The owner object
|
||||||
* @param string $ownerBaseClass The base class that the extension is applied to; this may be
|
|
||||||
* the class of owner, or it may be a parent. For example, if Versioned was applied to SiteTree,
|
|
||||||
* and then a Page object was instantiated, $owner would be a Page object, but $ownerBaseClass
|
|
||||||
* would be 'SiteTree'.
|
|
||||||
*/
|
*/
|
||||||
public function setOwner($owner, $ownerBaseClass = null)
|
public function setOwner($owner)
|
||||||
{
|
{
|
||||||
if ($owner) {
|
$this->ownerStack[] = $this->owner;
|
||||||
$this->ownerStack[] = $owner;
|
|
||||||
}
|
|
||||||
$this->owner = $owner;
|
$this->owner = $owner;
|
||||||
|
}
|
||||||
|
|
||||||
// Set ownerBaseClass
|
/**
|
||||||
if ($ownerBaseClass) {
|
* Temporarily modify the owner. The original owner is ensured to be restored
|
||||||
$this->ownerBaseClass = $ownerBaseClass;
|
*
|
||||||
} elseif (!$this->ownerBaseClass && $owner) {
|
* @param mixed $owner Owner to set
|
||||||
$this->ownerBaseClass = get_class($owner);
|
* @param callable $callback Callback to invoke
|
||||||
|
* @param array $args Args to pass to callback
|
||||||
|
* @return mixed
|
||||||
|
*/
|
||||||
|
public function withOwner($owner, callable $callback, $args = [])
|
||||||
|
{
|
||||||
|
try {
|
||||||
|
$this->setOwner($owner);
|
||||||
|
return $callback(...$args);
|
||||||
|
} finally {
|
||||||
|
$this->clearOwner();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -92,12 +89,7 @@ abstract class Extension
|
|||||||
if (empty($this->ownerStack)) {
|
if (empty($this->ownerStack)) {
|
||||||
throw new BadMethodCallException("clearOwner() called more than setOwner()");
|
throw new BadMethodCallException("clearOwner() called more than setOwner()");
|
||||||
}
|
}
|
||||||
array_pop($this->ownerStack);
|
$this->owner = array_pop($this->ownerStack);
|
||||||
if ($this->ownerStack) {
|
|
||||||
$this->owner = end($this->ownerStack);
|
|
||||||
} else {
|
|
||||||
$this->owner = null;
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@ -120,7 +112,7 @@ abstract class Extension
|
|||||||
*/
|
*/
|
||||||
public static function get_classname_without_arguments($extensionStr)
|
public static function get_classname_without_arguments($extensionStr)
|
||||||
{
|
{
|
||||||
$parts = explode('(', $extensionStr);
|
// Split out both args and service name
|
||||||
return $parts[0];
|
return strtok(strtok($extensionStr, '('), '.');
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -193,6 +193,16 @@ class Injector implements ContainerInterface
|
|||||||
*/
|
*/
|
||||||
protected $configLocator;
|
protected $configLocator;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Specify a service type singleton
|
||||||
|
*/
|
||||||
|
const SINGLETON = 'singleton';
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Specif ya service type prototype
|
||||||
|
*/
|
||||||
|
const PROTOTYPE = 'prototype';
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Create a new injector.
|
* Create a new injector.
|
||||||
*
|
*
|
||||||
@ -584,7 +594,7 @@ class Injector implements ContainerInterface
|
|||||||
$type = isset($spec['type']) ? $spec['type'] : null;
|
$type = isset($spec['type']) ? $spec['type'] : null;
|
||||||
}
|
}
|
||||||
|
|
||||||
if ($id && (!$type || $type != 'prototype')) {
|
if ($id && (!$type || $type !== self::PROTOTYPE)) {
|
||||||
// this ABSOLUTELY must be set before the object is injected.
|
// this ABSOLUTELY must be set before the object is injected.
|
||||||
// This prevents circular reference errors down the line
|
// This prevents circular reference errors down the line
|
||||||
$this->serviceCache[$id] = $object;
|
$this->serviceCache[$id] = $object;
|
||||||
@ -827,8 +837,8 @@ class Injector implements ContainerInterface
|
|||||||
*/
|
*/
|
||||||
public function getServiceName($name)
|
public function getServiceName($name)
|
||||||
{
|
{
|
||||||
// common case, get it overwith first
|
// Lazy load in spec (disable inheritance to check exact service name)
|
||||||
if (isset($this->specs[$name])) {
|
if ($this->getServiceSpec($name, false)) {
|
||||||
return $name;
|
return $name;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -919,10 +929,9 @@ class Injector implements ContainerInterface
|
|||||||
*
|
*
|
||||||
* @param string $name The name of the service to retrieve. If not a registered
|
* @param string $name The name of the service to retrieve. If not a registered
|
||||||
* service, then a class of the given name is instantiated
|
* service, then a class of the given name is instantiated
|
||||||
* @param boolean $asSingleton Whether to register the created object as a singleton
|
* @param bool $asSingleton If set to false a new instance will be returned.
|
||||||
* if no other configuration is found
|
* If true a singleton will be returned unless the spec is type=prototype'
|
||||||
* @param array $constructorArgs Optional set of arguments to pass as constructor arguments
|
* @param array $constructorArgs Args to pass in to the constructor. Note: Ignored for singletons
|
||||||
* if this object is to be created from scratch (with $asSingleton = false)
|
|
||||||
* @return mixed Instance of the specified object
|
* @return mixed Instance of the specified object
|
||||||
*/
|
*/
|
||||||
public function get($name, $asSingleton = true, $constructorArgs = [])
|
public function get($name, $asSingleton = true, $constructorArgs = [])
|
||||||
@ -939,70 +948,42 @@ class Injector implements ContainerInterface
|
|||||||
/**
|
/**
|
||||||
* Returns the service, or `null` if it doesnt' exist. See {@link get()} for main usage.
|
* Returns the service, or `null` if it doesnt' exist. See {@link get()} for main usage.
|
||||||
*
|
*
|
||||||
* @param string $name
|
* @param string $name The name of the service to retrieve. If not a registered
|
||||||
* @param boolean $asSingleton
|
* service, then a class of the given name is instantiated
|
||||||
* @param array $constructorArgs
|
* @param bool $asSingleton If set to false a new instance will be returned.
|
||||||
* @return mixed|null Instance of the specified object (if it exists)
|
* If true a singleton will be returned unless the spec is type=prototype'
|
||||||
|
* @param array $constructorArgs Args to pass in to the constructor. Note: Ignored for singletons
|
||||||
|
* @return mixed Instance of the specified object
|
||||||
*/
|
*/
|
||||||
protected function getNamedService($name, $asSingleton = true, $constructorArgs = [])
|
protected function getNamedService($name, $asSingleton = true, $constructorArgs = [])
|
||||||
{
|
{
|
||||||
// Allow service names of the form "%$ServiceName"
|
|
||||||
if (substr($name, 0, 2) == '%$') {
|
|
||||||
$name = substr($name, 2);
|
|
||||||
}
|
|
||||||
|
|
||||||
// Normalise service / args
|
// Normalise service / args
|
||||||
list($name, $constructorArgs) = $this->normaliseArguments($name, $constructorArgs);
|
list($name, $constructorArgs) = $this->normaliseArguments($name, $constructorArgs);
|
||||||
|
|
||||||
// reassign the name as it might actually be a compound name
|
// Resolve name with the appropriate spec, or a suitable mock for new services
|
||||||
if ($serviceName = $this->getServiceName($name)) {
|
list($name, $spec) = $this->getServiceNamedSpec($name, $constructorArgs);
|
||||||
// check to see what the type of bean is. If it's a prototype,
|
|
||||||
// we don't want to return the singleton version of it.
|
// Check if we are getting a prototype or singleton
|
||||||
$spec = $this->specs[$serviceName];
|
$type = $asSingleton
|
||||||
$type = isset($spec['type']) ? $spec['type'] : null;
|
? (isset($spec['type']) ? $spec['type'] : self::SINGLETON)
|
||||||
// if we're explicitly a prototype OR we're not wanting a singleton
|
: self::PROTOTYPE;
|
||||||
if (($type && $type == 'prototype') || !$asSingleton) {
|
|
||||||
if ($spec && $constructorArgs) {
|
// Return existing instance for singletons
|
||||||
$spec['constructor'] = $constructorArgs;
|
if ($type === self::SINGLETON && isset($this->serviceCache[$name])) {
|
||||||
} else {
|
return $this->serviceCache[$name];
|
||||||
// convert any _configured_ constructor args.
|
|
||||||
// we don't call this for get() calls where someone passes in
|
|
||||||
// constructor args, otherwise we end up calling convertServiceParams
|
|
||||||
// way too often
|
|
||||||
$this->updateSpecConstructor($spec);
|
|
||||||
}
|
|
||||||
return $this->instantiate($spec, $serviceName, !$type ? 'prototype' : $type);
|
|
||||||
} else {
|
|
||||||
if (!isset($this->serviceCache[$serviceName])) {
|
|
||||||
$this->updateSpecConstructor($spec);
|
|
||||||
$this->instantiate($spec, $serviceName);
|
|
||||||
}
|
|
||||||
return $this->serviceCache[$serviceName];
|
|
||||||
}
|
|
||||||
}
|
|
||||||
$config = $this->configLocator->locateConfigFor($name);
|
|
||||||
if ($config) {
|
|
||||||
$this->load(array($name => $config));
|
|
||||||
if (isset($this->specs[$name])) {
|
|
||||||
$spec = $this->specs[$name];
|
|
||||||
$this->updateSpecConstructor($spec);
|
|
||||||
if ($constructorArgs) {
|
|
||||||
$spec['constructor'] = $constructorArgs;
|
|
||||||
}
|
|
||||||
return $this->instantiate($spec, $name);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
// If we've got this far, we're dealing with a case of a user wanting
|
|
||||||
// to create an object based on its name. So, we need to fake its config
|
|
||||||
// if the user wants it managed as a singleton service style object
|
|
||||||
$spec = array('class' => $name, 'constructor' => $constructorArgs);
|
|
||||||
if ($asSingleton) {
|
|
||||||
// need to load the spec in; it'll be given the singleton type by default
|
|
||||||
$this->load(array($name => $spec));
|
|
||||||
return $this->instantiate($spec, $name);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
return $this->instantiate($spec);
|
// Update constructor args
|
||||||
|
if ($type === self::PROTOTYPE && $constructorArgs) {
|
||||||
|
// Passed in args are expected to already be normalised (no service references)
|
||||||
|
$spec['constructor'] = $constructorArgs;
|
||||||
|
} else {
|
||||||
|
// Resolve references in constructor args
|
||||||
|
$this->updateSpecConstructor($spec);
|
||||||
|
}
|
||||||
|
|
||||||
|
// Build instance
|
||||||
|
return $this->instantiate($spec, $name, $type);
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@ -1016,6 +997,11 @@ class Injector implements ContainerInterface
|
|||||||
*/
|
*/
|
||||||
protected function normaliseArguments($name, $args = [])
|
protected function normaliseArguments($name, $args = [])
|
||||||
{
|
{
|
||||||
|
// Allow service names of the form "%$ServiceName"
|
||||||
|
if (substr($name, 0, 2) == '%$') {
|
||||||
|
$name = substr($name, 2);
|
||||||
|
}
|
||||||
|
|
||||||
if (strstr($name, '(')) {
|
if (strstr($name, '(')) {
|
||||||
list($name, $extraArgs) = ClassInfo::parse_class_spec($name);
|
list($name, $extraArgs) = ClassInfo::parse_class_spec($name);
|
||||||
if ($args) {
|
if ($args) {
|
||||||
@ -1027,6 +1013,61 @@ class Injector implements ContainerInterface
|
|||||||
return [ $name, $args ];
|
return [ $name, $args ];
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Get or build a named service and specification
|
||||||
|
*
|
||||||
|
* @param string $name Service name
|
||||||
|
* @param array $constructorArgs Optional constructor args
|
||||||
|
* @return array
|
||||||
|
*/
|
||||||
|
protected function getServiceNamedSpec($name, $constructorArgs = [])
|
||||||
|
{
|
||||||
|
$spec = $this->getServiceSpec($name);
|
||||||
|
if ($spec) {
|
||||||
|
// Resolve to exact service name (in case inherited)
|
||||||
|
$name = $this->getServiceName($name);
|
||||||
|
} else {
|
||||||
|
// Late-generate config spec for non-configured spec
|
||||||
|
$spec = [
|
||||||
|
'class' => $name,
|
||||||
|
'constructor' => $constructorArgs,
|
||||||
|
];
|
||||||
|
}
|
||||||
|
return [$name, $spec];
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Search for spec, lazy-loading in from config locator.
|
||||||
|
* Falls back to parent service name if unloaded
|
||||||
|
*
|
||||||
|
* @param string $name
|
||||||
|
* @param bool $inherit Set to true to inherit from parent service if `.` suffixed
|
||||||
|
* E.g. 'Psr/Log/LoggerInterface.custom' would fail over to 'Psr/Log/LoggerInterface'
|
||||||
|
* @return mixed|object
|
||||||
|
*/
|
||||||
|
public function getServiceSpec($name, $inherit = true)
|
||||||
|
{
|
||||||
|
if (isset($this->specs[$name])) {
|
||||||
|
return $this->specs[$name];
|
||||||
|
}
|
||||||
|
|
||||||
|
// Lazy load
|
||||||
|
$config = $this->configLocator->locateConfigFor($name);
|
||||||
|
if ($config) {
|
||||||
|
$this->load([$name => $config]);
|
||||||
|
if (isset($this->specs[$name])) {
|
||||||
|
return $this->specs[$name];
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Fail over to parent service if allowed
|
||||||
|
if (!$inherit || !strpos($name, '.')) {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
return $this->getServiceSpec(substr($name, 0, strrpos($name, '.')));
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Magic method to return an item directly
|
* Magic method to return an item directly
|
||||||
*
|
*
|
||||||
|
@ -5,7 +5,6 @@ namespace SilverStripe\Dev\State;
|
|||||||
use LogicException;
|
use LogicException;
|
||||||
use SilverStripe\Core\Extension;
|
use SilverStripe\Core\Extension;
|
||||||
use SilverStripe\Core\Injector\Injector;
|
use SilverStripe\Core\Injector\Injector;
|
||||||
use SilverStripe\Dev\Debug;
|
|
||||||
use SilverStripe\Dev\SapphireTest;
|
use SilverStripe\Dev\SapphireTest;
|
||||||
use SilverStripe\ORM\DataObject;
|
use SilverStripe\ORM\DataObject;
|
||||||
|
|
||||||
@ -90,7 +89,10 @@ class ExtensionTestState implements TestState
|
|||||||
|
|
||||||
// clear singletons, they're caching old extension info
|
// clear singletons, they're caching old extension info
|
||||||
// which is used in DatabaseAdmin->doBuild()
|
// which is used in DatabaseAdmin->doBuild()
|
||||||
Injector::inst()->unregisterObjects(DataObject::class);
|
Injector::inst()->unregisterObjects([
|
||||||
|
DataObject::class,
|
||||||
|
Extension::class
|
||||||
|
]);
|
||||||
|
|
||||||
// If we have altered the schema, but SapphireTest::setUpBeforeClass() would not otherwise
|
// If we have altered the schema, but SapphireTest::setUpBeforeClass() would not otherwise
|
||||||
// reset the schema (if there were extra objects) then force a reset
|
// reset the schema (if there were extra objects) then force a reset
|
||||||
|
@ -23,21 +23,6 @@ use Exception;
|
|||||||
*/
|
*/
|
||||||
class Hierarchy extends DataExtension
|
class Hierarchy extends DataExtension
|
||||||
{
|
{
|
||||||
|
|
||||||
/**
|
|
||||||
* Cache for {@see numChildren()}
|
|
||||||
*
|
|
||||||
* @var int
|
|
||||||
*/
|
|
||||||
protected $_cache_numChildren = null;
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Cache for {@see Children()}
|
|
||||||
*
|
|
||||||
* @var SS_List
|
|
||||||
*/
|
|
||||||
protected $_cache_children = null;
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* The lower bounds for the amount of nodes to mark. If set, the logic will expand nodes until it reaches at least
|
* The lower bounds for the amount of nodes to mark. If set, the logic will expand nodes until it reaches at least
|
||||||
* this number, and then stops. Root nodes will always show regardless of this settting. Further nodes can be
|
* this number, and then stops. Root nodes will always show regardless of this settting. Further nodes can be
|
||||||
@ -86,6 +71,17 @@ class Hierarchy extends DataExtension
|
|||||||
*/
|
*/
|
||||||
private static $hide_from_cms_tree = array();
|
private static $hide_from_cms_tree = array();
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Prevent virtual page virtualising these fields
|
||||||
|
*
|
||||||
|
* @config
|
||||||
|
* @var array
|
||||||
|
*/
|
||||||
|
private static $non_virtual_fields = [
|
||||||
|
'_cache_children',
|
||||||
|
'_cache_numChildren',
|
||||||
|
];
|
||||||
|
|
||||||
public static function get_extra_config($class, $extension, $args)
|
public static function get_extra_config($class, $extension, $args)
|
||||||
{
|
{
|
||||||
return array(
|
return array(
|
||||||
@ -176,17 +172,19 @@ class Hierarchy extends DataExtension
|
|||||||
*/
|
*/
|
||||||
public function Children()
|
public function Children()
|
||||||
{
|
{
|
||||||
if ($this->_cache_children) {
|
$children = $this->owner->_cache_children;
|
||||||
return $this->_cache_children;
|
if ($children) {
|
||||||
|
return $children;
|
||||||
}
|
}
|
||||||
|
|
||||||
$this->_cache_children = $this
|
$children = $this
|
||||||
->owner
|
->owner
|
||||||
->stageChildren(false)
|
->stageChildren(false)
|
||||||
->filterByCallback(function (DataObject $record) {
|
->filterByCallback(function (DataObject $record) {
|
||||||
return $record->canView();
|
return $record->canView();
|
||||||
});
|
});
|
||||||
return $this->_cache_children;
|
$this->owner->_cache_children = $children;
|
||||||
|
return $children;
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@ -268,18 +266,21 @@ class Hierarchy extends DataExtension
|
|||||||
public function numChildren($cache = true)
|
public function numChildren($cache = true)
|
||||||
{
|
{
|
||||||
// Load if caching
|
// Load if caching
|
||||||
if ($cache && isset($this->_cache_numChildren)) {
|
if ($cache) {
|
||||||
return $this->_cache_numChildren;
|
$numChildren = $this->owner->_cache_numChildren;
|
||||||
|
if (isset($numChildren)) {
|
||||||
|
return $numChildren;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// We call stageChildren(), because Children() has canView() filtering
|
// We call stageChildren(), because Children() has canView() filtering
|
||||||
$children = (int)$this->owner->stageChildren(true)->Count();
|
$numChildren = (int)$this->owner->stageChildren(true)->Count();
|
||||||
|
|
||||||
// Save if caching
|
// Save if caching
|
||||||
if ($cache) {
|
if ($cache) {
|
||||||
$this->_cache_numChildren = $children;
|
$this->owner->_cache_numChildren = $numChildren;
|
||||||
}
|
}
|
||||||
return $children;
|
return $numChildren;
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@ -308,7 +309,8 @@ class Hierarchy extends DataExtension
|
|||||||
{
|
{
|
||||||
$hideFromHierarchy = $this->owner->config()->hide_from_hierarchy;
|
$hideFromHierarchy = $this->owner->config()->hide_from_hierarchy;
|
||||||
$hideFromCMSTree = $this->owner->config()->hide_from_cms_tree;
|
$hideFromCMSTree = $this->owner->config()->hide_from_cms_tree;
|
||||||
$staged = DataObject::get($this->ownerBaseClass)
|
$baseClass = $this->owner->baseClass();
|
||||||
|
$staged = DataObject::get($baseClass)
|
||||||
->filter('ParentID', (int)$this->owner->ID)
|
->filter('ParentID', (int)$this->owner->ID)
|
||||||
->exclude('ID', (int)$this->owner->ID);
|
->exclude('ID', (int)$this->owner->ID);
|
||||||
if ($hideFromHierarchy) {
|
if ($hideFromHierarchy) {
|
||||||
@ -374,11 +376,12 @@ class Hierarchy extends DataExtension
|
|||||||
if (empty($parentID)) {
|
if (empty($parentID)) {
|
||||||
return null;
|
return null;
|
||||||
}
|
}
|
||||||
$idSQL = $this->owner->getSchema()->sqlColumnForField($this->ownerBaseClass, 'ID');
|
$baseClass = $this->owner->baseClass();
|
||||||
return DataObject::get_one($this->ownerBaseClass, array(
|
$idSQL = $this->owner->getSchema()->sqlColumnForField($baseClass, 'ID');
|
||||||
array($idSQL => $parentID),
|
return DataObject::get_one($baseClass, [
|
||||||
|
[$idSQL => $parentID],
|
||||||
$filter
|
$filter
|
||||||
));
|
]);
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@ -424,13 +427,10 @@ class Hierarchy extends DataExtension
|
|||||||
* Flush all Hierarchy caches:
|
* Flush all Hierarchy caches:
|
||||||
* - Children (instance)
|
* - Children (instance)
|
||||||
* - NumChildren (instance)
|
* - NumChildren (instance)
|
||||||
* - Marked (global)
|
|
||||||
* - Expanded (global)
|
|
||||||
* - TreeOpened (global)
|
|
||||||
*/
|
*/
|
||||||
public function flushCache()
|
public function flushCache()
|
||||||
{
|
{
|
||||||
$this->_cache_children = null;
|
$this->owner->_cache_children = null;
|
||||||
$this->_cache_numChildren = null;
|
$this->owner->_cache_numChildren = null;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -616,11 +616,7 @@ class Group extends DataObject
|
|||||||
*/
|
*/
|
||||||
public function AllChildrenIncludingDeleted()
|
public function AllChildrenIncludingDeleted()
|
||||||
{
|
{
|
||||||
/** @var Hierarchy $extInstance */
|
$children = parent::AllChildrenIncludingDeleted();
|
||||||
$extInstance = $this->getExtensionInstance(Hierarchy::class);
|
|
||||||
$extInstance->setOwner($this);
|
|
||||||
$children = $extInstance->AllChildrenIncludingDeleted();
|
|
||||||
$extInstance->clearOwner();
|
|
||||||
|
|
||||||
$filteredChildren = new ArrayList();
|
$filteredChildren = new ArrayList();
|
||||||
|
|
||||||
|
@ -52,7 +52,14 @@ class ObjectTest extends SapphireTest
|
|||||||
$objs[] = new ObjectTest\T2();
|
$objs[] = new ObjectTest\T2();
|
||||||
|
|
||||||
// All these methods should exist and return true
|
// All these methods should exist and return true
|
||||||
$trueMethods = array('testMethod','otherMethod','someMethod','t1cMethod','normalMethod', 'failoverCallback');
|
$trueMethods = [
|
||||||
|
'testMethod',
|
||||||
|
'otherMethod',
|
||||||
|
'someMethod',
|
||||||
|
't1cMethod',
|
||||||
|
'normalMethod',
|
||||||
|
'failoverCallback'
|
||||||
|
];
|
||||||
|
|
||||||
foreach ($objs as $i => $obj) {
|
foreach ($objs as $i => $obj) {
|
||||||
foreach ($trueMethods as $method) {
|
foreach ($trueMethods as $method) {
|
||||||
@ -402,6 +409,14 @@ class ObjectTest extends SapphireTest
|
|||||||
array(Versioned::class, array('Stage', 'Live')),
|
array(Versioned::class, array('Stage', 'Live')),
|
||||||
ClassInfo::parse_class_spec("SilverStripe\\Versioned\\Versioned('Stage','Live')")
|
ClassInfo::parse_class_spec("SilverStripe\\Versioned\\Versioned('Stage','Live')")
|
||||||
);
|
);
|
||||||
|
// Case with service identifier
|
||||||
|
$this->assertEquals(
|
||||||
|
[
|
||||||
|
Versioned::class . '.versioned',
|
||||||
|
['Versioned'],
|
||||||
|
],
|
||||||
|
ClassInfo::parse_class_spec("SilverStripe\\Versioned\\Versioned.versioned('Versioned')")
|
||||||
|
);
|
||||||
// String with commas
|
// String with commas
|
||||||
$this->assertEquals(
|
$this->assertEquals(
|
||||||
array(Versioned::class, array('Stage,Live', 'Stage')),
|
array(Versioned::class, array('Stage,Live', 'Stage')),
|
||||||
@ -430,8 +445,13 @@ class ObjectTest extends SapphireTest
|
|||||||
);
|
);
|
||||||
// Nested array
|
// Nested array
|
||||||
$this->assertEquals(
|
$this->assertEquals(
|
||||||
array('Enum',array(array('Accepted', 'Pending', 'Declined', array('UnsubmittedA','UnsubmittedB')),
|
[
|
||||||
'Unsubmitted')),
|
'Enum',
|
||||||
|
[
|
||||||
|
['Accepted', 'Pending', 'Declined', ['UnsubmittedA', 'UnsubmittedB']],
|
||||||
|
'Unsubmitted'
|
||||||
|
]
|
||||||
|
],
|
||||||
ClassInfo::parse_class_spec(
|
ClassInfo::parse_class_spec(
|
||||||
"Enum(array('Accepted', 'Pending', 'Declined', array('UnsubmittedA','UnsubmittedB')), 'Unsubmitted')"
|
"Enum(array('Accepted', 'Pending', 'Declined', array('UnsubmittedA','UnsubmittedB')), 'Unsubmitted')"
|
||||||
)
|
)
|
||||||
@ -443,8 +463,13 @@ class ObjectTest extends SapphireTest
|
|||||||
);
|
);
|
||||||
// 5.4 Nested shorthand array
|
// 5.4 Nested shorthand array
|
||||||
$this->assertEquals(
|
$this->assertEquals(
|
||||||
array('Enum',array(array('Accepted', 'Pending', 'Declined', array('UnsubmittedA','UnsubmittedB')),
|
[
|
||||||
'Unsubmitted')),
|
'Enum',
|
||||||
|
[
|
||||||
|
['Accepted', 'Pending', 'Declined', ['UnsubmittedA', 'UnsubmittedB']],
|
||||||
|
'Unsubmitted'
|
||||||
|
]
|
||||||
|
],
|
||||||
ClassInfo::parse_class_spec(
|
ClassInfo::parse_class_spec(
|
||||||
"Enum(['Accepted', 'Pending', 'Declined', ['UnsubmittedA','UnsubmittedB']], 'Unsubmitted')"
|
"Enum(['Accepted', 'Pending', 'Declined', ['UnsubmittedA','UnsubmittedB']], 'Unsubmitted')"
|
||||||
)
|
)
|
||||||
|
@ -292,7 +292,11 @@ class DataExtensionTest extends SapphireTest
|
|||||||
$extension->clearOwner();
|
$extension->clearOwner();
|
||||||
$this->assertEquals($obj1, $extension->getOwner());
|
$this->assertEquals($obj1, $extension->getOwner());
|
||||||
|
|
||||||
// Clear original owner
|
// Clear pushed null
|
||||||
|
$extension->clearOwner();
|
||||||
|
$this->assertNull($extension->getOwner());
|
||||||
|
|
||||||
|
// Clear original null
|
||||||
$extension->clearOwner();
|
$extension->clearOwner();
|
||||||
$this->assertNull($extension->getOwner());
|
$this->assertNull($extension->getOwner());
|
||||||
|
|
||||||
|
Loading…
Reference in New Issue
Block a user