Virtual proxies, fine-tuned for use in lazy loading.
Install using composer:
composer require stratadox/proxy
An implementation of the Virtual Proxy pattern. Virtual proxies can take the place of "real" objects. They serve as placeholders for objects that are expensive to load.
The proxied objects may have to be retrieved from a database or remote web server. They may just require loads of memory, or require plenty of other such objects. When requests need only some of these objects, it can cause terrible performance problems to load all of them.
Using this package, you can provide your objects with proxies. These surrogates do not require database queries, and require only little memory. They satisfy the dependency demands of your class, without the overhead of the "actual" objects.
Virtual proxies like these are great for lazily loading the relationships of the entities in the domain model.
Let's say that you have a Shop. We all hope for the shop to have a lot of Customers. And for the Customers to place a lot of Orders. The more the better!
However... loading all those Customers, with all their orders, requires a lot of memory. Retrieving, organizing and sending the data may take a very long time.
In order to provide the Shop access to any customer, without loading all the customers, we can use CustomerProxy objects.
The proxies are subclasses of the real entities. That way, they satisfy all type checks. Proxies overwrite all public methods of the base class.
When one of these methods is called, the proxy gets loaded. This triggers the construction of the "expensive" object. Once loaded, the method that was called on the proxy is now called on the real object. All future calls upon the proxy get redirected immediately, without loading.
Proxy loaders implement the ProxyLoader interface:
<?php
use Stratadox\Proxy\ProxyLoader;
class MyLoader implements ProxyLoader
{
private $repository;
public function __construct(MyRepository $repository)
{
$this->repository = $repository;
}
public function loadTheInstance(array $data): object
{
return $this->repository->byId($data['id']);
}
}
Producing a basic factory is as simple as this:
<?php
use Stratadox\Proxy\BasicProxyFactory;
$proxyFactory = BasicProxyFactory::for(MyProxy::class, new MyLoader());
The proxy should be given the information needed by the loader:
<?php
/** @var \Stratadox\Proxy\ProxyFactory $proxyFactory */
$proxyFactory->create(['id' => 1]);
When the concrete type of the proxy class depends on the known data, one can use:
<?php
use Stratadox\Proxy\BasicProxyFactory;
use Stratadox\Proxy\CompositeProxyFactory;
// making the factory:
$proxyFactory = CompositeProxyFactory::decidingBy('type', [
'car' => BasicProxyFactory::for(CarProxy::class, $loader),
'painting' => BasicProxyFactory::for(PaintingProxy::class, $loader),
]);
// creating the proxies:
$carProxy = $proxyFactory->create(['type' => 'car', 'id' => 1]);
$paintingProxy = $proxyFactory->create(['type' => 'painting', 'id' => 5]);
This package only contains the behaviour for the virtual proxies. Proxy classes themselves are project-specific, and therefore not included. The proxy implementations simply use the Proxying trait and redirect calls. Classes for the proxies can be hand-crafted during development or, preferably, generated during deployment.
The module is no database, nor is it a data access tool. Client code is supposed to provide the mechanism through which the proxied entities are loaded.
In order to be able to make a proxy for a class, the class must:
- Not be declared
final
- Not access private properties of other objects*
- Not be compared by reference**
*for example, although normally valid, this would not work with proxies:
<?php
class Foo {
private $foo;
public function __construct(string $foo) {
$this->foo = $foo;
}
public function isEqual(Foo $other): bool {
return $this->foo === $other->foo;
}
}
**for example, although this would otherwise work, not so with proxies:
<?php
class Foo {
private $foo;
public function __construct(string $foo) {
$this->foo = $foo;
}
public function isEqual(Foo $other): bool {
return $this == $other;
}
}
Instead, compare other (potentially proxied) instances through their public interface:
<?php
class Foo {
private $foo;
public function __construct(string $foo) {
$this->foo = $foo;
}
public function foo(): string {
return $this->foo;
}
public function isEqual(Foo $other): bool {
return $this->foo() === $other->foo();
}
}
The placeholder or surrogate for the "real" object.
The expensive-to-load object that might eventually take the place of the proxy.