diff --git a/src/DataConverter/DataConverter.php b/src/DataConverter/DataConverter.php index d654c39e..47cb3e10 100644 --- a/src/DataConverter/DataConverter.php +++ b/src/DataConverter/DataConverter.php @@ -46,8 +46,12 @@ public function fromPayload(Payload $payload, $type) } $type = Type::create($type); - if ($type->getName() === Type::TYPE_VOID) { - return null; + if (\in_array($type->getName(), [Type::TYPE_VOID, Type::TYPE_NULL, Type::TYPE_FALSE, Type::TYPE_TRUE], true)) { + return match($type->getName()) { + Type::TYPE_VOID, Type::TYPE_NULL => null, + Type::TYPE_TRUE => true, + Type::TYPE_FALSE => false, + }; } return $this->converters[$encoding]->fromPayload($payload, $type); diff --git a/src/DataConverter/Type.php b/src/DataConverter/Type.php index 1bc89808..6d4cc266 100644 --- a/src/DataConverter/Type.php +++ b/src/DataConverter/Type.php @@ -26,6 +26,9 @@ final class Type public const TYPE_INT = 'int'; public const TYPE_FLOAT = 'float'; public const TYPE_VOID = 'void'; + public const TYPE_NULL = 'null'; + public const TYPE_TRUE = 'true'; + public const TYPE_FALSE = 'false'; /** * @var string @@ -46,7 +49,7 @@ public function __construct(string $name = Type::TYPE_ANY, bool $allowsNull = nu $this->name = $name; $this->allowsNull = $allowsNull ?? ( - $name === self::TYPE_ANY || $name === self::TYPE_VOID + $name === self::TYPE_ANY || $name === self::TYPE_VOID || $name === self::TYPE_NULL ); } diff --git a/tests/Fixtures/src/Activity/Php82TypesActivity.php b/tests/Fixtures/src/Activity/Php82TypesActivity.php new file mode 100644 index 00000000..035bc12e --- /dev/null +++ b/tests/Fixtures/src/Activity/Php82TypesActivity.php @@ -0,0 +1,38 @@ +<?php + +/** + * This file is part of Temporal package. + * + * For the full copyright and license information, please view the LICENSE + * file that was distributed with this source code. + */ + +declare(strict_types=1); + +namespace Temporal\Tests\Activity; + +use Temporal\Activity\ActivityInterface; +use Temporal\Activity\ActivityMethod; +/* +todo: uncomment this with min php 8.2 requirement or when we can skip activities loading depending on php version +#[ActivityInterface(prefix: "Php82.")] +class Php82TypesActivity +{ + #[ActivityMethod] + public function returnNull(null $value): null + { + return $value; + } + + #[ActivityMethod] + public function returnTrue(true $value): true + { + return $value; + } + + #[ActivityMethod] + public function returnFalse(false $value): false + { + return $value; + } +}*/ diff --git a/tests/Fixtures/src/Workflow/Php82TypesWorkflow.php b/tests/Fixtures/src/Workflow/Php82TypesWorkflow.php new file mode 100644 index 00000000..f9de6060 --- /dev/null +++ b/tests/Fixtures/src/Workflow/Php82TypesWorkflow.php @@ -0,0 +1,41 @@ +<?php + +/** + * This file is part of Temporal package. + * + * For the full copyright and license information, please view the LICENSE + * file that was distributed with this source code. + */ + +declare(strict_types=1); + +namespace Temporal\Tests\Workflow; + +use Temporal\Activity\ActivityOptions; +use Temporal\Common\RetryOptions; +use Temporal\Tests\Activity\Php82TypesActivity; +use Temporal\Workflow; +use Temporal\Workflow\WorkflowMethod; + +#[Workflow\WorkflowInterface] +class Php82TypesWorkflow +{ + #[WorkflowMethod(name: 'Php82TypesWorkflow')] + public function handler(): iterable + { + $simple = Workflow::newActivityStub( + Php82TypesActivity::class, + ActivityOptions::new() + ->withStartToCloseTimeout(5) + ->withRetryOptions( + RetryOptions::new()->withMaximumAttempts(2), + ), + ); + + return [ + yield $simple->returnNull(null), + yield $simple->returnTrue(true), + yield $simple->returnFalse(false), + ]; + } +} diff --git a/tests/Functional/Client/TypedStubTestCase.php b/tests/Functional/Client/TypedStubTestCase.php index 71ae7ba0..f3ffda95 100644 --- a/tests/Functional/Client/TypedStubTestCase.php +++ b/tests/Functional/Client/TypedStubTestCase.php @@ -17,6 +17,7 @@ use Temporal\Tests\Unit\Declaration\Fixture\WorkflowWithoutHandler; use Temporal\Tests\Workflow\ActivityReturnTypeWorkflow; use Temporal\Tests\Workflow\GeneratorWorkflow; +use Temporal\Tests\Workflow\Php82TypesWorkflow; use Temporal\Tests\Workflow\QueryWorkflow; use Temporal\Tests\Workflow\SignalledWorkflowReusable; use Temporal\Tests\Workflow\SignalledWorkflowWithInheritance; @@ -159,4 +160,19 @@ public function testSignalRunningWorkflowWithInheritedSignalViaParentInterface() $result = $workflowRun->getResult(); $this->assertEquals(['test1'], $result); } + + /** + * @requires PHP >= 8.2 + * todo: uncomment this with min php 8.2 requirement or when we can skip activities loading depending on php version + */ + // public function testPhp82Types(): void + // { + // $client = $this->createClient(); + // + // $workflow = $client->newWorkflowStub(Php82TypesWorkflow::class); + // $workflowRun = $client->start($workflow); + // $result = $workflowRun->getResult(); + // + // $this->assertSame([null, true, false], $result); + // } } diff --git a/tests/Unit/DataConverter/DataConverterTestCase.php b/tests/Unit/DataConverter/DataConverterTestCase.php index ead9699c..032a7ee4 100644 --- a/tests/Unit/DataConverter/DataConverterTestCase.php +++ b/tests/Unit/DataConverter/DataConverterTestCase.php @@ -37,6 +37,9 @@ public function typesDataProvider(): array Type::TYPE_VOID . ' => ' . Type::TYPE_ANY => [Type::TYPE_ANY, null], Type::TYPE_ARRAY . ' => ' . Type::TYPE_ANY => [Type::TYPE_ANY, []], Type::TYPE_OBJECT . ' => ' . Type::TYPE_ANY => [Type::TYPE_ANY, (object)[]], + Type::TYPE_NULL . ' => ' . Type::TYPE_ANY => [Type::TYPE_ANY, null], + Type::TYPE_TRUE . ' => ' . Type::TYPE_ANY => [Type::TYPE_ANY, true], + Type::TYPE_FALSE . ' => ' . Type::TYPE_ANY => [Type::TYPE_ANY, false], Type::TYPE_ARRAY => [Type::TYPE_ARRAY, [1, 2, 3]], Type::TYPE_OBJECT => [Type::TYPE_OBJECT, (object)['field' => 'value']], @@ -45,6 +48,9 @@ public function typesDataProvider(): array Type::TYPE_INT => [Type::TYPE_INT, 42], Type::TYPE_FLOAT => [Type::TYPE_FLOAT, 0.1], Type::TYPE_VOID => [Type::TYPE_VOID, null], + Type::TYPE_NULL => [Type::TYPE_NULL, null], + Type::TYPE_TRUE => [Type::TYPE_TRUE, true], + Type::TYPE_FALSE => [Type::TYPE_FALSE, false], Type::TYPE_ARRAY . ' (associative) => ' . Type::TYPE_ARRAY => [Type::TYPE_ARRAY, ['field' => 'value']], ]; @@ -60,7 +66,8 @@ public function negativeTypesDataProvider(): array Type::TYPE_ARRAY . ' => ' . Type::TYPE_STRING => [Type::TYPE_STRING, [1, 2, 3]], Type::TYPE_FLOAT . ' => ' . Type::TYPE_STRING => [Type::TYPE_STRING, .42], Type::TYPE_INT . ' => ' . Type::TYPE_STRING => [Type::TYPE_STRING, 42], - Type::TYPE_BOOL . ' => ' . Type::TYPE_STRING => [Type::TYPE_STRING, true], + Type::TYPE_TRUE . ' => ' . Type::TYPE_STRING => [Type::TYPE_STRING, true], + Type::TYPE_FALSE . ' => ' . Type::TYPE_STRING => [Type::TYPE_STRING, false], Type::TYPE_VOID . ' => ' . Type::TYPE_STRING => [Type::TYPE_STRING, null], Type::TYPE_OBJECT . ' => ' . Type::TYPE_BOOL => [Type::TYPE_BOOL, (object)['field' => 'value']], @@ -74,14 +81,16 @@ public function negativeTypesDataProvider(): array Type::TYPE_ARRAY . ' => ' . Type::TYPE_INT => [Type::TYPE_INT, [1, 2, 3]], Type::TYPE_FLOAT . ' => ' . Type::TYPE_INT => [Type::TYPE_INT, .42], Type::TYPE_STRING . ' => ' . Type::TYPE_INT => [Type::TYPE_INT, 'string'], - Type::TYPE_BOOL . ' => ' . Type::TYPE_INT => [Type::TYPE_INT, true], + Type::TYPE_TRUE . ' => ' . Type::TYPE_INT => [Type::TYPE_INT, true], + Type::TYPE_FALSE . ' => ' . Type::TYPE_INT => [Type::TYPE_INT, false], Type::TYPE_VOID . ' => ' . Type::TYPE_INT => [Type::TYPE_INT, null], Type::TYPE_OBJECT . ' => ' . Type::TYPE_FLOAT => [Type::TYPE_FLOAT, (object)['field' => 'value']], Type::TYPE_ARRAY . ' => ' . Type::TYPE_FLOAT => [Type::TYPE_FLOAT, [1, 2, 3]], Type::TYPE_INT . ' => ' . Type::TYPE_FLOAT => [Type::TYPE_FLOAT, 42], Type::TYPE_STRING . ' => ' . Type::TYPE_FLOAT => [Type::TYPE_FLOAT, 'string'], - Type::TYPE_BOOL . ' => ' . Type::TYPE_FLOAT => [Type::TYPE_FLOAT, true], + Type::TYPE_TRUE . ' => ' . Type::TYPE_FLOAT => [Type::TYPE_FLOAT, true], + Type::TYPE_FALSE . ' => ' . Type::TYPE_FLOAT => [Type::TYPE_FLOAT, false], Type::TYPE_VOID . ' => ' . Type::TYPE_FLOAT => [Type::TYPE_FLOAT, null], Type::TYPE_ARRAY . ' => ' . Type::TYPE_OBJECT => [Type::TYPE_OBJECT, [1, 2, 3]], @@ -90,11 +99,14 @@ public function negativeTypesDataProvider(): array Type::TYPE_STRING . ' => ' . Type::TYPE_OBJECT => [Type::TYPE_OBJECT, 'string'], Type::TYPE_BOOL . ' => ' . Type::TYPE_OBJECT => [Type::TYPE_OBJECT, true], Type::TYPE_VOID . ' => ' . Type::TYPE_OBJECT => [Type::TYPE_OBJECT, null], + Type::TYPE_TRUE . ' => ' . Type::TYPE_OBJECT => [Type::TYPE_OBJECT, true], + Type::TYPE_FALSE . ' => ' . Type::TYPE_OBJECT => [Type::TYPE_OBJECT, false], Type::TYPE_FLOAT . ' => ' . Type::TYPE_ARRAY => [Type::TYPE_ARRAY, .42], Type::TYPE_INT . ' => ' . Type::TYPE_ARRAY => [Type::TYPE_ARRAY, 42], Type::TYPE_STRING . ' => ' . Type::TYPE_ARRAY => [Type::TYPE_ARRAY, 'string'], - Type::TYPE_BOOL . ' => ' . Type::TYPE_ARRAY => [Type::TYPE_ARRAY, true], + Type::TYPE_TRUE . ' => ' . Type::TYPE_ARRAY => [Type::TYPE_ARRAY, true], + Type::TYPE_FALSE . ' => ' . Type::TYPE_ARRAY => [Type::TYPE_ARRAY, false], Type::TYPE_VOID . ' => ' . Type::TYPE_ARRAY => [Type::TYPE_ARRAY, null], ]; } @@ -111,7 +123,8 @@ public function nullableTypesDataProvider(): array Type::TYPE_FLOAT . ' => ' . Type::TYPE_VOID => [Type::TYPE_VOID, .42], Type::TYPE_INT . ' => ' . Type::TYPE_VOID => [Type::TYPE_VOID, 42], Type::TYPE_STRING . ' => ' . Type::TYPE_VOID => [Type::TYPE_VOID, 'string'], - Type::TYPE_BOOL . ' => ' . Type::TYPE_VOID => [Type::TYPE_VOID, true], + Type::TYPE_TRUE . ' => ' . Type::TYPE_VOID => [Type::TYPE_VOID, true], + Type::TYPE_FALSE . ' => ' . Type::TYPE_VOID => [Type::TYPE_VOID, false], ]; }