Compare commits
5 Commits
d6fdec2a8b
...
05c1a270e5
Author | SHA1 | Date | |
---|---|---|---|
|
05c1a270e5 | ||
|
5cf26579fc | ||
|
85a6212ef5 | ||
|
cbfb10ec10 | ||
|
e223ddfb4f |
2
.env
2
.env
@ -22,3 +22,5 @@ APP_SECRET=ff44e0bb0862dabdb2d0a1ba40f489d9
|
||||
###> doctrine/doctrine-bundle ###
|
||||
DATABASE_URL="mysql://iteo:iteo@mariadb:3306/iteo?serverVersion=10.11.2-MariaDB&charset=utf8mb4"
|
||||
###< doctrine/doctrine-bundle ###
|
||||
|
||||
APP_CRM_BASE_URL="http://crm.local"
|
||||
|
@ -3,6 +3,7 @@ default:
|
||||
default:
|
||||
contexts:
|
||||
- App\Tests\Behat\ApiCallContext
|
||||
- App\Tests\Behat\ClientsContext
|
||||
|
||||
extensions:
|
||||
FriendsOfBehat\SymfonyExtension:
|
||||
|
@ -18,11 +18,13 @@
|
||||
"symfony/dotenv": "7.0.*",
|
||||
"symfony/flex": "^2",
|
||||
"symfony/framework-bundle": "7.0.*",
|
||||
"symfony/http-client": "7.0.*",
|
||||
"symfony/property-access": "7.0.*",
|
||||
"symfony/property-info": "7.0.*",
|
||||
"symfony/runtime": "7.0.*",
|
||||
"symfony/serializer": "7.0.*",
|
||||
"symfony/uid": "7.0.*",
|
||||
"symfony/validator": "7.0.*",
|
||||
"symfony/yaml": "7.0.*"
|
||||
},
|
||||
"config": {
|
||||
@ -76,6 +78,7 @@
|
||||
},
|
||||
"require-dev": {
|
||||
"friends-of-behat/symfony-extension": "^2.0",
|
||||
"mockery/mockery": "^1.6",
|
||||
"phpunit/phpunit": "^9.5",
|
||||
"symfony/browser-kit": "7.0.*",
|
||||
"symfony/css-selector": "7.0.*",
|
||||
|
556
composer.lock
generated
556
composer.lock
generated
@ -4,7 +4,7 @@
|
||||
"Read more about it at https://getcomposer.org/doc/01-basic-usage.md#installing-dependencies",
|
||||
"This file is @generated automatically"
|
||||
],
|
||||
"content-hash": "e63cd12b3ea9a5ca7203ce3e59ec7678",
|
||||
"content-hash": "51cd32c43082b3312c47499f02f2c2ea",
|
||||
"packages": [
|
||||
{
|
||||
"name": "doctrine/cache",
|
||||
@ -3097,6 +3097,176 @@
|
||||
],
|
||||
"time": "2024-03-27T19:55:25+00:00"
|
||||
},
|
||||
{
|
||||
"name": "symfony/http-client",
|
||||
"version": "v7.0.6",
|
||||
"source": {
|
||||
"type": "git",
|
||||
"url": "https://github.com/symfony/http-client.git",
|
||||
"reference": "6e70473909f46fe5dd3b994a0f1b20ecb6b2f858"
|
||||
},
|
||||
"dist": {
|
||||
"type": "zip",
|
||||
"url": "https://api.github.com/repos/symfony/http-client/zipball/6e70473909f46fe5dd3b994a0f1b20ecb6b2f858",
|
||||
"reference": "6e70473909f46fe5dd3b994a0f1b20ecb6b2f858",
|
||||
"shasum": ""
|
||||
},
|
||||
"require": {
|
||||
"php": ">=8.2",
|
||||
"psr/log": "^1|^2|^3",
|
||||
"symfony/http-client-contracts": "^3.4.1",
|
||||
"symfony/service-contracts": "^2.5|^3"
|
||||
},
|
||||
"conflict": {
|
||||
"php-http/discovery": "<1.15",
|
||||
"symfony/http-foundation": "<6.4"
|
||||
},
|
||||
"provide": {
|
||||
"php-http/async-client-implementation": "*",
|
||||
"php-http/client-implementation": "*",
|
||||
"psr/http-client-implementation": "1.0",
|
||||
"symfony/http-client-implementation": "3.0"
|
||||
},
|
||||
"require-dev": {
|
||||
"amphp/amp": "^2.5",
|
||||
"amphp/http-client": "^4.2.1",
|
||||
"amphp/http-tunnel": "^1.0",
|
||||
"amphp/socket": "^1.1",
|
||||
"guzzlehttp/promises": "^1.4|^2.0",
|
||||
"nyholm/psr7": "^1.0",
|
||||
"php-http/httplug": "^1.0|^2.0",
|
||||
"psr/http-client": "^1.0",
|
||||
"symfony/dependency-injection": "^6.4|^7.0",
|
||||
"symfony/http-kernel": "^6.4|^7.0",
|
||||
"symfony/messenger": "^6.4|^7.0",
|
||||
"symfony/process": "^6.4|^7.0",
|
||||
"symfony/stopwatch": "^6.4|^7.0"
|
||||
},
|
||||
"type": "library",
|
||||
"autoload": {
|
||||
"psr-4": {
|
||||
"Symfony\\Component\\HttpClient\\": ""
|
||||
},
|
||||
"exclude-from-classmap": [
|
||||
"/Tests/"
|
||||
]
|
||||
},
|
||||
"notification-url": "https://packagist.org/downloads/",
|
||||
"license": [
|
||||
"MIT"
|
||||
],
|
||||
"authors": [
|
||||
{
|
||||
"name": "Nicolas Grekas",
|
||||
"email": "p@tchwork.com"
|
||||
},
|
||||
{
|
||||
"name": "Symfony Community",
|
||||
"homepage": "https://symfony.com/contributors"
|
||||
}
|
||||
],
|
||||
"description": "Provides powerful methods to fetch HTTP resources synchronously or asynchronously",
|
||||
"homepage": "https://symfony.com",
|
||||
"keywords": [
|
||||
"http"
|
||||
],
|
||||
"support": {
|
||||
"source": "https://github.com/symfony/http-client/tree/v7.0.6"
|
||||
},
|
||||
"funding": [
|
||||
{
|
||||
"url": "https://symfony.com/sponsor",
|
||||
"type": "custom"
|
||||
},
|
||||
{
|
||||
"url": "https://github.com/fabpot",
|
||||
"type": "github"
|
||||
},
|
||||
{
|
||||
"url": "https://tidelift.com/funding/github/packagist/symfony/symfony",
|
||||
"type": "tidelift"
|
||||
}
|
||||
],
|
||||
"time": "2024-04-01T20:49:44+00:00"
|
||||
},
|
||||
{
|
||||
"name": "symfony/http-client-contracts",
|
||||
"version": "v3.4.2",
|
||||
"source": {
|
||||
"type": "git",
|
||||
"url": "https://github.com/symfony/http-client-contracts.git",
|
||||
"reference": "b6b5c876b3a4ed74460e2c5ac53bbce2f12e2a7e"
|
||||
},
|
||||
"dist": {
|
||||
"type": "zip",
|
||||
"url": "https://api.github.com/repos/symfony/http-client-contracts/zipball/b6b5c876b3a4ed74460e2c5ac53bbce2f12e2a7e",
|
||||
"reference": "b6b5c876b3a4ed74460e2c5ac53bbce2f12e2a7e",
|
||||
"shasum": ""
|
||||
},
|
||||
"require": {
|
||||
"php": ">=8.1"
|
||||
},
|
||||
"type": "library",
|
||||
"extra": {
|
||||
"branch-alias": {
|
||||
"dev-main": "3.4-dev"
|
||||
},
|
||||
"thanks": {
|
||||
"name": "symfony/contracts",
|
||||
"url": "https://github.com/symfony/contracts"
|
||||
}
|
||||
},
|
||||
"autoload": {
|
||||
"psr-4": {
|
||||
"Symfony\\Contracts\\HttpClient\\": ""
|
||||
},
|
||||
"exclude-from-classmap": [
|
||||
"/Test/"
|
||||
]
|
||||
},
|
||||
"notification-url": "https://packagist.org/downloads/",
|
||||
"license": [
|
||||
"MIT"
|
||||
],
|
||||
"authors": [
|
||||
{
|
||||
"name": "Nicolas Grekas",
|
||||
"email": "p@tchwork.com"
|
||||
},
|
||||
{
|
||||
"name": "Symfony Community",
|
||||
"homepage": "https://symfony.com/contributors"
|
||||
}
|
||||
],
|
||||
"description": "Generic abstractions related to HTTP clients",
|
||||
"homepage": "https://symfony.com",
|
||||
"keywords": [
|
||||
"abstractions",
|
||||
"contracts",
|
||||
"decoupling",
|
||||
"interfaces",
|
||||
"interoperability",
|
||||
"standards"
|
||||
],
|
||||
"support": {
|
||||
"source": "https://github.com/symfony/http-client-contracts/tree/v3.4.2"
|
||||
},
|
||||
"funding": [
|
||||
{
|
||||
"url": "https://symfony.com/sponsor",
|
||||
"type": "custom"
|
||||
},
|
||||
{
|
||||
"url": "https://github.com/fabpot",
|
||||
"type": "github"
|
||||
},
|
||||
{
|
||||
"url": "https://tidelift.com/funding/github/packagist/symfony/symfony",
|
||||
"type": "tidelift"
|
||||
}
|
||||
],
|
||||
"time": "2024-04-01T18:51:09+00:00"
|
||||
},
|
||||
{
|
||||
"name": "symfony/http-foundation",
|
||||
"version": "v7.0.6",
|
||||
@ -4325,6 +4495,84 @@
|
||||
],
|
||||
"time": "2024-02-01T13:17:36+00:00"
|
||||
},
|
||||
{
|
||||
"name": "symfony/translation-contracts",
|
||||
"version": "v3.4.2",
|
||||
"source": {
|
||||
"type": "git",
|
||||
"url": "https://github.com/symfony/translation-contracts.git",
|
||||
"reference": "43810bdb2ddb5400e5c5e778e27b210a0ca83b6b"
|
||||
},
|
||||
"dist": {
|
||||
"type": "zip",
|
||||
"url": "https://api.github.com/repos/symfony/translation-contracts/zipball/43810bdb2ddb5400e5c5e778e27b210a0ca83b6b",
|
||||
"reference": "43810bdb2ddb5400e5c5e778e27b210a0ca83b6b",
|
||||
"shasum": ""
|
||||
},
|
||||
"require": {
|
||||
"php": ">=8.1"
|
||||
},
|
||||
"type": "library",
|
||||
"extra": {
|
||||
"branch-alias": {
|
||||
"dev-main": "3.4-dev"
|
||||
},
|
||||
"thanks": {
|
||||
"name": "symfony/contracts",
|
||||
"url": "https://github.com/symfony/contracts"
|
||||
}
|
||||
},
|
||||
"autoload": {
|
||||
"psr-4": {
|
||||
"Symfony\\Contracts\\Translation\\": ""
|
||||
},
|
||||
"exclude-from-classmap": [
|
||||
"/Test/"
|
||||
]
|
||||
},
|
||||
"notification-url": "https://packagist.org/downloads/",
|
||||
"license": [
|
||||
"MIT"
|
||||
],
|
||||
"authors": [
|
||||
{
|
||||
"name": "Nicolas Grekas",
|
||||
"email": "p@tchwork.com"
|
||||
},
|
||||
{
|
||||
"name": "Symfony Community",
|
||||
"homepage": "https://symfony.com/contributors"
|
||||
}
|
||||
],
|
||||
"description": "Generic abstractions related to translation",
|
||||
"homepage": "https://symfony.com",
|
||||
"keywords": [
|
||||
"abstractions",
|
||||
"contracts",
|
||||
"decoupling",
|
||||
"interfaces",
|
||||
"interoperability",
|
||||
"standards"
|
||||
],
|
||||
"support": {
|
||||
"source": "https://github.com/symfony/translation-contracts/tree/v3.4.2"
|
||||
},
|
||||
"funding": [
|
||||
{
|
||||
"url": "https://symfony.com/sponsor",
|
||||
"type": "custom"
|
||||
},
|
||||
{
|
||||
"url": "https://github.com/fabpot",
|
||||
"type": "github"
|
||||
},
|
||||
{
|
||||
"url": "https://tidelift.com/funding/github/packagist/symfony/symfony",
|
||||
"type": "tidelift"
|
||||
}
|
||||
],
|
||||
"time": "2024-01-23T14:51:35+00:00"
|
||||
},
|
||||
{
|
||||
"name": "symfony/uid",
|
||||
"version": "v7.0.3",
|
||||
@ -4399,6 +4647,100 @@
|
||||
],
|
||||
"time": "2024-01-23T15:02:46+00:00"
|
||||
},
|
||||
{
|
||||
"name": "symfony/validator",
|
||||
"version": "v7.0.6",
|
||||
"source": {
|
||||
"type": "git",
|
||||
"url": "https://github.com/symfony/validator.git",
|
||||
"reference": "a2df2c63b7944a162dee86ab8065f2f91b7d6e36"
|
||||
},
|
||||
"dist": {
|
||||
"type": "zip",
|
||||
"url": "https://api.github.com/repos/symfony/validator/zipball/a2df2c63b7944a162dee86ab8065f2f91b7d6e36",
|
||||
"reference": "a2df2c63b7944a162dee86ab8065f2f91b7d6e36",
|
||||
"shasum": ""
|
||||
},
|
||||
"require": {
|
||||
"php": ">=8.2",
|
||||
"symfony/polyfill-ctype": "~1.8",
|
||||
"symfony/polyfill-mbstring": "~1.0",
|
||||
"symfony/polyfill-php83": "^1.27",
|
||||
"symfony/translation-contracts": "^2.5|^3"
|
||||
},
|
||||
"conflict": {
|
||||
"doctrine/lexer": "<1.1",
|
||||
"symfony/dependency-injection": "<6.4",
|
||||
"symfony/doctrine-bridge": "<7.0",
|
||||
"symfony/expression-language": "<6.4",
|
||||
"symfony/http-kernel": "<6.4",
|
||||
"symfony/intl": "<6.4",
|
||||
"symfony/property-info": "<6.4",
|
||||
"symfony/translation": "<6.4.3|>=7.0,<7.0.3",
|
||||
"symfony/yaml": "<6.4"
|
||||
},
|
||||
"require-dev": {
|
||||
"egulias/email-validator": "^2.1.10|^3|^4",
|
||||
"symfony/cache": "^6.4|^7.0",
|
||||
"symfony/config": "^6.4|^7.0",
|
||||
"symfony/console": "^6.4|^7.0",
|
||||
"symfony/dependency-injection": "^6.4|^7.0",
|
||||
"symfony/expression-language": "^6.4|^7.0",
|
||||
"symfony/finder": "^6.4|^7.0",
|
||||
"symfony/http-client": "^6.4|^7.0",
|
||||
"symfony/http-foundation": "^6.4|^7.0",
|
||||
"symfony/http-kernel": "^6.4|^7.0",
|
||||
"symfony/intl": "^6.4|^7.0",
|
||||
"symfony/mime": "^6.4|^7.0",
|
||||
"symfony/property-access": "^6.4|^7.0",
|
||||
"symfony/property-info": "^6.4|^7.0",
|
||||
"symfony/translation": "^6.4.3|^7.0.3",
|
||||
"symfony/yaml": "^6.4|^7.0"
|
||||
},
|
||||
"type": "library",
|
||||
"autoload": {
|
||||
"psr-4": {
|
||||
"Symfony\\Component\\Validator\\": ""
|
||||
},
|
||||
"exclude-from-classmap": [
|
||||
"/Tests/"
|
||||
]
|
||||
},
|
||||
"notification-url": "https://packagist.org/downloads/",
|
||||
"license": [
|
||||
"MIT"
|
||||
],
|
||||
"authors": [
|
||||
{
|
||||
"name": "Fabien Potencier",
|
||||
"email": "fabien@symfony.com"
|
||||
},
|
||||
{
|
||||
"name": "Symfony Community",
|
||||
"homepage": "https://symfony.com/contributors"
|
||||
}
|
||||
],
|
||||
"description": "Provides tools to validate values",
|
||||
"homepage": "https://symfony.com",
|
||||
"support": {
|
||||
"source": "https://github.com/symfony/validator/tree/v7.0.6"
|
||||
},
|
||||
"funding": [
|
||||
{
|
||||
"url": "https://symfony.com/sponsor",
|
||||
"type": "custom"
|
||||
},
|
||||
{
|
||||
"url": "https://github.com/fabpot",
|
||||
"type": "github"
|
||||
},
|
||||
{
|
||||
"url": "https://tidelift.com/funding/github/packagist/symfony/symfony",
|
||||
"type": "tidelift"
|
||||
}
|
||||
],
|
||||
"time": "2024-03-28T09:20:36+00:00"
|
||||
},
|
||||
{
|
||||
"name": "symfony/var-dumper",
|
||||
"version": "v7.0.6",
|
||||
@ -4956,6 +5298,57 @@
|
||||
},
|
||||
"time": "2024-01-11T14:57:03+00:00"
|
||||
},
|
||||
{
|
||||
"name": "hamcrest/hamcrest-php",
|
||||
"version": "v2.0.1",
|
||||
"source": {
|
||||
"type": "git",
|
||||
"url": "https://github.com/hamcrest/hamcrest-php.git",
|
||||
"reference": "8c3d0a3f6af734494ad8f6fbbee0ba92422859f3"
|
||||
},
|
||||
"dist": {
|
||||
"type": "zip",
|
||||
"url": "https://api.github.com/repos/hamcrest/hamcrest-php/zipball/8c3d0a3f6af734494ad8f6fbbee0ba92422859f3",
|
||||
"reference": "8c3d0a3f6af734494ad8f6fbbee0ba92422859f3",
|
||||
"shasum": ""
|
||||
},
|
||||
"require": {
|
||||
"php": "^5.3|^7.0|^8.0"
|
||||
},
|
||||
"replace": {
|
||||
"cordoval/hamcrest-php": "*",
|
||||
"davedevelopment/hamcrest-php": "*",
|
||||
"kodova/hamcrest-php": "*"
|
||||
},
|
||||
"require-dev": {
|
||||
"phpunit/php-file-iterator": "^1.4 || ^2.0",
|
||||
"phpunit/phpunit": "^4.8.36 || ^5.7 || ^6.5 || ^7.0"
|
||||
},
|
||||
"type": "library",
|
||||
"extra": {
|
||||
"branch-alias": {
|
||||
"dev-master": "2.1-dev"
|
||||
}
|
||||
},
|
||||
"autoload": {
|
||||
"classmap": [
|
||||
"hamcrest"
|
||||
]
|
||||
},
|
||||
"notification-url": "https://packagist.org/downloads/",
|
||||
"license": [
|
||||
"BSD-3-Clause"
|
||||
],
|
||||
"description": "This is the PHP port of Hamcrest Matchers",
|
||||
"keywords": [
|
||||
"test"
|
||||
],
|
||||
"support": {
|
||||
"issues": "https://github.com/hamcrest/hamcrest-php/issues",
|
||||
"source": "https://github.com/hamcrest/hamcrest-php/tree/v2.0.1"
|
||||
},
|
||||
"time": "2020-07-09T08:09:16+00:00"
|
||||
},
|
||||
{
|
||||
"name": "masterminds/html5",
|
||||
"version": "2.9.0",
|
||||
@ -5023,6 +5416,89 @@
|
||||
},
|
||||
"time": "2024-03-31T07:05:07+00:00"
|
||||
},
|
||||
{
|
||||
"name": "mockery/mockery",
|
||||
"version": "1.6.11",
|
||||
"source": {
|
||||
"type": "git",
|
||||
"url": "https://github.com/mockery/mockery.git",
|
||||
"reference": "81a161d0b135df89951abd52296adf97deb0723d"
|
||||
},
|
||||
"dist": {
|
||||
"type": "zip",
|
||||
"url": "https://api.github.com/repos/mockery/mockery/zipball/81a161d0b135df89951abd52296adf97deb0723d",
|
||||
"reference": "81a161d0b135df89951abd52296adf97deb0723d",
|
||||
"shasum": ""
|
||||
},
|
||||
"require": {
|
||||
"hamcrest/hamcrest-php": "^2.0.1",
|
||||
"lib-pcre": ">=7.0",
|
||||
"php": ">=7.3"
|
||||
},
|
||||
"conflict": {
|
||||
"phpunit/phpunit": "<8.0"
|
||||
},
|
||||
"require-dev": {
|
||||
"phpunit/phpunit": "^8.5 || ^9.6.17",
|
||||
"symplify/easy-coding-standard": "^12.1.14"
|
||||
},
|
||||
"type": "library",
|
||||
"autoload": {
|
||||
"files": [
|
||||
"library/helpers.php",
|
||||
"library/Mockery.php"
|
||||
],
|
||||
"psr-4": {
|
||||
"Mockery\\": "library/Mockery"
|
||||
}
|
||||
},
|
||||
"notification-url": "https://packagist.org/downloads/",
|
||||
"license": [
|
||||
"BSD-3-Clause"
|
||||
],
|
||||
"authors": [
|
||||
{
|
||||
"name": "Pádraic Brady",
|
||||
"email": "padraic.brady@gmail.com",
|
||||
"homepage": "https://github.com/padraic",
|
||||
"role": "Author"
|
||||
},
|
||||
{
|
||||
"name": "Dave Marshall",
|
||||
"email": "dave.marshall@atstsolutions.co.uk",
|
||||
"homepage": "https://davedevelopment.co.uk",
|
||||
"role": "Developer"
|
||||
},
|
||||
{
|
||||
"name": "Nathanael Esayeas",
|
||||
"email": "nathanael.esayeas@protonmail.com",
|
||||
"homepage": "https://github.com/ghostwriter",
|
||||
"role": "Lead Developer"
|
||||
}
|
||||
],
|
||||
"description": "Mockery is a simple yet flexible PHP mock object framework",
|
||||
"homepage": "https://github.com/mockery/mockery",
|
||||
"keywords": [
|
||||
"BDD",
|
||||
"TDD",
|
||||
"library",
|
||||
"mock",
|
||||
"mock objects",
|
||||
"mockery",
|
||||
"stub",
|
||||
"test",
|
||||
"test double",
|
||||
"testing"
|
||||
],
|
||||
"support": {
|
||||
"docs": "https://docs.mockery.io/",
|
||||
"issues": "https://github.com/mockery/mockery/issues",
|
||||
"rss": "https://github.com/mockery/mockery/releases.atom",
|
||||
"security": "https://github.com/mockery/mockery/security/advisories",
|
||||
"source": "https://github.com/mockery/mockery"
|
||||
},
|
||||
"time": "2024-03-21T18:34:15+00:00"
|
||||
},
|
||||
{
|
||||
"name": "myclabs/deep-copy",
|
||||
"version": "1.11.1",
|
||||
@ -7171,84 +7647,6 @@
|
||||
],
|
||||
"time": "2024-02-22T20:27:20+00:00"
|
||||
},
|
||||
{
|
||||
"name": "symfony/translation-contracts",
|
||||
"version": "v3.4.2",
|
||||
"source": {
|
||||
"type": "git",
|
||||
"url": "https://github.com/symfony/translation-contracts.git",
|
||||
"reference": "43810bdb2ddb5400e5c5e778e27b210a0ca83b6b"
|
||||
},
|
||||
"dist": {
|
||||
"type": "zip",
|
||||
"url": "https://api.github.com/repos/symfony/translation-contracts/zipball/43810bdb2ddb5400e5c5e778e27b210a0ca83b6b",
|
||||
"reference": "43810bdb2ddb5400e5c5e778e27b210a0ca83b6b",
|
||||
"shasum": ""
|
||||
},
|
||||
"require": {
|
||||
"php": ">=8.1"
|
||||
},
|
||||
"type": "library",
|
||||
"extra": {
|
||||
"branch-alias": {
|
||||
"dev-main": "3.4-dev"
|
||||
},
|
||||
"thanks": {
|
||||
"name": "symfony/contracts",
|
||||
"url": "https://github.com/symfony/contracts"
|
||||
}
|
||||
},
|
||||
"autoload": {
|
||||
"psr-4": {
|
||||
"Symfony\\Contracts\\Translation\\": ""
|
||||
},
|
||||
"exclude-from-classmap": [
|
||||
"/Test/"
|
||||
]
|
||||
},
|
||||
"notification-url": "https://packagist.org/downloads/",
|
||||
"license": [
|
||||
"MIT"
|
||||
],
|
||||
"authors": [
|
||||
{
|
||||
"name": "Nicolas Grekas",
|
||||
"email": "p@tchwork.com"
|
||||
},
|
||||
{
|
||||
"name": "Symfony Community",
|
||||
"homepage": "https://symfony.com/contributors"
|
||||
}
|
||||
],
|
||||
"description": "Generic abstractions related to translation",
|
||||
"homepage": "https://symfony.com",
|
||||
"keywords": [
|
||||
"abstractions",
|
||||
"contracts",
|
||||
"decoupling",
|
||||
"interfaces",
|
||||
"interoperability",
|
||||
"standards"
|
||||
],
|
||||
"support": {
|
||||
"source": "https://github.com/symfony/translation-contracts/tree/v3.4.2"
|
||||
},
|
||||
"funding": [
|
||||
{
|
||||
"url": "https://symfony.com/sponsor",
|
||||
"type": "custom"
|
||||
},
|
||||
{
|
||||
"url": "https://github.com/fabpot",
|
||||
"type": "github"
|
||||
},
|
||||
{
|
||||
"url": "https://tidelift.com/funding/github/packagist/symfony/symfony",
|
||||
"type": "tidelift"
|
||||
}
|
||||
],
|
||||
"time": "2024-01-23T14:51:35+00:00"
|
||||
},
|
||||
{
|
||||
"name": "theseer/tokenizer",
|
||||
"version": "1.2.3",
|
||||
|
@ -9,6 +9,14 @@ framework:
|
||||
#esi: true
|
||||
#fragments: true
|
||||
|
||||
http_client:
|
||||
scoped_clients:
|
||||
crm.client:
|
||||
base_uri: '%env(APP_CRM_BASE_URL)%'
|
||||
headers:
|
||||
Accept: 'application/json'
|
||||
Content-Type: 'application/json'
|
||||
|
||||
when@test:
|
||||
framework:
|
||||
test: true
|
||||
|
11
config/packages/validator.yaml
Normal file
11
config/packages/validator.yaml
Normal file
@ -0,0 +1,11 @@
|
||||
framework:
|
||||
validation:
|
||||
# Enables validator auto-mapping support.
|
||||
# For instance, basic validation constraints will be inferred from Doctrine's metadata.
|
||||
#auto_mapping:
|
||||
# App\Entity\: []
|
||||
|
||||
when@test:
|
||||
framework:
|
||||
validation:
|
||||
not_compromised_password: false
|
@ -5,3 +5,8 @@ services:
|
||||
|
||||
App\Tests\Behat\:
|
||||
resource: '../tests/Behat/*'
|
||||
|
||||
crm.client:
|
||||
class: Symfony\Component\HttpClient\MockHttpClient
|
||||
arguments:
|
||||
$baseUri: "%env(APP_CRM_BASE_URL)%"
|
||||
|
138
features/accept-order.feature
Normal file
138
features/accept-order.feature
Normal file
@ -0,0 +1,138 @@
|
||||
Feature:
|
||||
Frontend is able to send requests that will be processed by the application.
|
||||
|
||||
Background:
|
||||
Given there exist following clients:
|
||||
| clientId | name | initialBalance | currentBalance | locked |
|
||||
| 018edd2e-894a-78d7-b10c-16e05ca933a3 | Kacper | 10000 | 200000 | no |
|
||||
| 018ede1e-a587-76a8-88f3-23987a8dade9 | Jacek | 10000 | 60000000 | yes |
|
||||
| 018ede4c-f5ac-7be9-9f14-7c00d2636812 | Mirek | 10000 | -10000 | no |
|
||||
|
||||
Scenario: Frontend is able to send new orders
|
||||
Given the request has the following body:
|
||||
"""
|
||||
{
|
||||
"orderId": "018eddae-ff52-7813-9f88-ada9e61a76f3",
|
||||
"clientId": "018edd2e-894a-78d7-b10c-16e05ca933a3",
|
||||
"products": [
|
||||
{ "productId": "p1", "quantity": 2, "price": 20.0, "weight": 100 },
|
||||
{ "productId": "p2", "quantity": 1, "price": 100.0, "weight": 200 },
|
||||
{ "productId": "p3", "quantity": 5, "price": 200.0, "weight": 100 },
|
||||
{ "productId": "p4", "quantity": 1, "price": 50.0, "weight": 400 },
|
||||
{ "productId": "p5", "quantity": 10, "price": 10.0, "weight": 100 }
|
||||
]
|
||||
}
|
||||
"""
|
||||
When I send a POST request to "/orders"
|
||||
Then the response status should be 200
|
||||
And client with id "018edd2e-894a-78d7-b10c-16e05ca933a3" should have balance of 71000
|
||||
|
||||
Scenario: Frontend must send properly formatted order request
|
||||
Given the request has the following body:
|
||||
"""
|
||||
{
|
||||
"orderId": "018eddae-ff52-7813-9f88-ada9e61a76f3"
|
||||
}
|
||||
"""
|
||||
When I send a POST request to "/orders"
|
||||
Then the response status should be 422
|
||||
|
||||
Scenario: Frontend must send valid client reference
|
||||
Given the request has the following body:
|
||||
"""
|
||||
{
|
||||
"orderId": "018eddae-ff52-7813-9f88-ada9e61a76f3",
|
||||
"clientId": "00000000-0000-0000-0000-000000000000",
|
||||
"products": [
|
||||
{ "productId": "p1", "quantity": 2, "price": 20.0, "weight": 100 },
|
||||
{ "productId": "p2", "quantity": 1, "price": 100.0, "weight": 200 },
|
||||
{ "productId": "p3", "quantity": 5, "price": 200.0, "weight": 100 },
|
||||
{ "productId": "p4", "quantity": 1, "price": 50.0, "weight": 400 },
|
||||
{ "productId": "p5", "quantity": 10, "price": 10.0, "weight": 100 }
|
||||
]
|
||||
}
|
||||
"""
|
||||
When I send a POST request to "/orders"
|
||||
Then the response status should be 422
|
||||
|
||||
Scenario: Locked clients should not be allowed
|
||||
Given the request has the following body:
|
||||
"""
|
||||
{
|
||||
"orderId": "018eddae-ff52-7813-9f88-ada9e61a76f3",
|
||||
"clientId": "018ede1e-a587-76a8-88f3-23987a8dade9",
|
||||
"products": [
|
||||
{ "productId": "p1", "quantity": 2, "price": 20.0, "weight": 100 },
|
||||
{ "productId": "p2", "quantity": 1, "price": 100.0, "weight": 200 },
|
||||
{ "productId": "p3", "quantity": 5, "price": 200.0, "weight": 100 },
|
||||
{ "productId": "p4", "quantity": 1, "price": 50.0, "weight": 400 },
|
||||
{ "productId": "p5", "quantity": 10, "price": 10.0, "weight": 100 }
|
||||
]
|
||||
}
|
||||
"""
|
||||
When I send a POST request to "/orders"
|
||||
Then the response status should be 403
|
||||
|
||||
Scenario: Orders above 24 tons should not be allowed
|
||||
Given the request has the following body:
|
||||
"""
|
||||
{
|
||||
"orderId": "018eddae-ff52-7813-9f88-ada9e61a76f3",
|
||||
"clientId": "018edd2e-894a-78d7-b10c-16e05ca933a3",
|
||||
"products": [
|
||||
{ "productId": "p1", "quantity": 2, "price": 20.0, "weight": 100 },
|
||||
{ "productId": "p2", "quantity": 1, "price": 100.0, "weight": 200 },
|
||||
{ "productId": "p3", "quantity": 5, "price": 200.0, "weight": 100 },
|
||||
{ "productId": "p4", "quantity": 1, "price": 50.0, "weight": 400 },
|
||||
{ "productId": "p5", "quantity": 10, "price": 10.0, "weight": 10000 }
|
||||
]
|
||||
}
|
||||
"""
|
||||
When I send a POST request to "/orders"
|
||||
Then the response status should be 400
|
||||
|
||||
Scenario: Clients with negative balance should not be allowed
|
||||
Given the request has the following body:
|
||||
"""
|
||||
{
|
||||
"orderId": "018eddae-ff52-7813-9f88-ada9e61a76f3",
|
||||
"clientId": "018ede4c-f5ac-7be9-9f14-7c00d2636812",
|
||||
"products": [
|
||||
{ "productId": "p1", "quantity": 2, "price": 20.0, "weight": 100 },
|
||||
{ "productId": "p2", "quantity": 1, "price": 100.0, "weight": 200 },
|
||||
{ "productId": "p3", "quantity": 5, "price": 200.0, "weight": 100 },
|
||||
{ "productId": "p4", "quantity": 1, "price": 50.0, "weight": 400 },
|
||||
{ "productId": "p5", "quantity": 10, "price": 10.0, "weight": 10000 }
|
||||
]
|
||||
}
|
||||
"""
|
||||
When I send a POST request to "/orders"
|
||||
Then the response status should be 400
|
||||
|
||||
Scenario: Orders below 5 products should not be allowed
|
||||
Given the request has the following body:
|
||||
"""
|
||||
{
|
||||
"orderId": "018eddae-ff52-7813-9f88-ada9e61a76f3",
|
||||
"clientId": "018edd2e-894a-78d7-b10c-16e05ca933a3",
|
||||
"products": [
|
||||
{ "productId": "p1", "quantity": 2, "price": 20.0, "weight": 100 }
|
||||
]
|
||||
}
|
||||
"""
|
||||
When I send a POST request to "/orders"
|
||||
Then the response status should be 400
|
||||
|
||||
Scenario: Invalid order should not deduce from balance
|
||||
Given the request has the following body:
|
||||
"""
|
||||
{
|
||||
"orderId": "018eddae-ff52-7813-9f88-ada9e61a76f3",
|
||||
"clientId": "018edd2e-894a-78d7-b10c-16e05ca933a3",
|
||||
"products": [
|
||||
{ "productId": "p1", "quantity": 2, "price": 20.0, "weight": 100 }
|
||||
]
|
||||
}
|
||||
"""
|
||||
When I send a POST request to "/orders"
|
||||
Then client with id "018edd2e-894a-78d7-b10c-16e05ca933a3" should have balance of 200000
|
@ -3,8 +3,9 @@ Feature:
|
||||
|
||||
Background:
|
||||
Given there exist following clients:
|
||||
| clientId | name | initialBalance | currentBalance |
|
||||
| 018edd2e-894a-78d7-b10c-16e05ca933a3 | Kacper | 10000 | 5000 |
|
||||
| clientId | name | initialBalance | currentBalance | locked |
|
||||
| 018edd2e-894a-78d7-b10c-16e05ca933a3 | Kacper | 10000 | 5000 | no |
|
||||
| 018ede1e-a587-76a8-88f3-23987a8dade9 | Jacek | 10000 | 60000000 | yes |
|
||||
|
||||
Scenario: CRM is able to create new clients
|
||||
Given the request has the following body:
|
||||
@ -20,6 +21,17 @@ Feature:
|
||||
And client with id "018edd37-c145-7143-ba91-c191084e4fba" should exist
|
||||
And client with id "018edd37-c145-7143-ba91-c191084e4fba" should have balance of 100000
|
||||
|
||||
Scenario: CRM is able to top up existing client
|
||||
Given the request has the following body:
|
||||
"""
|
||||
{
|
||||
"amount": 5000
|
||||
}
|
||||
"""
|
||||
When I send a POST request to "/clients/018edd2e-894a-78d7-b10c-16e05ca933a3/_top-up"
|
||||
Then the response status should be 200
|
||||
And client with id "018edd2e-894a-78d7-b10c-16e05ca933a3" should have balance of 10000
|
||||
|
||||
Scenario: CRM should not be able to override existing user
|
||||
Given the request has the following body:
|
||||
"""
|
||||
@ -32,3 +44,25 @@ Feature:
|
||||
When I send a POST request to "/clients"
|
||||
Then the response status should be 409
|
||||
And client with id "018edd2e-894a-78d7-b10c-16e05ca933a3" should exist
|
||||
|
||||
Scenario: CRM is able to lock existing client
|
||||
Given the request has the following body:
|
||||
"""
|
||||
{
|
||||
"lock": true
|
||||
}
|
||||
"""
|
||||
When I send a POST request to "/clients/018edd2e-894a-78d7-b10c-16e05ca933a3/_lock"
|
||||
Then the response status should be 200
|
||||
And client with id "018edd2e-894a-78d7-b10c-16e05ca933a3" should be locked
|
||||
|
||||
Scenario: CRM is able to unlock existing client
|
||||
Given the request has the following body:
|
||||
"""
|
||||
{
|
||||
"lock": false
|
||||
}
|
||||
"""
|
||||
When I send a POST request to "/clients/018ede1e-a587-76a8-88f3-23987a8dade9/_lock"
|
||||
Then the response status should be 200
|
||||
And client with id "018ede1e-a587-76a8-88f3-23987a8dade9" should not be locked
|
||||
|
18
src/Aggregate/AcceptOrderCommand.php
Normal file
18
src/Aggregate/AcceptOrderCommand.php
Normal file
@ -0,0 +1,18 @@
|
||||
<?php
|
||||
|
||||
namespace App\Aggregate;
|
||||
|
||||
use App\Contract\Order;
|
||||
use App\Entity\Client;
|
||||
use App\Validator as CustomConstraint;
|
||||
use Symfony\Component\Uid\Uuid;
|
||||
|
||||
readonly class AcceptOrderCommand
|
||||
{
|
||||
public function __construct(
|
||||
#[CustomConstraint\ValidOrder]
|
||||
public Order $order,
|
||||
public Client $client,
|
||||
) {
|
||||
}
|
||||
}
|
8
src/Contract/LockClientRequestDto.php
Normal file
8
src/Contract/LockClientRequestDto.php
Normal file
@ -0,0 +1,8 @@
|
||||
<?php
|
||||
|
||||
namespace App\Contract;
|
||||
|
||||
readonly class LockClientRequestDto
|
||||
{
|
||||
public function __construct(public bool $lock) {}
|
||||
}
|
71
src/Contract/Order.php
Normal file
71
src/Contract/Order.php
Normal file
@ -0,0 +1,71 @@
|
||||
<?php
|
||||
|
||||
namespace App\Contract;
|
||||
|
||||
use App\Service\PricingStrategy;
|
||||
use App\Service\TrivialPricingStrategy;
|
||||
use Symfony\Component\Uid\Uuid;
|
||||
|
||||
class Order
|
||||
{
|
||||
public function __construct(
|
||||
private Uuid $orderId,
|
||||
private Uuid $clientId,
|
||||
/** @var ProductEntry[] */
|
||||
private array $products,
|
||||
) {
|
||||
}
|
||||
|
||||
public function getOrderId(): Uuid
|
||||
{
|
||||
return $this->orderId;
|
||||
}
|
||||
|
||||
public function setOrderId(Uuid $orderId): void
|
||||
{
|
||||
$this->orderId = $orderId;
|
||||
}
|
||||
|
||||
public function getClientId(): Uuid
|
||||
{
|
||||
return $this->clientId;
|
||||
}
|
||||
|
||||
public function setClientId(Uuid $clientId): void
|
||||
{
|
||||
$this->clientId = $clientId;
|
||||
}
|
||||
|
||||
public function getProducts(): array
|
||||
{
|
||||
return array_values($this->products);
|
||||
}
|
||||
|
||||
public function setProducts(array $products): void
|
||||
{
|
||||
$this->products = array_combine(
|
||||
array_map(fn (ProductEntry $product) => $product->getProductId(), $products),
|
||||
array_values($products)
|
||||
);
|
||||
}
|
||||
|
||||
public function addProduct(ProductEntry $product): void
|
||||
{
|
||||
$this->products[$product->getProductId()] = $product;
|
||||
}
|
||||
|
||||
public function removeProduct(ProductEntry $product): void
|
||||
{
|
||||
$this->products = array_filter($this->products, fn (ProductEntry $other) => $other->getProductId() != $product->getProductId());
|
||||
}
|
||||
|
||||
public function getTotalWeight(): float
|
||||
{
|
||||
return array_sum(array_map(fn (ProductEntry $product) => $product->getTotalWeight(), $this->products));
|
||||
}
|
||||
|
||||
public function getTotalPrice(PricingStrategy $strategy): int
|
||||
{
|
||||
return $strategy->calculateTotalPriceOfOrder($this);
|
||||
}
|
||||
}
|
@ -1,45 +0,0 @@
|
||||
<?php
|
||||
|
||||
namespace App\Contract;
|
||||
|
||||
use Symfony\Component\Uid\Uuid;
|
||||
|
||||
class OrderDto
|
||||
{
|
||||
public function __construct(
|
||||
private Uuid $orderId,
|
||||
private Uuid $clientId,
|
||||
private array $products,
|
||||
) {
|
||||
}
|
||||
|
||||
public function getOrderId(): Uuid
|
||||
{
|
||||
return $this->orderId;
|
||||
}
|
||||
|
||||
public function setOrderId(Uuid $orderId): void
|
||||
{
|
||||
$this->orderId = $orderId;
|
||||
}
|
||||
|
||||
public function getClientId(): Uuid
|
||||
{
|
||||
return $this->clientId;
|
||||
}
|
||||
|
||||
public function setClientId(Uuid $clientId): void
|
||||
{
|
||||
$this->clientId = $clientId;
|
||||
}
|
||||
|
||||
public function getProducts(): array
|
||||
{
|
||||
return $this->products;
|
||||
}
|
||||
|
||||
public function setProducts(array $products): void
|
||||
{
|
||||
$this->products = $products;
|
||||
}
|
||||
}
|
@ -2,7 +2,10 @@
|
||||
|
||||
namespace App\Contract;
|
||||
|
||||
class ProductDto
|
||||
use App\Service\PricingStrategy;
|
||||
use App\Service\TrivialPricingStrategy;
|
||||
|
||||
class ProductEntry
|
||||
{
|
||||
public function __construct(
|
||||
private string $productId,
|
||||
@ -37,6 +40,11 @@ class ProductDto
|
||||
return $this->price;
|
||||
}
|
||||
|
||||
public function getPriceAsInt(): int
|
||||
{
|
||||
return round($this->getPrice() * 100);
|
||||
}
|
||||
|
||||
public function setPrice(float $price): void
|
||||
{
|
||||
$this->price = $price;
|
||||
@ -51,4 +59,14 @@ class ProductDto
|
||||
{
|
||||
$this->weight = $weight;
|
||||
}
|
||||
|
||||
public function getTotalWeight(): float
|
||||
{
|
||||
return $this->getWeight() * $this->getQuantity();
|
||||
}
|
||||
|
||||
public function getTotalPrice(PricingStrategy $strategy): float
|
||||
{
|
||||
return $strategy->calculateTotalPriceOfProductEntry($this);
|
||||
}
|
||||
}
|
8
src/Contract/TopUpRequestDto.php
Normal file
8
src/Contract/TopUpRequestDto.php
Normal file
@ -0,0 +1,8 @@
|
||||
<?php
|
||||
|
||||
namespace App\Contract;
|
||||
|
||||
readonly class TopUpRequestDto
|
||||
{
|
||||
public function __construct(public int $amount) {}
|
||||
}
|
@ -3,6 +3,8 @@
|
||||
namespace App\Controller;
|
||||
|
||||
use App\Contract\ClientDto;
|
||||
use App\Contract\LockClientRequestDto;
|
||||
use App\Contract\TopUpRequestDto;
|
||||
use App\Entity\Client;
|
||||
use App\Exception\ClientAlreadyExistsException;
|
||||
use App\Service\ClientService;
|
||||
@ -10,6 +12,7 @@ use Symfony\Component\HttpFoundation\Response;
|
||||
use Symfony\Component\HttpKernel\Attribute\AsController;
|
||||
use Symfony\Component\HttpKernel\Attribute\MapRequestPayload;
|
||||
use Symfony\Component\Routing\Attribute\Route;
|
||||
use Symfony\Component\Uid\Uuid;
|
||||
|
||||
#[Route(path: '/clients', name: 'clients_')]
|
||||
#[AsController]
|
||||
@ -37,4 +40,28 @@ readonly class ClientController
|
||||
|
||||
return new Response(status: Response::HTTP_CREATED);
|
||||
}
|
||||
|
||||
#[Route(path: '/{client}/_top-up', name: 'top_up', methods: ['POST'])]
|
||||
public function topUp(
|
||||
#[MapRequestPayload] TopUpRequestDto $topUpRequestDto,
|
||||
Client $client,
|
||||
): Response {
|
||||
$this->clientService->topUpBalance($client, $topUpRequestDto->amount);
|
||||
|
||||
return new Response();
|
||||
}
|
||||
|
||||
#[Route(path: '/{client}/_lock', name: 'lock', methods: ['POST'])]
|
||||
public function lock(
|
||||
#[MapRequestPayload] LockClientRequestDto $lockClientRequestDto,
|
||||
Client $client,
|
||||
): Response {
|
||||
if ($lockClientRequestDto->lock) {
|
||||
$this->clientService->lock($client);
|
||||
} else {
|
||||
$this->clientService->unlock($client);
|
||||
}
|
||||
|
||||
return new Response();
|
||||
}
|
||||
}
|
||||
|
58
src/Controller/OrderController.php
Normal file
58
src/Controller/OrderController.php
Normal file
@ -0,0 +1,58 @@
|
||||
<?php
|
||||
|
||||
declare(strict_types=1);
|
||||
|
||||
namespace App\Controller;
|
||||
|
||||
use App\Aggregate\AcceptOrderCommand;
|
||||
use App\Contract\Order;
|
||||
use App\Exception\ClientLockedException;
|
||||
use App\Exception\ClientNotExistsException;
|
||||
use App\Exception\OrderInvalidException;
|
||||
use App\Service\ClientService;
|
||||
use App\Service\OrderService;
|
||||
use Symfony\Component\HttpFoundation\JsonResponse;
|
||||
use Symfony\Component\HttpFoundation\Response;
|
||||
use Symfony\Component\HttpKernel\Attribute\AsController;
|
||||
use Symfony\Component\HttpKernel\Attribute\MapRequestPayload;
|
||||
use Symfony\Component\Routing\Attribute\Route;
|
||||
use Symfony\Component\Serializer\SerializerInterface;
|
||||
|
||||
#[AsController]
|
||||
#[Route(path: '/orders', name: 'orders_')]
|
||||
readonly class OrderController
|
||||
{
|
||||
public function __construct(
|
||||
private ClientService $clientService,
|
||||
private OrderService $orderService,
|
||||
private SerializerInterface $serializer,
|
||||
) {
|
||||
|
||||
}
|
||||
|
||||
#[Route(name: 'accept', methods: ['POST'])]
|
||||
public function accept(#[MapRequestPayload] Order $orderDto): Response
|
||||
{
|
||||
try {
|
||||
$command = new AcceptOrderCommand(
|
||||
order: $orderDto,
|
||||
client: $this->clientService->getClient($orderDto->getClientId()),
|
||||
);
|
||||
|
||||
$this->orderService->acceptOrder($command);
|
||||
|
||||
return new Response();
|
||||
} catch (ClientNotExistsException) {
|
||||
return new Response(status: Response::HTTP_UNPROCESSABLE_ENTITY);
|
||||
} catch (ClientLockedException) {
|
||||
return new Response(status: Response::HTTP_FORBIDDEN);
|
||||
} catch (OrderInvalidException $orderInvalidException) {
|
||||
return new JsonResponse(
|
||||
$this->serializer->serialize($orderInvalidException->getViolations(), format: 'json'),
|
||||
status: Response::HTTP_BAD_REQUEST,
|
||||
json: true
|
||||
);
|
||||
}
|
||||
}
|
||||
|
||||
}
|
@ -23,6 +23,9 @@ class Client
|
||||
|
||||
#[ORM\Column]
|
||||
private ?int $currentBalance = null,
|
||||
|
||||
#[ORM\Column(nullable: true)]
|
||||
private ?\DateTimeImmutable $lockedAt = null,
|
||||
) {
|
||||
$this->currentBalance ??= $this->initialBalance;
|
||||
}
|
||||
@ -56,4 +59,29 @@ class Client
|
||||
{
|
||||
return $this->currentBalance;
|
||||
}
|
||||
|
||||
public function deduceFromBalance(int $amount): void
|
||||
{
|
||||
$this->currentBalance -= $amount;
|
||||
}
|
||||
|
||||
public function topUpBalance(int $amount): void
|
||||
{
|
||||
$this->currentBalance += $amount;
|
||||
}
|
||||
|
||||
public function setLockedAt(?\DateTimeImmutable $dateTime)
|
||||
{
|
||||
$this->lockedAt = $dateTime;
|
||||
}
|
||||
|
||||
public function getLockedAt(): ?\DateTimeImmutable
|
||||
{
|
||||
return $this->lockedAt;
|
||||
}
|
||||
|
||||
public function isLocked(): bool
|
||||
{
|
||||
return $this->lockedAt !== null;
|
||||
}
|
||||
}
|
||||
|
7
src/Exception/ClientLockedException.php
Normal file
7
src/Exception/ClientLockedException.php
Normal file
@ -0,0 +1,7 @@
|
||||
<?php
|
||||
|
||||
namespace App\Exception;
|
||||
|
||||
class ClientLockedException extends \LogicException
|
||||
{
|
||||
}
|
7
src/Exception/ClientNotExistsException.php
Normal file
7
src/Exception/ClientNotExistsException.php
Normal file
@ -0,0 +1,7 @@
|
||||
<?php
|
||||
|
||||
namespace App\Exception;
|
||||
|
||||
class ClientNotExistsException extends \LogicException
|
||||
{
|
||||
}
|
28
src/Exception/OrderInvalidException.php
Normal file
28
src/Exception/OrderInvalidException.php
Normal file
@ -0,0 +1,28 @@
|
||||
<?php
|
||||
|
||||
namespace App\Exception;
|
||||
|
||||
use App\Contract\Order;
|
||||
use Symfony\Component\Validator\ConstraintViolationListInterface;
|
||||
|
||||
class OrderInvalidException extends \LogicException
|
||||
{
|
||||
private ConstraintViolationListInterface $violations;
|
||||
|
||||
public function getViolations(): ConstraintViolationListInterface
|
||||
{
|
||||
return $this->violations;
|
||||
}
|
||||
|
||||
public static function createFromViolations(Order $order, ConstraintViolationListInterface $violations): self
|
||||
{
|
||||
$exception = new self(sprintf(
|
||||
'Order "%s" is not valid, contains %d errors.',
|
||||
$order->getOrderId()->toRfc4122(),
|
||||
$violations->count()
|
||||
));
|
||||
$exception->violations = $violations;
|
||||
|
||||
return $exception;
|
||||
}
|
||||
}
|
@ -4,15 +4,35 @@ namespace App\Service;
|
||||
|
||||
use App\Entity\Client;
|
||||
use App\Exception\ClientAlreadyExistsException;
|
||||
use App\Exception\ClientNotExistsException;
|
||||
use App\Repository\ClientRepository;
|
||||
use Psr\Clock\ClockInterface;
|
||||
use Symfony\Component\Uid\Uuid;
|
||||
|
||||
class ClientService
|
||||
{
|
||||
public function __construct(
|
||||
public readonly ClientRepository $clientRepository
|
||||
public readonly ClientRepository $clientRepository,
|
||||
public readonly ClockInterface $clock,
|
||||
) {
|
||||
}
|
||||
|
||||
/**
|
||||
* @throws \App\Exception\ClientNotExistsException
|
||||
*/
|
||||
public function getClient(Uuid $clientId): Client
|
||||
{
|
||||
$client = $this->clientRepository->find($clientId);
|
||||
|
||||
if (!$client) {
|
||||
throw new ClientNotExistsException(
|
||||
message: sprintf('Client with id "%s" does not exist.', $clientId->toRfc4122())
|
||||
);
|
||||
}
|
||||
|
||||
return $client;
|
||||
}
|
||||
|
||||
public function createNewClient(Client $client): Client
|
||||
{
|
||||
$existing = $this->clientRepository->find($client->getClientId());
|
||||
@ -25,4 +45,32 @@ class ClientService
|
||||
|
||||
return $this->clientRepository->save($client);
|
||||
}
|
||||
|
||||
public function deduceFromBalance(Client $client, int $amount): void
|
||||
{
|
||||
$client->deduceFromBalance($amount);
|
||||
|
||||
$this->clientRepository->save($client);
|
||||
}
|
||||
|
||||
public function topUpBalance(Client $client, int $amount): void
|
||||
{
|
||||
$client->topUpBalance($amount);
|
||||
|
||||
$this->clientRepository->save($client);
|
||||
}
|
||||
|
||||
public function lock(Client $client): void
|
||||
{
|
||||
$client->setLockedAt($this->clock->now());
|
||||
|
||||
$this->clientRepository->save($client);
|
||||
}
|
||||
|
||||
public function unlock(Client $client): void
|
||||
{
|
||||
$client->setLockedAt(null);
|
||||
|
||||
$this->clientRepository->save($client);
|
||||
}
|
||||
}
|
||||
|
47
src/Service/OrderService.php
Normal file
47
src/Service/OrderService.php
Normal file
@ -0,0 +1,47 @@
|
||||
<?php
|
||||
|
||||
namespace App\Service;
|
||||
|
||||
use App\Aggregate\AcceptOrderCommand;
|
||||
use App\Exception\ClientLockedException;
|
||||
use App\Exception\OrderInvalidException;
|
||||
use Symfony\Component\Serializer\SerializerInterface;
|
||||
use Symfony\Component\Validator\Validator\ValidatorInterface;
|
||||
use Symfony\Contracts\HttpClient\HttpClientInterface;
|
||||
|
||||
readonly class OrderService
|
||||
{
|
||||
public function __construct(
|
||||
private HttpClientInterface $crmClient,
|
||||
private SerializerInterface $serializer,
|
||||
private PricingStrategy $pricingStrategy,
|
||||
private ClientService $clientService,
|
||||
private ValidatorInterface $validator,
|
||||
) {
|
||||
}
|
||||
|
||||
/**
|
||||
* @throws \App\Exception\ClientLockedException
|
||||
* @throws \App\Exception\OrderInvalidException
|
||||
*/
|
||||
public function acceptOrder(AcceptOrderCommand $command)
|
||||
{
|
||||
$violations = $this->validator->validate($command);
|
||||
|
||||
if ($violations->count() > 0) {
|
||||
throw OrderInvalidException::createFromViolations($command->order, $violations);
|
||||
}
|
||||
|
||||
if ($command->client->isLocked()) {
|
||||
throw new ClientLockedException(sprintf("Client '%s' is locked.", $command->client->getName()));
|
||||
}
|
||||
|
||||
$this->crmClient->request('POST', '/order', [
|
||||
'body' => $this->serializer->serialize($command->order, format: 'json')
|
||||
]);
|
||||
|
||||
$totalPrice = $command->order->getTotalPrice($this->pricingStrategy);
|
||||
|
||||
$this->clientService->deduceFromBalance($command->client, $totalPrice);
|
||||
}
|
||||
}
|
15
src/Service/PricingStrategy.php
Normal file
15
src/Service/PricingStrategy.php
Normal file
@ -0,0 +1,15 @@
|
||||
<?php
|
||||
|
||||
namespace App\Service;
|
||||
|
||||
use App\Contract\Order;
|
||||
use App\Contract\ProductEntry;
|
||||
|
||||
interface PricingStrategy
|
||||
{
|
||||
|
||||
public function calculateTotalPriceOfProductEntry(ProductEntry $product): int;
|
||||
|
||||
public function calculateTotalPriceOfOrder(Order $order): int;
|
||||
|
||||
}
|
21
src/Service/TrivialPricingStrategy.php
Normal file
21
src/Service/TrivialPricingStrategy.php
Normal file
@ -0,0 +1,21 @@
|
||||
<?php
|
||||
|
||||
namespace App\Service;
|
||||
|
||||
use App\Contract\Order;
|
||||
use App\Contract\ProductEntry;
|
||||
|
||||
class TrivialPricingStrategy implements PricingStrategy
|
||||
{
|
||||
public function calculateTotalPriceOfProductEntry(ProductEntry $product): int
|
||||
{
|
||||
$priceInBaseUnit = $product->getPriceAsInt();
|
||||
|
||||
return $priceInBaseUnit * $product->getQuantity();
|
||||
}
|
||||
|
||||
public function calculateTotalPriceOfOrder(Order $order): int
|
||||
{
|
||||
return array_sum(array_map($this->calculateTotalPriceOfProductEntry(...), $order->getProducts()));
|
||||
}
|
||||
}
|
14
src/Validator/ValidOrder.php
Normal file
14
src/Validator/ValidOrder.php
Normal file
@ -0,0 +1,14 @@
|
||||
<?php
|
||||
|
||||
namespace App\Validator;
|
||||
|
||||
use Symfony\Component\Validator\Constraint;
|
||||
|
||||
#[\Attribute(\Attribute::TARGET_CLASS | \Attribute::TARGET_PROPERTY)]
|
||||
class ValidOrder extends Constraint
|
||||
{
|
||||
public int $minProductCount = 5;
|
||||
|
||||
public float $maxWeight = 24_000;
|
||||
|
||||
}
|
80
src/Validator/ValidOrderValidator.php
Normal file
80
src/Validator/ValidOrderValidator.php
Normal file
@ -0,0 +1,80 @@
|
||||
<?php
|
||||
|
||||
namespace App\Validator;
|
||||
|
||||
use App\Aggregate\AcceptOrderCommand;
|
||||
use App\Contract\Order;
|
||||
use App\Entity\Client;
|
||||
use App\Service\ClientService;
|
||||
use Symfony\Component\Validator\Constraint;
|
||||
use Symfony\Component\Validator\ConstraintValidator;
|
||||
use Symfony\Component\Validator\Exception\UnexpectedValueException;
|
||||
|
||||
class ValidOrderValidator extends ConstraintValidator
|
||||
{
|
||||
public function __construct(private readonly ClientService $clientService)
|
||||
{
|
||||
}
|
||||
|
||||
public function validate($value, Constraint $constraint)
|
||||
{
|
||||
if (!$value instanceof Order) {
|
||||
throw new UnexpectedValueException($value, Order::class);
|
||||
}
|
||||
|
||||
if (!$constraint instanceof ValidOrder) {
|
||||
throw new UnexpectedValueException($constraint, ValidOrder::class);
|
||||
}
|
||||
|
||||
$client = $this->getClient($value);
|
||||
|
||||
$this->checkClientBalance($client);
|
||||
|
||||
$this->checkMinimumProductCount($value, $constraint);
|
||||
$this->checkMaximumWeight($value, $constraint);
|
||||
}
|
||||
|
||||
private function checkMinimumProductCount(Order $value, ValidOrder $constraint): void
|
||||
{
|
||||
if (count($value->getProducts()) < $constraint->minProductCount) {
|
||||
$this->context
|
||||
->buildViolation('The number of products in order should be no less than {{ min }}')
|
||||
->setParameter('{{ min }}', $constraint->minProductCount)
|
||||
->atPath('products')
|
||||
->addViolation();
|
||||
}
|
||||
}
|
||||
|
||||
private function checkMaximumWeight(Order $value, ValidOrder $constraint): void
|
||||
{
|
||||
if ($value->getTotalWeight() > $constraint->maxWeight) {
|
||||
$this->context
|
||||
->buildViolation('Total weight of products in order must not exceed {{ max }}kg')
|
||||
->setParameter('{{ max }}', $constraint->maxWeight)
|
||||
->atPath('products')
|
||||
->addViolation();
|
||||
}
|
||||
}
|
||||
|
||||
private function getClient(Order $value)
|
||||
{
|
||||
$root = $this->context->getRoot();
|
||||
|
||||
if ($root instanceof AcceptOrderCommand) {
|
||||
return $root->client;
|
||||
}
|
||||
|
||||
return $this->clientService->getClient($value->getClientId());
|
||||
}
|
||||
|
||||
private function checkClientBalance(Client $client): void
|
||||
{
|
||||
if ($client->getBalance() < 0) {
|
||||
$this->context
|
||||
->buildViolation('Total balance of the client is negative')
|
||||
->atPath('clientId')
|
||||
->addViolation();
|
||||
}
|
||||
}
|
||||
|
||||
}
|
12
symfony.lock
12
symfony.lock
@ -159,5 +159,17 @@
|
||||
"version": "7.0",
|
||||
"ref": "0df5844274d871b37fc3816c57a768ffc60a43a5"
|
||||
}
|
||||
},
|
||||
"symfony/validator": {
|
||||
"version": "7.0",
|
||||
"recipe": {
|
||||
"repo": "github.com/symfony/recipes",
|
||||
"branch": "main",
|
||||
"version": "7.0",
|
||||
"ref": "8c1c4e28d26a124b0bb273f537ca8ce443472bfd"
|
||||
},
|
||||
"files": [
|
||||
"config/packages/validator.yaml"
|
||||
]
|
||||
}
|
||||
}
|
||||
|
88
tests/App/Service/OrderServiceTest.php
Normal file
88
tests/App/Service/OrderServiceTest.php
Normal file
@ -0,0 +1,88 @@
|
||||
<?php
|
||||
|
||||
namespace App\Service;
|
||||
|
||||
use App\Aggregate\AcceptOrderCommand;
|
||||
use App\Contract\Order;
|
||||
use App\Contract\ProductEntry;
|
||||
use App\Entity\Client;
|
||||
use PHPUnit\Framework\TestCase;
|
||||
use Symfony\Component\Serializer\SerializerInterface;
|
||||
use Symfony\Component\Uid\Uuid;
|
||||
use Symfony\Component\Validator\Validator\ValidatorInterface;
|
||||
use Symfony\Contracts\HttpClient\HttpClientInterface;
|
||||
|
||||
class OrderServiceTest extends TestCase
|
||||
{
|
||||
private const ORDER_ID = '018eddae-ff52-7813-9f88-ada9e61a76f3';
|
||||
private const CLIENT_ID = '018edd2e-894a-78d7-b10c-16e05ca933a3';
|
||||
|
||||
private function createClient(int $balance): Client
|
||||
{
|
||||
return new Client(
|
||||
clientId: Uuid::fromRfc4122(self::CLIENT_ID),
|
||||
name: 'Stub',
|
||||
initialBalance: $balance,
|
||||
currentBalance: $balance
|
||||
);
|
||||
}
|
||||
|
||||
private function createValidOrderDto(): Order
|
||||
{
|
||||
return new Order(
|
||||
orderId: Uuid::fromRfc4122(self::ORDER_ID),
|
||||
clientId: Uuid::fromRfc4122(self::CLIENT_ID),
|
||||
products: [
|
||||
new ProductEntry('p1', 1, 100.00, 100),
|
||||
new ProductEntry('p2', 2, 300.00, 100),
|
||||
new ProductEntry('p3', 3, 200.00, 100),
|
||||
new ProductEntry('p4', 4, 400.00, 700),
|
||||
new ProductEntry('p5', 5, 800.00, 700),
|
||||
]
|
||||
);
|
||||
}
|
||||
|
||||
private function createOrderService(
|
||||
?HttpClientInterface $httpClientMock = null,
|
||||
?SerializerInterface $serializerMock = null,
|
||||
?PricingStrategy $pricingStrategyMock = null,
|
||||
?ClientService $clientServiceMock = null,
|
||||
?ValidatorInterface $validatorMock = null,
|
||||
) {
|
||||
$httpClientMock ??= $this->createMock(HttpClientInterface::class);
|
||||
$serializerMock ??= $this->createMock(SerializerInterface::class);
|
||||
$pricingStrategyMock ??= $this->createMock(PricingStrategy::class);
|
||||
$clientServiceMock ??= $this->createMock(ClientService::class);
|
||||
$validatorMock ??= $this->createMock(ValidatorInterface::class);
|
||||
|
||||
return new OrderService($httpClientMock, $serializerMock, $pricingStrategyMock, $clientServiceMock, $validatorMock);
|
||||
}
|
||||
|
||||
public function testCrmGetsCalledOnValidOrderRequest()
|
||||
{
|
||||
$httpClientMock = $this->createMock(HttpClientInterface::class);
|
||||
$httpClientMock->expects($this->once())->method('request')->with('POST', '/order');
|
||||
|
||||
$order = new AcceptOrderCommand(
|
||||
$this->createValidOrderDto(),
|
||||
$this->createClient(balance: 100_00)
|
||||
);
|
||||
|
||||
$sut = $this->createOrderService(httpClientMock: $httpClientMock);
|
||||
$sut->acceptOrder($order);
|
||||
}
|
||||
|
||||
public function testClientIsUpdatedAfterPlacingOrder()
|
||||
{
|
||||
$command = new AcceptOrderCommand(
|
||||
$this->createValidOrderDto(),
|
||||
$client = $this->createClient(10000_00),
|
||||
);
|
||||
|
||||
$clientServiceMock = $this->createMock(ClientService::class);
|
||||
$clientServiceMock->expects($this->once())->method('deduceFromBalance');
|
||||
|
||||
$sut = $this->createOrderService(clientServiceMock: $clientServiceMock);
|
||||
$sut->acceptOrder($command);
|
||||
}
|
||||
}
|
46
tests/App/Service/TrivialPricingStrategyTest.php
Normal file
46
tests/App/Service/TrivialPricingStrategyTest.php
Normal file
@ -0,0 +1,46 @@
|
||||
<?php
|
||||
|
||||
namespace App\Service;
|
||||
|
||||
use App\Contract\Order;
|
||||
use App\Contract\ProductEntry;
|
||||
use PHPUnit\Framework\TestCase;
|
||||
use Symfony\Component\Uid\Uuid;
|
||||
|
||||
class TrivialPricingStrategyTest extends TestCase
|
||||
{
|
||||
|
||||
public function testCalculatingSingularProductPrice()
|
||||
{
|
||||
$product = new ProductEntry(productId: 'p1', quantity: 1, price: 2.00, weight: 100);
|
||||
|
||||
$sut = new TrivialPricingStrategy();
|
||||
|
||||
$this->assertEquals(200, $sut->calculateTotalPriceOfProductEntry($product));
|
||||
}
|
||||
|
||||
public function testCalculatingMultipleProductPrice()
|
||||
{
|
||||
$product = new ProductEntry(productId: 'p1', quantity: 5, price: 2.00, weight: 100);
|
||||
|
||||
$sut = new TrivialPricingStrategy();
|
||||
|
||||
$this->assertEquals(1000, $sut->calculateTotalPriceOfProductEntry($product));
|
||||
}
|
||||
|
||||
public function testCalculatingOrderPrice()
|
||||
{
|
||||
$order = new Order(
|
||||
orderId: Uuid::v4(),
|
||||
clientId: Uuid::v4(),
|
||||
products: [
|
||||
new ProductEntry(productId: 'p1', quantity: 5, price: 2.00, weight: 100),
|
||||
new ProductEntry(productId: 'p2', quantity: 2, price: 1.25, weight: 100),
|
||||
]
|
||||
);
|
||||
|
||||
$sut = new TrivialPricingStrategy();
|
||||
|
||||
$this->assertEquals(1250, $sut->calculateTotalPriceOfOrder($order));
|
||||
}
|
||||
}
|
@ -47,6 +47,7 @@ class ClientsContext implements Context
|
||||
name: $row['name'],
|
||||
initialBalance: intval($row['initialBalance']),
|
||||
currentBalance: intval($row['currentBalance']),
|
||||
lockedAt: $row['locked'] === 'yes' ? new \DateTimeImmutable() : null,
|
||||
);
|
||||
|
||||
$this->clientRepository->save($client);
|
||||
@ -69,6 +70,22 @@ class ClientsContext implements Context
|
||||
Assert::assertNotNull($this->getClient($clientId));
|
||||
}
|
||||
|
||||
/**
|
||||
* @Given /^client with id "([^"]+)" should be locked$/
|
||||
*/
|
||||
public function clientWithIdIsLocked(string $clientId)
|
||||
{
|
||||
Assert::assertTrue($this->getClient($clientId)->isLocked());
|
||||
}
|
||||
|
||||
/**
|
||||
* @Given /^client with id "([^"]+)" should not be locked$/
|
||||
*/
|
||||
public function clientWithIdIsNotLocked(string $clientId)
|
||||
{
|
||||
Assert::assertFalse($this->getClient($clientId)->isLocked());
|
||||
}
|
||||
|
||||
/**
|
||||
* @Given /^client with id "([^"]+)" should have balance of (\d+)$/
|
||||
*/
|
||||
@ -76,7 +93,7 @@ class ClientsContext implements Context
|
||||
{
|
||||
$client = $this->getClient($clientId);
|
||||
|
||||
Assert::assertEquals($client->getBalance(), $balance);
|
||||
Assert::assertEquals($balance, $client->getBalance());
|
||||
}
|
||||
|
||||
}
|
||||
|
Loading…
Reference in New Issue
Block a user