2012-04-11 20:00:21 +02:00
|
|
|
|
<?php
|
|
|
|
|
|
2012-04-13 11:03:05 +02:00
|
|
|
|
/*
|
|
|
|
|
* This file is part of the NelmioApiDocBundle.
|
|
|
|
|
*
|
|
|
|
|
* (c) Nelmio <hello@nelm.io>
|
|
|
|
|
*
|
|
|
|
|
* For the full copyright and license information, please view the LICENSE
|
|
|
|
|
* file that was distributed with this source code.
|
|
|
|
|
*/
|
|
|
|
|
|
2012-04-12 18:37:42 +02:00
|
|
|
|
namespace Nelmio\ApiDocBundle\Formatter;
|
2012-04-11 20:00:21 +02:00
|
|
|
|
|
2012-04-12 18:37:42 +02:00
|
|
|
|
use Nelmio\ApiDocBundle\Annotation\ApiDoc;
|
Unified data types [actualType and subType]
This is the result of https://github.com/nelmio/NelmioApiDocBundle/issues/410.
This PR aims to provide a uniform way of declaring data-types of parameters for
parsers and handlers to follow. In turn, this would allow formatters to
determine data-types in a cleaner and less volatile manner. (See use-case that
can be improved with this PR:
https://github.com/nelmio/NelmioApiDocBundle/blob/master/Formatter/AbstractFormatter.php#L103)
This is possible by the addition two properties to each property item in
`response`, and `parameters` fields in each API endpoint produced by the
`ApiDocExtractor`:
* `actualType` Contains a value from one of the `DataTypes` class constants.
* `subType` Can contain either `null`, or any other `DataTypes` class constant.
This is relevant when the `actualType` is a `DataTypes::COLLECTION`, wherein
`subType` would specify the type of the collection items. It is also relevant
when `actualType` is a `DataTypes::MODEL`, wherein `subType` would contain an
identifier of the model (the FQCN or anything the parser would wish to specify)
Examples:
```php
array(
'id' => array(
'dataType' => 'integer',
'actualType' => DataTypes::INTEGER,
'subType' => null,
),
'profile' => array(
'dataType' => 'object (Profile)',
'actualType' => DataTypes::MODEL,
'subType' => 'Foo\Entity\Profile',
'children' => array(
'name' => array(
'dataType' => 'string',
'actualType' => DataTypes::STRING,
'subType' => null,
),
'birthDate' => array(
'dataType' => 'date',
'actualType' => DataTypes::DATE,
'subType' => null,
),
)
),
'languages' => array(
'dataType' => 'array of strings',
'actualType' => DataTypes::COLLECTION,
'subType' => DataTypes::STRING,
),
'roles' => array(
'dataType' => 'array of choices',
'actualType' => DataTypes::COLLECTION,
'subType' => DataTypes::ENUM,
),
'groups' => array(
'dataType' => 'array of objects (Group)',
'actualType' => DataTypes::COLLECTION,
'subType' => 'Foo\Entity\Group',
),
'profileRevisions' => array(
'dataType' => 'array of objects (Profile)',
'actualType' => DataTypes::COLLECTION,
'subType' => 'Foo\Entity\Profile',
),
'address' => array(
'dataType' => 'object (a_type_a_custom_JMS_serializer_handler_handles)',
'actualType' => DataTypes::MODEL,
'subType' => 'a_type_a_custom_JMS_serializer_handler_handles',
),
);
```
When a formatter omits the `dataType` property or leaves it blank, it is
inferred within `ApiDocExtractor` before everything is passed to formatters.
2014-06-17 17:05:00 -07:00
|
|
|
|
use Nelmio\ApiDocBundle\DataTypes;
|
2012-04-11 20:00:21 +02:00
|
|
|
|
|
2012-04-12 01:28:36 +02:00
|
|
|
|
abstract class AbstractFormatter implements FormatterInterface
|
2012-04-11 20:00:21 +02:00
|
|
|
|
{
|
2012-04-12 12:48:36 +02:00
|
|
|
|
/**
|
|
|
|
|
* {@inheritdoc}
|
|
|
|
|
*/
|
2012-07-20 00:58:58 +02:00
|
|
|
|
public function formatOne(ApiDoc $annotation)
|
2012-04-12 01:28:36 +02:00
|
|
|
|
{
|
2012-11-15 21:58:09 +01:00
|
|
|
|
return $this->renderOne(
|
|
|
|
|
$this->processAnnotation($annotation->toArray())
|
|
|
|
|
);
|
2012-04-12 01:28:36 +02:00
|
|
|
|
}
|
|
|
|
|
|
2012-04-12 12:48:36 +02:00
|
|
|
|
/**
|
|
|
|
|
* {@inheritdoc}
|
|
|
|
|
*/
|
|
|
|
|
public function format(array $collection)
|
|
|
|
|
{
|
2012-11-15 21:58:09 +01:00
|
|
|
|
return $this->render(
|
|
|
|
|
$this->processCollection($collection)
|
|
|
|
|
);
|
2012-04-12 12:48:36 +02:00
|
|
|
|
}
|
|
|
|
|
|
2012-04-12 17:24:38 +02:00
|
|
|
|
/**
|
|
|
|
|
* Format a single array of data
|
|
|
|
|
*
|
2012-05-23 00:33:01 +02:00
|
|
|
|
* @param array $data
|
2012-04-12 17:24:38 +02:00
|
|
|
|
* @return string|array
|
|
|
|
|
*/
|
2012-05-23 00:33:01 +02:00
|
|
|
|
abstract protected function renderOne(array $data);
|
2012-04-12 12:48:36 +02:00
|
|
|
|
|
2012-04-12 17:24:38 +02:00
|
|
|
|
/**
|
|
|
|
|
* Format a set of resource sections.
|
|
|
|
|
*
|
2012-05-23 00:33:01 +02:00
|
|
|
|
* @param array $collection
|
2012-04-12 17:24:38 +02:00
|
|
|
|
* @return string|array
|
|
|
|
|
*/
|
2012-05-23 00:33:01 +02:00
|
|
|
|
abstract protected function render(array $collection);
|
2012-08-31 11:56:48 -04:00
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* Compresses nested parameters into a flat by changing the parameter
|
|
|
|
|
* names to strings which contain the nested property names, for example:
|
|
|
|
|
* `user[group][name]`
|
|
|
|
|
*
|
|
|
|
|
*
|
|
|
|
|
* @param array $data
|
|
|
|
|
* @param string $parentName
|
|
|
|
|
* @param boolean $ignoreNestedReadOnly
|
|
|
|
|
* @return array
|
|
|
|
|
*/
|
|
|
|
|
protected function compressNestedParameters(array $data, $parentName = null, $ignoreNestedReadOnly = false)
|
|
|
|
|
{
|
|
|
|
|
$newParams = array();
|
|
|
|
|
foreach ($data as $name => $info) {
|
|
|
|
|
$newName = $this->getNewName($name, $info, $parentName);
|
|
|
|
|
|
|
|
|
|
$newParams[$newName] = array(
|
2014-06-27 10:07:03 +02:00
|
|
|
|
'dataType' => $info['dataType'],
|
|
|
|
|
'readonly' => array_key_exists('readonly', $info) ? $info['readonly'] : null,
|
|
|
|
|
'required' => $info['required'],
|
|
|
|
|
'default' => array_key_exists('default', $info) ? $info['default'] : null,
|
|
|
|
|
'description' => array_key_exists('description', $info) ? $info['description'] : null,
|
|
|
|
|
'format' => array_key_exists('format', $info) ? $info['format'] : null,
|
|
|
|
|
'sinceVersion' => array_key_exists('sinceVersion', $info) ? $info['sinceVersion'] : null,
|
|
|
|
|
'untilVersion' => array_key_exists('untilVersion', $info) ? $info['untilVersion'] : null,
|
|
|
|
|
'actualType' => array_key_exists('actualType', $info) ? $info['actualType'] : null,
|
|
|
|
|
'subType' => array_key_exists('subType', $info) ? $info['subType'] : null,
|
2012-08-31 11:56:48 -04:00
|
|
|
|
);
|
|
|
|
|
|
|
|
|
|
if (isset($info['children']) && (!$info['readonly'] || !$ignoreNestedReadOnly)) {
|
|
|
|
|
foreach ($this->compressNestedParameters($info['children'], $newName, $ignoreNestedReadOnly) as $nestedItemName => $nestedItemData) {
|
|
|
|
|
$newParams[$nestedItemName] = $nestedItemData;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return $newParams;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* Returns a new property name, taking into account whether or not the property
|
|
|
|
|
* is an array of some other data type.
|
|
|
|
|
*
|
|
|
|
|
* @param string $name
|
|
|
|
|
* @param array $data
|
|
|
|
|
* @param string $parentName
|
|
|
|
|
* @return string
|
|
|
|
|
*/
|
|
|
|
|
protected function getNewName($name, $data, $parentName = null)
|
|
|
|
|
{
|
2014-06-27 10:07:03 +02:00
|
|
|
|
$array = '';
|
2012-08-31 11:56:48 -04:00
|
|
|
|
$newName = ($parentName) ? sprintf("%s[%s]", $parentName, $name) : $name;
|
2014-06-27 10:07:03 +02:00
|
|
|
|
|
|
|
|
|
if (isset($data['actualType']) && $data['actualType'] == DataTypes::COLLECTION
|
|
|
|
|
&& isset($data['subType']) && $data['subType'] !== null
|
|
|
|
|
) {
|
|
|
|
|
$array = '[]';
|
|
|
|
|
}
|
2012-08-31 11:56:48 -04:00
|
|
|
|
|
|
|
|
|
return sprintf("%s%s", $newName, $array);
|
|
|
|
|
}
|
|
|
|
|
|
2012-11-15 21:58:09 +01:00
|
|
|
|
/**
|
|
|
|
|
* @param array $annotation
|
|
|
|
|
* @return array
|
|
|
|
|
*/
|
|
|
|
|
protected function processAnnotation($annotation)
|
|
|
|
|
{
|
|
|
|
|
if (isset($annotation['parameters'])) {
|
|
|
|
|
$annotation['parameters'] = $this->compressNestedParameters($annotation['parameters'], null, true);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (isset($annotation['response'])) {
|
|
|
|
|
$annotation['response'] = $this->compressNestedParameters($annotation['response']);
|
|
|
|
|
}
|
|
|
|
|
|
2013-09-05 19:04:17 +02:00
|
|
|
|
$annotation['id'] = strtolower($annotation['method']).'-'.str_replace('/', '-', $annotation['uri']);
|
|
|
|
|
|
2012-11-15 21:58:09 +01:00
|
|
|
|
return $annotation;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* @param array[ApiDoc] $collection
|
|
|
|
|
* @return array
|
|
|
|
|
*/
|
|
|
|
|
protected function processCollection(array $collection)
|
|
|
|
|
{
|
|
|
|
|
$array = array();
|
|
|
|
|
foreach ($collection as $coll) {
|
2013-02-21 15:46:59 -03:00
|
|
|
|
$array[$coll['annotation']->getSection()][$coll['resource']][] = $coll['annotation']->toArray();
|
2012-11-15 21:58:09 +01:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
$processedCollection = array();
|
2013-02-21 15:46:59 -03:00
|
|
|
|
foreach ($array as $section => $resources) {
|
|
|
|
|
foreach ($resources as $path => $annotations) {
|
|
|
|
|
foreach ($annotations as $annotation) {
|
2013-03-17 16:40:31 +01:00
|
|
|
|
if ($section) {
|
2013-02-21 15:46:59 -03:00
|
|
|
|
$processedCollection[$section][$path][] = $this->processAnnotation($annotation);
|
|
|
|
|
} else {
|
2013-02-25 17:27:09 -03:00
|
|
|
|
$processedCollection['_others'][$path][] = $this->processAnnotation($annotation);
|
2013-02-21 15:46:59 -03:00
|
|
|
|
}
|
|
|
|
|
}
|
2012-11-15 21:58:09 +01:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2013-03-17 16:40:31 +01:00
|
|
|
|
ksort($processedCollection);
|
|
|
|
|
|
2012-11-15 21:58:09 +01:00
|
|
|
|
return $processedCollection;
|
|
|
|
|
}
|
2012-04-11 20:00:21 +02:00
|
|
|
|
}
|