From f361479edf28a366e2c8273082c65d19cad5b980 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Wojciech=20=C5=9Alawski?= Date: Fri, 26 Aug 2016 09:57:04 +0200 Subject: [PATCH] Fix #12000, #12170 --- CHANGELOG.md | 1 + phalcon/mvc/model.zep | 24 +++++++++++++++++++-- tests/_data/models/Robots.php | 1 + tests/unit/Mvc/ModelTest.php | 39 +++++++++++++++++++++++++++++++++++ 4 files changed, 63 insertions(+), 2 deletions(-) diff --git a/CHANGELOG.md b/CHANGELOG.md index ab730497075..2afd88ba409 100755 --- a/CHANGELOG.md +++ b/CHANGELOG.md @@ -1,4 +1,5 @@ # [3.0.2](https://github.com/phalcon/cphalcon/releases/tag/v3.0.2) (2016-XX-XX) +- Fixed saving snapshot data while caching model [#12170](https://github.com/phalcon/cphalcon/issues/12170), [#12000](https://github.com/phalcon/cphalcon/issues/12000) # [3.0.1](https://github.com/phalcon/cphalcon/releases/tag/v3.0.1) (2016-08-24) - Fixed `Phalcon\Cache\Backend\Redis::flush` in order to flush cache correctly diff --git a/phalcon/mvc/model.zep b/phalcon/mvc/model.zep index bae9dfd631f..d6a5294d05c 100644 --- a/phalcon/mvc/model.zep +++ b/phalcon/mvc/model.zep @@ -4303,7 +4303,18 @@ abstract class Model implements EntityInterface, ModelInterface, ResultInterface /** * Use the standard serialize function to serialize the array data */ - return serialize(this->toArray()); + var attributes, snapshot; + + let attributes = this->toArray(); + if ( this->_modelsManager)->isKeepingSnapshots(this) { + let snapshot = this->_snapshot; + // if attributes is not the same as snapshot then save snapshot too + if attributes != snapshot { + return serialize(["_attributes": attributes, "_snapshot": snapshot]); + } + } + + return serialize(attributes); } /** @@ -4311,9 +4322,18 @@ abstract class Model implements EntityInterface, ModelInterface, ResultInterface */ public function unserialize(string! data) { - var attributes, dependencyInjector, manager, key, value; + var attributes, dependencyInjector, manager, key, value, snapshot; let attributes = unserialize(data); + if ( this->_modelsManager)->isKeepingSnapshots(this) { + if fetch snapshot, attributes["_snapshot"] { + let this->_snapshot = snapshot; + let attributes = attributes["_attributes"]; + } + else { + let this->_snapshot = attributes; + } + } if typeof attributes == "array" { /** diff --git a/tests/_data/models/Robots.php b/tests/_data/models/Robots.php index 262bafb1b5e..7bc29725f46 100644 --- a/tests/_data/models/Robots.php +++ b/tests/_data/models/Robots.php @@ -25,6 +25,7 @@ class Robots extends Model { public function initialize() { + $this->keepSnapshots(true); $this->hasMany('id',RobotsParts::class, 'robots_id', ['foreignKey' => true, 'reusable' => false, 'alias' => 'parts']); } } diff --git a/tests/unit/Mvc/ModelTest.php b/tests/unit/Mvc/ModelTest.php index 96b161d10bc..7fcb8c5077a 100644 --- a/tests/unit/Mvc/ModelTest.php +++ b/tests/unit/Mvc/ModelTest.php @@ -2,6 +2,10 @@ namespace Phalcon\Test\Unit\Mvc; +use Phalcon\Cache\Backend; +use Phalcon\Cache\Backend\Apc; +use Phalcon\Cache\Frontend\Data; +use Phalcon\Test\Models\Robots; use Phalcon\Test\Models\Users; use Phalcon\Test\Models\Customers; use Phalcon\Test\Models\Packages; @@ -187,4 +191,39 @@ function () { } ); } + + /** + * Tests serializing model while using cache and keeping snapshots + * + * The snapshot should be saved while using cache + * + * @issue 12170, 12000 + * @author Wojciech Ĺšlawski + * @since 2016-08-26 + */ + public function testSerializeSnapshotCache() + { + $this->specify( + 'Snapshot data should be saved while saving model to cache', + function () { + $cache = new Apc(new Data(['lifetime' => 20])); + $robot = Robots::findFirst(); + expect($robot)->isInstanceOf(Robots::class); + expect($robot->getSnapshotData())->notEmpty(); + $cache->save('robot', $robot); + /** @var Robots $robot */ + $robot = $cache->get('robot'); + expect($robot)->isInstanceOf(Robots::class); + expect($robot->getSnapshotData())->notEmpty(); + expect($robot->getSnapshotData())->equals($robot->toArray()); + $robot->text = 'abc'; + $cache->save('robot', $robot); + /** @var Robots $robot */ + $robot = $cache->get('robot'); + expect($robot)->isInstanceOf(Robots::class); + expect($robot->getSnapshotData())->notEmpty(); + expect($robot->getSnapshotData())->notEquals($robot->toArray()); + } + ); + } }