diff --git a/.gitignore b/.gitignore index c86af6f..00d6c0f 100644 --- a/.gitignore +++ b/.gitignore @@ -8,3 +8,4 @@ /.phpunit.result.cache php-cs-fixer.phar +xdebug_remote.log diff --git a/Makefile b/Makefile index ee99b2a..becbf1d 100644 --- a/Makefile +++ b/Makefile @@ -15,6 +15,9 @@ DOCKER_PHP=docker-compose run --rm php DOCKER_NODE=docker-compose run --rm -w /app node endif +CONTAINER_NAME=$$(echo $$(pwd) | tr / _ | cut -c 2-) # lets have unique container name when dealing with lot of forks +UID=$(shell id -u) + all: @echo "the following commands are available:" @echo "" @@ -32,6 +35,12 @@ all: check-style: php-cs-fixer.phar PHP_CS_FIXER_IGNORE_ENV=1 ./php-cs-fixer.phar fix src/ --diff --dry-run +cli: + COMPOSE_PROJECT_NAME=$(CONTAINER_NAME) docker-compose exec --user=$(UID) php bash # lets have unique container name when dealing with lot of forks + +cli_root: + COMPOSE_PROJECT_NAME=$(CONTAINER_NAME) docker-compose exec --user="root" php bash + fix-style: php-cs-fixer.phar $(DOCKER_PHP) vendor/bin/indent --tabs composer.json $(DOCKER_PHP) vendor/bin/indent --spaces .php_cs.dist @@ -81,5 +90,13 @@ coverage: .php-openapi-covA .php-openapi-covB .php-openapi-covB: grep -rhPo '^class \w+' src/spec/ | awk '{print $$2}' |grep -v '^Type$$' | sort > $@ -.PHONY: all check-style fix-style install test lint coverage +build-docker: + COMPOSE_PROJECT_NAME=$(CONTAINER_NAME) docker-compose build + +start-docker: + COMPOSE_PROJECT_NAME=$(CONTAINER_NAME) docker-compose up -d + +stop-docker: + COMPOSE_PROJECT_NAME=$(CONTAINER_NAME) docker-compose down --remove-orphans +.PHONY: all check-style fix-style install test lint coverage build-docker start-docker stop-docker diff --git a/doc/array merge recursive distinct.md b/doc/array merge recursive distinct.md new file mode 100644 index 0000000..a29881e --- /dev/null +++ b/doc/array merge recursive distinct.md @@ -0,0 +1,69 @@ +### array merge recursive distinct + +While resolving `allOf`s (pull request https://github.com/cebe/php-openapi/pull/208), if a duplicate property is found + +```yaml +components: + schemas: + User: + type: object + required: + - id + - name # <-------------------------------------------------------------- + properties: + id: + type: integer + name: # <-------------------------------------------------------------- + type: string + maxLength: 10 # <-------------------------------------------------------------- + Pet: + type: object + required: + - id2 + - name # <-------------------------------------------------------------- + properties: + id2: + type: integer + name: # <-------------------------------------------------------------- + type: string + maxLength: 12 # <-------------------------------------------------------------- + Post: + type: object + properties: + id: + type: integer + content: + type: string + user: + allOf: + - $ref: '#/components/schemas/User' + - $ref: '#/components/schemas/Pet' + - x-faker: true +``` + +then property from the last component schema will be considered: + +```yaml +Post: + type: object + properties: + id: + type: integer + content: + type: string + user: + type: object + required: + - id + - name # <-------------------------------------------------------------- + - id2 + properties: + id: + type: integer + name: # <-------------------------------------------------------------- + type: string + maxLength: 12 # <-------------------------------------------------------------- + id2: + type: integer + x-faker: true +``` \ No newline at end of file diff --git a/src/Helper.php b/src/Helper.php new file mode 100644 index 0000000..9e66674 --- /dev/null +++ b/src/Helper.php @@ -0,0 +1,58 @@ + and contributors + * @license https://github.com/cebe/php-openapi/blob/master/LICENSE + */ + +namespace cebe\openapi; + +/** + * Helper class containing widely used custom functions used in library + */ +class Helper +{ + /** + * Thanks https://www.php.net/manual/en/function.array-merge-recursive.php#96201 + * + * Merges any number of arrays / parameters recursively, replacing + * entries with string keys with values from latter arrays. + * If the entry or the next value to be assigned is an array, then it + * automagically treats both arguments as an array. + * Numeric entries are appended, not replaced, but only if they are + * unique + * + * Function call example: `$result = array_merge_recursive_distinct(a1, a2, ... aN);` + * More documentation is present at [array merge recursive distinct.md](../../../doc/array merge recursive distinct.md) file + * @return array + */ + public static function arrayMergeRecursiveDistinct() + { + $arrays = func_get_args(); + $base = array_shift($arrays); + if (!is_array($base)) { + $base = empty($base) ? [] : [$base]; + } + foreach ($arrays as $append) { + if (!is_array($append)) { + $append = [$append]; + } + foreach ($append as $key => $value) { + if (!array_key_exists($key, $base) and !is_numeric($key)) { + $base[$key] = $append[$key]; + continue; + } + if (is_array($value) || is_array($base[$key])) { + $base[$key] = static::arrayMergeRecursiveDistinct($base[$key], $append[$key]); + } elseif (is_numeric($key)) { + if (!in_array($value, $base)) { + $base[] = $value; + } + } else { + $base[$key] = $value; + } + } + } + return $base; + } +} diff --git a/src/Reader.php b/src/Reader.php index 99ee5c3..2568ddd 100644 --- a/src/Reader.php +++ b/src/Reader.php @@ -74,6 +74,10 @@ public static function readFromYaml(string $yaml, string $baseType = OpenApi::cl * Since version 1.5.0 this can be a string indicating the reference resolving mode: * - `inline` only resolve references to external files. * - `all` resolve all references except recursive references. + * @param bool $resolveAllOfs whether to automatically resolve all `allOf`s automatically. It will + * only work if [[$resolveReferences]] is `true` or [[ReferenceContext::RESOLVE_MODE_ALL]]. The + * concept of resolving all `allOf`s is explained at https://github.com/cebe/php-openapi/pull/208 + * in detail with example. * @return SpecObjectInterface|OpenApi the OpenApi object instance. * The type of the returned object depends on the `$baseType` argument. * @throws TypeErrorException in case invalid spec data is supplied. @@ -81,7 +85,7 @@ public static function readFromYaml(string $yaml, string $baseType = OpenApi::cl * @throws IOException when the file is not readable. * @throws InvalidJsonPointerSyntaxException in case an invalid JSON pointer string is passed to the spec references. */ - public static function readFromJsonFile(string $fileName, string $baseType = OpenApi::class, $resolveReferences = true): SpecObjectInterface + public static function readFromJsonFile(string $fileName, string $baseType = OpenApi::class, $resolveReferences = true, bool $resolveAllOfs = false): SpecObjectInterface { $fileContent = file_get_contents($fileName); if ($fileContent === false) { @@ -101,6 +105,9 @@ public static function readFromJsonFile(string $fileName, string $baseType = Ope } $spec->resolveReferences(); } + if ($resolveAllOfs && ($resolveReferences === true || $resolveReferences === ReferenceContext::RESOLVE_MODE_ALL)) { + $spec->resolveAllOf(); + } return $spec; } @@ -121,13 +128,14 @@ public static function readFromJsonFile(string $fileName, string $baseType = Ope * Since version 1.5.0 this can be a string indicating the reference resolving mode: * - `inline` only resolve references to external files. * - `all` resolve all references except recursive references. + * @param bool $resolveAllOfs whether to automatically resolve all `allOf`s automatically. It will only work if [[$resolveReferences]] is `true` or [[ReferenceContext::RESOLVE_MODE_ALL]]. The concept of resolving all `allOf`s is explained at https://github.com/cebe/php-openapi/pull/208 in detail with example. * @return SpecObjectInterface|OpenApi the OpenApi object instance. * The type of the returned object depends on the `$baseType` argument. * @throws TypeErrorException in case invalid spec data is supplied. * @throws UnresolvableReferenceException in case references could not be resolved. * @throws IOException when the file is not readable. */ - public static function readFromYamlFile(string $fileName, string $baseType = OpenApi::class, $resolveReferences = true): SpecObjectInterface + public static function readFromYamlFile(string $fileName, string $baseType = OpenApi::class, $resolveReferences = true, bool $resolveAllOfs = false): SpecObjectInterface { $fileContent = file_get_contents($fileName); if ($fileContent === false) { @@ -147,6 +155,9 @@ public static function readFromYamlFile(string $fileName, string $baseType = Ope } $spec->resolveReferences(); } + if ($resolveAllOfs && ($resolveReferences === true || $resolveReferences === ReferenceContext::RESOLVE_MODE_ALL)) { + $spec->resolveAllOf(); + } return $spec; } } diff --git a/src/SpecBaseObject.php b/src/SpecBaseObject.php index 1de429b..a3fadf6 100644 --- a/src/SpecBaseObject.php +++ b/src/SpecBaseObject.php @@ -12,6 +12,7 @@ use cebe\openapi\json\JsonPointer; use cebe\openapi\json\JsonReference; use cebe\openapi\spec\Reference; +use cebe\openapi\spec\Schema; use cebe\openapi\spec\Type; /** @@ -31,6 +32,7 @@ abstract class SpecBaseObject implements SpecObjectInterface, DocumentContextInt private $_recursingReferences = false; private $_recursingReferenceContext = false; private $_recursingDocumentContext = false; + private $_recursingAllOf = false; private $_baseDocument; private $_jsonPointer; @@ -525,4 +527,86 @@ public function getExtensions(): array } return $extensions; } + + public function getProperties(): array + { + return $this->_properties; + } + + public function mergeProperties($properties) + { + $this->_properties = Helper::arrayMergeRecursiveDistinct($this->_properties, $properties); + } + + public function resolveAllOf() + { + // avoid recursion to get stuck in a loop + if ($this->_recursingAllOf) { + return; + } + $this->_recursingAllOf = true; + + foreach ($this->_properties as $property => $value) { + $this->handleMergingOfAllAllOfs($property, $value); + $this->removeAllOfKey($property, $value); + } + $this->_recursingAllOf = false; + } + + private function mergeAllAllOfsInToSingleObject(): self + { + $allOfs = $this->allOf; + /** @var static $first */ + $first = $this->allOf[0]; + unset($allOfs[0]); + foreach ($allOfs as $allOf) { + /** @var Schema $allOf */ + $first->mergeProperties($allOf->getProperties()); + } + return $first; + } + + private function handleMergingOfAllAllOfs(string $property, $value): void + { + if ($property === 'allOf' && !empty($value)) { + $this->_properties[$property] = $this->mergeAllAllOfsInToSingleObject(); + } elseif ($value instanceof SpecObjectInterface && method_exists($value, 'resolveAllOf')) { + $value->resolveAllOf(); + } elseif (is_array($value)) { + foreach ($value as $k => $item) { + if ($k === 'allOf' && !empty($item)) { + $this->_properties[$property][$k] = $this->mergeAllAllOfsInToSingleObject(); + } elseif ($item instanceof SpecObjectInterface && method_exists($item, 'resolveAllOf')) { + $item->resolveAllOf(); + } + } + } + } + + private function removeAllOfKey(string $property, $value): void + { + if ($property === 'properties' && !empty($value)) { + foreach ($value as $k => $v) { + if (!empty($v->allOf)) { + $temp = $v->allOf; + $this->_properties[$property][$k] = $temp; + } + } + } elseif ($value instanceof SpecObjectInterface && method_exists($value, 'resolveAllOf')) { + $value->resolveAllOf(); + } elseif (is_array($value)) { + foreach ($value as $arrayValueKey => $item) { + if ($arrayValueKey === 'properties' && !empty($item)) { + foreach ($item as $itemKey => $itemValue) { + if (!empty($itemValue->allOf)) { + $tempIn = $itemValue->allOf; + $this->_properties[$property][$arrayValueKey][$itemKey] = $tempIn; + } + } + } elseif ($item instanceof SpecObjectInterface && method_exists($item, 'resolveAllOf')) { + $item->resolveAllOf(); + } + } + } + } } diff --git a/src/spec/OpenApi.php b/src/spec/OpenApi.php index 29d38b3..d22cd57 100644 --- a/src/spec/OpenApi.php +++ b/src/spec/OpenApi.php @@ -7,7 +7,6 @@ namespace cebe\openapi\spec; -use cebe\openapi\exceptions\TypeErrorException; use cebe\openapi\SpecBaseObject; /** diff --git a/tests/HelperTest.php b/tests/HelperTest.php new file mode 100644 index 0000000..2794223 --- /dev/null +++ b/tests/HelperTest.php @@ -0,0 +1,92 @@ +assertSame(['id', 'name', 'id2', 'name2'], $result); + + $result = Helper::arrayMergeRecursiveDistinct(['id', 'name'], ['id2', 'name']); + $this->assertSame(['id', 'name', 'id2'], $result); + + $result = Helper::arrayMergeRecursiveDistinct(['type' => 'object'], ['x-faker' => true]); + $this->assertSame(['type' => 'object', 'x-faker' => true], $result); + + $result = Helper::arrayMergeRecursiveDistinct([ + 'properties' => [ + 'id' => [ + 'type' => 'integer' + ], + 'name' => [ + 'type' => 'string' + ], + ] + ], [ + 'properties' => [ + 'id2' => [ + 'type' => 'integer' + ], + 'name2' => [ + 'type' => 'string' + ], + ] + ]); + $this->assertSame([ + 'properties' => [ + 'id' => [ + 'type' => 'integer' + ], + 'name' => [ + 'type' => 'string' + ], + 'id2' => [ + 'type' => 'integer' + ], + 'name2' => [ + 'type' => 'string' + ], + ] + ], $result); + + $result = Helper::arrayMergeRecursiveDistinct([ + 'properties' => [ + 'id' => [ + 'type' => 'integer' + ], + 'name' => [ + 'type' => 'string', + 'maxLength' => 10 + ], + ] + ], [ + 'properties' => [ + 'id2' => [ + 'type' => 'integer' + ], + 'name' => [ + 'type' => 'string', + 'maxLength' => 12 + ], + ] + ]); + $this->assertSame([ + 'properties' => [ + 'id' => [ + 'type' => 'integer' + ], + 'name' => [ + 'type' => 'string', + 'maxLength' => 12 + ], + 'id2' => [ + 'type' => 'integer' + ] + ] + ], $result); + } + +} diff --git a/tests/spec/OpenApiTest.php b/tests/spec/OpenApiTest.php index 20b568e..a089442 100644 --- a/tests/spec/OpenApiTest.php +++ b/tests/spec/OpenApiTest.php @@ -230,6 +230,7 @@ public function testSpecs($openApiFile) if ($openapi->externalDocs !== null) { $this->assertInstanceOf(\cebe\openapi\spec\ExternalDocumentation::class, $openapi->externalDocs); } - } + + } diff --git a/tests/spec/SchemaTest.php b/tests/spec/SchemaTest.php index 1600b3b..7d49b71 100644 --- a/tests/spec/SchemaTest.php +++ b/tests/spec/SchemaTest.php @@ -1,8 +1,9 @@ assertEquals('string', $person->properties['name']->type); $this->assertEquals('string', $person->properties['$ref']->type); } + + // https://github.com/cebe/yii2-openapi/issues/165 + public function test165ResolveAllOf() + { + $unresolvedAllOfOpenApi = Reader::readFromYamlFile(__DIR__ . '/data/resolve_all_of.yml'); + $this->assertInstanceOf(SpecBaseObject::class, $unresolvedAllOfOpenApi->components->schemas['Post']->properties['user']); + $this->assertIsArray($unresolvedAllOfOpenApi->components->schemas['Post']->properties['user']->allOf); + $this->assertNotEmpty($unresolvedAllOfOpenApi->components->schemas['Post']->properties['user']->allOf); + + $openApi = Reader::readFromYamlFile(__DIR__ . '/data/resolve_all_of.yml', OpenApi::class, true, true); + $result = $openApi->validate(); + $this->assertTrue($result); + $this->assertEquals([], $openApi->getErrors()); + + $this->assertInstanceOf(SpecBaseObject::class, $openApi->components->schemas['Post']->properties['user']); + $this->assertObjectNotHasProperty('allOf', $openApi->components->schemas['Post']->properties['user']); + + $this->assertFalse($openApi->components->schemas['Post']->properties['user']->{'x-faker'}); + $this->assertTrue($openApi->components->schemas['Post']->properties['user']->{'x-faker2'}); + $expected = require_once __DIR__ . '/data/resolve_all_of_expected.php'; + +// $this->assertSame( +// json_decode(json_encode($openApi->components->schemas['Post']->getSerializableData()), true) +// , [] +// ); + + $this->assertSame( + json_decode(json_encode($openApi->components->schemas['Post']->getSerializableData()), true) + , $expected + ); + } + + // https://github.com/cebe/yii2-openapi/issues/165 + public function test165ResolveNestedAllOfWithReference() + { + $openApi = Reader::readFromYamlFile(__DIR__ . '/data/resolve_nested_all_of_with_reference.yml', OpenApi::class, true, true); + $result = $openApi->validate(); + $this->assertTrue($result); + $this->assertEquals([], $openApi->getErrors()); + + $expected = require_once __DIR__ . '/data/resolve_nested_all_of_with_reference.php'; + $this->assertSame( + json_decode(json_encode($openApi->components->schemas['Pet']->getSerializableData()), true) + , $expected + ); + } } diff --git a/tests/spec/data/resolve_all_of.yml b/tests/spec/data/resolve_all_of.yml new file mode 100644 index 0000000..a8e58fa --- /dev/null +++ b/tests/spec/data/resolve_all_of.yml @@ -0,0 +1,71 @@ + +openapi: 3.0.3 + +info: + title: Resolve allOf + version: 1.0.0 + +components: + schemas: + User: + type: object + required: + - id + - name + properties: + id: + type: integer + name: + type: string + maxLength: 10 + Pet: + type: object + required: + - id2 + - name2 + properties: + id2: + type: integer + name2: + type: string + maxLength: 12 + Fruit: + type: object + description: The Fruit description + required: + - id3 + - name + properties: + id3: + type: integer + name: + type: string + maxLength: 14 + Post: + type: object + properties: + id: + type: integer + content: + type: string + user: + allOf: + - $ref: '#/components/schemas/User' + - $ref: '#/components/schemas/Pet' + - $ref: '#/components/schemas/Fruit' + - x-faker: false + - x-faker2: true + - type: object + properties: + id4: + type: integer + name4: + type: string + - description: The last user description + +paths: + '/': + get: + responses: + '200': + description: OK diff --git a/tests/spec/data/resolve_all_of_expected.php b/tests/spec/data/resolve_all_of_expected.php new file mode 100644 index 0000000..02cef27 --- /dev/null +++ b/tests/spec/data/resolve_all_of_expected.php @@ -0,0 +1,52 @@ + 'object', + 'properties' => [ + 'id' => [ + 'type' => 'integer' + ], + 'content' => [ + 'type' => 'string' + ], + 'user' => [ + 'required' => [ + 'id', + 'name', + 'id2', + 'name2', + 'id3' + ], + 'type' => 'object', + 'properties' => [ + 'id' => [ + 'type' => 'integer' + ], + 'name' => [ + 'maxLength' => 14, + 'type' => 'string' + ], + 'id2' => [ + 'type' => 'integer' + ], + 'name2' => [ + 'maxLength' => 12, + 'type' => 'string' + ], + 'id3' => [ + 'type' => 'integer' + ], + 'id4' => [ + 'type' => 'integer' + ], + 'name4' => [ + 'type' => 'string' + ] + + ], + 'description' => 'The last user description', + 'x-faker' => false, + 'x-faker2' => true, + ], + ], +]; diff --git a/tests/spec/data/resolve_nested_all_of_with_reference.php b/tests/spec/data/resolve_nested_all_of_with_reference.php new file mode 100644 index 0000000..ea1bea2 --- /dev/null +++ b/tests/spec/data/resolve_nested_all_of_with_reference.php @@ -0,0 +1,60 @@ + [ + 'id2', + 'name', + ], + 'type' => 'object', + 'properties' => [ + 'id2' => [ + 'type' => 'integer', + ], + 'name' => [ + 'maxLength' => 12, + 'type' => 'string', + ], + 'physical' => [ + 'type' => 'object', + 'properties' => [ + 'weight' => [ + 'type' => 'integer', + ], + 'dimension' => [ + 'type' => 'object', + 'properties' => [ + 'height' => [ + 'type' => 'integer', + ], + 'length' => [ + 'type' => 'integer', + ], + 'width' => [ + 'type' => 'integer', + ], + 'miscellaneous' => [ + 'type' => 'object', + 'properties' => [ + 'owner' => [ + 'required' => [ + 0 => 'id', + 1 => 'name', + ], + 'type' => 'object', + 'properties' => [ + 'id' => [ + 'type' => 'integer', + ], + 'name' => [ + 'maxLength' => 10, + 'type' => 'string', + ], + ], + ], + ], + ], + ], + ], + ], + ], + ], +]; \ No newline at end of file diff --git a/tests/spec/data/resolve_nested_all_of_with_reference.yml b/tests/spec/data/resolve_nested_all_of_with_reference.yml new file mode 100644 index 0000000..2ca4467 --- /dev/null +++ b/tests/spec/data/resolve_nested_all_of_with_reference.yml @@ -0,0 +1,59 @@ +openapi: 3.0.3 + +info: + title: resolve_nested_all_of_with_reference + version: 1.0.0 + +components: + schemas: + User: + type: object + required: + - id + - name + properties: + id: + type: integer + name: + type: string + maxLength: 10 + + Pet: + type: object + required: + - id2 + - name + properties: + id2: + type: integer + name: + type: string + maxLength: 12 + physical: + allOf: + - type: object + properties: + weight: + type: integer + dimension: + allOf: + - type: object + properties: + height: + type: integer + length: + type: integer + width: + type: integer + miscellaneous: + type: object + properties: + owner: + $ref: '#/components/schemas/User' + +paths: + '/': + get: + responses: + '200': + description: OK