diff --git a/DependencyInjection/RegisterJmsParserPass.php b/DependencyInjection/RegisterJmsParserPass.php new file mode 100644 index 0000000..7c61460 --- /dev/null +++ b/DependencyInjection/RegisterJmsParserPass.php @@ -0,0 +1,21 @@ +hasDefinition('jms_serializer.serializer')) { + $loader->load('services.jms.xml'); + } + } +} diff --git a/Extractor/ApiDocExtractor.php b/Extractor/ApiDocExtractor.php index 820651a..d392a55 100644 --- a/Extractor/ApiDocExtractor.php +++ b/Extractor/ApiDocExtractor.php @@ -230,6 +230,7 @@ class ApiDocExtractor foreach ($this->parsers as $parser) { if ($parser->supports($input)) { $parameters = $parser->parse($input); + break; } } diff --git a/NelmioApiDocBundle.php b/NelmioApiDocBundle.php index fbfe504..ad9a5ce 100644 --- a/NelmioApiDocBundle.php +++ b/NelmioApiDocBundle.php @@ -4,6 +4,7 @@ namespace Nelmio\ApiDocBundle; use Symfony\Component\HttpKernel\Bundle\Bundle; use Symfony\Component\DependencyInjection\ContainerBuilder; +use Nelmio\ApiDocBundle\DependencyInjection\RegisterJmsParserPass; use Nelmio\ApiDocBundle\DependencyInjection\RegisterExtractorParsersPass; class NelmioApiDocBundle extends Bundle @@ -12,6 +13,7 @@ class NelmioApiDocBundle extends Bundle { parent::build($container); + $container->addCompilerPass(new RegisterJmsParserPass()); $container->addCompilerPass(new RegisterExtractorParsersPass()); } } diff --git a/Parser/JmsMetadataParser.php b/Parser/JmsMetadataParser.php new file mode 100644 index 0000000..24594ec --- /dev/null +++ b/Parser/JmsMetadataParser.php @@ -0,0 +1,85 @@ + + * + * For the full copyright and license information, please view the LICENSE + * file that was distributed with this source code. + */ + +namespace Nelmio\ApiDocBundle\Parser; + +use Metadata\MetadataFactoryInterface; + +/** + * Uses the JMS metadata factory to extract input/output model information + */ +class JmsMetadataParser implements ParserInterface +{ + + /** + * Constructor, requires JMS Metadata factory + */ + public function __construct(MetadataFactoryInterface $factory) + { + $this->factory = $factory; + } + + /** + * {@inheritdoc} + */ + public function supports($input) + { + if ($meta = $this->factory->getMetadataForClass($input)) { + return true; + } + + return false; + } + + /** + * {@inheritdoc} + */ + public function parse($input) + { + $meta = $this->factory->getMetadataForClass($input); + + if (null === $meta) { + throw new \InvalidArgumentException(sprintf("No metadata found for class %s", $input)); + } + + $params = array(); + + //iterate over property metadata + foreach ($meta->propertyMetadata as $item) { + + if (!is_null($item->type)) { + $name = isset($item->serializedName) ? $item->serializedName : $item->name; + + //TODO: check for nested type + + $params[$name] = array( + 'dataType' => $item->type, + 'required' => false, //TODO: can't think of a good way to specify this one, JMS doesn't have a setting for this + 'description' => $this->getDescription($input, $item->name), + 'readonly' => $item->readOnly + ); + } + } + + return $params; + } + + protected function getDescription($className, $propertyName) + { + $description = "No description."; + + //TODO: regex comment to get description - or move doc comment parsing functionality from `ApiDocExtractor` to a new location + //in order to reuse it here + + return $description; + } + +} diff --git a/README.md b/README.md index ae19234..46f8dd4 100644 --- a/README.md +++ b/README.md @@ -99,8 +99,8 @@ The following properties are available: * `filters`: an array of filters; -* `input`: the input type associated to the method, currently this only supports Form Types, useful for POST|PUT methods, either as FQCN or - as form type (if it is registered in the form factory in the container) +* `input`: the input type associated to the method, currently this supports Form Types, and classes with JMS Serializer + metadata, useful for POST|PUT methods, either as FQCN or as form type (if it is registered in the form factory in the container) Each _filter_ has to define a `name` parameter, but other parameters are free. Filters are often optional parameters, and you can document them as you want, but keep in mind to be consistent for the whole documentation. @@ -110,6 +110,8 @@ and determines for each parameter its data type, and if it's required or not. For Form Types, you can add an extra option named `description` on each field: +For classes parsed with JMS metadata, description will be taken from the properties doc comment, if available. + ``` php + + + + Nelmio\ApiDocBundle\Parser\JmsMetadataParser + + + + + + diff --git a/Tests/Extractor/ApiDocExtratorTest.php b/Tests/Extractor/ApiDocExtratorTest.php index b092643..a15de3a 100644 --- a/Tests/Extractor/ApiDocExtratorTest.php +++ b/Tests/Extractor/ApiDocExtratorTest.php @@ -22,7 +22,7 @@ class ApiDocExtractorTest extends WebTestCase $data = $extractor->all(); $this->assertTrue(is_array($data)); - $this->assertCount(10, $data); + $this->assertCount(11, $data); foreach ($data as $d) { $this->assertTrue(is_array($d)); diff --git a/Tests/Fixtures/Controller/TestController.php b/Tests/Fixtures/Controller/TestController.php index d39a02a..16c8e04 100644 --- a/Tests/Fixtures/Controller/TestController.php +++ b/Tests/Fixtures/Controller/TestController.php @@ -90,4 +90,14 @@ class TestController public function zActionWithQueryParamAction() { } + + /** + * @ApiDoc( + * description="Testing JMS", + * input="Nelmio\ApiDocBundle\Tests\Fixtures\Model\JmsTest" + * ) + */ + public function jmsInputTestAction() + { + } } diff --git a/Tests/Fixtures/Model/JmsTest.php b/Tests/Fixtures/Model/JmsTest.php new file mode 100644 index 0000000..ff44443 --- /dev/null +++ b/Tests/Fixtures/Model/JmsTest.php @@ -0,0 +1,33 @@ + 'Action without HTTP verb', ), 3 => + array( + 'method' => 'POST', + 'uri' => '/jms-input-test', + 'parameters' => + array( + 'foo' => + array( + 'dataType' => 'string', + 'required' => false, + 'description' => 'No description.', + 'readonly' => false + ), + 'bar' => + array( + 'dataType' => 'DateTime', + 'required' => false, + 'description' => 'No description.', + 'readonly' => true + ), + 'number' => + array( + 'dataType' => 'double', + 'required' => false, + 'description' => 'No description.', + 'readonly' => false + ), + 'arr' => + array( + 'dataType' => 'array', + 'required' => false, + 'description' => 'No description.', + 'readonly' => false + ) + ), + 'description' => 'Testing JMS' + ), + 4 => array( 'method' => 'ANY', 'uri' => '/my-commented/{id}/{page}', @@ -189,7 +226,7 @@ class SimpleFormatterTest extends WebTestCase 'description' => 'This method is useful to test if the getDocComment works.', 'documentation' => "This method is useful to test if the getDocComment works.\nAnd, it supports multilines until the first '@' char." ), - 4 => + 5 => array( 'method' => 'ANY', 'uri' => '/yet-another/{id}', @@ -198,7 +235,7 @@ class SimpleFormatterTest extends WebTestCase 'id' => array('type' => '', 'description' => '', 'requirement' => '\d+') ), ), - 5 => + 6 => array( 'method' => 'GET', 'uri' => '/z-action-with-query-param', diff --git a/composer.json b/composer.json index 2f8d743..b198ee1 100644 --- a/composer.json +++ b/composer.json @@ -25,7 +25,8 @@ "symfony/browser-kit": "2.1.*", "symfony/validator": "2.1.*", "symfony/yaml": "2.1.*", - "friendsofsymfony/rest-bundle": "dev-master" + "friendsofsymfony/rest-bundle": "dev-master", + "jms/serializer-bundle": "0.9.*" }, "minimum-stability": "dev", "autoload": {