refactor(commercial) : RG-1.03 distributor/broker by category code
ClientProcessor::hasCategoryCode (ex hasCategoryType) verifie le code de la Category (DISTRIBUTEUR/COURTIER) et non plus le type. Filtre liste/export renomme categoryType -> categoryCode (filtre sur category.code). Tests RG-1.03 distributor + courtier ajoutes ; factory de test adaptee au type unique CLIENT.
This commit is contained in:
@@ -17,13 +17,17 @@ use DateTimeImmutable;
|
||||
* Base des tests fonctionnels du module Commercial (M1 — repertoire clients).
|
||||
*
|
||||
* Etend la base Core : ajoute des factories pour seeder vite des categories
|
||||
* typees (DISTRIBUTEUR / COURTIER / SECTEUR) et des clients, plus un helper
|
||||
* d'authentification admin.
|
||||
* codees (DISTRIBUTEUR / COURTIER / SECTEUR...) sous le type unique CLIENT et
|
||||
* des clients, plus un helper d'authentification admin.
|
||||
*
|
||||
* Refonte taxonomie ERP-78 : il n'y a plus qu'un type CLIENT ; le code metier
|
||||
* vit desormais sur la Category. `createCategory($code)` est un fetch-or-create
|
||||
* PAR CODE (idempotent) sous CLIENT — deux clients d'un meme test partagent ainsi
|
||||
* la categorie de meme code sans violer l'index unique partiel uq_category_code.
|
||||
*
|
||||
* Cleanup : tearDown purge clients, categories `test_cli_cat_*` et users/roles
|
||||
* `test_*`. Les category_types business sont fetch-or-create (idempotents) et
|
||||
* laisses en place (pas de DELETE pour ne pas entrer en conflit avec d'autres
|
||||
* suites). Pas de DAMA en local -> purge manuelle obligatoire.
|
||||
* `test_*`. Le type CLIENT est fetch-or-create (idempotent) et laisse en place.
|
||||
* Pas de DAMA en local -> purge manuelle obligatoire.
|
||||
*
|
||||
* @internal
|
||||
*/
|
||||
@@ -31,6 +35,14 @@ abstract class AbstractCommercialApiTestCase extends AbstractApiTestCase
|
||||
{
|
||||
protected const string TEST_CATEGORY_PREFIX = 'test_cli_cat_';
|
||||
|
||||
/**
|
||||
* Codes pilotant les RG (RG-1.03 distributor/broker, RG-1.29 adresse) : ils
|
||||
* doivent matcher exactement, donc createCategory() les fetch-or-create par
|
||||
* code. Les autres codes sont traites comme de simples libelles generiques et
|
||||
* produisent une categorie a code UNIQUE (cf. createCategory).
|
||||
*/
|
||||
private const array RG_EXACT_CODES = ['DISTRIBUTEUR', 'COURTIER'];
|
||||
|
||||
protected function tearDown(): void
|
||||
{
|
||||
$this->cleanupCommercialTestData();
|
||||
@@ -43,20 +55,20 @@ abstract class AbstractCommercialApiTestCase extends AbstractApiTestCase
|
||||
}
|
||||
|
||||
/**
|
||||
* Recupere (ou cree) un CategoryType par son code metier. Idempotent : la
|
||||
* Recupere (ou cree) le type unique CLIENT (refonte ERP-78). Idempotent : la
|
||||
* contrainte d'unicite sur category_type.code interdit les doublons.
|
||||
*/
|
||||
protected function createCategoryType(string $code): CategoryType
|
||||
protected function clientCategoryType(): CategoryType
|
||||
{
|
||||
$em = $this->getEm();
|
||||
$existing = $em->getRepository(CategoryType::class)->findOneBy(['code' => $code]);
|
||||
$existing = $em->getRepository(CategoryType::class)->findOneBy(['code' => 'CLIENT']);
|
||||
if (null !== $existing) {
|
||||
return $existing;
|
||||
}
|
||||
|
||||
$type = new CategoryType();
|
||||
$type->setCode($code);
|
||||
$type->setLabel(ucfirst(strtolower($code)));
|
||||
$type->setCode('CLIENT');
|
||||
$type->setLabel('Client');
|
||||
$em->persist($type);
|
||||
$em->flush();
|
||||
|
||||
@@ -64,15 +76,38 @@ abstract class AbstractCommercialApiTestCase extends AbstractApiTestCase
|
||||
}
|
||||
|
||||
/**
|
||||
* Cree une Category de test rattachee a un type metier donne (code).
|
||||
* Cree une Category de test sous le type unique CLIENT (ERP-78).
|
||||
*
|
||||
* - Code RG (DISTRIBUTEUR / COURTIER) : fetch-or-create par code EXACT — le
|
||||
* code doit matcher la regle de gestion, et l'appel repete dans un test
|
||||
* renvoie la meme categorie (pas de violation de uq_category_code).
|
||||
* - Autre code (SECTEUR, AUTRE, ...) : simple libelle generique -> categorie
|
||||
* a code UNIQUE (suffixe aleatoire). Garantit que deux categories
|
||||
* « generiques » d'un meme test sont DISTINCTES (ex: detection de
|
||||
* changement de categorie dans les tests RBAC).
|
||||
*/
|
||||
protected function createCategory(string $typeCode = 'SECTEUR'): Category
|
||||
protected function createCategory(string $code = 'SECTEUR'): Category
|
||||
{
|
||||
$em = $this->getEm();
|
||||
$suffix = substr(bin2hex(random_bytes(4)), 0, 8);
|
||||
$em = $this->getEm();
|
||||
|
||||
if (in_array($code, self::RG_EXACT_CODES, true)) {
|
||||
$existing = $em->getRepository(Category::class)->findOneBy(['code' => $code, 'deletedAt' => null]);
|
||||
if (null !== $existing) {
|
||||
return $existing;
|
||||
}
|
||||
|
||||
$effectiveCode = $code;
|
||||
$name = self::TEST_CATEGORY_PREFIX.strtolower($code);
|
||||
} else {
|
||||
$suffix = substr(bin2hex(random_bytes(4)), 0, 8);
|
||||
$effectiveCode = strtoupper($code).'_'.strtoupper($suffix);
|
||||
$name = self::TEST_CATEGORY_PREFIX.strtolower($code).'_'.$suffix;
|
||||
}
|
||||
|
||||
$category = new Category();
|
||||
$category->setName(self::TEST_CATEGORY_PREFIX.$suffix);
|
||||
$category->setCategoryType($this->createCategoryType($typeCode));
|
||||
$category->setName($name);
|
||||
$category->setCode($effectiveCode);
|
||||
$category->setCategoryType($this->clientCategoryType());
|
||||
$em->persist($category);
|
||||
$em->flush();
|
||||
|
||||
@@ -81,9 +116,10 @@ abstract class AbstractCommercialApiTestCase extends AbstractApiTestCase
|
||||
|
||||
/**
|
||||
* Seede directement un Client en base (sans passer par l'API), pour les
|
||||
* tests de liste / archivage. Le client porte une categorie SECTEUR.
|
||||
* tests de liste / archivage. Le client porte une categorie du code donne
|
||||
* (defaut SECTEUR — categorie generique non interdite sur adresse).
|
||||
*/
|
||||
protected function seedClient(string $companyName, bool $isArchived = false, string $categoryTypeCode = 'SECTEUR'): ClientEntity
|
||||
protected function seedClient(string $companyName, bool $isArchived = false, string $categoryCode = 'SECTEUR'): ClientEntity
|
||||
{
|
||||
$em = $this->getEm();
|
||||
$client = new ClientEntity();
|
||||
@@ -93,7 +129,7 @@ abstract class AbstractCommercialApiTestCase extends AbstractApiTestCase
|
||||
$client->setLastName('Seed');
|
||||
$client->setPhonePrimary('0102030405');
|
||||
$client->setEmail(strtolower(str_replace(' ', '', $companyName)).'@seed.test');
|
||||
$client->addCategory($this->createCategory($categoryTypeCode));
|
||||
$client->addCategory($this->createCategory($categoryCode));
|
||||
$client->setIsArchived($isArchived);
|
||||
if ($isArchived) {
|
||||
$client->setArchivedAt(new DateTimeImmutable());
|
||||
|
||||
Reference in New Issue
Block a user