Implement ObjectStorageTrait

This commit is contained in:
Matias Griese
2017-03-06 18:58:55 +02:00
parent 34cff72fce
commit 9ba038b2b3

View File

@@ -0,0 +1,360 @@
<?php
namespace Grav\Common\Object;
use Grav\Common\Object\Storage\StorageInterface;
/**
* Abstract base class for stored objects.
*
* @property string $id
*/
trait ObjectStorageTrait
{
/**
* If you don't have global instance ids, override this in extending class.
* @var array
*/
static protected $instances = [];
/**
* If you don't have global storage, override this in extending class.
* @var StorageInterface
*/
static protected $storage;
/**
* Readonly object.
* @var bool
*/
protected $readonly = false;
/**
* @var bool
*/
protected $initialized = false;
/**
* @param StorageInterface $storage
*/
static public function setStorage(StorageInterface $storage)
{
static::$storage = $storage;
}
/**
* Returns the global instance to the object.
*
* Note that using array of fields will always make a query, but it's very useful feature if you want to search one
* item by using arbitrary set of matching fields. If there are more than one matching object, first one gets returned.
*
* @param string|array $keys An optional primary key value to load the object by, or an array of fields to match.
* @param boolean $reload Force object to reload.
*
* @return ObjectInterface
*/
static public function instance($keys = null, $reload = false)
{
// If we are creating or loading a new item or we load instance by alternative keys, we need to create a new object.
if (!$keys || is_array($keys) || (is_scalar($keys) && !isset(static::$instances[$keys]))) {
$c = get_called_class();
/** @var ObjectStorageTrait|ObjectInterface $instance */
$instance = new $c();
if (!$instance->load($keys)) {
return $instance;
}
// Instance exists in storage: make sure that we return the global instance.
$keys = $instance->getInstanceId();
$reload = false;
}
// Return global instance from the identifier.
$instance = static::$instances[$keys];
if ($reload) {
$instance->load();
}
return $instance;
}
/**
* Removes all or selected instances from the object cache.
*
* @param null|string|array $ids An optional primary key or list of keys.
*/
static public function freeInstances($ids = null)
{
if ($ids === null) {
$ids = array_keys(static::$instances);
}
$ids = (array) $ids;
foreach ($ids as $id) {
unset(static::$instances[$id]);
}
}
/**
* Override this function if you need to initialize object right after creating it.
*
* Can be used for example if the fields need to be converted from json strings to array.
*
* @return bool True if initialization was done, false if object was already initialized.
*/
public function initialize()
{
$initialized = $this->initialized;
$this->initialized = true;
return !$initialized;
}
/**
* Convert instance to a read only object.
*
* @return $this
*/
public function readonly()
{
$this->readonly = true;
return $this;
}
/**
* Returns true if the object has been stored.
*
* @return boolean True if object exists in storage.
*/
public function isSaved()
{
return $this->getStorage()->exists($this->getStorageKey());
}
/**
* Method to load object from the storage.
*
* @param mixed $keys An optional primary key value to load the object by, or an array of fields to match. If not
* set the instance key value is used.
* @param bool $getKey Internal parameter, please do not use.
*
* @return bool True on success, false if the object doesn't exist.
*/
public function load($keys = null, $getKey = true)
{
if ($getKey) {
if (is_scalar($keys)) {
$keys = ['id' => (string) $keys];
}
// Fetch internal key.
$key = $this->getStorageKey($keys);
} else {
// Internal key was passed.
$key = $keys;
$keys = [];
}
// Get storage.
$storage = $this->getStorage();
// Load the object based on the keys.
$this->items = $storage->load($key);
$this->initialize();
$id = $this->getInstanceId();
if ($id) {
if (!isset(static::$instances[$id])) {
static::$instances[$id] = $this;
}
}
return $this->isSaved();
}
/**
* Method to save the object to the storage.
*
* Before saving the object, this method checks if object can be safely saved.
*
* @param bool $includeChildren
* @return bool True on success.
*/
public function save($includeChildren = false)
{
// Check the object.
if ($this->readonly || !$this->check($includeChildren) || !$this->onBeforeSave()) {
return false;
}
try {
// Get storage.
$storage = $this->getStorage();
$key = $this->getStorageKey();
$exists = $this->getStorage()->exists($key);
// Get data to be saved.
$data = $this->prepareSave();
// Save the object.
$id = $storage->save($key, $data);
} catch (\Exception $e) {
return false;
}
if (!$id) {
return false;
}
// If item was created, load the object (making sure it has been properly initialized).
if (!$exists) {
$this->load($id, false);
}
if ($includeChildren) {
$this->saveChildren();
}
$this->onAfterSave();
return true;
}
/**
* Method to delete the object from the database.
*
* @param bool $includeChildren
* @return bool True on success.
*/
public function delete($includeChildren = false)
{
if ($this->readonly || !$this->isSaved() || !$this->onBeforeDelete()) {
return false;
}
if ($includeChildren) {
$this->deleteChildren();
}
// Get storage.
$storage = $this->getStorage();
if (!$storage->delete($this->getStorageKey())) {
return false;
}
$this->onAfterDelete();
return true;
}
/**
* Method to perform sanity checks on the instance properties to ensure they are safe to store in the storage.
*
* Child classes should override this method to make sure the data they are storing in the storage is safe and as
* expected before saving the object.
*
* @return bool True if the instance is sane and able to be stored in the storage.
*/
public function check($includeChildren = false)
{
$result = true;
if ($includeChildren) {
foreach ($this as $field => $value) {
if (is_object($value) && method_exists($value, 'check')) {
$result = $result && $value->check();
}
}
}
return $result;
}
// Internal functions
/**
* @return bool
*/
protected function onBeforeSave()
{
return true;
}
protected function onAfterSave()
{
}
/**
* @return bool
*/
protected function onBeforeDelete()
{
return true;
}
protected function onAfterDelete()
{
}
protected function saveChildren()
{
foreach ($this as $field => $value) {
if (is_object($value) && method_exists($value, 'save')) {
$value->save(true);
}
}
}
protected function deleteChildren()
{
foreach ($this as $field => $value) {
if (is_object($value) && method_exists($value, 'delete')) {
$value->delete(true);
}
}
}
protected function prepareSave(array $data = null)
{
if ($data === null) {
$data = $this->toArray();
}
foreach ($data as $field => $value) {
if (is_object($value) && method_exists($value, 'save')) {
unset($data[$field]);
}
}
return $data;
}
/**
* Method to get the storage key for the object.
*
* @param array
* @return string
*/
abstract public function getStorageKey(array $keys = []);
public function getInstanceId(array $keys = [])
{
return $this->getStorageKey($keys);
}
//abstract public function setStorageKey(array $keys = []);
/**
* @return StorageInterface
*/
protected function getStorage()
{
return static::$storage;
}
}