Skip to content

Commit

Permalink
Refactor and cleanup
Browse files Browse the repository at this point in the history
  • Loading branch information
DerManoMann committed Nov 1, 2022
1 parent fe02aba commit cf6d828
Show file tree
Hide file tree
Showing 16 changed files with 277 additions and 75 deletions.
2 changes: 2 additions & 0 deletions Examples/polymorphism/polymorphism-3.1.0.yaml
Original file line number Diff line number Diff line change
Expand Up @@ -45,6 +45,7 @@ components:
-
$ref: '#/components/schemas/EmployeeResponsible'
EmployeeResponsible:
type: object
allOf:
-
$ref: '#/components/schemas/Responsible'
Expand All @@ -56,6 +57,7 @@ components:
type: string
const: Virtual
FlResponsible:
type: object
allOf:
-
$ref: '#/components/schemas/Responsible'
Expand Down
2 changes: 2 additions & 0 deletions Examples/polymorphism/polymorphism.yaml
Original file line number Diff line number Diff line change
Expand Up @@ -46,6 +46,7 @@ components:
-
$ref: '#/components/schemas/EmployeeResponsible'
EmployeeResponsible:
type: object
allOf:
-
$ref: '#/components/schemas/Responsible'
Expand All @@ -59,6 +60,7 @@ components:
enum:
- Virtual
FlResponsible:
type: object
allOf:
-
$ref: '#/components/schemas/Responsible'
Expand Down
1 change: 1 addition & 0 deletions Examples/using-interfaces/using-interfaces.yaml
Original file line number Diff line number Diff line change
Expand Up @@ -67,6 +67,7 @@ components:
example: blue
Product:
title: 'Product model'
type: object
allOf:
-
$ref: '#/components/schemas/ProductInterface'
Expand Down
4 changes: 4 additions & 0 deletions Examples/using-traits/using-traits.yaml
Original file line number Diff line number Diff line change
Expand Up @@ -46,6 +46,7 @@ components:
schemas:
BellsAndWhistles:
title: 'Bells and Whistles trait'
type: object
allOf:
-
$ref: '#/components/schemas/Bells'
Expand Down Expand Up @@ -86,6 +87,7 @@ components:
type: object
Product:
title: 'Product model'
type: object
allOf:
-
$ref: '#/components/schemas/Colour'
Expand All @@ -103,6 +105,7 @@ components:
example: gong
SimpleProduct:
title: 'SimpleProduct model'
type: object
allOf:
-
$ref: '#/components/schemas/Bells'
Expand All @@ -115,6 +118,7 @@ components:
example: 1
TrickyProduct:
title: 'TrickyProduct model'
type: object
allOf:
-
$ref: '#/components/schemas/SimpleProduct'
Expand Down
22 changes: 0 additions & 22 deletions src/Processors/AugmentSchemas.php
Original file line number Diff line number Diff line change
Expand Up @@ -57,28 +57,6 @@ public function __invoke(Analysis $analysis)
continue;
}

if (!Generator::isDefault($annotation->allOf)) {
$schema = null;
foreach ($annotation->allOf as $nestedSchema) {
if (!Generator::isDefault($nestedSchema->ref)) {
continue;
}

$schema = $nestedSchema;
}

if ($schema === null) {
$schema = new OA\Schema([
'_context' => new Context(['generated' => true], $annotation->_context),
]);
$analysis->addAnnotation($schema, $schema->_context);
$annotation->allOf[] = $schema;
}

$schema->merge([$property], true);
break;
}

$annotation->merge([$property], true);
break;
}
Expand Down
22 changes: 6 additions & 16 deletions src/Processors/Concerns/MergePropertiesTrait.php
Original file line number Diff line number Diff line change
Expand Up @@ -35,21 +35,11 @@ protected function inheritFrom(Analysis $analysis, OA\Schema $schema, OA\Schema
$analysis->addAnnotation($refSchema, $refSchema->_context);
}

protected function mergeAnnotations(OA\Schema $schema, array $annotations, array &$existing): void
{
foreach ($annotations as $annotation) {
if ($annotation instanceof OA\Property && !in_array($annotation->_context->property, $existing, true)) {
$existing[] = $annotation->_context->property;
$schema->merge([$annotation], true);
}
}
}

protected function mergeProperties(OA\Schema $schema, array $from, array &$existing): void
{
foreach ($from['properties'] as $method) {
if (is_iterable($method->annotations)) {
foreach ($method->annotations as $annotation) {
foreach ($from['properties'] as $context) {
if (is_iterable($context->annotations)) {
foreach ($context->annotations as $annotation) {
if ($annotation instanceof OA\Property && !in_array($annotation->_context->property, $existing, true)) {
$existing[] = $annotation->_context->property;
$schema->merge([$annotation], true);
Expand All @@ -61,9 +51,9 @@ protected function mergeProperties(OA\Schema $schema, array $from, array &$exist

protected function mergeMethods(OA\Schema $schema, array $from, array &$existing): void
{
foreach ($from['methods'] as $method) {
if (is_iterable($method->annotations)) {
foreach ($method->annotations as $annotation) {
foreach ($from['methods'] as $context) {
if (is_iterable($context->annotations)) {
foreach ($context->annotations as $annotation) {
if ($annotation instanceof OA\Property && !in_array($annotation->_context->property, $existing, true)) {
$existing[] = $annotation->_context->property;
$schema->merge([$annotation], true);
Expand Down
12 changes: 1 addition & 11 deletions src/Processors/ExpandClasses.php
Original file line number Diff line number Diff line change
Expand Up @@ -9,16 +9,14 @@
use OpenApi\Analysis;
use OpenApi\Annotations as OA;
use OpenApi\Attributes as OAT;
use OpenApi\Context;
use OpenApi\Generator;

/**
* Iterate over the chain of ancestors of a schema and:
* - if the ancestor has a schema
* => inherit from the ancestor if it has a schema (allOf) and stop.
* - else
* => merge ancestor properties into the schema
* => merge ancestor trait properties into the schema (recursively).
* => merge ancestor properties into the schema.
*/
class ExpandClasses
{
Expand All @@ -44,14 +42,6 @@ public function __invoke(Analysis $analysis)
} else {
$this->mergeMethods($schema, $ancestor, $existing);
$this->mergeProperties($schema, $ancestor, $existing);

$traits = $analysis->getTraitsOfClass($schema->_context->fullyQualifiedName($ancestor['class']), true);
foreach ($traits as $definition) {
/** @var Context $context */
foreach ($definition['properties'] as $context) {
$this->mergeAnnotations($schema, $context->annotations, $existing);
}
}
}
}
}
Expand Down
41 changes: 37 additions & 4 deletions src/Processors/ExpandTraits.php
Original file line number Diff line number Diff line change
Expand Up @@ -25,18 +25,51 @@ public function __invoke(Analysis $analysis)
/** @var OA\Schema[] $schemas */
$schemas = $analysis->getAnnotationsOfType([OA\Schema::class, OAT\Schema::class], true);

// do regular trait inheritance / merge
foreach ($schemas as $schema) {
if ($schema->_context->is('class') || $schema->_context->is('trait')) {
$source = $schema->_context->class ?: $schema->_context->trait;
$traits = $analysis->getTraitsOfClass($schema->_context->fullyQualifiedName($source), true);
if ($schema->_context->is('trait')) {
$traits = $analysis->getTraitsOfClass($schema->_context->fullyQualifiedName($schema->_context->trait), true);
$existing = [];
foreach ($traits as $trait) {
$traitSchema = $analysis->getSchemaForSource($trait['context']->fullyQualifiedName($trait['trait']));
if ($traitSchema) {
$refPath = !Generator::isDefault($traitSchema->schema) ? $traitSchema->schema : $trait['trait'];
$this->inheritFrom($analysis, $schema, $traitSchema, $refPath, $trait['context']);
} else {
if ($schema->_context->is('class')) {
$this->mergeMethods($schema, $trait, $existing);
$this->mergeProperties($schema, $trait, $existing);
}
}
}
}

foreach ($schemas as $schema) {
if ($schema->_context->is('class') && !$schema->_aux) {
// look at class traits
$traits = $analysis->getTraitsOfClass($schema->_context->fullyQualifiedName($schema->_context->class), true);
$existing = [];
foreach ($traits as $trait) {
$traitSchema = $analysis->getSchemaForSource($trait['context']->fullyQualifiedName($trait['trait']));
if ($traitSchema) {
$refPath = !Generator::isDefault($traitSchema->schema) ? $traitSchema->schema : $trait['trait'];
$this->inheritFrom($analysis, $schema, $traitSchema, $refPath, $trait['context']);
} else {
$this->mergeMethods($schema, $trait, $existing);
$this->mergeProperties($schema, $trait, $existing);
}
}

// also merge ancestor traits of non schema parents
$ancestors = $analysis->getSuperClasses($schema->_context->fullyQualifiedName($schema->_context->class));
$existing = [];
foreach ($ancestors as $ancestor) {
$ancestorSchema = $analysis->getSchemaForSource($ancestor['context']->fullyQualifiedName($ancestor['class']));
if ($ancestorSchema) {
// stop here as we inherit everything above
break;
} else {
$traits = $analysis->getTraitsOfClass($schema->_context->fullyQualifiedName($ancestor['class']), true);
foreach ($traits as $trait) {
$this->mergeMethods($schema, $trait, $existing);
$this->mergeProperties($schema, $trait, $existing);
}
Expand Down
5 changes: 3 additions & 2 deletions tests/Fixtures/Apis/basic.yaml
Original file line number Diff line number Diff line change
Expand Up @@ -129,6 +129,7 @@ components:
Product:
title: Product
description: 'A Product.'
type: object
allOf:
-
$ref: '#/components/schemas/NameTrait'
Expand All @@ -144,12 +145,12 @@ components:
example: null
colour:
$ref: '#/components/schemas/Colour'
releasedAt:
type: string
id:
description: 'The id.'
format: int64
example: 1
releasedAt:
type: string
kind:
description: 'The kind.'
type: string
Expand Down
37 changes: 20 additions & 17 deletions tests/Fixtures/Scratch/MergeTraits.php
Original file line number Diff line number Diff line change
@@ -1,25 +1,13 @@
<?php
<?php declare(strict_types=1);

/**
* @license Apache 2.0
*/

namespace OpenApi\Tests\Fixtures\Scratch;

use OpenApi\Annotations as OA;

/**
* @OA\Info(title="API", version="1.0")
*
* @OA\Get(
* path="/api/endpoint",
* @OA\Response(
* response=200,
* description="successful operation",
* @OA\JsonContent(ref="#/components/schemas/Address")
* )
* )
*/
class Endpoint
{
}

trait HasId
{
/**
Expand Down Expand Up @@ -74,3 +62,18 @@ class Address extends Model
/** @OA\Property */
public string $street;
}

/**
* @OA\Info(title="API", version="1.0")
* @OA\Get(
* path="/api/endpoint",
* @OA\Response(
* response=200,
* description="successful operation",
* @OA\JsonContent(ref="#/components/schemas/Address")
* )
* )
*/
class Endpoint
{
}
37 changes: 37 additions & 0 deletions tests/Fixtures/Scratch/MergeTraits.yaml
Original file line number Diff line number Diff line change
@@ -0,0 +1,37 @@
openapi: 3.0.0
info:
title: API
version: '1.0'
paths:
/api/endpoint:
get:
responses:
'200':
description: 'successful operation'
content:
application/json:
schema:
$ref: '#/components/schemas/Address'
components:
schemas:
Address:
required:
- street
properties:
id:
type: integer
format: int64
readOnly: true
created_at:
type: string
format: date-time
readOnly: true
updated_at:
type: string
format: date-time
readOnly: true
street:
type: string
type: object
xml:
name: Address
Loading

0 comments on commit cf6d828

Please sign in to comment.