From 22e03941b90bd9532cadcb935201ed76166dce87 Mon Sep 17 00:00:00 2001 From: Akolzin Dmitry Date: Wed, 3 Feb 2021 18:08:03 +0300 Subject: [PATCH] add callback argument value resolver --- .github/workflows/ci.yml | 34 ++++++++++ .gitignore | 4 ++ ArgumentResolver/AbstractValueResolver.php | 27 ++++++++ ArgumentResolver/CallbackValueResolver.php | 68 +++++++++++++++++++ DependencyInjection/Configuration.php | 41 +++++++++++ .../RetailCrmServiceExtension.php | 36 ++++++++++ RetailCrmServiceBundle.php | 9 +++ .../DependencyInjection/ConfigurationTest.php | 38 +++++++++++ composer.json | 39 +++++++++++ phpunit.xml.dist | 18 +++++ 10 files changed, 314 insertions(+) create mode 100644 .github/workflows/ci.yml create mode 100644 ArgumentResolver/AbstractValueResolver.php create mode 100644 ArgumentResolver/CallbackValueResolver.php create mode 100644 DependencyInjection/Configuration.php create mode 100644 DependencyInjection/RetailCrmServiceExtension.php create mode 100644 RetailCrmServiceBundle.php create mode 100644 Tests/DependencyInjection/ConfigurationTest.php create mode 100644 composer.json create mode 100644 phpunit.xml.dist diff --git a/.github/workflows/ci.yml b/.github/workflows/ci.yml new file mode 100644 index 0000000..6df4c2b --- /dev/null +++ b/.github/workflows/ci.yml @@ -0,0 +1,34 @@ +name: ci + +on: + push: + branches: + - '**' + tags-ignore: + - '*.*' + pull_request: + +jobs: + test: + runs-on: ubuntu-latest + strategy: + matrix: + php-version: ['7.3', '7.4', '8.0'] + steps: + - uses: actions/checkout@v2 + - name: Setup PHP ${{ matrix.php-version }} + uses: shivammathur/setup-php@v2 + with: + php-version: ${{ matrix.php-version }} + coverage: pcov + - name: Composer cache + uses: actions/cache@v2 + with: + path: ${{ env.HOME }}/.composer/cache + key: ${{ runner.os }}-php-${{ hashFiles('**/composer.lock') }} + - name: Install dependencies + run: composer install -o + - name: Run tests + run: composer run tests + - name: Coverage + run: bash <(curl -s https://codecov.io/bash) diff --git a/.gitignore b/.gitignore index 3dab634..5202207 100644 --- a/.gitignore +++ b/.gitignore @@ -30,6 +30,7 @@ !bin/console !bin/symfony_requirements /vendor/ +composer.lock # Assets and user uploads /web/bundles/ @@ -38,6 +39,7 @@ # PHPUnit /app/phpunit.xml /phpunit.xml +.phpunit.result.cache # Build data /build/ @@ -50,3 +52,5 @@ # Embedded web-server pid file /.web-server-pid + +.idea diff --git a/ArgumentResolver/AbstractValueResolver.php b/ArgumentResolver/AbstractValueResolver.php new file mode 100644 index 0000000..638db72 --- /dev/null +++ b/ArgumentResolver/AbstractValueResolver.php @@ -0,0 +1,27 @@ +validator = $validator; + } + + /** + * @param object $data + */ + protected function validate(object $data): void + { + $errors = $this->validator->validate($data); + if (0 !== count($errors)) { + throw new InvalidArgumentException($errors); + } + } +} diff --git a/ArgumentResolver/CallbackValueResolver.php b/ArgumentResolver/CallbackValueResolver.php new file mode 100644 index 0000000..7d0a30c --- /dev/null +++ b/ArgumentResolver/CallbackValueResolver.php @@ -0,0 +1,68 @@ +serializer = $serializer; + $this->requestSchema = $requestSchema; + } + + /** + * {@inheritdoc } + */ + public function supports(Request $request, ArgumentMetadata $argument): bool + { + if (empty($this->requestSchema) || $request->getMethod() !== Request::METHOD_POST) { + return false; + } + + return null !== $this->search($request, $argument); + } + + /** + * {@inheritdoc } + */ + public function resolve(Request $request, ArgumentMetadata $argument): Generator + { + $parameter = $this->search($request, $argument); + $data = $this->serializer->deserialize($request->request->get($parameter), $argument->getType(), 'json'); + $this->validate($data); + + yield $data; + } + + private function search(Request $request, ArgumentMetadata $argument): ?string + { + foreach ($this->requestSchema as $callback) { + if (!$argument->getName() === $callback['type']) { + continue; + } + + foreach ($callback['params'] as $param) { + if ($request->request->has($param)) { + return $param; + } + } + } + + return null; + } +} diff --git a/DependencyInjection/Configuration.php b/DependencyInjection/Configuration.php new file mode 100644 index 0000000..93acfc1 --- /dev/null +++ b/DependencyInjection/Configuration.php @@ -0,0 +1,41 @@ +getRootNode(); + + $rootNode + ->children() + ->arrayNode('request_schema') + ->children() + ->arrayNode('callback') + ->arrayPrototype() + ->children() + ->scalarNode('type')->isRequired()->end() + ->arrayNode('params') + ->isRequired()->scalarPrototype()->end() + ->end() + ->end() + ->end() + ->end() + ->arrayNode('client') + ->scalarPrototype()->end() + ->end() + ->end() + ->end() + ->end(); + + return $treeBuilder; + } +} diff --git a/DependencyInjection/RetailCrmServiceExtension.php b/DependencyInjection/RetailCrmServiceExtension.php new file mode 100644 index 0000000..3df53f0 --- /dev/null +++ b/DependencyInjection/RetailCrmServiceExtension.php @@ -0,0 +1,36 @@ +getConfiguration($configs, $container); + $config = $this->processConfiguration($configuration, $configs); + + $container->setParameter( + 'retail_crm_service.request_schema.callback', + $config['request_schema']['callback'] + ); + + $container->setParameter( + 'retail_crm_service.request_schema.client', + $config['request_schema']['client'] + ); + + $container + ->register(CallbackValueResolver::class) + ->setArgument('$requestSchema', '%retail_crm_service.request_schema.callback%') + ->addTag('controller.argument_value_resolver', ['priority' => 50]) + ->setAutowired(true); + } +} diff --git a/RetailCrmServiceBundle.php b/RetailCrmServiceBundle.php new file mode 100644 index 0000000..ec08b5a --- /dev/null +++ b/RetailCrmServiceBundle.php @@ -0,0 +1,9 @@ + [ + 'callback' => [ + [ + 'type' => 'type', + 'params' => ['param'] + ] + ], + 'client' => [ + 'type1', + 'type2' + ] + ] + ] + ]; + + $config = $processor->processConfiguration(new Configuration(), $configs); + + static::assertArrayHasKey('request_schema', $config); + static::assertArrayHasKey('callback', $config['request_schema']); + static::assertArrayHasKey('client', $config['request_schema']); + } +} diff --git a/composer.json b/composer.json new file mode 100644 index 0000000..a0d6499 --- /dev/null +++ b/composer.json @@ -0,0 +1,39 @@ +{ + "name": "retailcrm/service-bundle", + "description": "Core bundle for RetailCRM integration services", + "type": "symfony-bundle", + "license": "MIT", + "authors": [ + { + "name": "RetailCRM", + "email": "support@retailcrm.pro" + } + ], + "minimum-stability": "stable", + "require": { + "php": ">=7.3", + "symfony/framework-bundle": "^4.0|^5.0", + "symfony/serializer": "^5.2", + "symfony/http-kernel": "^4.0|^5.0", + "symfony/validator": "^4.0|^5.0" + }, + "autoload": { + "psr-4": { + "RetailCrm\\ServiceBundle\\": "" + }, + "exclude-from-classmap": [ + "Tests/" + ] + }, + "autoload-dev": { + "psr-4": { + "RetailCrm\\ServiceBundle\\Tests\\": "Tests/" + } + }, + "require-dev": { + "phpunit/phpunit": "^8.0 || ^9.0" + }, + "scripts": { + "tests": "./vendor/bin/phpunit -c phpunit.xml.dist" + } +} diff --git a/phpunit.xml.dist b/phpunit.xml.dist new file mode 100644 index 0000000..6f6d82a --- /dev/null +++ b/phpunit.xml.dist @@ -0,0 +1,18 @@ + + + + + + ./Tests + + +