Compare commits

...

5 Commits

Author SHA1 Message Date
Kacper Donat
05c1a270e5 36: Add order validation 2024-04-14 22:42:40 +02:00
Kacper Donat
5cf26579fc TASK-40: Add ability to lock-up clients 2024-04-14 21:49:23 +02:00
Kacper Donat
85a6212ef5 TASK-39: Add ability to top-up balance of given client
This commit also refactors a bit logic behind deducing from balance to
have better separation of concerns.
2024-04-14 21:25:43 +02:00
Kacper Donat
cbfb10ec10 TASK-38: Deduce from balance after order 2024-04-14 21:15:42 +02:00
Kacper Donat
e223ddfb4f TASK-35: Send orders to CRM 2024-04-14 20:17:21 +02:00
31 changed files with 1350 additions and 129 deletions

2
.env
View File

@ -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"

View File

@ -3,6 +3,7 @@ default:
default:
contexts:
- App\Tests\Behat\ApiCallContext
- App\Tests\Behat\ClientsContext
extensions:
FriendsOfBehat\SymfonyExtension:

View File

@ -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
View File

@ -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",

View File

@ -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

View 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

View File

@ -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)%"

View 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

View File

@ -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

View 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,
) {
}
}

View 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
View 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);
}
}

View File

@ -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;
}
}

View File

@ -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);
}
}

View File

@ -0,0 +1,8 @@
<?php
namespace App\Contract;
readonly class TopUpRequestDto
{
public function __construct(public int $amount) {}
}

View File

@ -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();
}
}

View 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
);
}
}
}

View File

@ -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;
}
}

View File

@ -0,0 +1,7 @@
<?php
namespace App\Exception;
class ClientLockedException extends \LogicException
{
}

View File

@ -0,0 +1,7 @@
<?php
namespace App\Exception;
class ClientNotExistsException extends \LogicException
{
}

View 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;
}
}

View File

@ -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);
}
}

View 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);
}
}

View 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;
}

View 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()));
}
}

View 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;
}

View 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();
}
}
}

View File

@ -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"
]
}
}

View 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);
}
}

View 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));
}
}

View File

@ -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());
}
}