From 21366a8c43819457d458e772a4e86d0df9233b5e Mon Sep 17 00:00:00 2001
From: Djordy Koert
Date: Thu, 30 Nov 2023 22:47:39 +0100
Subject: [PATCH] Feat: Change nullable & type for OpenApi v3.1.0 (#1502)
---
docs/reference/annotations.md | 13 +++--
docs/reference/attributes.md | 68 ++++++++++++++++++-------
src/Annotations/Schema.php | 11 +++-
src/Annotations/SecurityScheme.php | 2 +-
src/Attributes/AdditionalProperties.php | 3 +-
src/Attributes/Items.php | 3 +-
src/Attributes/JsonContent.php | 3 +-
src/Attributes/Property.php | 3 +-
src/Attributes/Schema.php | 3 +-
src/Attributes/SecurityScheme.php | 11 ++--
src/Attributes/XmlContent.php | 3 +-
src/Processors/AugmentProperties.php | 6 ++-
src/Processors/AugmentSchemas.php | 2 +-
tests/Fixtures/Scratch/Nullable.php | 3 ++
tests/Fixtures/Scratch/Nullable.yaml | 4 ++
tests/Fixtures/Scratch/Types31.php | 33 ++++++++++++
tests/Fixtures/Scratch/Types31.yaml | 30 +++++++++++
17 files changed, 161 insertions(+), 40 deletions(-)
create mode 100644 tests/Fixtures/Scratch/Types31.php
create mode 100644 tests/Fixtures/Scratch/Types31.yaml
diff --git a/docs/reference/annotations.md b/docs/reference/annotations.md
index fc0d71d68..a06d805fe 100644
--- a/docs/reference/annotations.md
+++ b/docs/reference/annotations.md
@@ -956,10 +956,12 @@ An object instance is valid against this property if its number of properties is
required : string[]
An object instance is valid against this property if its property set contains all elements in this property's
array value.
- type : string
+ type : string|non-empty-array<string>
The type of the schema/property.
-The value MUST be one of "string", "number", "integer", "boolean", "array" or "object".
+OpenApi v3.0: The value MUST be one of "string", "number", "integer", "boolean", "array" or "object".
+
+Since OpenApi v3.1 an array of types may be used.
format : string
The extending format for the previously mentioned type. See Data Type Formats for further details.
collectionFormat : string
@@ -1040,7 +1042,10 @@ To represent examples that cannot naturally be represented in JSON or YAML, a st
contain the example with escaping where necessary.
nullable : bool
Allows sending a null value for the defined schema.
-Default value is false.
+Default value is false.
+
+This must not be used when using OpenApi version 3.1,
+instead make the "type" property an array and add "null" as a possible type.
deprecated : bool
Specifies that a schema is deprecated and should be transitioned out of usage.
Default value is false.
@@ -1093,7 +1098,7 @@ defined by this property's value.
The relative or absolute path to a security scheme.
See: Using refs
securityScheme : string
The key into OpenApi->security array.
- type : string
+ type : string|non-empty-array<string>
The type of the security scheme.
description : string
A short description for security scheme.
diff --git a/docs/reference/attributes.md b/docs/reference/attributes.md
index fe9beb263..ea06132bc 100644
--- a/docs/reference/attributes.md
+++ b/docs/reference/attributes.md
@@ -44,10 +44,12 @@ array value.
A collection of properties to define for an object.
Each property is represented as an instance of the Property class.
- type : string|null
+ type : string|non-empty-array<string>|null
The type of the schema/property.
-The value MUST be one of "string", "number", "integer", "boolean", "array" or "object".
+OpenApi v3.0: The value MUST be one of "string", "number", "integer", "boolean", "array" or "object".
+
+Since OpenApi v3.1 an array of types may be used.
format : string|null
The extending format for the previously mentioned type. See Data Type Formats for further details.
items : OpenApi\Attributes\Items|null
@@ -139,7 +141,10 @@ To represent examples that cannot naturally be represented in JSON or YAML, a st
contain the example with escaping where necessary.
nullable : bool|null
Allows sending a null value for the defined schema.
-Default value is false.
+Default value is false.
+
+This must not be used when using OpenApi version 3.1,
+instead make the "type" property an array and add "null" as a possible type.
deprecated : bool|null
Specifies that a schema is deprecated and should be transitioned out of usage.
Default value is false.
@@ -1002,10 +1007,12 @@ array value.
A collection of properties to define for an object.
Each property is represented as an instance of the Property class.
- type : string|null
+ type : string|non-empty-array<string>|null
The type of the schema/property.
-The value MUST be one of "string", "number", "integer", "boolean", "array" or "object".
+OpenApi v3.0: The value MUST be one of "string", "number", "integer", "boolean", "array" or "object".
+
+Since OpenApi v3.1 an array of types may be used.
format : string|null
The extending format for the previously mentioned type. See Data Type Formats for further details.
items : OpenApi\Attributes\Items|null
@@ -1097,7 +1104,10 @@ To represent examples that cannot naturally be represented in JSON or YAML, a st
contain the example with escaping where necessary.
nullable : bool|null
Allows sending a null value for the defined schema.
-Default value is false.
+Default value is false.
+
+This must not be used when using OpenApi version 3.1,
+instead make the "type" property an array and add "null" as a possible type.
deprecated : bool|null
Specifies that a schema is deprecated and should be transitioned out of usage.
Default value is false.
@@ -1160,10 +1170,12 @@ array value.
A collection of properties to define for an object.
Each property is represented as an instance of the Property class.
- type : string|null
+ type : string|non-empty-array<string>|null
The type of the schema/property.
-The value MUST be one of "string", "number", "integer", "boolean", "array" or "object".
+OpenApi v3.0: The value MUST be one of "string", "number", "integer", "boolean", "array" or "object".
+
+Since OpenApi v3.1 an array of types may be used.
format : string|null
The extending format for the previously mentioned type. See Data Type Formats for further details.
items : OpenApi\Attributes\Items|null
@@ -1255,7 +1267,10 @@ To represent examples that cannot naturally be represented in JSON or YAML, a st
contain the example with escaping where necessary.
nullable : bool|null
Allows sending a null value for the defined schema.
-Default value is false.
+Default value is false.
+
+This must not be used when using OpenApi version 3.1,
+instead make the "type" property an array and add "null" as a possible type.
deprecated : bool|null
Specifies that a schema is deprecated and should be transitioned out of usage.
Default value is false.
@@ -2024,10 +2039,12 @@ array value.
A collection of properties to define for an object.
Each property is represented as an instance of the Property class.
- type : string|null
+ type : string|non-empty-array<string>|null
The type of the schema/property.
-The value MUST be one of "string", "number", "integer", "boolean", "array" or "object".
+OpenApi v3.0: The value MUST be one of "string", "number", "integer", "boolean", "array" or "object".
+
+Since OpenApi v3.1 an array of types may be used.
format : string|null
The extending format for the previously mentioned type. See Data Type Formats for further details.
items : OpenApi\Attributes\Items|null
@@ -2119,7 +2136,10 @@ To represent examples that cannot naturally be represented in JSON or YAML, a st
contain the example with escaping where necessary.
nullable : bool|null
Allows sending a null value for the defined schema.
-Default value is false.
+Default value is false.
+
+This must not be used when using OpenApi version 3.1,
+instead make the "type" property an array and add "null" as a possible type.
deprecated : bool|null
Specifies that a schema is deprecated and should be transitioned out of usage.
Default value is false.
@@ -2470,10 +2490,12 @@ array value.
A collection of properties to define for an object.
Each property is represented as an instance of the Property class.
- type : string|null
+ type : string|non-empty-array<string>|null
The type of the schema/property.
-The value MUST be one of "string", "number", "integer", "boolean", "array" or "object".
+OpenApi v3.0: The value MUST be one of "string", "number", "integer", "boolean", "array" or "object".
+
+Since OpenApi v3.1 an array of types may be used.
format : string|null
The extending format for the previously mentioned type. See Data Type Formats for further details.
items : OpenApi\Attributes\Items|null
@@ -2565,7 +2587,10 @@ To represent examples that cannot naturally be represented in JSON or YAML, a st
contain the example with escaping where necessary.
nullable : bool|null
Allows sending a null value for the defined schema.
-Default value is false.
+Default value is false.
+
+This must not be used when using OpenApi version 3.1,
+instead make the "type" property an array and add "null" as a possible type.
deprecated : bool|null
Specifies that a schema is deprecated and should be transitioned out of usage.
Default value is false.
@@ -2610,7 +2635,7 @@ These will be ignored but can be used for custom processing.
The relative or absolute path to a security scheme.
See: Using refs
securityScheme : string|null
The key into OpenApi->security array.
- type : string|null
+ type : string|non-empty-array<string>|null
The type of the security scheme.
description : string|null
A short description for security scheme.
@@ -2918,10 +2943,12 @@ array value.
A collection of properties to define for an object.
Each property is represented as an instance of the Property class.
- type : string|null
+ type : string|non-empty-array<string>|null
The type of the schema/property.
-The value MUST be one of "string", "number", "integer", "boolean", "array" or "object".
+OpenApi v3.0: The value MUST be one of "string", "number", "integer", "boolean", "array" or "object".
+
+Since OpenApi v3.1 an array of types may be used.
format : string|null
The extending format for the previously mentioned type. See Data Type Formats for further details.
items : OpenApi\Attributes\Items|null
@@ -3013,7 +3040,10 @@ To represent examples that cannot naturally be represented in JSON or YAML, a st
contain the example with escaping where necessary.
nullable : bool|null
Allows sending a null value for the defined schema.
-Default value is false.
+Default value is false.
+
+This must not be used when using OpenApi version 3.1,
+instead make the "type" property an array and add "null" as a possible type.
deprecated : bool|null
Specifies that a schema is deprecated and should be transitioned out of usage.
Default value is false.
diff --git a/src/Annotations/Schema.php b/src/Annotations/Schema.php
index afcf9fdd5..d0deeed05 100644
--- a/src/Annotations/Schema.php
+++ b/src/Annotations/Schema.php
@@ -91,9 +91,11 @@ class Schema extends AbstractAnnotation
/**
* The type of the schema/property.
*
- * The value MUST be one of "string", "number", "integer", "boolean", "array" or "object".
+ * OpenApi v3.0: The value MUST be one of "string", "number", "integer", "boolean", "array" or "object".
*
- * @var string
+ * Since OpenApi v3.1 an array of types may be used.
+ *
+ * @var string|non-empty-array
*/
public $type = Generator::UNDEFINED;
@@ -325,7 +327,12 @@ class Schema extends AbstractAnnotation
* Allows sending a null value for the defined schema.
* Default value is false.
*
+ * This must not be used when using OpenApi version 3.1,
+ * instead make the "type" property an array and add "null" as a possible type.
+ *
* @var bool
+ *
+ * @see https://www.openapis.org/blog/2021/02/16/migrating-from-openapi-3-0-to-3-1-0
*/
public $nullable = Generator::UNDEFINED;
diff --git a/src/Annotations/SecurityScheme.php b/src/Annotations/SecurityScheme.php
index 2d30e5bc6..631fcec8b 100644
--- a/src/Annotations/SecurityScheme.php
+++ b/src/Annotations/SecurityScheme.php
@@ -34,7 +34,7 @@ class SecurityScheme extends AbstractAnnotation
/**
* The type of the security scheme.
*
- * @var string
+ * @var string|non-empty-array
*/
public $type = Generator::UNDEFINED;
diff --git a/src/Attributes/AdditionalProperties.php b/src/Attributes/AdditionalProperties.php
index ae9ff9445..c3e275dad 100644
--- a/src/Attributes/AdditionalProperties.php
+++ b/src/Attributes/AdditionalProperties.php
@@ -12,6 +12,7 @@
class AdditionalProperties extends \OpenApi\Annotations\AdditionalProperties
{
/**
+ * @param string|non-empty-array|null $type
* @param string|class-string|object|null $ref
* @param string[] $required
* @param Property[] $properties
@@ -34,7 +35,7 @@ public function __construct(
?int $minProperties = null,
?array $required = null,
?array $properties = null,
- ?string $type = null,
+ string|array|null $type = null,
?string $format = null,
?Items $items = null,
?string $collectionFormat = null,
diff --git a/src/Attributes/Items.php b/src/Attributes/Items.php
index 8ab82d967..6659fc855 100644
--- a/src/Attributes/Items.php
+++ b/src/Attributes/Items.php
@@ -12,6 +12,7 @@
class Items extends \OpenApi\Annotations\Items
{
/**
+ * @param string|non-empty-array|null $type
* @param string|class-string|object|null $ref
* @param string[] $required
* @param Property[] $properties
@@ -34,7 +35,7 @@ public function __construct(
?int $minProperties = null,
?array $required = null,
?array $properties = null,
- ?string $type = null,
+ string|array|null $type = null,
?string $format = null,
?Items $items = null,
?string $collectionFormat = null,
diff --git a/src/Attributes/JsonContent.php b/src/Attributes/JsonContent.php
index 2b35183db..342044c78 100644
--- a/src/Attributes/JsonContent.php
+++ b/src/Attributes/JsonContent.php
@@ -12,6 +12,7 @@
class JsonContent extends \OpenApi\Annotations\JsonContent
{
/**
+ * @param string|non-empty-array|null $type
* @param string|class-string|object|null $ref
* @param array $examples
* @param string[] $required
@@ -36,7 +37,7 @@ public function __construct(
?int $minProperties = null,
?array $required = null,
?array $properties = null,
- ?string $type = null,
+ string|array|null $type = null,
?string $format = null,
?Items $items = null,
?string $collectionFormat = null,
diff --git a/src/Attributes/Property.php b/src/Attributes/Property.php
index d30af6b43..e27372d6b 100644
--- a/src/Attributes/Property.php
+++ b/src/Attributes/Property.php
@@ -12,6 +12,7 @@
class Property extends \OpenApi\Annotations\Property
{
/**
+ * @param string|non-empty-array|null $type
* @param string|class-string|object|null $ref
* @param string[] $required
* @param Property[] $properties
@@ -35,7 +36,7 @@ public function __construct(
?int $minProperties = null,
?array $required = null,
?array $properties = null,
- ?string $type = null,
+ string|array|null $type = null,
?string $format = null,
?Items $items = null,
?string $collectionFormat = null,
diff --git a/src/Attributes/Schema.php b/src/Attributes/Schema.php
index 1998f274a..f09c5e3c0 100644
--- a/src/Attributes/Schema.php
+++ b/src/Attributes/Schema.php
@@ -12,6 +12,7 @@
class Schema extends \OpenApi\Annotations\Schema
{
/**
+ * @param string|non-empty-array|null $type
* @param string|class-string|object|null $ref
* @param string[] $required
* @param Property[] $properties
@@ -35,7 +36,7 @@ public function __construct(
?int $minProperties = null,
?array $required = null,
?array $properties = null,
- ?string $type = null,
+ string|array|null $type = null,
?string $format = null,
?Items $items = null,
?string $collectionFormat = null,
diff --git a/src/Attributes/SecurityScheme.php b/src/Attributes/SecurityScheme.php
index ea98cab90..e43c274bd 100644
--- a/src/Attributes/SecurityScheme.php
+++ b/src/Attributes/SecurityScheme.php
@@ -12,15 +12,16 @@
class SecurityScheme extends \OpenApi\Annotations\SecurityScheme
{
/**
- * @param string|class-string|object|null $ref
- * @param Flow[] $flows
- * @param array|null $x
- * @param Attachable[]|null $attachables
+ * @param string|non-empty-array|null $type
+ * @param string|class-string|object|null $ref
+ * @param Flow[] $flows
+ * @param array|null $x
+ * @param Attachable[]|null $attachables
*/
public function __construct(
string|object|null $ref = null,
?string $securityScheme = null,
- ?string $type = null,
+ string|array|null $type = null,
?string $description = null,
?string $name = null,
?string $in = null,
diff --git a/src/Attributes/XmlContent.php b/src/Attributes/XmlContent.php
index f7a416478..dd944cab1 100644
--- a/src/Attributes/XmlContent.php
+++ b/src/Attributes/XmlContent.php
@@ -12,6 +12,7 @@
class XmlContent extends \OpenApi\Annotations\XmlContent
{
/**
+ * @param string|non-empty-array|null $type
* @param string|class-string|object|null $ref
* @param array $examples
* @param string[] $required
@@ -36,7 +37,7 @@ public function __construct(
?int $minProperties = null,
?array $required = null,
?array $properties = null,
- ?string $type = null,
+ string|array|null $type = null,
?string $format = null,
?Items $items = null,
?string $collectionFormat = null,
diff --git a/src/Processors/AugmentProperties.php b/src/Processors/AugmentProperties.php
index 395ac8c28..68e575c0f 100644
--- a/src/Processors/AugmentProperties.php
+++ b/src/Processors/AugmentProperties.php
@@ -51,7 +51,9 @@ public function __invoke(Analysis $analysis)
if (Generator::isDefault($property->type)) {
$this->augmentType($analysis, $property, $context, $refs, $typeAndDescription['type']);
} else {
- $this->mapNativeType($property, $property->type);
+ if (!is_array($property->type)) {
+ $this->mapNativeType($property, $property->type);
+ }
}
if (Generator::isDefault($property->description) && $typeAndDescription['description']) {
@@ -129,7 +131,7 @@ protected function augmentType(Analysis $analysis, OA\Property $property, Contex
if (Generator::isDefault($property->ref) && array_key_exists($refKey, $refs)) {
$this->applyRef($analysis, $property, $refs[$refKey]);
} else {
- if ($typeSchema = $analysis->getSchemaForSource($context->type)) {
+ if (is_string($context->type) && $typeSchema = $analysis->getSchemaForSource($context->type)) {
if (Generator::isDefault($property->format)) {
$property->ref = OA\Components::ref($typeSchema);
$property->type = Generator::UNDEFINED;
diff --git a/src/Processors/AugmentSchemas.php b/src/Processors/AugmentSchemas.php
index 47952b0c6..a07b2aa30 100644
--- a/src/Processors/AugmentSchemas.php
+++ b/src/Processors/AugmentSchemas.php
@@ -103,7 +103,7 @@ protected function augmentType(Analysis $analysis, array $schemas): void
$schema->type = 'object';
}
} else {
- if ($typeSchema = $analysis->getSchemaForSource($schema->type)) {
+ if (is_string($schema->type) && $typeSchema = $analysis->getSchemaForSource($schema->type)) {
if (Generator::isDefault($schema->format)) {
$schema->ref = OA\Components::ref($typeSchema);
$schema->type = Generator::UNDEFINED;
diff --git a/tests/Fixtures/Scratch/Nullable.php b/tests/Fixtures/Scratch/Nullable.php
index f86289285..f840e646f 100644
--- a/tests/Fixtures/Scratch/Nullable.php
+++ b/tests/Fixtures/Scratch/Nullable.php
@@ -44,6 +44,9 @@ class Nullable
#[OAT\Property]
public MyDateTime|null $anotherdate;
+
+ #[OAT\Property(type: ['string', 'null'])]
+ public ?string $description;
}
#[OAT\Get(
diff --git a/tests/Fixtures/Scratch/Nullable.yaml b/tests/Fixtures/Scratch/Nullable.yaml
index 0971e208a..b6c546ce1 100644
--- a/tests/Fixtures/Scratch/Nullable.yaml
+++ b/tests/Fixtures/Scratch/Nullable.yaml
@@ -47,4 +47,8 @@ components:
$ref: '#/components/schemas/MyDateTime'
-
type: 'null'
+ description:
+ type:
+ - string
+ - 'null'
type: object
diff --git a/tests/Fixtures/Scratch/Types31.php b/tests/Fixtures/Scratch/Types31.php
new file mode 100644
index 000000000..04a473f0b
--- /dev/null
+++ b/tests/Fixtures/Scratch/Types31.php
@@ -0,0 +1,33 @@
+