diff --git a/.gitignore b/.gitignore index 660fc15e..cc31cac1 100644 --- a/.gitignore +++ b/.gitignore @@ -2,3 +2,4 @@ composer.lock /phpunit.xml .phpunit.result.cache +.idea diff --git a/src/Serializers/Native.php b/src/Serializers/Native.php index 4a04b981..54e19d02 100644 --- a/src/Serializers/Native.php +++ b/src/Serializers/Native.php @@ -194,15 +194,31 @@ public function __unserialize($data) $this->closure = $this->closure->bindTo($this->code['this'], $this->code['scope']); - if (! empty($this->code['objects'])) { - foreach ($this->code['objects'] as $item) { - $item['property']->setValue($item['instance'], $item['object']->getClosure()); - } - } + $this->bindObjectsIfNeeded(); $this->code = $this->code['function']; } + private function bindObjectsIfNeeded() { + if (!empty($this->code['objects'])) { + $this->bindObjects(); + } + } + + private function bindObjects() { + foreach ($this->code['objects'] as $item) { + $item['property']->setValue($item['instance'], $this->calculateObjectValue($item)); + } + } + + private function calculateObjectValue($item) { + return $this->isSerializableClosure($item['object']) ? $item['object'] : $item['object']->getClosure(); + } + + private function isSerializableClosure($object): bool { + return $object instanceof SerializableClosure || $object instanceof UnsignedSerializableClosure; + } + /** * Ensures the given closures are serializable. * diff --git a/tests/Fixtures/ClassWithPublicProperty.php b/tests/Fixtures/ClassWithPublicProperty.php new file mode 100644 index 00000000..494fe7ff --- /dev/null +++ b/tests/Fixtures/ClassWithPublicProperty.php @@ -0,0 +1,14 @@ +closure = $closure; + } +} diff --git a/tests/SerializerTest.php b/tests/SerializerTest.php index 9c2e70bd..7a5481a8 100644 --- a/tests/SerializerTest.php +++ b/tests/SerializerTest.php @@ -6,6 +6,7 @@ use Laravel\SerializableClosure\Serializers\Signed; use Laravel\SerializableClosure\Support\ReflectionClosure; use Laravel\SerializableClosure\UnsignedSerializableClosure; +use Tests\Fixtures\ClassWithPublicProperty; use Tests\Fixtures\Model; test('closure with simple const', function () { @@ -485,6 +486,32 @@ function () { new CarbonImmutable, ]); +test('SerializableClosure in the class property', function () { + SerializableClosure::setSecretKey('foo'); + + $innerClosure = new ClassWithPublicProperty(new SerializableClosure(function () {})); + $outerClosure = new SerializableClosure(function () use ($innerClosure) { + return $innerClosure->closure; + }); + + $unSerializedOuterClosure = unserialize(serialize($outerClosure)); + + expect($outerClosure())->toBeInstanceOf(SerializableClosure::class); + expect($unSerializedOuterClosure())->toBeInstanceOf(SerializableClosure::class); +}); + +test('UnsignedSerializableClosure in the class property', function () { + $innerClosure = new ClassWithPublicProperty(SerializableClosure::unsigned(function () {})); + $outerClosure = SerializableClosure::unsigned(function () use ($innerClosure) { + return $innerClosure->closure; + }); + + $unSerializedOuterClosure = unserialize(serialize($outerClosure)); + + expect($outerClosure())->toBeInstanceOf(UnsignedSerializableClosure::class); + expect($unSerializedOuterClosure())->toBeInstanceOf(UnsignedSerializableClosure::class); +}); + function serializer_php_74_switch_statement_test_is_two($a) { return $a === 2;