efded9fd40
Auto Tag Develop / tag (push) Successful in 12s
## Objectif Introduit un `CategoryType` dédié **ADRESSE** (module Catalog) consommé par le champ « Catégorie » des blocs adresse, en remplacement de la réutilisation détournée des types CLIENT / FOURNISSEUR. ## Changements **Backend** - Migration de seed du type ADRESSE + 6 catégories : Siège, Contact issues, Facturation, Livraison, Approvisionnement, Méthaniseur (idempotente, réversible) ; fixtures alignées. - `ClientAddress` : validation blacklist (DISTRIBUTEUR/COURTIER) remplacée par une whitelist « catégories de type ADRESSE uniquement ». - `SupplierAddress` : type requis FOURNISSEUR → ADRESSE (le bloc principal fournisseur reste en FOURNISSEUR). **Frontend** - Ref dédiée `addressCategories` (`?typeCode=ADRESSE`) dans les composables référentiels client et fournisseur. - Pages new/edit client et fournisseur câblées sur les blocs adresse. **Tests** - `CategoryAdresseSeedTest` (miroir du test PRESTATAIRE). - Adaptation des tests d'adresse client/fournisseur (sémantique whitelist ADRESSE) + helper `createAddressCategory()`. ## Vérifications - Back : suites Catalog + Architecture + adresse/fournisseur vertes (le flake JWT connu du hook est sans rapport, tests verts en isolation). - Front : Vitest vert (composables référentiels + ciblés). - php-cs-fixer : 0 correction ; eslint : OK. Reviewed-on: #147 Co-authored-by: tristan <tristan@yuno.malio.fr> Co-committed-by: tristan <tristan@yuno.malio.fr>
448 lines
17 KiB
PHP
448 lines
17 KiB
PHP
<?php
|
|
|
|
declare(strict_types=1);
|
|
|
|
namespace App\Tests\Module\Commercial\Api;
|
|
|
|
use App\Module\Sites\Domain\Entity\Site;
|
|
|
|
/**
|
|
* Tests fonctionnels de l'onglet Adresse.
|
|
*
|
|
* RG-1.09 (code postal) et RG-1.10 (>= 1 site) sont DEJA couverts par
|
|
* ClientSubResourceApiTest (ERP-57) et ne sont pas reduplique ici. Ce fichier
|
|
* cible :
|
|
* - RG-1.06 / RG-1.07 / RG-1.08 : exclusivite is_prospect vs
|
|
* is_delivery / is_billing ;
|
|
* - RG-1.11 : billing_email obligatoire ssi is_billing ;
|
|
* - categorie d'adresse : seules les categories de type ADRESSE sont acceptees
|
|
* (-> 422 sinon), au moins une est obligatoire.
|
|
*
|
|
* Depuis ERP-76, ces regles sont portees par des Assert\Callback sur l'entite
|
|
* ClientAddress (mirror applicatif des CHECK Postgres) : la combinaison invalide
|
|
* est donc rejetee en 422 AVANT la base, et non plus par une violation CHECK
|
|
* remontant en 500. Les CHECK BDD restent en filet de securite (non testes ici,
|
|
* inatteignables tant que les validators applicatifs passent en premier).
|
|
*
|
|
* @internal
|
|
*/
|
|
final class ClientAddressTest extends AbstractCommercialApiTestCase
|
|
{
|
|
private const string LD = 'application/ld+json';
|
|
|
|
/**
|
|
* RG-1.06 / RG-1.07 : une adresse de prospection ne peut pas etre une
|
|
* adresse de livraison -> 422 (Assert\Callback, mirror du CHECK
|
|
* chk_client_address_prospect_exclusive).
|
|
*/
|
|
public function testProspectAddressCannotBeDelivery(): void
|
|
{
|
|
$this->skipIfSitesModuleDisabled();
|
|
$client = $this->createAdminClient();
|
|
$seed = $this->seedClient('Prospect Delivery');
|
|
|
|
$client->request('POST', '/api/clients/'.$seed->getId().'/addresses', [
|
|
'headers' => ['Content-Type' => self::LD],
|
|
'json' => [
|
|
'isProspect' => true,
|
|
'isDelivery' => true,
|
|
'postalCode' => '86100',
|
|
'city' => 'Châtellerault',
|
|
'street' => '1 rue du Test',
|
|
'sites' => [$this->firstSiteIri()],
|
|
],
|
|
]);
|
|
|
|
self::assertResponseStatusCodeSame(422);
|
|
}
|
|
|
|
/**
|
|
* RG-1.06 / RG-1.08 : une adresse de prospection ne peut pas etre une
|
|
* adresse de facturation -> 422. On fournit billingEmail pour que la seule
|
|
* violation possible soit l'exclusivite prospect/billing.
|
|
*/
|
|
public function testProspectAddressCannotBeBilling(): void
|
|
{
|
|
$this->skipIfSitesModuleDisabled();
|
|
$client = $this->createAdminClient();
|
|
$seed = $this->seedClient('Prospect Billing');
|
|
|
|
$client->request('POST', '/api/clients/'.$seed->getId().'/addresses', [
|
|
'headers' => ['Content-Type' => self::LD],
|
|
'json' => [
|
|
'isProspect' => true,
|
|
'isBilling' => true,
|
|
'billingEmail' => 'facturation@test.fr',
|
|
'postalCode' => '86100',
|
|
'city' => 'Châtellerault',
|
|
'street' => '1 rue du Test',
|
|
'sites' => [$this->firstSiteIri()],
|
|
],
|
|
]);
|
|
|
|
self::assertResponseStatusCodeSame(422);
|
|
}
|
|
|
|
/**
|
|
* RG-1.11 : une adresse de facturation exige un billingEmail -> 422.
|
|
*/
|
|
public function testBillingAddressRequiresBillingEmail(): void
|
|
{
|
|
$this->skipIfSitesModuleDisabled();
|
|
$client = $this->createAdminClient();
|
|
$seed = $this->seedClient('Billing No Email');
|
|
|
|
$client->request('POST', '/api/clients/'.$seed->getId().'/addresses', [
|
|
'headers' => ['Content-Type' => self::LD],
|
|
'json' => [
|
|
'isBilling' => true,
|
|
'postalCode' => '86100',
|
|
'city' => 'Châtellerault',
|
|
'street' => '1 rue du Test',
|
|
'sites' => [$this->firstSiteIri()],
|
|
],
|
|
]);
|
|
|
|
self::assertResponseStatusCodeSame(422);
|
|
}
|
|
|
|
/**
|
|
* RG-1.11 (cas chaine vide) : une adresse de facturation avec un billingEmail
|
|
* vide ("") doit etre rejetee en 422, et NON passer la validation pour finir
|
|
* en 500 sur le CHECK BDD. Le ClientAddressProcessor normalise "" -> null
|
|
* APRES la validation : le callback doit donc traiter "" comme « absent ».
|
|
*/
|
|
public function testBillingAddressRejectsEmptyBillingEmail(): void
|
|
{
|
|
$this->skipIfSitesModuleDisabled();
|
|
$client = $this->createAdminClient();
|
|
$seed = $this->seedClient('Billing Empty Email');
|
|
|
|
$client->request('POST', '/api/clients/'.$seed->getId().'/addresses', [
|
|
'headers' => ['Content-Type' => self::LD],
|
|
'json' => [
|
|
'isBilling' => true,
|
|
'billingEmail' => '',
|
|
'postalCode' => '86100',
|
|
'city' => 'Châtellerault',
|
|
'street' => '1 rue du Test',
|
|
'sites' => [$this->firstSiteIri()],
|
|
],
|
|
]);
|
|
|
|
self::assertResponseStatusCodeSame(422);
|
|
}
|
|
|
|
/**
|
|
* RG-1.11 (sens inverse) : une adresse NON facturable ne peut pas porter un
|
|
* billingEmail -> 422.
|
|
*/
|
|
public function testNonBillingAddressRejectsBillingEmail(): void
|
|
{
|
|
$this->skipIfSitesModuleDisabled();
|
|
$client = $this->createAdminClient();
|
|
$seed = $this->seedClient('Non Billing With Email');
|
|
|
|
$client->request('POST', '/api/clients/'.$seed->getId().'/addresses', [
|
|
'headers' => ['Content-Type' => self::LD],
|
|
'json' => [
|
|
'isDelivery' => true,
|
|
'isBilling' => false,
|
|
'billingEmail' => 'parasite@test.fr',
|
|
'postalCode' => '86100',
|
|
'city' => 'Châtellerault',
|
|
'street' => '1 rue du Test',
|
|
'sites' => [$this->firstSiteIri()],
|
|
],
|
|
]);
|
|
|
|
self::assertResponseStatusCodeSame(422);
|
|
}
|
|
|
|
/**
|
|
* RG-1.11 (sens inverse, cas chaine vide) : une adresse NON facturable avec
|
|
* un billingEmail vide ("") est ACCEPTEE (201). Le "" equivaut a « pas
|
|
* d'email » : il ne doit pas declencher la violation « email interdit hors
|
|
* facturation » (sinon un champ simplement vide serait refuse a tort).
|
|
*/
|
|
public function testNonBillingAddressAcceptsEmptyBillingEmail(): void
|
|
{
|
|
$this->skipIfSitesModuleDisabled();
|
|
$client = $this->createAdminClient();
|
|
$seed = $this->seedClient('Non Billing Empty Email');
|
|
$category = $this->createAddressCategory();
|
|
|
|
$client->request('POST', '/api/clients/'.$seed->getId().'/addresses', [
|
|
'headers' => ['Content-Type' => self::LD],
|
|
'json' => [
|
|
'isDelivery' => true,
|
|
'isBilling' => false,
|
|
'billingEmail' => '',
|
|
'postalCode' => '86100',
|
|
'city' => 'Châtellerault',
|
|
'street' => '1 rue du Test',
|
|
'sites' => [$this->firstSiteIri()],
|
|
'categories' => ['/api/categories/'.$category->getId()],
|
|
],
|
|
]);
|
|
|
|
self::assertResponseStatusCodeSame(201);
|
|
}
|
|
|
|
/**
|
|
* ERP-119 : une adresse de facturation accepte un 2e email (optionnel, max 2).
|
|
*/
|
|
public function testBillingAddressAcceptsTwoEmails(): void
|
|
{
|
|
$this->skipIfSitesModuleDisabled();
|
|
$client = $this->createAdminClient();
|
|
$seed = $this->seedClient('Billing Two Emails');
|
|
$category = $this->createAddressCategory();
|
|
|
|
$client->request('POST', '/api/clients/'.$seed->getId().'/addresses', [
|
|
'headers' => ['Content-Type' => self::LD],
|
|
'json' => [
|
|
'isBilling' => true,
|
|
'billingEmail' => 'facturation@test.fr',
|
|
'billingEmailSecondary' => 'compta@test.fr',
|
|
'postalCode' => '86100',
|
|
'city' => 'Châtellerault',
|
|
'street' => '1 rue du Test',
|
|
'sites' => [$this->firstSiteIri()],
|
|
'categories' => ['/api/categories/'.$category->getId()],
|
|
],
|
|
]);
|
|
|
|
self::assertResponseStatusCodeSame(201);
|
|
}
|
|
|
|
/**
|
|
* ERP-119 : le 2e email de facturation, comme le principal, n'est autorise que
|
|
* sur une adresse de facturation -> 422 avec violation sur billingEmailSecondary.
|
|
*/
|
|
public function testSecondaryBillingEmailRejectedOnNonBillingAddress(): void
|
|
{
|
|
$this->skipIfSitesModuleDisabled();
|
|
$client = $this->createAdminClient();
|
|
$seed = $this->seedClient('Secondary Email Non Billing');
|
|
$category = $this->createAddressCategory();
|
|
|
|
$body = $client->request('POST', '/api/clients/'.$seed->getId().'/addresses', [
|
|
'headers' => ['Content-Type' => self::LD, 'Accept' => self::LD],
|
|
'json' => [
|
|
'isDelivery' => true,
|
|
'billingEmailSecondary' => 'compta@test.fr',
|
|
'postalCode' => '86100',
|
|
'city' => 'Châtellerault',
|
|
'street' => '1 rue du Test',
|
|
'sites' => [$this->firstSiteIri()],
|
|
'categories' => ['/api/categories/'.$category->getId()],
|
|
],
|
|
])->toArray(false);
|
|
|
|
self::assertResponseStatusCodeSame(422);
|
|
$byPath = $this->violationsByPath($body);
|
|
self::assertArrayHasKey('billingEmailSecondary', $byPath);
|
|
}
|
|
|
|
/**
|
|
* Une categorie qui n'est PAS de type ADRESSE (ici une categorie CLIENT) est
|
|
* refusee sur une adresse -> 422 avec violation sur le champ `categories`.
|
|
*/
|
|
public function testAddressRejectsNonAddressCategory(): void
|
|
{
|
|
$this->skipIfSitesModuleDisabled();
|
|
$client = $this->createAdminClient();
|
|
$seed = $this->seedClient('Address Non Address Cat');
|
|
// Categorie de type CLIENT (et non ADRESSE) -> doit etre refusee sur l'adresse.
|
|
$category = $this->createCategory('SECTEUR');
|
|
|
|
$client->request('POST', '/api/clients/'.$seed->getId().'/addresses', [
|
|
'headers' => ['Content-Type' => self::LD],
|
|
'json' => [
|
|
'isDelivery' => true,
|
|
'postalCode' => '86100',
|
|
'city' => 'Châtellerault',
|
|
'street' => '1 rue du Test',
|
|
'sites' => [$this->firstSiteIri()],
|
|
'categories' => ['/api/categories/'.$category->getId()],
|
|
],
|
|
]);
|
|
|
|
self::assertResponseStatusCodeSame(422);
|
|
self::assertStringContainsString(
|
|
'Type de catégorie non autorisé (ADRESSE attendu).',
|
|
(string) $client->getResponse()->getContent(false),
|
|
);
|
|
}
|
|
|
|
/**
|
|
* Une categorie de type ADRESSE est acceptee sur une adresse -> 201.
|
|
*/
|
|
public function testAddressAcceptsAddressCategory(): void
|
|
{
|
|
$this->skipIfSitesModuleDisabled();
|
|
$client = $this->createAdminClient();
|
|
$seed = $this->seedClient('Address Address Cat');
|
|
$category = $this->createAddressCategory();
|
|
|
|
$client->request('POST', '/api/clients/'.$seed->getId().'/addresses', [
|
|
'headers' => ['Content-Type' => self::LD],
|
|
'json' => [
|
|
'isDelivery' => true,
|
|
'postalCode' => '86100',
|
|
'city' => 'Châtellerault',
|
|
'street' => '1 rue du Test',
|
|
'sites' => [$this->firstSiteIri()],
|
|
'categories' => ['/api/categories/'.$category->getId()],
|
|
],
|
|
]);
|
|
|
|
self::assertResponseStatusCodeSame(201);
|
|
}
|
|
|
|
/**
|
|
* Spec-front § Adresse : au moins une categorie est obligatoire sur une
|
|
* adresse. POST sans categorie (mais avec site) -> 422.
|
|
*/
|
|
public function testAddressRequiresAtLeastOneCategory(): void
|
|
{
|
|
$this->skipIfSitesModuleDisabled();
|
|
$client = $this->createAdminClient();
|
|
$seed = $this->seedClient('Address No Cat');
|
|
|
|
$client->request('POST', '/api/clients/'.$seed->getId().'/addresses', [
|
|
'headers' => ['Content-Type' => self::LD],
|
|
'json' => [
|
|
'isDelivery' => true,
|
|
'postalCode' => '86100',
|
|
'city' => 'Châtellerault',
|
|
'street' => '1 rue du Test',
|
|
'sites' => [$this->firstSiteIri()],
|
|
],
|
|
]);
|
|
|
|
self::assertResponseStatusCodeSame(422);
|
|
}
|
|
|
|
/**
|
|
* RG (ERP-119) : au moins un type d'adresse (Prospection / Livraison /
|
|
* Facturation) est obligatoire. POST sans aucun drapeau de type -> 422, avec
|
|
* une violation portee sur `isProspect` (mappee sous le select « Type
|
|
* d'adresse » cote front via ClientAddressBlock).
|
|
*/
|
|
public function testAddressRequiresAtLeastOneType(): void
|
|
{
|
|
$this->skipIfSitesModuleDisabled();
|
|
$client = $this->createAdminClient();
|
|
$seed = $this->seedClient('Address No Type');
|
|
$category = $this->createAddressCategory();
|
|
|
|
$body = $client->request('POST', '/api/clients/'.$seed->getId().'/addresses', [
|
|
'headers' => ['Content-Type' => self::LD, 'Accept' => self::LD],
|
|
'json' => [
|
|
'postalCode' => '86100',
|
|
'city' => 'Châtellerault',
|
|
'street' => '1 rue du Test',
|
|
'sites' => [$this->firstSiteIri()],
|
|
'categories' => ['/api/categories/'.$category->getId()],
|
|
],
|
|
])->toArray(false);
|
|
|
|
self::assertResponseStatusCodeSame(422);
|
|
$byPath = $this->violationsByPath($body);
|
|
self::assertArrayHasKey('isProspect', $byPath);
|
|
self::assertSame('Le type d\'adresse est obligatoire.', $byPath['isProspect']);
|
|
}
|
|
|
|
/**
|
|
* Nouveaux types d'adresse (ERP-119) : Courtier et Distributeur sont acceptes
|
|
* comme types autonomes (avec site + categorie). is_broker / is_distributor.
|
|
*/
|
|
public function testBrokerAddressAccepted(): void
|
|
{
|
|
$this->skipIfSitesModuleDisabled();
|
|
$client = $this->createAdminClient();
|
|
$seed = $this->seedClient('Address Broker Type');
|
|
$category = $this->createAddressCategory();
|
|
|
|
$client->request('POST', '/api/clients/'.$seed->getId().'/addresses', [
|
|
'headers' => ['Content-Type' => self::LD],
|
|
'json' => [
|
|
'isBroker' => true,
|
|
'postalCode' => '86100',
|
|
'city' => 'Châtellerault',
|
|
'street' => '1 rue du Test',
|
|
'sites' => [$this->firstSiteIri()],
|
|
'categories' => ['/api/categories/'.$category->getId()],
|
|
],
|
|
]);
|
|
|
|
self::assertResponseStatusCodeSame(201);
|
|
}
|
|
|
|
public function testDistributorAddressAccepted(): void
|
|
{
|
|
$this->skipIfSitesModuleDisabled();
|
|
$client = $this->createAdminClient();
|
|
$seed = $this->seedClient('Address Distributor Type');
|
|
$category = $this->createAddressCategory();
|
|
|
|
$client->request('POST', '/api/clients/'.$seed->getId().'/addresses', [
|
|
'headers' => ['Content-Type' => self::LD],
|
|
'json' => [
|
|
'isDistributor' => true,
|
|
'postalCode' => '86100',
|
|
'city' => 'Châtellerault',
|
|
'street' => '1 rue du Test',
|
|
'sites' => [$this->firstSiteIri()],
|
|
'categories' => ['/api/categories/'.$category->getId()],
|
|
],
|
|
]);
|
|
|
|
self::assertResponseStatusCodeSame(201);
|
|
}
|
|
|
|
/**
|
|
* Courtier / Distributeur sont des types AUTONOMES exclusifs : les combiner avec
|
|
* un autre usage (ici Livraison) -> 422, violation sur isProspect (mappee sous le
|
|
* select Type d'adresse). Miroir applicatif du CHECK chk_client_address_broker_exclusive.
|
|
*/
|
|
public function testExclusiveAddressTypeRejected(): void
|
|
{
|
|
$this->skipIfSitesModuleDisabled();
|
|
$client = $this->createAdminClient();
|
|
$seed = $this->seedClient('Address Broker Mix');
|
|
$category = $this->createAddressCategory();
|
|
|
|
$body = $client->request('POST', '/api/clients/'.$seed->getId().'/addresses', [
|
|
'headers' => ['Content-Type' => self::LD, 'Accept' => self::LD],
|
|
'json' => [
|
|
'isBroker' => true,
|
|
'isDelivery' => true,
|
|
'postalCode' => '86100',
|
|
'city' => 'Châtellerault',
|
|
'street' => '1 rue du Test',
|
|
'sites' => [$this->firstSiteIri()],
|
|
'categories' => ['/api/categories/'.$category->getId()],
|
|
],
|
|
])->toArray(false);
|
|
|
|
self::assertResponseStatusCodeSame(422);
|
|
$byPath = $this->violationsByPath($body);
|
|
self::assertArrayHasKey('isProspect', $byPath);
|
|
self::assertSame('Une adresse Courtier ne peut pas avoir d\'autre type.', $byPath['isProspect']);
|
|
}
|
|
|
|
/**
|
|
* Retourne l'IRI du premier site seede (fixtures Sites).
|
|
*/
|
|
private function firstSiteIri(): string
|
|
{
|
|
$site = $this->getEm()->getRepository(Site::class)->findOneBy([]);
|
|
self::assertNotNull($site, 'Aucun site seede : impossible de tester les adresses.');
|
|
|
|
return '/api/sites/'.$site->getId();
|
|
}
|
|
}
|