Compare commits
9 Commits
| Author | SHA1 | Date | |
|---|---|---|---|
| dfd6e5251d | |||
| 583d634a83 | |||
| ee1521384e | |||
| 79dffccc79 | |||
| 1ff335b3fe | |||
| fa47517028 | |||
| 402c83d40d | |||
| 50e6e14b91 | |||
| 00bd02858c |
@@ -3,6 +3,14 @@ lexik_jwt_authentication:
|
|||||||
public_key: '%env(resolve:JWT_PUBLIC_KEY)%'
|
public_key: '%env(resolve:JWT_PUBLIC_KEY)%'
|
||||||
pass_phrase: '%env(JWT_PASSPHRASE)%'
|
pass_phrase: '%env(JWT_PASSPHRASE)%'
|
||||||
token_ttl: '%env(int:JWT_TOKEN_TTL)%'
|
token_ttl: '%env(int:JWT_TOKEN_TTL)%'
|
||||||
|
# Tolerance d'horloge (en secondes) appliquee a la validation des claims
|
||||||
|
# temporels iat / nbf / exp (LooseValidAt cote lcobucci). Sans cette marge
|
||||||
|
# (defaut 0), un recul d'horloge entre la signature (/login_check) et la
|
||||||
|
# requete suivante rend iat/nbf « dans le futur » -> « Invalid JWT Token »
|
||||||
|
# (401). Observe en dev sous WSL2/Docker (horloge CLOCK_REALTIME non
|
||||||
|
# monotone) : flakes intermittents de la suite PHPUnit (ERP-98). Benefice
|
||||||
|
# aussi en prod si les noeuds derivent legerement entre eux.
|
||||||
|
clock_skew: 15
|
||||||
remove_token_from_body_when_cookies_used: true
|
remove_token_from_body_when_cookies_used: true
|
||||||
token_extractors:
|
token_extractors:
|
||||||
authorization_header:
|
authorization_header:
|
||||||
|
|||||||
+1
-1
@@ -1,2 +1,2 @@
|
|||||||
parameters:
|
parameters:
|
||||||
app.version: '0.1.62'
|
app.version: '0.1.66'
|
||||||
|
|||||||
@@ -118,6 +118,8 @@ Aucun pattern soft delete existant dans Starseed (vérifié, aucune entité ne p
|
|||||||
|
|
||||||
Index unique partiel sur `(LOWER(name), category_type_id) WHERE deleted_at IS NULL`. Permet de recréer une catégorie avec le même `(name, type)` après suppression logique. Postgres supporte nativement (`CREATE UNIQUE INDEX ... WHERE`). Pattern propre, pas besoin de validator applicatif maison côté unicité — la contrainte SQL fait le job.
|
Index unique partiel sur `(LOWER(name), category_type_id) WHERE deleted_at IS NULL`. Permet de recréer une catégorie avec le même `(name, type)` après suppression logique. Postgres supporte nativement (`CREATE UNIQUE INDEX ... WHERE`). Pattern propre, pas besoin de validator applicatif maison côté unicité — la contrainte SQL fait le job.
|
||||||
|
|
||||||
|
> **🔗 Évolution ERP-78 (refonte taxonomie M1)** : `Category` porte désormais une colonne **`code`** (`VARCHAR(50)`, NOT NULL), slug MAJUSCULE auto-généré du nom (figé à la création, lecture seule via l'API), avec un **second index unique partiel** `uq_category_code (code) WHERE deleted_at IS NULL`. Ce code est la clé métier stable utilisée par le M1 Commercial (RG-1.03 / RG-1.29). Détail : `docs/specs/M1-clients/spec-back.md` § 3.3.
|
||||||
|
|
||||||
### 2.5 Audit & traces temporelles — deux niveaux complémentaires
|
### 2.5 Audit & traces temporelles — deux niveaux complémentaires
|
||||||
|
|
||||||
Deux mécanismes **indépendants** cohabitent :
|
Deux mécanismes **indépendants** cohabitent :
|
||||||
|
|||||||
@@ -465,26 +465,32 @@ CREATE TABLE client_rib (
|
|||||||
CREATE INDEX idx_client_rib_client ON client_rib(client_id);
|
CREATE INDEX idx_client_rib_client ON client_rib(client_id);
|
||||||
```
|
```
|
||||||
|
|
||||||
### 3.3 Seed `CategoryType` (extension du M0)
|
### 3.3 Seed taxonomie — type unique `CLIENT` + `Category.code` (refonte ERP-78)
|
||||||
|
|
||||||
Au M0, la table `category_type` a été créée mais reste vide (HP-1 du M0). Le M1 lève cette restriction avec un seed initial des **types métier** dont le module Tiers a besoin :
|
> **⚠ Refonte ERP-78 (décision produit 01/06) — le modèle ci-dessous remplace l'ancien.**
|
||||||
|
> Historique : à l'origine (#38), `DISTRIBUTEUR` / `COURTIER` / `SECTEUR` / `AUTRE` étaient des **`category_type`**. Le modèle a été **inversé** :
|
||||||
|
>
|
||||||
|
> - **UN SEUL `category_type` : `CLIENT`** (code `CLIENT`, label « Client »).
|
||||||
|
> - `Distributeur` / `Courtier` / `Secteur` / `Autre` (+ catégories métier fines) sont désormais des **`Category`** rattachées au type `CLIENT`.
|
||||||
|
> - Le filtrage métier ne se fait plus sur le **type** mais sur un **`code` stable porté par la `Category`** (NOT NULL, unique parmi les actifs — index partiel `uq_category_code`). Le code est un **slug MAJUSCULE auto-généré du nom** (`CategoryCodeGenerator`), figé à la création, et exposé en **lecture seule** (groupe `category:read`). Les codes `DISTRIBUTEUR` / `COURTIER` (anciennement portés par le type) sont reportés sur les `Category` correspondantes.
|
||||||
|
|
||||||
|
Seed cible (migration corrective `Version20260602100000`, namespace racine) :
|
||||||
|
|
||||||
```sql
|
```sql
|
||||||
INSERT INTO category_type (code, label, position) VALUES
|
-- Type unique
|
||||||
('DISTRIBUTEUR', 'Distributeur', 10),
|
INSERT INTO category_type (code, label) VALUES ('CLIENT', 'Client') ON CONFLICT (code) DO NOTHING;
|
||||||
('COURTIER', 'Courtier', 20),
|
-- Catégories système sous CLIENT (codes stables pilotant les RG)
|
||||||
('SECTEUR', 'Secteur', 30),
|
-- Distributeur -> DISTRIBUTEUR, Courtier -> COURTIER, Secteur -> SECTEUR, Autre -> AUTRE
|
||||||
('AUTRE', 'Autre', 99);
|
|
||||||
```
|
```
|
||||||
|
|
||||||
> **Note** : le CRUD admin de `CategoryType` reste HP (cf. M0).
|
> **Note** : le CRUD admin de `CategoryType` reste HP (cf. M0). Le `code` de `Category` n'est PAS saisissable via l'API (auto-généré côté serveur).
|
||||||
>
|
>
|
||||||
> **Seed en DEUX endroits (décision 29/05, vérifiée empiriquement)** : le `make db-reset` lance les fixtures, dont le purger Doctrine **vide `category_type`** (entité M0 mappée) avant `load()` → un seed posé uniquement en migration disparaît en dev/test. Donc :
|
> **Seed en DEUX endroits (décision 29/05, vérifiée empiriquement)** : le `make db-reset` lance les fixtures, dont le purger Doctrine **vide `category` / `category_type`** (entités M0 mappées) avant `load()` → un seed posé uniquement en migration disparaît en dev/test. Donc :
|
||||||
> 1. **Migration** (`ON CONFLICT (code) DO NOTHING`) → sert en **prod** (pas de fixtures).
|
> 1. **Migration** (`ON CONFLICT` / guards `NOT EXISTS`) → sert en **prod** (pas de fixtures).
|
||||||
> 2. **Fixture Commercial idempotente** (ex. `CommercialReferentialFixtures`) re-seedant les 4 types → survit au `db-reset`, satisfait le critère « 4 types présents après db-reset ».
|
> 2. **Fixtures idempotentes** (`CategoryTypeFixtures` → type CLIENT ; `CategoryFixtures` → catégories codées sous CLIENT) → survivent au `db-reset`.
|
||||||
>
|
>
|
||||||
> ⚠ **À venir en ERP-54** : `tva_mode` / `payment_delay` / `payment_type` / `bank` ne sont pas encore des entités mappées au M1.0 → le purger ne les touche pas, leur seed migration survit. **Dès qu'ERP-54 crée leurs entités, ils seront purgés au db-reset** → il faudra les ajouter à la même fixture référentielle.
|
> ⚠ **À venir en ERP-54** : `tva_mode` / `payment_delay` / `payment_type` / `bank` ne sont pas encore des entités mappées au M1.0 → le purger ne les touche pas, leur seed migration survit. **Dès qu'ERP-54 crée leurs entités, ils seront purgés au db-reset** → il faudra les ajouter à la même fixture référentielle.
|
||||||
> 🔗 **Coordination ERP-68** : ERP-53 pose la fixture référentielle minimale (4 category_types). ERP-68 l'**étend** (clients de démo, ~12-15) sans la dupliquer.
|
> 🔗 **Coordination ERP-68** : ERP-78 (cette refonte) atterrit avant ERP-68. `CategoryFixtures` / `ClientFixtures` ont été adaptées au type unique CLIENT + codes (les tiers distributeur/courtier portent les `Category` de code DISTRIBUTEUR/COURTIER).
|
||||||
|
|
||||||
### 3.4 Entité `Client` — squelette
|
### 3.4 Entité `Client` — squelette
|
||||||
|
|
||||||
@@ -742,7 +748,7 @@ class Client implements TimestampableInterface, BlamableInterface
|
|||||||
- **Security** : `is_granted('commercial.clients.view')`
|
- **Security** : `is_granted('commercial.clients.view')`
|
||||||
- **Query params** :
|
- **Query params** :
|
||||||
- `includeArchived=true|false` (default `false`)
|
- `includeArchived=true|false` (default `false`)
|
||||||
- `categoryType=<code>` (filtre par type de catégorie via `SearchFilter`)
|
- `categoryCode=<code>` (filtre les clients ayant ≥ 1 `Category` de ce code stable — ERP-78 ; ex. `DISTRIBUTEUR`, `COURTIER`)
|
||||||
- `search=<text>` (recherche fuzzy sur companyName + lastName + email)
|
- `search=<text>` (recherche fuzzy sur companyName + lastName + email)
|
||||||
- **Tri par défaut** : `companyName ASC`
|
- **Tri par défaut** : `companyName ASC`
|
||||||
- **Pagination** : front via `<MalioDataTable>` (volumétrie cible faible). Pas de pagination serveur au M1.
|
- **Pagination** : front via `<MalioDataTable>` (volumétrie cible faible). Pas de pagination serveur au M1.
|
||||||
@@ -881,7 +887,7 @@ Cf. § 2.6. Pattern Shared standard.
|
|||||||
|
|
||||||
- **RG-1.01** : Au moins l'un des champs `firstName` (Prénom du contact principal) ou `lastName` (Nom du contact principal) doit être renseigné. Sinon → 422.
|
- **RG-1.01** : Au moins l'un des champs `firstName` (Prénom du contact principal) ou `lastName` (Nom du contact principal) doit être renseigné. Sinon → 422.
|
||||||
- **RG-1.02** : Le champ `phoneSecondary` est optionnel et apparaît au clic sur un bouton `+` côté front. Maximum 2 téléphones (primary + secondary). Comportement purement front au niveau UI ; côté serveur, les 2 colonnes existent et sont distinctes.
|
- **RG-1.02** : Le champ `phoneSecondary` est optionnel et apparaît au clic sur un bouton `+` côté front. Maximum 2 téléphones (primary + secondary). Comportement purement front au niveau UI ; côté serveur, les 2 colonnes existent et sont distinctes.
|
||||||
- **RG-1.03** : Les champs `distributor` et `broker` sont **mutuellement exclusifs** (au plus une seule des deux est renseignée). Tentative d'envoyer les deux → 422. Contrainte CHECK en base également : `NOT (distributor_id IS NOT NULL AND broker_id IS NOT NULL)`. La liste front de `distributor` = clients ayant au moins une catégorie de type `DISTRIBUTEUR` ; idem pour `broker` avec `COURTIER`.
|
- **RG-1.03** : Les champs `distributor` et `broker` sont **mutuellement exclusifs** (au plus une seule des deux est renseignée). Tentative d'envoyer les deux → 422. Contrainte CHECK en base également : `NOT (distributor_id IS NOT NULL AND broker_id IS NOT NULL)`. Un `distributor` référencé doit porter une **`Category` de code `DISTRIBUTEUR`** ; un `broker` une **`Category` de code `COURTIER`** — sinon 422. _(Refonte ERP-78 : le filtrage se fait sur le `code` de la `Category`, plus sur le type — `ClientProcessor::hasCategoryCode`.)_ La liste front de `distributor` = clients ayant une catégorie de code `DISTRIBUTEUR`, via `GET /api/clients?categoryCode=DISTRIBUTEUR` ; idem `broker` avec `COURTIER`.
|
||||||
|
|
||||||
### Onglet Information
|
### Onglet Information
|
||||||
|
|
||||||
@@ -946,9 +952,9 @@ Cf. § 2.6. Pattern Shared standard.
|
|||||||
|
|
||||||
- **RG-1.28** : Si un PATCH contient des champs de **plusieurs groupes** de sérialisation et que l'utilisateur **n'a pas toutes les permissions** correspondantes, le `ClientProcessor` renvoie **403 Forbidden sur l'ensemble du payload** (mode strict — pas de filtrage silencieux). Le front est responsable de ne JAMAIS envoyer de champs hors-permission (les onglets masqués via `usePermissions()` ne génèrent pas de payload). Cette règle protège contre les appels API directs malveillants. Exemple : un Bureau qui envoie `{ "companyName": "...", "siren": "..." }` → 403, le message d'erreur précise « Champ `siren` requiert la permission `commercial.clients.accounting.manage` ».
|
- **RG-1.28** : Si un PATCH contient des champs de **plusieurs groupes** de sérialisation et que l'utilisateur **n'a pas toutes les permissions** correspondantes, le `ClientProcessor` renvoie **403 Forbidden sur l'ensemble du payload** (mode strict — pas de filtrage silencieux). Le front est responsable de ne JAMAIS envoyer de champs hors-permission (les onglets masqués via `usePermissions()` ne génèrent pas de payload). Cette règle protège contre les appels API directs malveillants. Exemple : un Bureau qui envoie `{ "companyName": "...", "siren": "..." }` → 403, le message d'erreur précise « Champ `siren` requiert la permission `commercial.clients.accounting.manage` ».
|
||||||
|
|
||||||
### Catégorie sur ClientAddress (filtrage par type)
|
### Catégorie sur ClientAddress (filtrage par code)
|
||||||
|
|
||||||
- **RG-1.29** : Le `<MalioSelectCheckbox>` Catégorie de l'onglet Adresse n'expose **que** les `Category` dont `categoryType.code IN ('SECTEUR', 'AUTRE')`. Les types `DISTRIBUTEUR` et `COURTIER` qualifient une **relation entre clients** (cf. RG-1.03) et n'ont pas de sens sur une adresse physique. Implémentation : `ClientAddressProvider` filtre côté serveur via paramètre de requête à l'endpoint `GET /api/categories?categoryType.code[]=SECTEUR&categoryType.code[]=AUTRE` (SearchFilter API Platform). Côté validation du POST/PATCH : si l'utilisateur tente de poster une catégorie de type DISTRIBUTEUR ou COURTIER sur une adresse → **422** avec violation `categories: "Type de catégorie non autorisé sur une adresse."`.
|
- **RG-1.29** _(refonte ERP-78)_ : sur une adresse, les `Category` de **code `DISTRIBUTEUR` ou `COURTIER`** sont **interdites** — elles qualifient une **relation entre clients** (cf. RG-1.03) et n'ont pas de sens sur une adresse physique. **Toute autre** catégorie (type unique CLIENT) est autorisée. Validation du POST/PATCH : poster une catégorie de code DISTRIBUTEUR/COURTIER sur une adresse → **422** avec violation `categories: "Type de catégorie non autorisé sur une adresse."` (`ClientAddress::validateCategoryCodes`). Côté front, le `<MalioSelectCheckbox>` Catégorie de l'onglet Adresse exclut les `Category` de code `DISTRIBUTEUR` / `COURTIER` (le `code` est exposé en lecture sur `/api/categories`).
|
||||||
|
|
||||||
## 8. Tests à automatiser
|
## 8. Tests à automatiser
|
||||||
|
|
||||||
@@ -957,7 +963,7 @@ Cf. § 2.6. Pattern Shared standard.
|
|||||||
- [ ] **RG-1.01** : POST sans firstName ni lastName → 422
|
- [ ] **RG-1.01** : POST sans firstName ni lastName → 422
|
||||||
- [ ] **RG-1.02** : POST avec phoneSecondary rempli → persistance OK ; PATCH ajoutant un 3e téléphone → côté API, 2 colonnes uniquement (test que le payload ne peut pas créer un 3e)
|
- [ ] **RG-1.02** : POST avec phoneSecondary rempli → persistance OK ; PATCH ajoutant un 3e téléphone → côté API, 2 colonnes uniquement (test que le payload ne peut pas créer un 3e)
|
||||||
- [ ] **RG-1.03** : POST avec distributor ET broker → 422 ; POST distributor seul → 201
|
- [ ] **RG-1.03** : POST avec distributor ET broker → 422 ; POST distributor seul → 201
|
||||||
- [ ] **RG-1.03** : POST distributor référençant un client SANS catégorie de type DISTRIBUTEUR → 422 (validation custom)
|
- [ ] **RG-1.03** : POST distributor référençant un client SANS catégorie de code DISTRIBUTEUR → 422 (validation custom `ClientProcessor::hasCategoryCode`)
|
||||||
- [ ] **RG-1.04** : PATCH onglet Information par un user Commerciale avec champs incomplets → 422 ; même PATCH par Admin → 200
|
- [ ] **RG-1.04** : PATCH onglet Information par un user Commerciale avec champs incomplets → 422 ; même PATCH par Admin → 200
|
||||||
- [ ] **RG-1.05** : POST contact sans firstName ni lastName → 422 (BDD CHECK lève une exception)
|
- [ ] **RG-1.05** : POST contact sans firstName ni lastName → 422 (BDD CHECK lève une exception)
|
||||||
- [ ] **RG-1.06/07/08** : POST adresse avec isProspect=true ET isDelivery=true → 422 / CHECK
|
- [ ] **RG-1.06/07/08** : POST adresse avec isProspect=true ET isDelivery=true → 422 / CHECK
|
||||||
|
|||||||
@@ -96,8 +96,8 @@ C'est le 1er bloc à remplir. Sans validation de ce formulaire, les onglets ne s
|
|||||||
| **Téléphone secondaire** | `<MalioInputText>` (masque tel) | Non | Apparaît au clic sur le bouton `+` (RG-1.02). Max 2 — bouton `+` disparaît une fois rempli. |
|
| **Téléphone secondaire** | `<MalioInputText>` (masque tel) | Non | Apparaît au clic sur le bouton `+` (RG-1.02). Max 2 — bouton `+` disparaît une fois rempli. |
|
||||||
| **Email** | `<MalioInputText>` type email | Oui | RG-1.21 (lowercase) |
|
| **Email** | `<MalioInputText>` type email | Oui | RG-1.21 (lowercase) |
|
||||||
| **Distributeur / Courtier** | `<MalioSelect>` | Non | Valeurs : `Dépend du distributeur` / `Dépend du courtier` / `Aucun`. RG-1.03 conditionne les 2 champs suivants. |
|
| **Distributeur / Courtier** | `<MalioSelect>` | Non | Valeurs : `Dépend du distributeur` / `Dépend du courtier` / `Aucun`. RG-1.03 conditionne les 2 champs suivants. |
|
||||||
| **Nom du distributeur** | `<MalioSelect>` | Conditionnel | Visible si « Dépend du distributeur ». Liste = clients ayant ≥ 1 catégorie de type `DISTRIBUTEUR`. RG-1.03. |
|
| **Nom du distributeur** | `<MalioSelect>` | Conditionnel | Visible si « Dépend du distributeur ». Liste = clients ayant ≥ 1 catégorie de **code** `DISTRIBUTEUR` (ERP-78), via `GET /api/clients?categoryCode=DISTRIBUTEUR`. RG-1.03. |
|
||||||
| **Nom du courtier** | `<MalioSelect>` | Conditionnel | Visible si « Dépend du courtier ». Liste = clients ayant ≥ 1 catégorie de type `COURTIER`. RG-1.03. |
|
| **Nom du courtier** | `<MalioSelect>` | Conditionnel | Visible si « Dépend du courtier ». Liste = clients ayant ≥ 1 catégorie de **code** `COURTIER` (ERP-78), via `GET /api/clients?categoryCode=COURTIER`. RG-1.03. |
|
||||||
| **Prestation de triage** | `<MalioCheckbox>` | Non | — |
|
| **Prestation de triage** | `<MalioCheckbox>` | Non | — |
|
||||||
|
|
||||||
**Action** : « Valider » (`<MalioButton>`) → POST `/api/clients` ([`spec-back.md` § 4.3](./spec-back.md)). Si succès, on passe automatiquement à l'onglet « Information ».
|
**Action** : « Valider » (`<MalioButton>`) → POST `/api/clients` ([`spec-back.md` § 4.3](./spec-back.md)). Si succès, on passe automatiquement à l'onglet « Information ».
|
||||||
@@ -150,7 +150,7 @@ Saisir une ou plusieurs adresses du client, rattachées à un ou plusieurs sites
|
|||||||
| **Prospect** | `<MalioCheckbox>` | Non | RG-1.06 — masque Adresse de livraison + Facturation si coché |
|
| **Prospect** | `<MalioCheckbox>` | Non | RG-1.06 — masque Adresse de livraison + Facturation si coché |
|
||||||
| **Adresse de livraison** | `<MalioCheckbox>` | Non | RG-1.07 — masque Prospect si coché |
|
| **Adresse de livraison** | `<MalioCheckbox>` | Non | RG-1.07 — masque Prospect si coché |
|
||||||
| **Facturation** | `<MalioCheckbox>` | Non | RG-1.08 — masque Prospect si coché ; affiche le champ Email (RG-1.11) |
|
| **Facturation** | `<MalioCheckbox>` | Non | RG-1.08 — masque Prospect si coché ; affiche le champ Email (RG-1.11) |
|
||||||
| **Catégorie** | `<MalioSelectCheckbox>` (multi) | Oui | Liste des `Category` de **type SECTEUR + AUTRE** uniquement (cf. décision Q5 — DISTRIBUTEUR et COURTIER qualifient une relation entre clients, pas un lieu) |
|
| **Catégorie** | `<MalioSelectCheckbox>` (multi) | Oui | Liste des `Category` **hors codes `DISTRIBUTEUR` / `COURTIER`** (ERP-78 — ces codes qualifient une relation entre clients, pas un lieu). Le front exclut ces 2 codes du select (le `code` est exposé en lecture sur `/api/categories`). |
|
||||||
| **Pays** | `<MalioSelect>` | Oui | Préremplie « France » |
|
| **Pays** | `<MalioSelect>` | Oui | Préremplie « France » |
|
||||||
| **Code postal** | `<MalioInputText>` (masque numérique) | Oui | RG-1.09 — déclenche autocomplete ville via BAN |
|
| **Code postal** | `<MalioInputText>` (masque numérique) | Oui | RG-1.09 — déclenche autocomplete ville via BAN |
|
||||||
| **Ville** | `<MalioSelect>` | Oui | RG-1.09 — alimentée par api-adresse.data.gouv.fr suivant le CP |
|
| **Ville** | `<MalioSelect>` | Oui | RG-1.09 — alimentée par api-adresse.data.gouv.fr suivant le CP |
|
||||||
@@ -268,7 +268,7 @@ Le composant `Code postal` + `Ville` + `Adresse` est branché sur **api-adresse.
|
|||||||
|
|
||||||
| # | Zone d'ombre V0 | Résolution (cf. `spec-back.md`) |
|
| # | Zone d'ombre V0 | Résolution (cf. `spec-back.md`) |
|
||||||
|---|---|---|
|
|---|---|---|
|
||||||
| 1 | Catégorie en multi-select non clarifiée (1 ou n par client) | **M2M `client_category`** validée. CategoryType seedé avec `DISTRIBUTEUR`, `COURTIER`, `SECTEUR`, `AUTRE` (HP-3 du M0 levé). |
|
| 1 | Catégorie en multi-select non clarifiée (1 ou n par client) | **M2M `client_category`** validée. Refonte ERP-78 : type unique `CLIENT` ; `Distributeur`/`Courtier`/`Secteur`/`Autre` (+ catégories métier) sont des `Category` portant un `code` stable (HP-3 du M0 levé). |
|
||||||
| 2 | Distributeur / Courtier : liste de quoi ? | **Auto-référence Client** via 2 FK nullables `distributor_id` et `broker_id` (cf. RG-1.03). Une seule des deux est remplie à la fois. |
|
| 2 | Distributeur / Courtier : liste de quoi ? | **Auto-référence Client** via 2 FK nullables `distributor_id` et `broker_id` (cf. RG-1.03). Une seule des deux est remplie à la fois. |
|
||||||
| 3 | Onglet « Comptabilité » : qui édite ? | **Admin et Compta** peuvent éditer l'onglet Comptabilité (`commercial.clients.accounting.manage`). Bureau / Commerciale ne voient pas l'onglet. Compta ne peut pas créer un client (pas de `manage` global), mais peut éditer la partie comptable d'un client existant. |
|
| 3 | Onglet « Comptabilité » : qui édite ? | **Admin et Compta** peuvent éditer l'onglet Comptabilité (`commercial.clients.accounting.manage`). Bureau / Commerciale ne voient pas l'onglet. Compta ne peut pas créer un client (pas de `manage` global), mais peut éditer la partie comptable d'un client existant. |
|
||||||
| 4 | Workflow par onglet | **Sauvegarde incrémentale**. POST formulaire principal crée le `Client` (status implicite « actif »). Chaque onglet validé = PATCH partiel par groupe de sérialisation dédié. Pas d'état « draft ». |
|
| 4 | Workflow par onglet | **Sauvegarde incrémentale**. POST formulaire principal crée le `Client` (status implicite « actif »). Chaque onglet validé = PATCH partiel par groupe de sérialisation dédié. Pas d'état « draft ». |
|
||||||
|
|||||||
@@ -44,7 +44,63 @@
|
|||||||
},
|
},
|
||||||
"commercial": {
|
"commercial": {
|
||||||
"title": "Commercial",
|
"title": "Commercial",
|
||||||
"welcome": "Module Commercial"
|
"welcome": "Module Commercial",
|
||||||
|
"clients": {
|
||||||
|
"title": "Répertoire clients",
|
||||||
|
"add": "Ajouter",
|
||||||
|
"export": "Exporter",
|
||||||
|
"empty": "Aucun client pour l'instant.",
|
||||||
|
"column": {
|
||||||
|
"companyName": "Nom",
|
||||||
|
"categories": "Catégories",
|
||||||
|
"sites": "Site",
|
||||||
|
"lastActivity": "Dernière activité"
|
||||||
|
},
|
||||||
|
"filters": {
|
||||||
|
"title": "Filtres",
|
||||||
|
"search": "Recherche",
|
||||||
|
"categories": "Catégories",
|
||||||
|
"sites": "Sites",
|
||||||
|
"status": "Statut",
|
||||||
|
"archivedOnly": "Voir les archivés",
|
||||||
|
"apply": "Voir les résultats",
|
||||||
|
"reset": "Réinitialiser"
|
||||||
|
},
|
||||||
|
"tab": {
|
||||||
|
"information": "Information",
|
||||||
|
"contact": "Contact",
|
||||||
|
"address": "Adresse",
|
||||||
|
"transport": "Transport",
|
||||||
|
"accounting": "Comptabilité",
|
||||||
|
"statistics": "Statistiques",
|
||||||
|
"reports": "Rapports",
|
||||||
|
"exchanges": "Échanges"
|
||||||
|
},
|
||||||
|
"action": {
|
||||||
|
"edit": "Modifier",
|
||||||
|
"archive": "Archiver",
|
||||||
|
"restore": "Restaurer"
|
||||||
|
},
|
||||||
|
"toast": {
|
||||||
|
"createSuccess": "Client créé avec succès",
|
||||||
|
"updateSuccess": "Client mis à jour avec succès",
|
||||||
|
"archiveSuccess": "Client archivé avec succès",
|
||||||
|
"restoreSuccess": "Client restauré avec succès",
|
||||||
|
"error": "Une erreur est survenue. Réessayez.",
|
||||||
|
"exportError": "L'export du répertoire clients a échoué. Réessayez."
|
||||||
|
},
|
||||||
|
"validation": {
|
||||||
|
"informationRequiredForCommercial": "Les informations de l'entreprise sont obligatoires pour le rôle Commerciale.",
|
||||||
|
"contactRequired": "Au moins un contact (nom ou prénom) est obligatoire.",
|
||||||
|
"siteRequired": "Au moins un site Starseed doit être rattaché à l'adresse.",
|
||||||
|
"billingEmailRequired": "L'email de facturation est obligatoire pour une adresse de facturation.",
|
||||||
|
"bankRequiredForTransfer": "La banque est obligatoire pour un règlement par virement.",
|
||||||
|
"ribRequiredForLcr": "Au moins un RIB complet est obligatoire pour un règlement par LCR.",
|
||||||
|
"phoneFormat": "Format de téléphone invalide (attendu : XX XX XX XX XX).",
|
||||||
|
"emailFormat": "Format d'email invalide.",
|
||||||
|
"addressCategoryForbidden": "Une catégorie « Distributeur » ou « Courtier » ne peut pas qualifier une adresse."
|
||||||
|
}
|
||||||
|
}
|
||||||
},
|
},
|
||||||
"auth": {
|
"auth": {
|
||||||
"login": "Connexion",
|
"login": "Connexion",
|
||||||
|
|||||||
@@ -0,0 +1,85 @@
|
|||||||
|
import { describe, it, expect, vi, beforeEach } from 'vitest'
|
||||||
|
import type { HydraCollection } from '~/shared/utils/api'
|
||||||
|
import type { Client } from '../useClientsRepository'
|
||||||
|
|
||||||
|
// `useApi` est un auto-import Nuxt : on le stubbe globalement pour intercepter
|
||||||
|
// les appels declenches par usePaginatedList (que useClientsRepository enveloppe)
|
||||||
|
// et controler les reponses. Meme pattern que useCategoriesAdmin.spec.ts.
|
||||||
|
const mockGet = vi.hoisted(() => vi.fn())
|
||||||
|
vi.stubGlobal('useApi', () => ({
|
||||||
|
get: mockGet,
|
||||||
|
post: vi.fn(),
|
||||||
|
put: vi.fn(),
|
||||||
|
patch: vi.fn(),
|
||||||
|
delete: vi.fn(),
|
||||||
|
}))
|
||||||
|
|
||||||
|
// Import APRES le stub pour que useApi soit bien resolu au top-level du module.
|
||||||
|
const { useClientsRepository } = await import('../useClientsRepository')
|
||||||
|
|
||||||
|
/** Envelope Hydra minimale (la liste reelle des membres importe peu ici). */
|
||||||
|
function makeHydra(total: number): HydraCollection<Client> {
|
||||||
|
return { totalItems: total, member: [] }
|
||||||
|
}
|
||||||
|
|
||||||
|
describe('useClientsRepository', () => {
|
||||||
|
beforeEach(() => {
|
||||||
|
mockGet.mockReset()
|
||||||
|
// 25 items → 3 pages a 10/page : permet de tester la navigation page 2.
|
||||||
|
mockGet.mockResolvedValue(makeHydra(25))
|
||||||
|
})
|
||||||
|
|
||||||
|
it('cible la ressource /clients en page 1 par defaut', async () => {
|
||||||
|
const repo = useClientsRepository()
|
||||||
|
await repo.fetch()
|
||||||
|
|
||||||
|
expect(mockGet).toHaveBeenLastCalledWith(
|
||||||
|
'/clients',
|
||||||
|
{ page: 1, itemsPerPage: 10 },
|
||||||
|
expect.objectContaining({ toast: false }),
|
||||||
|
)
|
||||||
|
})
|
||||||
|
|
||||||
|
it('pousse les filtres du drawer (categories multi, sites, archives) et retombe en page 1', async () => {
|
||||||
|
const repo = useClientsRepository()
|
||||||
|
await repo.fetch()
|
||||||
|
await repo.goToPage(2)
|
||||||
|
expect(repo.currentPage.value).toBe(2)
|
||||||
|
|
||||||
|
await repo.setFilters(
|
||||||
|
{
|
||||||
|
search: 'acme',
|
||||||
|
'categoryCode[]': ['DISTRIBUTEUR', 'COURTIER'],
|
||||||
|
'siteId[]': ['1', '2'],
|
||||||
|
archivedOnly: true,
|
||||||
|
},
|
||||||
|
{ replace: true },
|
||||||
|
)
|
||||||
|
|
||||||
|
expect(repo.currentPage.value).toBe(1)
|
||||||
|
expect(mockGet).toHaveBeenLastCalledWith(
|
||||||
|
'/clients',
|
||||||
|
{
|
||||||
|
search: 'acme',
|
||||||
|
'categoryCode[]': ['DISTRIBUTEUR', 'COURTIER'],
|
||||||
|
'siteId[]': ['1', '2'],
|
||||||
|
archivedOnly: true,
|
||||||
|
page: 1,
|
||||||
|
itemsPerPage: 10,
|
||||||
|
},
|
||||||
|
expect.objectContaining({ toast: false }),
|
||||||
|
)
|
||||||
|
})
|
||||||
|
|
||||||
|
it('repasse a une query propre apres reinitialisation des filtres', async () => {
|
||||||
|
const repo = useClientsRepository()
|
||||||
|
await repo.setFilters({ search: 'acme', archivedOnly: true }, { replace: true })
|
||||||
|
await repo.setFilters({}, { replace: true })
|
||||||
|
|
||||||
|
expect(mockGet).toHaveBeenLastCalledWith(
|
||||||
|
'/clients',
|
||||||
|
{ page: 1, itemsPerPage: 10 },
|
||||||
|
expect.objectContaining({ toast: false }),
|
||||||
|
)
|
||||||
|
})
|
||||||
|
})
|
||||||
@@ -0,0 +1,53 @@
|
|||||||
|
import { usePaginatedList } from '~/shared/composables/usePaginatedList'
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Site Starseed rattache a une adresse du client, tel qu'embarque en LISTE
|
||||||
|
* (groupe site:read) pour la colonne « Site(s) » du Repertoire (badges colores).
|
||||||
|
*/
|
||||||
|
export interface ClientSite {
|
||||||
|
id: number
|
||||||
|
name: string
|
||||||
|
color: string
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Categorie rattachee au client, embarquee en LISTE (groupe category:read).
|
||||||
|
* Seul le `code` (stable, MAJUSCULE — ERP-78) est affiche dans la colonne
|
||||||
|
* « Catégories ». Les autres champs sont presents mais non utilises ici.
|
||||||
|
*/
|
||||||
|
export interface ClientCategory {
|
||||||
|
code: string
|
||||||
|
name?: string
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Vue MINIMALE d'un client pour le Repertoire (datatable). Volontairement
|
||||||
|
* partielle : seuls les champs des colonnes + l'id (navigation) sont types ici.
|
||||||
|
* Le detail complet (onglets) est hors perimetre de cet ecran (ERP-62).
|
||||||
|
*/
|
||||||
|
export interface Client {
|
||||||
|
id: number
|
||||||
|
companyName: string
|
||||||
|
categories: ClientCategory[]
|
||||||
|
sites: ClientSite[]
|
||||||
|
/** Date ISO de derniere modification (default:read) — colonne « Dernière activité ». */
|
||||||
|
updatedAt: string | null
|
||||||
|
isArchived: boolean
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Repertoire clients (ERP-62) — simple enveloppe de `usePaginatedList<Client>`
|
||||||
|
* sur la ressource `/clients` (RG-13 : pagination serveur obligatoire ; jamais
|
||||||
|
* de chargement integral en memoire).
|
||||||
|
*
|
||||||
|
* Les filtres (recherche, categories, sites, archives) sont pilotes par la page
|
||||||
|
* via `setFilters` du composable partage — la remise en page 1 est garantie.
|
||||||
|
*
|
||||||
|
* Volontairement PAR INSTANCE (pas de singleton module-level) : l'etat tableau
|
||||||
|
* est propre a l'ecran Repertoire et meurt avec lui, comme tout consommateur de
|
||||||
|
* `usePaginatedList` (cf. sites.vue / categories.vue). Aucun reset au logout a
|
||||||
|
* gerer.
|
||||||
|
*/
|
||||||
|
export function useClientsRepository() {
|
||||||
|
return usePaginatedList<Client>({ url: '/clients' })
|
||||||
|
}
|
||||||
@@ -0,0 +1,421 @@
|
|||||||
|
<template>
|
||||||
|
<div>
|
||||||
|
<PageHeader>
|
||||||
|
{{ t('commercial.clients.title') }}
|
||||||
|
<template #actions>
|
||||||
|
<!-- gap-12 = 48px d'espacement entre Ajouter et Filtres. -->
|
||||||
|
<div class="flex items-center gap-12">
|
||||||
|
<MalioButton
|
||||||
|
v-if="canManage"
|
||||||
|
variant="secondary"
|
||||||
|
:label="t('commercial.clients.add')"
|
||||||
|
icon-name="mdi:add-bold"
|
||||||
|
icon-position="left"
|
||||||
|
@click="goToCreate"
|
||||||
|
/>
|
||||||
|
<!-- Bouton Filtres a DROITE d'Ajouter : meme design que
|
||||||
|
l'audit-log. Le compteur reflete les filtres actifs. -->
|
||||||
|
<MalioButton
|
||||||
|
v-if="canView"
|
||||||
|
variant="tertiary"
|
||||||
|
:label="filterButtonLabel"
|
||||||
|
icon-name="mdi:tune"
|
||||||
|
icon-position="left"
|
||||||
|
icon-size="24"
|
||||||
|
button-class="w-[184px] justify-start gap-4 text-black"
|
||||||
|
@click="openFilters"
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
</template>
|
||||||
|
</PageHeader>
|
||||||
|
|
||||||
|
<!-- Datatable branchee sur usePaginatedList via useClientsRepository :
|
||||||
|
pagination serveur, tri companyName ASC par defaut (cote back). -->
|
||||||
|
<MalioDataTable
|
||||||
|
:columns="columns"
|
||||||
|
:items="rows"
|
||||||
|
:total-items="totalItems"
|
||||||
|
:page="currentPage"
|
||||||
|
:per-page="itemsPerPage"
|
||||||
|
:per-page-options="itemsPerPageOptions"
|
||||||
|
row-clickable
|
||||||
|
table-class="table-fixed"
|
||||||
|
:empty-message="t('commercial.clients.empty')"
|
||||||
|
@row-click="onRowClick"
|
||||||
|
@update:page="goToPage"
|
||||||
|
@update:per-page="setItemsPerPage"
|
||||||
|
>
|
||||||
|
<!-- Categories : codes stables separes par une virgule (ERP-78). -->
|
||||||
|
<template #cell-categories="{ item }">
|
||||||
|
{{ formatCategories(item) }}
|
||||||
|
</template>
|
||||||
|
|
||||||
|
<!-- Sites : badges colores (name + color), agreges des adresses. -->
|
||||||
|
<template #cell-sites="{ item }">
|
||||||
|
<span class="flex flex-wrap gap-1">
|
||||||
|
<span
|
||||||
|
v-for="site in (item.sites as ClientSite[])"
|
||||||
|
:key="site.id"
|
||||||
|
class="inline-flex items-center rounded-full px-2 py-0.5 text-xs font-medium text-white"
|
||||||
|
:style="{ backgroundColor: site.color }"
|
||||||
|
>
|
||||||
|
{{ site.name }}
|
||||||
|
</span>
|
||||||
|
</span>
|
||||||
|
</template>
|
||||||
|
|
||||||
|
<!-- Derniere activite : date de derniere modification (updatedAt). -->
|
||||||
|
<template #cell-lastActivity="{ item }">
|
||||||
|
{{ formatLastActivity(item) }}
|
||||||
|
</template>
|
||||||
|
</MalioDataTable>
|
||||||
|
|
||||||
|
<div class="flex justify-center mt-6">
|
||||||
|
<MalioButton
|
||||||
|
v-if="canView"
|
||||||
|
variant="primary"
|
||||||
|
:label="t('commercial.clients.export')"
|
||||||
|
:disabled="exporting"
|
||||||
|
@click="exportXlsx"
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<!-- Drawer de filtres : etat BROUILLON, applique uniquement au clic sur
|
||||||
|
« Appliquer ». Meme pattern que l'audit-log. Etat 100 % local, jamais
|
||||||
|
dans l'URL (regle ABSOLUE n°6). -->
|
||||||
|
<MalioDrawer
|
||||||
|
v-model="filterDrawerOpen"
|
||||||
|
drawer-class="max-w-[450px]"
|
||||||
|
body-class="p-0"
|
||||||
|
footer-class="justify-between border-t border-black p-6"
|
||||||
|
>
|
||||||
|
<template #header>
|
||||||
|
<h2 class="text-[24px] font-bold uppercase">{{ t('commercial.clients.filters.title') }}</h2>
|
||||||
|
</template>
|
||||||
|
|
||||||
|
<MalioAccordion>
|
||||||
|
<!-- Recherche : nom societe + contact + email (param `search`). -->
|
||||||
|
<MalioAccordionItem :title="t('commercial.clients.filters.search')" value="search">
|
||||||
|
<MalioInputText
|
||||||
|
v-model="draftSearch"
|
||||||
|
icon-name="mdi:magnify"
|
||||||
|
/>
|
||||||
|
</MalioAccordionItem>
|
||||||
|
|
||||||
|
<!-- Categories : cases a cocher (multi). Valeur = code stable. -->
|
||||||
|
<MalioAccordionItem :title="t('commercial.clients.filters.categories')" value="categories">
|
||||||
|
<div class="flex flex-col">
|
||||||
|
<MalioCheckbox
|
||||||
|
v-for="opt in categoryOptions"
|
||||||
|
:id="`filter-category-${opt.value}`"
|
||||||
|
:key="opt.value"
|
||||||
|
:label="opt.label"
|
||||||
|
:model-value="draftCategoryCodes.includes(opt.value)"
|
||||||
|
@update:model-value="(val: boolean) => toggleCategory(opt.value, val)"
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
</MalioAccordionItem>
|
||||||
|
|
||||||
|
<!-- Sites : cases a cocher (multi). Valeur = id du site. -->
|
||||||
|
<MalioAccordionItem :title="t('commercial.clients.filters.sites')" value="sites">
|
||||||
|
<div class="flex flex-col">
|
||||||
|
<MalioCheckbox
|
||||||
|
v-for="opt in siteOptions"
|
||||||
|
:id="`filter-site-${opt.value}`"
|
||||||
|
:key="opt.value"
|
||||||
|
:label="opt.label"
|
||||||
|
:model-value="draftSiteIds.includes(opt.value)"
|
||||||
|
@update:model-value="(val: boolean) => toggleSite(opt.value, val)"
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
</MalioAccordionItem>
|
||||||
|
|
||||||
|
<!-- Statut : bool unique. Coche = archives uniquement, sinon actifs. -->
|
||||||
|
<MalioAccordionItem :title="t('commercial.clients.filters.status')" value="status">
|
||||||
|
<MalioCheckbox
|
||||||
|
id="filter-archived-only"
|
||||||
|
:label="t('commercial.clients.filters.archivedOnly')"
|
||||||
|
:model-value="draftArchivedOnly"
|
||||||
|
@update:model-value="(val: boolean) => draftArchivedOnly = val"
|
||||||
|
/>
|
||||||
|
</MalioAccordionItem>
|
||||||
|
</MalioAccordion>
|
||||||
|
|
||||||
|
<template #footer>
|
||||||
|
<MalioButton
|
||||||
|
variant="tertiary"
|
||||||
|
:label="t('commercial.clients.filters.reset')"
|
||||||
|
button-class="w-m-btn-action"
|
||||||
|
@click="resetFilters"
|
||||||
|
/>
|
||||||
|
<MalioButton
|
||||||
|
variant="primary"
|
||||||
|
:label="t('commercial.clients.filters.apply')"
|
||||||
|
button-class="w-[170px]"
|
||||||
|
@click="applyFilters"
|
||||||
|
/>
|
||||||
|
</template>
|
||||||
|
</MalioDrawer>
|
||||||
|
</div>
|
||||||
|
</template>
|
||||||
|
|
||||||
|
<script setup lang="ts">
|
||||||
|
import { computed, onMounted, ref } from 'vue'
|
||||||
|
import type { Client, ClientSite } from '~/modules/commercial/composables/useClientsRepository'
|
||||||
|
|
||||||
|
interface FilterOption {
|
||||||
|
value: string
|
||||||
|
label: string
|
||||||
|
}
|
||||||
|
|
||||||
|
const { t } = useI18n()
|
||||||
|
const api = useApi()
|
||||||
|
const router = useRouter()
|
||||||
|
const toast = useToast()
|
||||||
|
const { can } = usePermissions()
|
||||||
|
|
||||||
|
useHead({ title: t('commercial.clients.title') })
|
||||||
|
|
||||||
|
// Bouton « Ajouter » reserve a `manage` (POST /clients garde manage seul →
|
||||||
|
// Compta / Usine ne creent pas). « Exporter » et « Filtres » suivent `view`.
|
||||||
|
const canManage = computed(() => can('commercial.clients.manage'))
|
||||||
|
const canView = computed(() => can('commercial.clients.view'))
|
||||||
|
|
||||||
|
const {
|
||||||
|
items: clients,
|
||||||
|
totalItems,
|
||||||
|
currentPage,
|
||||||
|
itemsPerPage,
|
||||||
|
itemsPerPageOptions,
|
||||||
|
fetch: loadClients,
|
||||||
|
goToPage,
|
||||||
|
setItemsPerPage,
|
||||||
|
setFilters,
|
||||||
|
} = useClientsRepository()
|
||||||
|
|
||||||
|
// Mappe les clients en objets « plats » pour MalioDataTable (items typees
|
||||||
|
// Record<string, unknown>[]) : un objet litteral porte une signature d'index
|
||||||
|
// implicite, contrairement a l'interface Client. Meme pattern que sites.vue.
|
||||||
|
const rows = computed(() => clients.value.map(client => ({
|
||||||
|
id: client.id,
|
||||||
|
companyName: client.companyName,
|
||||||
|
categories: client.categories,
|
||||||
|
sites: client.sites,
|
||||||
|
updatedAt: client.updatedAt,
|
||||||
|
})))
|
||||||
|
|
||||||
|
const columns = [
|
||||||
|
{ key: 'companyName', label: t('commercial.clients.column.companyName') },
|
||||||
|
{ key: 'categories', label: t('commercial.clients.column.categories') },
|
||||||
|
{ key: 'sites', label: t('commercial.clients.column.sites') },
|
||||||
|
{ key: 'lastActivity', label: t('commercial.clients.column.lastActivity') },
|
||||||
|
]
|
||||||
|
|
||||||
|
/** Codes des categories du client, separes par une virgule (ERP-78). */
|
||||||
|
function formatCategories(item: Record<string, unknown>): string {
|
||||||
|
const categories = (item.categories as Client['categories']) ?? []
|
||||||
|
return categories.map(c => c.code).join(', ')
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Derniere activite : faute de suivi d'activite metier au M1, on affiche la
|
||||||
|
* date de derniere modification de la fiche (updatedAt, expose en liste via
|
||||||
|
* default:read). Format court francais jj/mm/aaaa.
|
||||||
|
*/
|
||||||
|
function formatLastActivity(item: Record<string, unknown>): string {
|
||||||
|
const value = item.updatedAt as string | null | undefined
|
||||||
|
if (!value) {
|
||||||
|
return ''
|
||||||
|
}
|
||||||
|
|
||||||
|
// Garde-fou date invalide : un updatedAt mal forme donnerait « Invalid Date ».
|
||||||
|
const date = new Date(value)
|
||||||
|
if (Number.isNaN(date.getTime())) {
|
||||||
|
return ''
|
||||||
|
}
|
||||||
|
|
||||||
|
return date.toLocaleDateString('fr-FR')
|
||||||
|
}
|
||||||
|
|
||||||
|
/** Clic sur une ligne → ecran Consultation (route a plat /clients/{id}). */
|
||||||
|
function onRowClick(item: Record<string, unknown>): void {
|
||||||
|
router.push(`/clients/${item.id}`)
|
||||||
|
}
|
||||||
|
|
||||||
|
function goToCreate(): void {
|
||||||
|
router.push('/clients/new')
|
||||||
|
}
|
||||||
|
|
||||||
|
// ── Filtres (drawer) ────────────────────────────────────────────────────────
|
||||||
|
// Deux niveaux d'etat (pattern audit-log) :
|
||||||
|
// - APPLIED : pilote la liste/l'export + le compteur du bouton. Modifie
|
||||||
|
// uniquement au clic « Appliquer » / « Réinitialiser ».
|
||||||
|
// - DRAFT : edite librement dans le drawer ; recopie vers applied a la validation.
|
||||||
|
const filterDrawerOpen = ref(false)
|
||||||
|
|
||||||
|
const draftSearch = ref('')
|
||||||
|
const draftCategoryCodes = ref<string[]>([])
|
||||||
|
const draftSiteIds = ref<string[]>([])
|
||||||
|
const draftArchivedOnly = ref(false)
|
||||||
|
|
||||||
|
const appliedSearch = ref('')
|
||||||
|
const appliedCategoryCodes = ref<string[]>([])
|
||||||
|
const appliedSiteIds = ref<string[]>([])
|
||||||
|
const appliedArchivedOnly = ref(false)
|
||||||
|
|
||||||
|
// Options des selects multi, chargees une fois (referentiels courts).
|
||||||
|
const categoryOptions = ref<FilterOption[]>([])
|
||||||
|
const siteOptions = ref<FilterOption[]>([])
|
||||||
|
|
||||||
|
const activeFilterCount = computed(() => {
|
||||||
|
let count = 0
|
||||||
|
if (appliedSearch.value.trim() !== '') count++
|
||||||
|
if (appliedCategoryCodes.value.length > 0) count++
|
||||||
|
if (appliedSiteIds.value.length > 0) count++
|
||||||
|
if (appliedArchivedOnly.value) count++
|
||||||
|
return count
|
||||||
|
})
|
||||||
|
|
||||||
|
const filterButtonLabel = computed(() => {
|
||||||
|
const base = t('commercial.clients.filters.title')
|
||||||
|
return activeFilterCount.value > 0 ? `${base} (${activeFilterCount.value})` : base
|
||||||
|
})
|
||||||
|
|
||||||
|
// Recopie l'etat applique vers le brouillon puis ouvre le drawer : la
|
||||||
|
// reouverture reflete les filtres actifs.
|
||||||
|
function openFilters(): void {
|
||||||
|
draftSearch.value = appliedSearch.value
|
||||||
|
draftCategoryCodes.value = [...appliedCategoryCodes.value]
|
||||||
|
draftSiteIds.value = [...appliedSiteIds.value]
|
||||||
|
draftArchivedOnly.value = appliedArchivedOnly.value
|
||||||
|
filterDrawerOpen.value = true
|
||||||
|
}
|
||||||
|
|
||||||
|
function toggleCategory(code: string, selected: boolean): void {
|
||||||
|
draftCategoryCodes.value = selected
|
||||||
|
? [...draftCategoryCodes.value, code]
|
||||||
|
: draftCategoryCodes.value.filter(c => c !== code)
|
||||||
|
}
|
||||||
|
|
||||||
|
function toggleSite(id: string, selected: boolean): void {
|
||||||
|
draftSiteIds.value = selected
|
||||||
|
? [...draftSiteIds.value, id]
|
||||||
|
: draftSiteIds.value.filter(s => s !== id)
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Construit le payload de filtres serveur a partir de l'etat applique. Cles
|
||||||
|
* `categoryCode[]` / `siteId[]` pour que PHP les parse en tableaux (OR cote back).
|
||||||
|
* Les filtres vides sont omis pour une query propre.
|
||||||
|
*/
|
||||||
|
function buildFilterPayload(): Record<string, string | string[] | boolean> {
|
||||||
|
const payload: Record<string, string | string[] | boolean> = {}
|
||||||
|
if (appliedSearch.value.trim() !== '') payload.search = appliedSearch.value.trim()
|
||||||
|
if (appliedCategoryCodes.value.length > 0) payload['categoryCode[]'] = [...appliedCategoryCodes.value]
|
||||||
|
if (appliedSiteIds.value.length > 0) payload['siteId[]'] = [...appliedSiteIds.value]
|
||||||
|
if (appliedArchivedOnly.value) payload.archivedOnly = true
|
||||||
|
return payload
|
||||||
|
}
|
||||||
|
|
||||||
|
// « Appliquer » : recopie brouillon → applied, pousse les filtres (retombe en
|
||||||
|
// page 1 via usePaginatedList) et ferme le drawer.
|
||||||
|
function applyFilters(): void {
|
||||||
|
appliedSearch.value = draftSearch.value.trim()
|
||||||
|
appliedCategoryCodes.value = [...draftCategoryCodes.value]
|
||||||
|
appliedSiteIds.value = [...draftSiteIds.value]
|
||||||
|
appliedArchivedOnly.value = draftArchivedOnly.value
|
||||||
|
|
||||||
|
setFilters(buildFilterPayload(), { replace: true })
|
||||||
|
filterDrawerOpen.value = false
|
||||||
|
}
|
||||||
|
|
||||||
|
// « Réinitialiser » : vide brouillon ET applied, recharge la liste complete.
|
||||||
|
// Le drawer reste ouvert pour montrer le formulaire vide.
|
||||||
|
function resetFilters(): void {
|
||||||
|
draftSearch.value = ''
|
||||||
|
draftCategoryCodes.value = []
|
||||||
|
draftSiteIds.value = []
|
||||||
|
draftArchivedOnly.value = false
|
||||||
|
|
||||||
|
appliedSearch.value = ''
|
||||||
|
appliedCategoryCodes.value = []
|
||||||
|
appliedSiteIds.value = []
|
||||||
|
appliedArchivedOnly.value = false
|
||||||
|
|
||||||
|
setFilters({}, { replace: true })
|
||||||
|
}
|
||||||
|
|
||||||
|
/** Charge les referentiels du drawer (categories + sites) via ?pagination=false. */
|
||||||
|
async function loadFilterOptions(): Promise<void> {
|
||||||
|
const [cats, sites] = await Promise.all([
|
||||||
|
api.get<{ member?: Array<{ code: string, name: string }> }>(
|
||||||
|
'/categories',
|
||||||
|
{ pagination: 'false' },
|
||||||
|
{ headers: { Accept: 'application/ld+json' }, toast: false },
|
||||||
|
),
|
||||||
|
api.get<{ member?: Array<{ id: number, name: string }> }>(
|
||||||
|
'/sites',
|
||||||
|
{ pagination: 'false' },
|
||||||
|
{ headers: { Accept: 'application/ld+json' }, toast: false },
|
||||||
|
),
|
||||||
|
])
|
||||||
|
|
||||||
|
categoryOptions.value = (cats.member ?? []).map(c => ({ value: c.code, label: c.name }))
|
||||||
|
siteOptions.value = (sites.member ?? []).map(s => ({ value: String(s.id), label: s.name }))
|
||||||
|
}
|
||||||
|
|
||||||
|
// ── Export XLSX ─────────────────────────────────────────────────────────────
|
||||||
|
// Memes filtres que la vue. La colonne SIREN n'est dans le fichier que si
|
||||||
|
// l'utilisateur a accounting.view (gere cote back).
|
||||||
|
const exporting = ref(false)
|
||||||
|
|
||||||
|
async function exportXlsx(): Promise<void> {
|
||||||
|
if (exporting.value) {
|
||||||
|
return
|
||||||
|
}
|
||||||
|
exporting.value = true
|
||||||
|
try {
|
||||||
|
// useApi type ses options en JSON ; l'export renvoie un binaire, donc on
|
||||||
|
// force responseType:'blob' (transmis tel quel a ofetch au runtime). Cast
|
||||||
|
// contenu faute d'overload blob sur le client partage — a generaliser via
|
||||||
|
// un ticket dedie si d'autres exports binaires arrivent.
|
||||||
|
const blob = await api.get<Blob>('/clients/export.xlsx', buildFilterPayload(), {
|
||||||
|
responseType: 'blob',
|
||||||
|
toast: false,
|
||||||
|
} as unknown as Parameters<typeof api.get>[2])
|
||||||
|
|
||||||
|
triggerDownload(blob, 'repertoire-clients.xlsx')
|
||||||
|
}
|
||||||
|
catch {
|
||||||
|
toast.error({
|
||||||
|
title: t('commercial.clients.toast.error'),
|
||||||
|
message: t('commercial.clients.toast.exportError'),
|
||||||
|
})
|
||||||
|
}
|
||||||
|
finally {
|
||||||
|
exporting.value = false
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/** Declenche le telechargement d'un blob via un lien temporaire. */
|
||||||
|
function triggerDownload(blob: Blob, filename: string): void {
|
||||||
|
const url = URL.createObjectURL(blob)
|
||||||
|
const link = document.createElement('a')
|
||||||
|
link.href = url
|
||||||
|
link.download = filename
|
||||||
|
document.body.appendChild(link)
|
||||||
|
link.click()
|
||||||
|
link.remove()
|
||||||
|
URL.revokeObjectURL(url)
|
||||||
|
}
|
||||||
|
|
||||||
|
onMounted(() => {
|
||||||
|
loadClients()
|
||||||
|
// Echec du chargement des referentiels non bloquant : la liste s'affiche,
|
||||||
|
// l'utilisateur perd juste les options de filtre.
|
||||||
|
loadFilterOptions().catch(() => {
|
||||||
|
categoryOptions.value = []
|
||||||
|
siteOptions.value = []
|
||||||
|
})
|
||||||
|
})
|
||||||
|
</script>
|
||||||
@@ -0,0 +1,23 @@
|
|||||||
|
import { describe, it, expect } from 'vitest'
|
||||||
|
import { formatPhoneFR } from '../phone'
|
||||||
|
|
||||||
|
describe('formatPhoneFR', () => {
|
||||||
|
it('formate un numero 10 chiffres en XX XX XX XX XX', () => {
|
||||||
|
expect(formatPhoneFR('0612345678')).toBe('06 12 34 56 78')
|
||||||
|
})
|
||||||
|
|
||||||
|
it('tolere une saisie deja pointee ou espacee', () => {
|
||||||
|
expect(formatPhoneFR('06.12.34.56.78')).toBe('06 12 34 56 78')
|
||||||
|
expect(formatPhoneFR('06 12 34 56 78')).toBe('06 12 34 56 78')
|
||||||
|
})
|
||||||
|
|
||||||
|
it('retourne une chaine vide pour une valeur vide ou nulle', () => {
|
||||||
|
expect(formatPhoneFR('')).toBe('')
|
||||||
|
expect(formatPhoneFR(null)).toBe('')
|
||||||
|
expect(formatPhoneFR(undefined)).toBe('')
|
||||||
|
})
|
||||||
|
|
||||||
|
it('groupe par 2 meme un nombre impair de chiffres (dernier groupe seul)', () => {
|
||||||
|
expect(formatPhoneFR('123')).toBe('12 3')
|
||||||
|
})
|
||||||
|
})
|
||||||
@@ -0,0 +1,23 @@
|
|||||||
|
/**
|
||||||
|
* Formatage d'un numero de telephone francais en groupes de 2 chiffres
|
||||||
|
* (`XX XX XX XX XX`).
|
||||||
|
*
|
||||||
|
* Helper PARTAGE volontaire : les telephones sont presents un peu partout dans
|
||||||
|
* l'app (fiches clients, contacts, fournisseurs, prestataires...). Introduit ici
|
||||||
|
* comme util transverse stable plutot que duplique a chaque ecran. La signature
|
||||||
|
* `formatPhoneFR(value): string` est coordonnee avec ERP-66, qui pourra enrichir
|
||||||
|
* l'implementation (validation, indicatif international) sans casser les appelants.
|
||||||
|
*
|
||||||
|
* - Ne garde que les chiffres puis groupe par 2 (tolere une saisie deja espacee
|
||||||
|
* ou pointee, ex: `06.12.34.56.78` ou `0612345678`).
|
||||||
|
* - Retourne une chaine vide si la valeur est vide/nulle (cellule vide propre).
|
||||||
|
*/
|
||||||
|
export function formatPhoneFR(value: string | null | undefined): string {
|
||||||
|
const digits = (value ?? '').replace(/\D/g, '')
|
||||||
|
if (digits.length === 0) {
|
||||||
|
return ''
|
||||||
|
}
|
||||||
|
|
||||||
|
// Groupe par paquets de 2 ; un dernier groupe impair reste tel quel.
|
||||||
|
return digits.match(/.{1,2}/g)?.join(' ') ?? digits
|
||||||
|
}
|
||||||
@@ -208,6 +208,8 @@ migration-migrate:
|
|||||||
# exprimables via les attributs Doctrine ORM (fonctionnel + partiel), donc
|
# exprimables via les attributs Doctrine ORM (fonctionnel + partiel), donc
|
||||||
# ils disparaissent apres schema:update. On les recree par dbal:run-sql :
|
# ils disparaissent apres schema:update. On les recree par dbal:run-sql :
|
||||||
# - `uq_category_name_type_active` (M0 Catalog) : tests RG-1.07.
|
# - `uq_category_name_type_active` (M0 Catalog) : tests RG-1.07.
|
||||||
|
# - `uq_category_code` (Catalog ERP-78) : unicite du code categorie parmi
|
||||||
|
# les actifs (slug du nom), pilote RG-1.03/1.29.
|
||||||
# - `uq_client_company_name_active` (M1 Commercial) : unicite nom societe
|
# - `uq_client_company_name_active` (M1 Commercial) : unicite nom societe
|
||||||
# parmi actifs non archives/non supprimes (RG-1.16), tests ERP-55.
|
# parmi actifs non archives/non supprimes (RG-1.16), tests ERP-55.
|
||||||
# Sans ces restores, les POST doublons remontent 201 au lieu de 409.
|
# Sans ces restores, les POST doublons remontent 201 au lieu de 409.
|
||||||
@@ -225,6 +227,7 @@ test-db-setup:
|
|||||||
$(SYMFONY_CONSOLE) --env=test --no-interaction app:sync-permissions
|
$(SYMFONY_CONSOLE) --env=test --no-interaction app:sync-permissions
|
||||||
$(SYMFONY_CONSOLE) --env=test --no-interaction app:seed-rbac
|
$(SYMFONY_CONSOLE) --env=test --no-interaction app:seed-rbac
|
||||||
$(SYMFONY_CONSOLE) --env=test dbal:run-sql "CREATE UNIQUE INDEX IF NOT EXISTS uq_category_name_type_active ON category (LOWER(name), category_type_id) WHERE deleted_at IS NULL"
|
$(SYMFONY_CONSOLE) --env=test dbal:run-sql "CREATE UNIQUE INDEX IF NOT EXISTS uq_category_name_type_active ON category (LOWER(name), category_type_id) WHERE deleted_at IS NULL"
|
||||||
|
$(SYMFONY_CONSOLE) --env=test dbal:run-sql "CREATE UNIQUE INDEX IF NOT EXISTS uq_category_code ON category (code) WHERE deleted_at IS NULL"
|
||||||
$(SYMFONY_CONSOLE) --env=test dbal:run-sql "CREATE UNIQUE INDEX IF NOT EXISTS uq_client_company_name_active ON client (LOWER(company_name)) WHERE is_archived = FALSE AND deleted_at IS NULL"
|
$(SYMFONY_CONSOLE) --env=test dbal:run-sql "CREATE UNIQUE INDEX IF NOT EXISTS uq_client_company_name_active ON client (LOWER(company_name)) WHERE is_archived = FALSE AND deleted_at IS NULL"
|
||||||
|
|
||||||
fixtures:
|
fixtures:
|
||||||
|
|||||||
@@ -39,20 +39,40 @@ final class Version20260528120000 extends AbstractMigration
|
|||||||
|
|
||||||
public function up(Schema $schema): void
|
public function up(Schema $schema): void
|
||||||
{
|
{
|
||||||
// Ne commente que les tables deja presentes a ce stade de la chaine de
|
// Ne commente que les tables ET colonnes deja presentes a ce stade de la
|
||||||
// migrations. Les modules crees plus tard (ex: M1 Commercial, 06-01)
|
// chaine de migrations. Les tables des modules crees plus tard (M1
|
||||||
// figurent desormais dans le catalogue partage mais leurs tables
|
// Commercial, 06-01) ET les colonnes ajoutees ensuite sur une table
|
||||||
// n'existent pas encore ici : elles posent leurs propres COMMENT dans
|
// existante (ex: category.code, ERP-78 06-02) figurent desormais dans le
|
||||||
// leur migration dediee (regle ABSOLUE n°12). Garde-fou indispensable,
|
// catalogue partage mais n'existent pas encore ici : elles posent leur
|
||||||
// sinon l'ajout d'un module au catalogue casse ce retrofit avec un
|
// propre COMMENT dans leur migration dediee (regle ABSOLUE n°12). Garde-fou
|
||||||
// "relation X does not exist".
|
// indispensable (table + colonne), sinon enrichir le catalogue casse ce
|
||||||
$existingTables = array_values(array_filter(
|
// retrofit avec un "relation/column X does not exist".
|
||||||
array_keys(ColumnCommentsCatalog::comments()),
|
foreach (ColumnCommentsCatalog::comments() as $table => $entries) {
|
||||||
static fn (string $table): bool => $schema->hasTable($table),
|
if (!$schema->hasTable($table)) {
|
||||||
));
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
foreach (ColumnCommentsCatalog::toSqlStatements($existingTables) as $sql) {
|
$dbTable = $schema->getTable($table);
|
||||||
$this->addSql($sql);
|
$quotedTable = '"'.str_replace('"', '""', $table).'"';
|
||||||
|
|
||||||
|
foreach ($entries as $column => $description) {
|
||||||
|
if ('_table' === $column) {
|
||||||
|
$this->addSql(sprintf('COMMENT ON TABLE %s IS $_$%s$_$', $quotedTable, $description));
|
||||||
|
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!$dbTable->hasColumn($column)) {
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
$this->addSql(sprintf(
|
||||||
|
'COMMENT ON COLUMN %s.%s IS $_$%s$_$',
|
||||||
|
$quotedTable,
|
||||||
|
'"'.str_replace('"', '""', $column).'"',
|
||||||
|
$description,
|
||||||
|
));
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -0,0 +1,189 @@
|
|||||||
|
<?php
|
||||||
|
|
||||||
|
declare(strict_types=1);
|
||||||
|
|
||||||
|
namespace DoctrineMigrations;
|
||||||
|
|
||||||
|
use App\Shared\Infrastructure\Database\CategoryCodeSql;
|
||||||
|
use Doctrine\DBAL\Schema\Schema;
|
||||||
|
use Doctrine\Migrations\AbstractMigration;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* ERP-78 — Refonte de la taxonomie Categories (M0/M1).
|
||||||
|
*
|
||||||
|
* Modele AVANT (merge via Version20260527164000 + Version20260601000000) :
|
||||||
|
* DISTRIBUTEUR / COURTIER / SECTEUR / AUTRE sont des `category_type`.
|
||||||
|
*
|
||||||
|
* Modele APRES (decision produit 01/06) :
|
||||||
|
* - UN SEUL `category_type` : CLIENT (code CLIENT, label « Client ») ;
|
||||||
|
* - Distributeur / Courtier / Secteur / Autre (+ categories metier fines)
|
||||||
|
* deviennent des `Category` rattachees au type CLIENT ;
|
||||||
|
* - filtrage metier sur un `code` stable porte par la `Category` (et non plus
|
||||||
|
* par le type) : on reporte les codes DISTRIBUTEUR / COURTIER sur la categorie
|
||||||
|
* correspondante. RG-1.03 (distributor/broker) et RG-1.29 (categorie interdite
|
||||||
|
* sur adresse) s'appuient desormais sur `category.code`.
|
||||||
|
*
|
||||||
|
* Migration CORRECTIVE et NOUVELLE : la migration mergee Version20260601000000
|
||||||
|
* (qui a pu tourner en CI / chez d'autres devs) n'est PAS editee.
|
||||||
|
*
|
||||||
|
* Namespace racine `DoctrineMigrations` (regle ABSOLUE n°11) et NON modulaire
|
||||||
|
* Catalog : avec plusieurs migrations_paths, Doctrine Migrations 3.x trie par
|
||||||
|
* FQCN alphabetique (AlphabeticalComparator). Introduire la 1re migration
|
||||||
|
* modulaire `App\Module\Catalog\...` la ferait trier AVANT toutes les
|
||||||
|
* `DoctrineMigrations\...` sur base vide -> elle s'executerait avant la creation
|
||||||
|
* des tables et le seed dont elle depend. Le namespace racine garantit l'ordre
|
||||||
|
* par timestamp.
|
||||||
|
*
|
||||||
|
* Idempotence : `ADD COLUMN IF NOT EXISTS`, `INSERT ... ON CONFLICT` / guards
|
||||||
|
* `NOT EXISTS`, `CREATE UNIQUE INDEX IF NOT EXISTS`. En prod la table `category`
|
||||||
|
* est vide (aucune fixture metier) : l'ajout de `code NOT NULL` est sur. En
|
||||||
|
* dev/test, le purger Doctrine vide `category`/`category_type` avant les
|
||||||
|
* fixtures, qui reproduisent le meme etat final (cf. CategoryTypeFixtures /
|
||||||
|
* CategoryFixtures).
|
||||||
|
*/
|
||||||
|
final class Version20260602100000 extends AbstractMigration
|
||||||
|
{
|
||||||
|
/**
|
||||||
|
* Categories systeme reportees depuis les anciens types : nom => code.
|
||||||
|
* Le code est la cle metier stable (RG-1.03 / RG-1.29).
|
||||||
|
*/
|
||||||
|
private const array SYSTEM_CATEGORIES = [
|
||||||
|
'Distributeur' => 'DISTRIBUTEUR',
|
||||||
|
'Courtier' => 'COURTIER',
|
||||||
|
'Secteur' => 'SECTEUR',
|
||||||
|
'Autre' => 'AUTRE',
|
||||||
|
];
|
||||||
|
|
||||||
|
/** Anciens codes de `category_type` devenus inutiles. */
|
||||||
|
private const array LEGACY_TYPE_CODES = ['DISTRIBUTEUR', 'COURTIER', 'SECTEUR', 'AUTRE'];
|
||||||
|
|
||||||
|
public function getDescription(): string
|
||||||
|
{
|
||||||
|
return 'ERP-78 : Category.code + type unique CLIENT (categories Distributeur/Courtier/Secteur/Autre codees, anciens types supprimes).';
|
||||||
|
}
|
||||||
|
|
||||||
|
public function up(Schema $schema): void
|
||||||
|
{
|
||||||
|
// 1. Colonne `code` (nullable d'abord pour pouvoir backfiller, NOT NULL ensuite).
|
||||||
|
$this->addSql('ALTER TABLE category ADD COLUMN IF NOT EXISTS code VARCHAR(50) DEFAULT NULL');
|
||||||
|
|
||||||
|
// 2. Type unique CLIENT (idempotent via l'index unique uq_category_type_code).
|
||||||
|
$this->addSql(<<<'SQL'
|
||||||
|
INSERT INTO category_type (code, label) VALUES ('CLIENT', 'Client')
|
||||||
|
ON CONFLICT (code) DO NOTHING
|
||||||
|
SQL);
|
||||||
|
|
||||||
|
// 3. Re-pointer toute categorie pre-existante (rattachee a un ancien type)
|
||||||
|
// vers le type CLIENT, en lui donnant un code derive du nom si absent.
|
||||||
|
// En prod la table est vide -> no-op ; defensif pour les envs qui
|
||||||
|
// auraient deja seede des categories sous les anciens types. Le slug
|
||||||
|
// SQL est le miroir EXACT de CategoryCodeGenerator::slugify (cf.
|
||||||
|
// CategoryCodeSql + CategoryCodeSqlSlugTest) : un nom accentue produit
|
||||||
|
// le meme code que la generation applicative (« Independant » ->
|
||||||
|
// INDEPENDANT, et non IND_PENDANT).
|
||||||
|
$this->addSql(
|
||||||
|
'UPDATE category c '
|
||||||
|
."SET category_type_id = (SELECT id FROM category_type WHERE code = 'CLIENT'), "
|
||||||
|
.'code = COALESCE(c.code, '.CategoryCodeSql::slugExpression('c.name').') '
|
||||||
|
.'WHERE c.category_type_id IN (SELECT id FROM category_type WHERE code IN (:legacyCodes))',
|
||||||
|
['legacyCodes' => self::LEGACY_TYPE_CODES],
|
||||||
|
['legacyCodes' => \Doctrine\DBAL\ArrayParameterType::STRING],
|
||||||
|
);
|
||||||
|
|
||||||
|
// 4. Creer les 4 categories systeme sous CLIENT (si leur code est libre
|
||||||
|
// parmi les actifs). created_at/updated_at NOT NULL -> now() ; le blame
|
||||||
|
// reste null (seed hors contexte HTTP, libelle « Systeme » cote front).
|
||||||
|
foreach (self::SYSTEM_CATEGORIES as $name => $code) {
|
||||||
|
$this->addSql(<<<'SQL'
|
||||||
|
INSERT INTO category (name, code, category_type_id, created_at, updated_at)
|
||||||
|
SELECT :name, :code, ct.id, NOW(), NOW()
|
||||||
|
FROM category_type ct
|
||||||
|
WHERE ct.code = 'CLIENT'
|
||||||
|
AND NOT EXISTS (
|
||||||
|
SELECT 1 FROM category c WHERE c.code = :code AND c.deleted_at IS NULL
|
||||||
|
)
|
||||||
|
SQL, ['name' => $name, 'code' => $code]);
|
||||||
|
}
|
||||||
|
|
||||||
|
// 5. Backfill defensif : toute categorie encore sans code recoit un slug
|
||||||
|
// de son nom (garantit que le SET NOT NULL passe). Meme expression de
|
||||||
|
// slug fidele au generateur applicatif (CategoryCodeSql).
|
||||||
|
$this->addSql(
|
||||||
|
'UPDATE category SET code = '.CategoryCodeSql::slugExpression('name').' WHERE code IS NULL',
|
||||||
|
);
|
||||||
|
|
||||||
|
// 6. Index unique partiel sur le code parmi les actifs (non exprimable en
|
||||||
|
// ORM : recree aussi dans `test-db-setup` apres schema:update).
|
||||||
|
$this->addSql('CREATE UNIQUE INDEX IF NOT EXISTS uq_category_code ON category (code) WHERE deleted_at IS NULL');
|
||||||
|
|
||||||
|
// 7. Code desormais obligatoire.
|
||||||
|
$this->addSql('ALTER TABLE category ALTER COLUMN code SET NOT NULL');
|
||||||
|
|
||||||
|
// 8. Documentation SQL (regle ABSOLUE n°12). Dollar-quoting Postgres.
|
||||||
|
$this->addSql(<<<'SQL'
|
||||||
|
COMMENT ON COLUMN category.code IS $_$Code technique stable (slug MAJUSCULE du nom, <= 50) — unique parmi les actifs (uq_category_code). Fige a la creation. DISTRIBUTEUR/COURTIER pilotent RG-1.03/1.29.$_$
|
||||||
|
SQL);
|
||||||
|
|
||||||
|
// 9. Supprimer les anciens types devenus orphelins (aucune categorie ne
|
||||||
|
// les reference plus apres le re-pointage de l'etape 3). Le guard
|
||||||
|
// NOT EXISTS evite de casser sur la FK RESTRICT category.category_type_id.
|
||||||
|
$this->addSql(<<<'SQL'
|
||||||
|
DELETE FROM category_type
|
||||||
|
WHERE code IN (:legacyCodes)
|
||||||
|
AND NOT EXISTS (
|
||||||
|
SELECT 1 FROM category c WHERE c.category_type_id = category_type.id
|
||||||
|
)
|
||||||
|
SQL, ['legacyCodes' => self::LEGACY_TYPE_CODES], ['legacyCodes' => \Doctrine\DBAL\ArrayParameterType::STRING]);
|
||||||
|
|
||||||
|
// 10. Realigner la doc SQL de client_address_category (migration mergee
|
||||||
|
// Version20260601000000, non editable) sur le nouveau modele RG-1.29.
|
||||||
|
$this->addSql(<<<'SQL'
|
||||||
|
COMMENT ON TABLE client_address_category IS $_$Jointure M2M client_address <-> category — codes DISTRIBUTEUR/COURTIER interdits sur une adresse (RG-1.29).$_$
|
||||||
|
SQL);
|
||||||
|
$this->addSql(<<<'SQL'
|
||||||
|
COMMENT ON COLUMN client_address_category.category_id IS $_$FK -> category.id, ON DELETE RESTRICT — categorie d adresse (tout code sauf DISTRIBUTEUR/COURTIER, RG-1.29).$_$
|
||||||
|
SQL);
|
||||||
|
}
|
||||||
|
|
||||||
|
public function down(Schema $schema): void
|
||||||
|
{
|
||||||
|
// Best-effort : rollback du modele CLIENT vers les 4 anciens types.
|
||||||
|
// 1. Retirer l'index unique sur le code.
|
||||||
|
$this->addSql('DROP INDEX IF EXISTS uq_category_code');
|
||||||
|
|
||||||
|
// 2. Recreer les 4 anciens types.
|
||||||
|
$this->addSql(<<<'SQL'
|
||||||
|
INSERT INTO category_type (code, label) VALUES
|
||||||
|
('DISTRIBUTEUR', 'Distributeur'),
|
||||||
|
('COURTIER', 'Courtier'),
|
||||||
|
('SECTEUR', 'Secteur'),
|
||||||
|
('AUTRE', 'Autre')
|
||||||
|
ON CONFLICT (code) DO NOTHING
|
||||||
|
SQL);
|
||||||
|
|
||||||
|
// 3. Re-pointer les categories systeme (par code) vers leur type d'origine.
|
||||||
|
// Codes inlines : constantes controlees (self::SYSTEM_CATEGORIES), pas
|
||||||
|
// d'entree utilisateur — evite le binding d'un parametre nomme repete.
|
||||||
|
foreach (self::SYSTEM_CATEGORIES as $name => $code) {
|
||||||
|
$this->addSql(sprintf(
|
||||||
|
"UPDATE category SET category_type_id = (SELECT id FROM category_type WHERE code = '%s') WHERE code = '%s'",
|
||||||
|
$code,
|
||||||
|
$code,
|
||||||
|
));
|
||||||
|
}
|
||||||
|
|
||||||
|
// 4. Supprimer le type CLIENT s'il ne reference plus aucune categorie.
|
||||||
|
$this->addSql(<<<'SQL'
|
||||||
|
DELETE FROM category_type
|
||||||
|
WHERE code = 'CLIENT'
|
||||||
|
AND NOT EXISTS (
|
||||||
|
SELECT 1 FROM category c WHERE c.category_type_id = category_type.id
|
||||||
|
)
|
||||||
|
SQL);
|
||||||
|
|
||||||
|
// 5. Retirer la colonne code (les categories libres sans type d'origine
|
||||||
|
// restent sous CLIENT si encore presentes — rollback uniquement
|
||||||
|
// pertinent en prod ou seules les 4 categories systeme existent).
|
||||||
|
$this->addSql('ALTER TABLE category DROP COLUMN IF EXISTS code');
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -0,0 +1,77 @@
|
|||||||
|
<?php
|
||||||
|
|
||||||
|
declare(strict_types=1);
|
||||||
|
|
||||||
|
namespace App\Module\Catalog\Application\Service;
|
||||||
|
|
||||||
|
use App\Module\Catalog\Domain\Repository\CategoryRepositoryInterface;
|
||||||
|
use Symfony\Component\String\Slugger\AsciiSlugger;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Genere le code technique stable d'une Category a partir de son nom (ERP-78).
|
||||||
|
*
|
||||||
|
* Regle (decision produit 02/06) : `code` est obligatoire et auto-genere — un
|
||||||
|
* slug MAJUSCULE du nom, sans accent, separateurs non alphanumeriques reduits a
|
||||||
|
* `_`, borne a 50 caracteres (longueur colonne). Exemples :
|
||||||
|
* - « Distributeur » -> DISTRIBUTEUR
|
||||||
|
* - « Agro-alimentaire » -> AGRO_ALIMENTAIRE
|
||||||
|
* - « Transport/Logistique » -> TRANSPORT_LOGISTIQUE
|
||||||
|
*
|
||||||
|
* Le code est FIGE a la creation (jamais recalcule sur renommage) afin de rester
|
||||||
|
* une cle deterministe stable entre environnements (RG-1.03 / RG-1.29 cote M1).
|
||||||
|
*
|
||||||
|
* Unicite : l'index partiel `uq_category_code` (WHERE deleted_at IS NULL) impose
|
||||||
|
* l'unicite parmi les categories actives. Deux noms distincts peuvent produire
|
||||||
|
* le meme slug (« Agro alimentaire » / « Agro-alimentaire ») : on suffixe alors
|
||||||
|
* le code par `_2`, `_3`... jusqu'a obtenir un code libre.
|
||||||
|
*/
|
||||||
|
final class CategoryCodeGenerator
|
||||||
|
{
|
||||||
|
/** Longueur maximale de la colonne `category.code`. */
|
||||||
|
private const int MAX_LENGTH = 50;
|
||||||
|
|
||||||
|
private readonly AsciiSlugger $slugger;
|
||||||
|
|
||||||
|
public function __construct(
|
||||||
|
private readonly CategoryRepositoryInterface $categoryRepository,
|
||||||
|
) {
|
||||||
|
$this->slugger = new AsciiSlugger();
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Slug brut (sans garantie d'unicite) — utile pour les seeds deterministes.
|
||||||
|
*/
|
||||||
|
public function slugify(string $name): string
|
||||||
|
{
|
||||||
|
$slug = $this->slugger->slug($name, '_')->upper()->toString();
|
||||||
|
|
||||||
|
// Borne a la longueur colonne, puis retire un eventuel `_` terminal
|
||||||
|
// introduit par la troncature.
|
||||||
|
$slug = substr($slug, 0, self::MAX_LENGTH);
|
||||||
|
$slug = trim($slug, '_');
|
||||||
|
|
||||||
|
// Garde-fou : un nom uniquement compose de caracteres non alphanumeriques
|
||||||
|
// (theorique, le nom est NotBlank + Length>=2) donnerait un slug vide.
|
||||||
|
return '' === $slug ? 'CATEGORY' : $slug;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Code unique parmi les categories actives : slug du nom, suffixe `_N` en
|
||||||
|
* cas de collision. `$excludeId` ignore la categorie courante (PATCH).
|
||||||
|
*/
|
||||||
|
public function generateUnique(string $name, ?int $excludeId = null): string
|
||||||
|
{
|
||||||
|
$base = $this->slugify($name);
|
||||||
|
$candidate = $base;
|
||||||
|
$suffix = 2;
|
||||||
|
|
||||||
|
while ($this->categoryRepository->existsActiveByCode($candidate, $excludeId)) {
|
||||||
|
$suffixStr = '_'.$suffix;
|
||||||
|
// Retronque la base pour que `base + suffixe` tienne dans 50 caracteres.
|
||||||
|
$candidate = substr($base, 0, self::MAX_LENGTH - strlen($suffixStr)).$suffixStr;
|
||||||
|
++$suffix;
|
||||||
|
}
|
||||||
|
|
||||||
|
return $candidate;
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -74,10 +74,11 @@ use Symfony\Component\Validator\Constraints as Assert;
|
|||||||
)]
|
)]
|
||||||
#[ORM\Entity(repositoryClass: DoctrineCategoryRepository::class)]
|
#[ORM\Entity(repositoryClass: DoctrineCategoryRepository::class)]
|
||||||
#[ORM\Table(name: 'category')]
|
#[ORM\Table(name: 'category')]
|
||||||
// Index nommes pour matcher la migration (cf. Role/Permission/Site). L'index
|
// Index nommes pour matcher la migration (cf. Role/Permission/Site). Les index
|
||||||
// unique partiel `uq_category_name_type_active` (LOWER(name), category_type_id
|
// uniques partiels `uq_category_name_type_active` (LOWER(name), category_type_id
|
||||||
// WHERE deleted_at IS NULL) reste possede par la seule migration : Doctrine ORM
|
// WHERE deleted_at IS NULL) et `uq_category_code` (code WHERE deleted_at IS NULL)
|
||||||
// ne sait pas exprimer un index fonctionnel + partiel via attribut.
|
// restent possedes par la seule migration : Doctrine ORM ne sait pas exprimer un
|
||||||
|
// index partiel via attribut.
|
||||||
#[ORM\Index(name: 'idx_category_deleted_at', columns: ['deleted_at'])]
|
#[ORM\Index(name: 'idx_category_deleted_at', columns: ['deleted_at'])]
|
||||||
#[ORM\Index(name: 'idx_category_type_id', columns: ['category_type_id'])]
|
#[ORM\Index(name: 'idx_category_type_id', columns: ['category_type_id'])]
|
||||||
#[ORM\Index(name: 'idx_category_created_by', columns: ['created_by'])]
|
#[ORM\Index(name: 'idx_category_created_by', columns: ['created_by'])]
|
||||||
@@ -109,6 +110,16 @@ class Category implements TimestampableInterface, BlamableInterface, CategoryInt
|
|||||||
#[Groups(['category:read', 'category:write'])]
|
#[Groups(['category:read', 'category:write'])]
|
||||||
private ?string $name = null;
|
private ?string $name = null;
|
||||||
|
|
||||||
|
// Code technique stable (slug MAJUSCULE du nom) — NOT NULL + unique parmi les
|
||||||
|
// actifs (index partiel `uq_category_code` possede par la migration). Genere
|
||||||
|
// par le CategoryProcessor a la creation puis fige (jamais recalcule sur
|
||||||
|
// renommage) : sert de cle metier deterministe (RG-1.03 / RG-1.29). Lecture
|
||||||
|
// seule cote API (hors groupe category:write) : le front filtre dessus mais
|
||||||
|
// ne le saisit pas.
|
||||||
|
#[ORM\Column(length: 50)]
|
||||||
|
#[Groups(['category:read'])]
|
||||||
|
private ?string $code = null;
|
||||||
|
|
||||||
#[ORM\ManyToOne(targetEntity: CategoryType::class)]
|
#[ORM\ManyToOne(targetEntity: CategoryType::class)]
|
||||||
#[ORM\JoinColumn(name: 'category_type_id', referencedColumnName: 'id', nullable: false, onDelete: 'RESTRICT')]
|
#[ORM\JoinColumn(name: 'category_type_id', referencedColumnName: 'id', nullable: false, onDelete: 'RESTRICT')]
|
||||||
#[Assert\NotNull(message: 'Type de catégorie obligatoire.')]
|
#[Assert\NotNull(message: 'Type de catégorie obligatoire.')]
|
||||||
@@ -141,6 +152,21 @@ class Category implements TimestampableInterface, BlamableInterface, CategoryInt
|
|||||||
return $this;
|
return $this;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Implemente CategoryInterface : code technique stable de la categorie.
|
||||||
|
*/
|
||||||
|
public function getCode(): ?string
|
||||||
|
{
|
||||||
|
return $this->code;
|
||||||
|
}
|
||||||
|
|
||||||
|
public function setCode(string $code): static
|
||||||
|
{
|
||||||
|
$this->code = $code;
|
||||||
|
|
||||||
|
return $this;
|
||||||
|
}
|
||||||
|
|
||||||
public function getCategoryType(): ?CategoryType
|
public function getCategoryType(): ?CategoryType
|
||||||
{
|
{
|
||||||
return $this->categoryType;
|
return $this->categoryType;
|
||||||
|
|||||||
@@ -13,6 +13,13 @@ interface CategoryRepositoryInterface
|
|||||||
|
|
||||||
public function save(Category $category): void;
|
public function save(Category $category): void;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Vrai si une categorie active (deleted_at IS NULL) porte deja ce code.
|
||||||
|
* `$excludeId` exclut une categorie precise du test (cas PATCH). Sert a
|
||||||
|
* garantir l'unicite du code generee par le CategoryCodeGenerator (ERP-78).
|
||||||
|
*/
|
||||||
|
public function existsActiveByCode(string $code, ?int $excludeId = null): bool;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Construit un QueryBuilder de liste avec filtre soft-delete et tri par defaut.
|
* Construit un QueryBuilder de liste avec filtre soft-delete et tri par defaut.
|
||||||
* - $includeDeleted = false : exclut les categories soft-deleted (RG-1.08)
|
* - $includeDeleted = false : exclut les categories soft-deleted (RG-1.08)
|
||||||
|
|||||||
+17
-4
@@ -7,6 +7,7 @@ namespace App\Module\Catalog\Infrastructure\ApiPlatform\State\Processor;
|
|||||||
use ApiPlatform\Metadata\DeleteOperationInterface;
|
use ApiPlatform\Metadata\DeleteOperationInterface;
|
||||||
use ApiPlatform\Metadata\Operation;
|
use ApiPlatform\Metadata\Operation;
|
||||||
use ApiPlatform\State\ProcessorInterface;
|
use ApiPlatform\State\ProcessorInterface;
|
||||||
|
use App\Module\Catalog\Application\Service\CategoryCodeGenerator;
|
||||||
use App\Module\Catalog\Domain\Entity\Category;
|
use App\Module\Catalog\Domain\Entity\Category;
|
||||||
use DateTimeImmutable;
|
use DateTimeImmutable;
|
||||||
use Doctrine\DBAL\Exception\UniqueConstraintViolationException;
|
use Doctrine\DBAL\Exception\UniqueConstraintViolationException;
|
||||||
@@ -16,10 +17,13 @@ use Symfony\Component\HttpKernel\Exception\HttpException;
|
|||||||
/**
|
/**
|
||||||
* Processor Category : applique les regles de gestion en ecriture.
|
* Processor Category : applique les regles de gestion en ecriture.
|
||||||
*
|
*
|
||||||
* - POST / PATCH : trim du nom (RG-1.03) puis delegation au persist_processor
|
* - POST / PATCH : trim du nom (RG-1.03) ; a la CREATION, generation du `code`
|
||||||
* Doctrine ORM. Toute UniqueConstraintViolationException remontee par Postgres
|
* technique stable (slug MAJUSCULE du nom, unique parmi les actifs — ERP-78)
|
||||||
* (collision sur l'index partiel uq_category_name_type_active) est traduite
|
* via CategoryCodeGenerator ; puis delegation au persist_processor Doctrine
|
||||||
* en HTTP 409 avec le message attendu par la spec (RG-1.07).
|
* ORM. Le code est FIGE a la creation (jamais recalcule sur PATCH). Toute
|
||||||
|
* UniqueConstraintViolationException remontee par Postgres (collision sur
|
||||||
|
* l'index partiel uq_category_name_type_active) est traduite en HTTP 409 avec
|
||||||
|
* le message attendu par la spec (RG-1.07).
|
||||||
* - DELETE : soft delete (RG-1.12). On NE delegue PAS au remove_processor ;
|
* - DELETE : soft delete (RG-1.12). On NE delegue PAS au remove_processor ;
|
||||||
* on pose deletedAt = now() puis on delegue au persist_processor pour que
|
* on pose deletedAt = now() puis on delegue au persist_processor pour que
|
||||||
* le UPDATE Doctrine parte et que le TimestampableBlamableSubscriber mette
|
* le UPDATE Doctrine parte et que le TimestampableBlamableSubscriber mette
|
||||||
@@ -32,6 +36,7 @@ final class CategoryProcessor implements ProcessorInterface
|
|||||||
public function __construct(
|
public function __construct(
|
||||||
#[Autowire(service: 'api_platform.doctrine.orm.state.persist_processor')]
|
#[Autowire(service: 'api_platform.doctrine.orm.state.persist_processor')]
|
||||||
private readonly ProcessorInterface $persistProcessor,
|
private readonly ProcessorInterface $persistProcessor,
|
||||||
|
private readonly CategoryCodeGenerator $codeGenerator,
|
||||||
) {}
|
) {}
|
||||||
|
|
||||||
public function process(mixed $data, Operation $operation, array $uriVariables = [], array $context = []): mixed
|
public function process(mixed $data, Operation $operation, array $uriVariables = [], array $context = []): mixed
|
||||||
@@ -62,6 +67,14 @@ final class CategoryProcessor implements ProcessorInterface
|
|||||||
$data->setName(trim($data->getName()));
|
$data->setName(trim($data->getName()));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// ERP-78 : le code est genere a la CREATION puis fige. On le (re)genere
|
||||||
|
// uniquement s'il est absent (POST, ou entite seedee sans code) — un PATCH
|
||||||
|
// sur une categorie existante conserve son code. Genere depuis le nom
|
||||||
|
// (NotBlank, deja trimme), unique parmi les actifs.
|
||||||
|
if (null === $data->getCode() && null !== $data->getName()) {
|
||||||
|
$data->setCode($this->codeGenerator->generateUnique($data->getName(), $data->getId()));
|
||||||
|
}
|
||||||
|
|
||||||
try {
|
try {
|
||||||
return $this->persistProcessor->process($data, $operation, $uriVariables, $context);
|
return $this->persistProcessor->process($data, $operation, $uriVariables, $context);
|
||||||
} catch (UniqueConstraintViolationException $e) {
|
} catch (UniqueConstraintViolationException $e) {
|
||||||
|
|||||||
@@ -14,19 +14,19 @@ use RuntimeException;
|
|||||||
use Symfony\Component\DependencyInjection\Attribute\Autowire;
|
use Symfony\Component\DependencyInjection\Attribute\Autowire;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Fixtures dev/test du module Catalog : ~12 categories de demonstration reparties
|
* Fixtures dev/test du module Catalog : ~11 categories de demonstration, toutes
|
||||||
* sur les 4 types metier (DISTRIBUTEUR / COURTIER / SECTEUR / AUTRE). Alimente le
|
* rattachees au type unique CLIENT (refonte taxonomie ERP-78). Chaque categorie
|
||||||
* repertoire clients (ClientFixtures, module Commercial) avec des donnees
|
* porte un `code` stable. Alimente le repertoire clients (ClientFixtures, module
|
||||||
* realistes couvrant les categorisations RG-1.03 (DISTRIBUTEUR/COURTIER) et
|
* Commercial) avec des donnees realistes couvrant RG-1.03 (codes DISTRIBUTEUR /
|
||||||
* RG-1.29 (SECTEUR/AUTRE sur adresse).
|
* COURTIER) et RG-1.29 (codes interdits sur adresse).
|
||||||
*
|
*
|
||||||
* Depend de CategoryTypeFixtures : les 4 CategoryType doivent etre seedes avant
|
* Depend de CategoryTypeFixtures : le type CLIENT doit etre seede avant de
|
||||||
* de pouvoir y rattacher des Category.
|
* pouvoir y rattacher des Category.
|
||||||
*
|
*
|
||||||
* Idempotence : lookup par (name, categoryType) parmi les categories non
|
* Idempotence : lookup par `code` parmi les categories non supprimees (deletedAt
|
||||||
* supprimees (deletedAt null), coherent avec l'index unique partiel
|
* null), coherent avec l'index unique partiel uq_category_code (code WHERE
|
||||||
* uq_category_name_type_active (LOWER(name), category_type_id WHERE deleted_at
|
* deleted_at IS NULL). Rejouable sans doublon meme si le purger Doctrine est
|
||||||
* IS NULL). Rejouable sans doublon meme si le purger Doctrine est desactive.
|
* desactive.
|
||||||
*
|
*
|
||||||
* Audit / Blamable : persist hors contexte HTTP -> created_by / updated_by
|
* Audit / Blamable : persist hors contexte HTTP -> created_by / updated_by
|
||||||
* restent null (« Systeme » cote front), c'est attendu.
|
* restent null (« Systeme » cote front), c'est attendu.
|
||||||
@@ -34,39 +34,33 @@ use Symfony\Component\DependencyInjection\Attribute\Autowire;
|
|||||||
* Portee : DONNEES DE DEMONSTRATION (dev uniquement). En environnement `test`,
|
* Portee : DONNEES DE DEMONSTRATION (dev uniquement). En environnement `test`,
|
||||||
* la fixture ne charge rien : les tests seedent et nettoient leurs propres
|
* la fixture ne charge rien : les tests seedent et nettoient leurs propres
|
||||||
* categories (prefixe dedie) et comptent sur une table `category` vierge — y
|
* categories (prefixe dedie) et comptent sur une table `category` vierge — y
|
||||||
* injecter 12 categories de demo casserait comptages et cleanups FK
|
* injecter des categories de demo casserait comptages et cleanups FK
|
||||||
* (client_category). Cf. ClientFixtures (meme garde-fou).
|
* (client_category). Cf. ClientFixtures (meme garde-fou).
|
||||||
*/
|
*/
|
||||||
class CategoryFixtures extends Fixture implements DependentFixtureInterface
|
class CategoryFixtures extends Fixture implements DependentFixtureInterface
|
||||||
{
|
{
|
||||||
|
/** Code du type unique (cf. CategoryTypeFixtures, migration ERP-78). */
|
||||||
|
private const string CLIENT_TYPE_CODE = 'CLIENT';
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Source unique des categories de demonstration : code de type metier =>
|
* Source unique des categories de demonstration : nom => code stable. Les 4
|
||||||
* liste de noms. Les noms sont stockes tels quels (l'unicite est
|
* premieres (Distributeur / Courtier / Secteur / Autre) sont les categories
|
||||||
* case-insensitive cote index).
|
* « systeme » reportees des anciens types ; leurs codes pilotent les RG.
|
||||||
*
|
*
|
||||||
* @var array<string, list<string>>
|
* @var array<string, string>
|
||||||
*/
|
*/
|
||||||
private const CATEGORIES = [
|
private const CATEGORIES = [
|
||||||
'SECTEUR' => [
|
'Distributeur' => 'DISTRIBUTEUR',
|
||||||
'BTP',
|
'Courtier' => 'COURTIER',
|
||||||
'Industrie',
|
'Secteur' => 'SECTEUR',
|
||||||
'Agro-alimentaire',
|
'Autre' => 'AUTRE',
|
||||||
'Transport/Logistique',
|
'BTP' => 'BTP',
|
||||||
'Services',
|
'Industrie' => 'INDUSTRIE',
|
||||||
],
|
'Agro-alimentaire' => 'AGRO_ALIMENTAIRE',
|
||||||
'DISTRIBUTEUR' => [
|
'Transport/Logistique' => 'TRANSPORT_LOGISTIQUE',
|
||||||
'Distributeur Grand Sud-Ouest',
|
'Services' => 'SERVICES',
|
||||||
'Distributeur National Premium',
|
'Association' => 'ASSOCIATION',
|
||||||
'Grossiste régional',
|
'Indépendant' => 'INDEPENDANT',
|
||||||
],
|
|
||||||
'COURTIER' => [
|
|
||||||
'Cabinet de courtage Léonard',
|
|
||||||
'Cabinet de courtage Bernard',
|
|
||||||
],
|
|
||||||
'AUTRE' => [
|
|
||||||
'Indépendant',
|
|
||||||
'Association',
|
|
||||||
],
|
|
||||||
];
|
];
|
||||||
|
|
||||||
public function __construct(
|
public function __construct(
|
||||||
@@ -90,41 +84,39 @@ class CategoryFixtures extends Fixture implements DependentFixtureInterface
|
|||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
// Index des types metier par code (CategoryTypeFixtures les a seedes).
|
$clientType = null;
|
||||||
$typesByCode = [];
|
|
||||||
foreach ($this->categoryTypeRepository->findAllOrderedByLabel() as $type) {
|
foreach ($this->categoryTypeRepository->findAllOrderedByLabel() as $type) {
|
||||||
$typesByCode[$type->getCode()] = $type;
|
if (self::CLIENT_TYPE_CODE === $type->getCode()) {
|
||||||
|
$clientType = $type;
|
||||||
|
|
||||||
|
break;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
foreach (self::CATEGORIES as $typeCode => $names) {
|
if (!$clientType instanceof CategoryType) {
|
||||||
$type = $typesByCode[$typeCode] ?? null;
|
// Misconfiguration : CategoryTypeFixtures n'a pas tourne avant.
|
||||||
if (!$type instanceof CategoryType) {
|
throw new RuntimeException(
|
||||||
// Misconfiguration : CategoryTypeFixtures n'a pas tourne avant.
|
'CategoryTypeFixtures doit avoir seede le type "CLIENT" avant CategoryFixtures.',
|
||||||
throw new RuntimeException(sprintf(
|
);
|
||||||
'CategoryTypeFixtures doit avoir seede le type "%s" avant CategoryFixtures.',
|
}
|
||||||
$typeCode,
|
|
||||||
));
|
|
||||||
}
|
|
||||||
|
|
||||||
foreach ($names as $name) {
|
foreach (self::CATEGORIES as $name => $code) {
|
||||||
$this->ensureCategory($manager, $name, $type);
|
$this->ensureCategory($manager, $name, $code, $clientType);
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
$manager->flush();
|
$manager->flush();
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Cree la categorie (name, type) si elle n'existe pas encore parmi les
|
* Cree la categorie (name, code) sous le type CLIENT si son code n'existe pas
|
||||||
* categories actives, sinon la laisse en place. Lookup aligne sur l'index
|
* encore parmi les categories actives, sinon la laisse en place. Lookup
|
||||||
* unique partiel (nom + type, hors soft-deleted).
|
* aligne sur l'index unique partiel uq_category_code.
|
||||||
*/
|
*/
|
||||||
private function ensureCategory(ObjectManager $manager, string $name, CategoryType $type): void
|
private function ensureCategory(ObjectManager $manager, string $name, string $code, CategoryType $type): void
|
||||||
{
|
{
|
||||||
$existing = $manager->getRepository(Category::class)->findOneBy([
|
$existing = $manager->getRepository(Category::class)->findOneBy([
|
||||||
'name' => $name,
|
'code' => $code,
|
||||||
'categoryType' => $type,
|
'deletedAt' => null,
|
||||||
'deletedAt' => null,
|
|
||||||
]);
|
]);
|
||||||
|
|
||||||
if (null !== $existing) {
|
if (null !== $existing) {
|
||||||
@@ -133,6 +125,7 @@ class CategoryFixtures extends Fixture implements DependentFixtureInterface
|
|||||||
|
|
||||||
$category = new Category();
|
$category = new Category();
|
||||||
$category->setName($name);
|
$category->setName($name);
|
||||||
|
$category->setCode($code);
|
||||||
$category->setCategoryType($type);
|
$category->setCategoryType($type);
|
||||||
$manager->persist($category);
|
$manager->persist($category);
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -10,17 +10,19 @@ use Doctrine\Bundle\FixturesBundle\Fixture;
|
|||||||
use Doctrine\Persistence\ObjectManager;
|
use Doctrine\Persistence\ObjectManager;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Fixtures du module Catalog : seed des types de categorie metier (M1).
|
* Fixtures du module Catalog : seed du type de categorie (M1).
|
||||||
*
|
*
|
||||||
* La table `category_type` est creee vide au M0 ; le M1 la peuple avec les 4
|
* Refonte taxonomie ERP-78 : le modele n'a plus qu'UN SEUL `category_type`,
|
||||||
* types DISTRIBUTEUR / COURTIER / SECTEUR / AUTRE (cf. spec M1 § 3.3).
|
* CLIENT (code CLIENT, label « Client »). Distributeur / Courtier / Secteur /
|
||||||
|
* Autre (et les categories metier fines) sont desormais des `Category` codees
|
||||||
|
* rattachees a ce type (cf. CategoryFixtures + migration Version20260602100000).
|
||||||
*
|
*
|
||||||
* Pourquoi une fixture EN PLUS du seed de la migration (Version20260601000000) :
|
* Pourquoi une fixture EN PLUS du seed de la migration : `category_type` est une
|
||||||
* `category_type` est une entite managee par l ORM, donc le purger Doctrine la
|
* entite managee par l ORM, donc le purger Doctrine la vide avant chaque
|
||||||
* vide avant chaque `doctrine:fixtures:load`. Sans cette fixture, les 4 types
|
* `doctrine:fixtures:load`. Sans cette fixture, le type CLIENT seede par la
|
||||||
* seedes par la migration disparaitraient apres `make db-reset` / setup de test.
|
* migration disparaitrait apres `make db-reset` / setup de test. Le seed
|
||||||
* Le seed migration couvre la prod (ou les fixtures ne tournent pas) ; cette
|
* migration couvre la prod (ou les fixtures ne tournent pas) ; cette fixture
|
||||||
* fixture re-aligne dev et test. Les deux chemins produisent un etat identique.
|
* re-aligne dev et test. Les deux chemins produisent un etat identique.
|
||||||
*
|
*
|
||||||
* Idempotence : lookup par `code` parmi les types existants avant insertion,
|
* Idempotence : lookup par `code` parmi les types existants avant insertion,
|
||||||
* sur le modele d AppFixtures::ensureSystemRole. Rejouable sans doublon meme
|
* sur le modele d AppFixtures::ensureSystemRole. Rejouable sans doublon meme
|
||||||
@@ -29,14 +31,11 @@ use Doctrine\Persistence\ObjectManager;
|
|||||||
class CategoryTypeFixtures extends Fixture
|
class CategoryTypeFixtures extends Fixture
|
||||||
{
|
{
|
||||||
/**
|
/**
|
||||||
* Source unique des 4 types metier : code technique => libelle FR.
|
* Source unique du type : code technique => libelle FR. Doit rester aligne
|
||||||
* Doit rester aligne sur le seed de la migration Version20260601000000.
|
* sur le seed de la migration Version20260602100000 (type unique CLIENT).
|
||||||
*/
|
*/
|
||||||
private const TYPES = [
|
private const TYPES = [
|
||||||
'DISTRIBUTEUR' => 'Distributeur',
|
'CLIENT' => 'Client',
|
||||||
'COURTIER' => 'Courtier',
|
|
||||||
'SECTEUR' => 'Secteur',
|
|
||||||
'AUTRE' => 'Autre',
|
|
||||||
];
|
];
|
||||||
|
|
||||||
public function __construct(
|
public function __construct(
|
||||||
|
|||||||
@@ -31,6 +31,23 @@ class DoctrineCategoryRepository extends ServiceEntityRepository implements Cate
|
|||||||
$this->getEntityManager()->flush();
|
$this->getEntityManager()->flush();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public function existsActiveByCode(string $code, ?int $excludeId = null): bool
|
||||||
|
{
|
||||||
|
$qb = $this->createQueryBuilder('c')
|
||||||
|
->select('1')
|
||||||
|
->andWhere('c.code = :code')
|
||||||
|
->andWhere('c.deletedAt IS NULL')
|
||||||
|
->setParameter('code', $code)
|
||||||
|
->setMaxResults(1)
|
||||||
|
;
|
||||||
|
|
||||||
|
if (null !== $excludeId) {
|
||||||
|
$qb->andWhere('c.id != :excludeId')->setParameter('excludeId', $excludeId);
|
||||||
|
}
|
||||||
|
|
||||||
|
return [] !== $qb->getQuery()->getResult();
|
||||||
|
}
|
||||||
|
|
||||||
public function createListQueryBuilder(bool $includeDeleted = false): QueryBuilder
|
public function createListQueryBuilder(bool $includeDeleted = false): QueryBuilder
|
||||||
{
|
{
|
||||||
$qb = $this->createQueryBuilder('c')
|
$qb = $this->createQueryBuilder('c')
|
||||||
|
|||||||
@@ -15,6 +15,7 @@ use App\Module\Commercial\Infrastructure\Doctrine\DoctrineClientRepository;
|
|||||||
use App\Shared\Domain\Attribute\Auditable;
|
use App\Shared\Domain\Attribute\Auditable;
|
||||||
use App\Shared\Domain\Contract\BlamableInterface;
|
use App\Shared\Domain\Contract\BlamableInterface;
|
||||||
use App\Shared\Domain\Contract\CategoryInterface;
|
use App\Shared\Domain\Contract\CategoryInterface;
|
||||||
|
use App\Shared\Domain\Contract\SiteInterface;
|
||||||
use App\Shared\Domain\Contract\TimestampableInterface;
|
use App\Shared\Domain\Contract\TimestampableInterface;
|
||||||
use App\Shared\Domain\Trait\TimestampableBlamableTrait;
|
use App\Shared\Domain\Trait\TimestampableBlamableTrait;
|
||||||
use DateTimeImmutable;
|
use DateTimeImmutable;
|
||||||
@@ -58,27 +59,39 @@ use Symfony\Component\Validator\Constraints as Assert;
|
|||||||
operations: [
|
operations: [
|
||||||
new GetCollection(
|
new GetCollection(
|
||||||
security: "is_granted('commercial.clients.view')",
|
security: "is_granted('commercial.clients.view')",
|
||||||
normalizationContext: ['groups' => ['client:read', 'default:read']],
|
// La liste embarque les categories (avec leur code, groupe
|
||||||
|
// category:read) et les sites agreges des adresses (groupe
|
||||||
|
// site:read) pour alimenter les colonnes « Catégories » et
|
||||||
|
// « Site(s) » du Repertoire (ERP-62). Cf. getSites() plus bas.
|
||||||
|
normalizationContext: ['groups' => ['client:read', 'default:read', 'category:read', 'site:read']],
|
||||||
provider: ClientProvider::class,
|
provider: ClientProvider::class,
|
||||||
),
|
),
|
||||||
new Get(
|
new Get(
|
||||||
security: "is_granted('commercial.clients.view')",
|
security: "is_granted('commercial.clients.view')",
|
||||||
// Detail : client + sous-collections embarquees. Le groupe
|
// Detail : client + sous-collections embarquees.
|
||||||
// client:read:accounting est ajoute par le context builder selon la
|
// - client:read:accounting est ajoute par le context builder selon la
|
||||||
// permission, donc absent ici volontairement.
|
// permission (gate les scalaires comptables ET les RIB embarques),
|
||||||
|
// donc absent ici volontairement.
|
||||||
|
// - client_rib:read N'EST PLUS dans le contexte : le contenu des RIB
|
||||||
|
// embarques est desormais porte par client:read:accounting (gate),
|
||||||
|
// ce qui retire la fuite IBAN/BIC vers les users sans accounting.view.
|
||||||
|
// - category:read et site:read sont indispensables pour embarquer le
|
||||||
|
// code/libelle des categories et des sites (sinon stub IRI nu) :
|
||||||
|
// Category.code/name vivent sous category:read, Site.name sous site:read.
|
||||||
normalizationContext: ['groups' => [
|
normalizationContext: ['groups' => [
|
||||||
'client:read',
|
'client:read',
|
||||||
'client:item:read',
|
'client:item:read',
|
||||||
'client_contact:read',
|
'client_contact:read',
|
||||||
'client_address:read',
|
'client_address:read',
|
||||||
'client_rib:read',
|
'category:read',
|
||||||
|
'site:read',
|
||||||
'default:read',
|
'default:read',
|
||||||
]],
|
]],
|
||||||
provider: ClientProvider::class,
|
provider: ClientProvider::class,
|
||||||
),
|
),
|
||||||
new Post(
|
new Post(
|
||||||
security: "is_granted('commercial.clients.manage')",
|
security: "is_granted('commercial.clients.manage')",
|
||||||
normalizationContext: ['groups' => ['client:read', 'default:read']],
|
normalizationContext: ['groups' => ['client:read', 'default:read', 'category:read', 'site:read']],
|
||||||
denormalizationContext: ['groups' => ['client:write:main']],
|
denormalizationContext: ['groups' => ['client:write:main']],
|
||||||
processor: ClientProcessor::class,
|
processor: ClientProcessor::class,
|
||||||
),
|
),
|
||||||
@@ -96,7 +109,7 @@ use Symfony\Component\Validator\Constraints as Assert;
|
|||||||
// autoriser/refuser onglet par onglet (RG-1.22 / RG-1.28) : les
|
// autoriser/refuser onglet par onglet (RG-1.22 / RG-1.28) : les
|
||||||
// champs accounting exigent accounting.manage, isArchived exige
|
// champs accounting exigent accounting.manage, isArchived exige
|
||||||
// archive, le reste (main/information) exige manage.
|
// archive, le reste (main/information) exige manage.
|
||||||
normalizationContext: ['groups' => ['client:read', 'default:read']],
|
normalizationContext: ['groups' => ['client:read', 'default:read', 'category:read', 'site:read']],
|
||||||
denormalizationContext: ['groups' => [
|
denormalizationContext: ['groups' => [
|
||||||
'client:write:main',
|
'client:write:main',
|
||||||
'client:write:information',
|
'client:write:information',
|
||||||
@@ -651,8 +664,38 @@ class Client implements TimestampableInterface, BlamableInterface
|
|||||||
return $this;
|
return $this;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Sites distincts rattaches a au moins une adresse du client (RG-1.10).
|
||||||
|
* Le Client ne porte pas de sites en propre : ils vivent sur les adresses.
|
||||||
|
* Agrege en lecture seule pour la colonne « Site(s) » du Repertoire (badges
|
||||||
|
* colores) — expose en LISTE via le groupe client:read (les adresses
|
||||||
|
* completes restent reservees au detail, client:item:read).
|
||||||
|
*
|
||||||
|
* @return list<SiteInterface>
|
||||||
|
*/
|
||||||
|
#[Groups(['client:read'])]
|
||||||
|
public function getSites(): array
|
||||||
|
{
|
||||||
|
$sites = [];
|
||||||
|
foreach ($this->addresses as $address) {
|
||||||
|
foreach ($address->getSites() as $site) {
|
||||||
|
// Deduplication par identite d'objet : un meme site peut etre
|
||||||
|
// rattache a plusieurs adresses du client.
|
||||||
|
$sites[spl_object_id($site)] = $site;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return array_values($sites);
|
||||||
|
}
|
||||||
|
|
||||||
|
// Embed gate sur le groupe COMPTABLE (et non client:item:read comme contacts/
|
||||||
|
// adresses) : client:read:accounting n'est ajoute au contexte que si l'user a
|
||||||
|
// accounting.view (ClientReadGroupContextBuilder). Resultat : la cle `ribs` est
|
||||||
|
// TOTALEMENT ABSENTE du detail pour un user sans accounting.view (ex. Commerciale),
|
||||||
|
// au meme titre que les scalaires comptables — corrige la fuite de RIB ou la
|
||||||
|
// Commerciale recevait IBAN/BIC en clair.
|
||||||
/** @return Collection<int, ClientRib> */
|
/** @return Collection<int, ClientRib> */
|
||||||
#[Groups(['client:item:read'])]
|
#[Groups(['client:read:accounting'])]
|
||||||
public function getRibs(): Collection
|
public function getRibs(): Collection
|
||||||
{
|
{
|
||||||
return $this->ribs;
|
return $this->ribs;
|
||||||
|
|||||||
@@ -22,6 +22,7 @@ use Doctrine\Common\Collections\ArrayCollection;
|
|||||||
use Doctrine\Common\Collections\Collection;
|
use Doctrine\Common\Collections\Collection;
|
||||||
use Doctrine\ORM\Mapping as ORM;
|
use Doctrine\ORM\Mapping as ORM;
|
||||||
use Symfony\Component\Serializer\Attribute\Groups;
|
use Symfony\Component\Serializer\Attribute\Groups;
|
||||||
|
use Symfony\Component\Serializer\Attribute\SerializedName;
|
||||||
use Symfony\Component\Validator\Constraints as Assert;
|
use Symfony\Component\Validator\Constraints as Assert;
|
||||||
use Symfony\Component\Validator\Context\ExecutionContextInterface;
|
use Symfony\Component\Validator\Context\ExecutionContextInterface;
|
||||||
|
|
||||||
@@ -39,7 +40,7 @@ use Symfony\Component\Validator\Context\ExecutionContextInterface;
|
|||||||
* - sites : SiteInterface (module Sites) via resolve_target_entities
|
* - sites : SiteInterface (module Sites) via resolve_target_entities
|
||||||
* - contacts : ClientContact (meme module)
|
* - contacts : ClientContact (meme module)
|
||||||
* - categories : CategoryInterface (module Catalog) via resolve_target_entities
|
* - categories : CategoryInterface (module Catalog) via resolve_target_entities
|
||||||
* — limitees aux types SECTEUR/AUTRE (RG-1.29, validateCategoryTypes, ERP-76)
|
* — codes DISTRIBUTEUR/COURTIER interdits (RG-1.29, validateCategoryCodes, ERP-78)
|
||||||
*
|
*
|
||||||
* Audite (#[Auditable]) + Timestampable/Blamable.
|
* Audite (#[Auditable]) + Timestampable/Blamable.
|
||||||
*
|
*
|
||||||
@@ -87,8 +88,12 @@ class ClientAddress implements TimestampableInterface, BlamableInterface
|
|||||||
{
|
{
|
||||||
use TimestampableBlamableTrait;
|
use TimestampableBlamableTrait;
|
||||||
|
|
||||||
/** RG-1.29 : seuls ces types de categorie qualifient une adresse physique. */
|
/**
|
||||||
private const array ALLOWED_CATEGORY_TYPES = ['SECTEUR', 'AUTRE'];
|
* RG-1.29 (ERP-78) : ces codes de categorie decrivent une relation entre
|
||||||
|
* clients (distributeur / courtier) et n'ont pas de sens sur une adresse.
|
||||||
|
* Toute autre categorie du type CLIENT est autorisee.
|
||||||
|
*/
|
||||||
|
private const array FORBIDDEN_CATEGORY_CODES = ['DISTRIBUTEUR', 'COURTIER'];
|
||||||
|
|
||||||
#[ORM\Id]
|
#[ORM\Id]
|
||||||
#[ORM\GeneratedValue]
|
#[ORM\GeneratedValue]
|
||||||
@@ -100,16 +105,23 @@ class ClientAddress implements TimestampableInterface, BlamableInterface
|
|||||||
#[ORM\JoinColumn(name: 'client_id', referencedColumnName: 'id', nullable: false, onDelete: 'CASCADE')]
|
#[ORM\JoinColumn(name: 'client_id', referencedColumnName: 'id', nullable: false, onDelete: 'CASCADE')]
|
||||||
private ?Client $client = null;
|
private ?Client $client = null;
|
||||||
|
|
||||||
|
// Groupe d'ECRITURE uniquement sur la propriete (denormalisation PATCH/POST).
|
||||||
|
// Le groupe de LECTURE est porte par le getter isProspect()/isDelivery()/
|
||||||
|
// isBilling() avec SerializedName : sans cela, Symfony strip le prefixe "is"
|
||||||
|
// des getters booleens et exposerait les cles "prospect"/"delivery"/"billing"
|
||||||
|
// — en pratique le #[Groups] etant sur la propriete `isX` et le getter
|
||||||
|
// derivant l'attribut `x`, la cle etait totalement DROPPEE du JSON (meme bug
|
||||||
|
// que Client::isArchived). Pattern corrige : Groups + SerializedName sur le getter.
|
||||||
#[ORM\Column(name: 'is_prospect', options: ['default' => false])]
|
#[ORM\Column(name: 'is_prospect', options: ['default' => false])]
|
||||||
#[Groups(['client_address:read', 'client_address:write'])]
|
#[Groups(['client_address:write'])]
|
||||||
private bool $isProspect = false;
|
private bool $isProspect = false;
|
||||||
|
|
||||||
#[ORM\Column(name: 'is_delivery', options: ['default' => false])]
|
#[ORM\Column(name: 'is_delivery', options: ['default' => false])]
|
||||||
#[Groups(['client_address:read', 'client_address:write'])]
|
#[Groups(['client_address:write'])]
|
||||||
private bool $isDelivery = false;
|
private bool $isDelivery = false;
|
||||||
|
|
||||||
#[ORM\Column(name: 'is_billing', options: ['default' => false])]
|
#[ORM\Column(name: 'is_billing', options: ['default' => false])]
|
||||||
#[Groups(['client_address:read', 'client_address:write'])]
|
#[Groups(['client_address:write'])]
|
||||||
private bool $isBilling = false;
|
private bool $isBilling = false;
|
||||||
|
|
||||||
#[ORM\Column(length: 80, options: ['default' => 'France'])]
|
#[ORM\Column(length: 80, options: ['default' => 'France'])]
|
||||||
@@ -165,7 +177,7 @@ class ClientAddress implements TimestampableInterface, BlamableInterface
|
|||||||
#[Groups(['client_address:read', 'client_address:write'])]
|
#[Groups(['client_address:read', 'client_address:write'])]
|
||||||
private Collection $contacts;
|
private Collection $contacts;
|
||||||
|
|
||||||
// RG-1.29 : categories de type SECTEUR/AUTRE uniquement (validateCategoryTypes).
|
// RG-1.29 : categories de code DISTRIBUTEUR/COURTIER interdites (validateCategoryCodes).
|
||||||
/** @var Collection<int, CategoryInterface> */
|
/** @var Collection<int, CategoryInterface> */
|
||||||
#[ORM\ManyToMany(targetEntity: CategoryInterface::class)]
|
#[ORM\ManyToMany(targetEntity: CategoryInterface::class)]
|
||||||
#[ORM\JoinTable(name: 'client_address_category')]
|
#[ORM\JoinTable(name: 'client_address_category')]
|
||||||
@@ -232,18 +244,19 @@ class ClientAddress implements TimestampableInterface, BlamableInterface
|
|||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* RG-1.29 : seules les categories de type SECTEUR / AUTRE qualifient une
|
* RG-1.29 (ERP-78) : une adresse interdit les categories de code
|
||||||
* adresse physique. Les types DISTRIBUTEUR / COURTIER decrivent une relation
|
* DISTRIBUTEUR / COURTIER — elles decrivent une relation entre clients
|
||||||
* entre clients (RG-1.03) et n'ont pas de sens sur une adresse -> 422 avec
|
* (RG-1.03) et n'ont pas de sens sur une adresse physique -> 422 avec
|
||||||
* violation sur le champ `categories`. S'appuie sur
|
* violation sur le champ `categories`. Toute autre categorie (type unique
|
||||||
* CategoryInterface::getCategoryTypeCode() (pas d'import du module Catalog).
|
* CLIENT) est acceptee. S'appuie sur CategoryInterface::getCode() (pas
|
||||||
|
* d'import du module Catalog — regle ABSOLUE n°1).
|
||||||
*/
|
*/
|
||||||
#[Assert\Callback]
|
#[Assert\Callback]
|
||||||
public function validateCategoryTypes(ExecutionContextInterface $context): void
|
public function validateCategoryCodes(ExecutionContextInterface $context): void
|
||||||
{
|
{
|
||||||
foreach ($this->categories as $category) {
|
foreach ($this->categories as $category) {
|
||||||
if ($category instanceof CategoryInterface
|
if ($category instanceof CategoryInterface
|
||||||
&& !in_array($category->getCategoryTypeCode(), self::ALLOWED_CATEGORY_TYPES, true)) {
|
&& in_array($category->getCode(), self::FORBIDDEN_CATEGORY_CODES, true)) {
|
||||||
$context->buildViolation('Type de catégorie non autorisé sur une adresse.')
|
$context->buildViolation('Type de catégorie non autorisé sur une adresse.')
|
||||||
->atPath('categories')
|
->atPath('categories')
|
||||||
->addViolation()
|
->addViolation()
|
||||||
@@ -271,6 +284,12 @@ class ClientAddress implements TimestampableInterface, BlamableInterface
|
|||||||
return $this;
|
return $this;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Groupe de lecture + nom serialise explicite (cf. note sur la propriete) :
|
||||||
|
// sans SerializedName, Symfony exposerait la cle "prospect" (strip du prefixe
|
||||||
|
// "is" sur les getters) et, le groupe etant declare sur la propriete `isProspect`,
|
||||||
|
// droppait silencieusement la cle du JSON.
|
||||||
|
#[Groups(['client_address:read'])]
|
||||||
|
#[SerializedName('isProspect')]
|
||||||
public function isProspect(): bool
|
public function isProspect(): bool
|
||||||
{
|
{
|
||||||
return $this->isProspect;
|
return $this->isProspect;
|
||||||
@@ -283,6 +302,8 @@ class ClientAddress implements TimestampableInterface, BlamableInterface
|
|||||||
return $this;
|
return $this;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
#[Groups(['client_address:read'])]
|
||||||
|
#[SerializedName('isDelivery')]
|
||||||
public function isDelivery(): bool
|
public function isDelivery(): bool
|
||||||
{
|
{
|
||||||
return $this->isDelivery;
|
return $this->isDelivery;
|
||||||
@@ -295,6 +316,8 @@ class ClientAddress implements TimestampableInterface, BlamableInterface
|
|||||||
return $this;
|
return $this;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
#[Groups(['client_address:read'])]
|
||||||
|
#[SerializedName('isBilling')]
|
||||||
public function isBilling(): bool
|
public function isBilling(): bool
|
||||||
{
|
{
|
||||||
return $this->isBilling;
|
return $this->isBilling;
|
||||||
|
|||||||
@@ -79,10 +79,17 @@ class ClientRib implements TimestampableInterface, BlamableInterface
|
|||||||
{
|
{
|
||||||
use TimestampableBlamableTrait;
|
use TimestampableBlamableTrait;
|
||||||
|
|
||||||
|
// Double groupe de lecture :
|
||||||
|
// - `client_rib:read` : sous-ressource autonome GET /api/client_ribs/{id}
|
||||||
|
// (deja securisee par commercial.clients.accounting.view).
|
||||||
|
// - `client:read:accounting` : embed des RIB sous le detail Client, ajoute
|
||||||
|
// DYNAMIQUEMENT par ClientReadGroupContextBuilder uniquement si l'user a
|
||||||
|
// accounting.view. Ce double marquage gate les RIB embarques au meme titre
|
||||||
|
// que les scalaires comptables (RG : la Commerciale ne voit aucun RIB).
|
||||||
#[ORM\Id]
|
#[ORM\Id]
|
||||||
#[ORM\GeneratedValue]
|
#[ORM\GeneratedValue]
|
||||||
#[ORM\Column]
|
#[ORM\Column]
|
||||||
#[Groups(['client_rib:read'])]
|
#[Groups(['client_rib:read', 'client:read:accounting'])]
|
||||||
private ?int $id = null;
|
private ?int $id = null;
|
||||||
|
|
||||||
#[ORM\ManyToOne(targetEntity: Client::class, inversedBy: 'ribs')]
|
#[ORM\ManyToOne(targetEntity: Client::class, inversedBy: 'ribs')]
|
||||||
@@ -92,23 +99,23 @@ class ClientRib implements TimestampableInterface, BlamableInterface
|
|||||||
#[ORM\Column(length: 120)]
|
#[ORM\Column(length: 120)]
|
||||||
#[Assert\NotBlank]
|
#[Assert\NotBlank]
|
||||||
#[Assert\Length(max: 120, normalizer: 'trim')]
|
#[Assert\Length(max: 120, normalizer: 'trim')]
|
||||||
#[Groups(['client_rib:read', 'client_rib:write'])]
|
#[Groups(['client_rib:read', 'client:read:accounting', 'client_rib:write'])]
|
||||||
private ?string $label = null;
|
private ?string $label = null;
|
||||||
|
|
||||||
#[ORM\Column(length: 20)]
|
#[ORM\Column(length: 20)]
|
||||||
#[Assert\NotBlank]
|
#[Assert\NotBlank]
|
||||||
#[Assert\Bic]
|
#[Assert\Bic]
|
||||||
#[Groups(['client_rib:read', 'client_rib:write'])]
|
#[Groups(['client_rib:read', 'client:read:accounting', 'client_rib:write'])]
|
||||||
private ?string $bic = null;
|
private ?string $bic = null;
|
||||||
|
|
||||||
#[ORM\Column(length: 34)]
|
#[ORM\Column(length: 34)]
|
||||||
#[Assert\NotBlank]
|
#[Assert\NotBlank]
|
||||||
#[Assert\Iban]
|
#[Assert\Iban]
|
||||||
#[Groups(['client_rib:read', 'client_rib:write'])]
|
#[Groups(['client_rib:read', 'client:read:accounting', 'client_rib:write'])]
|
||||||
private ?string $iban = null;
|
private ?string $iban = null;
|
||||||
|
|
||||||
#[ORM\Column(options: ['default' => 0])]
|
#[ORM\Column(options: ['default' => 0])]
|
||||||
#[Groups(['client_rib:read', 'client_rib:write'])]
|
#[Groups(['client_rib:read', 'client:read:accounting', 'client_rib:write'])]
|
||||||
private int $position = 0;
|
private int $position = 0;
|
||||||
|
|
||||||
public function getId(): ?int
|
public function getId(): ?int
|
||||||
|
|||||||
@@ -16,20 +16,31 @@ interface ClientRepositoryInterface
|
|||||||
/**
|
/**
|
||||||
* Construit un QueryBuilder de liste pour le repertoire clients.
|
* Construit un QueryBuilder de liste pour le repertoire clients.
|
||||||
* - Exclut toujours les clients soft-deletes (deleted_at IS NOT NULL, RG-1.24).
|
* - Exclut toujours les clients soft-deletes (deleted_at IS NOT NULL, RG-1.24).
|
||||||
* - Exclut les archives sauf si $includeArchived = true (RG-1.25).
|
* - Archivage (RG-1.25) :
|
||||||
|
* - $archivedOnly = true -> uniquement les archives (is_archived = true) ;
|
||||||
|
* - sinon $includeArchived = true -> actifs + archives (echappatoire) ;
|
||||||
|
* - sinon (defaut) -> uniquement les actifs (is_archived = false).
|
||||||
|
* $archivedOnly a la priorite sur $includeArchived.
|
||||||
* - Tri par defaut : companyName ASC (RG-1.26).
|
* - Tri par defaut : companyName ASC (RG-1.26).
|
||||||
* - $search : recherche fuzzy insensible a la casse sur companyName +
|
* - $search : recherche fuzzy insensible a la casse sur companyName +
|
||||||
* lastName + email (metacaracteres LIKE echappes). Ignore si null/vide.
|
* lastName + email (metacaracteres LIKE echappes). Ignore si null/vide.
|
||||||
* - $categoryType : restreint aux clients possedant au moins une categorie
|
* - $categoryCodes : restreint aux clients possedant au moins une categorie
|
||||||
* du type donne (code). Ignore si null/vide.
|
* dont le code est dans la liste (OR — ERP-78). Liste vide = pas de filtre.
|
||||||
|
* - $siteIds : restreint aux clients ayant au moins une adresse rattachee a
|
||||||
|
* l'un des sites donnes (OR — RG-1.10). Liste vide = pas de filtre.
|
||||||
*
|
*
|
||||||
* Filtrage centralise ICI (et non dans les providers/controllers) pour que
|
* Filtrage centralise ICI (et non dans les providers/controllers) pour que
|
||||||
* la liste paginee (ClientProvider) et l'export (ClientExportController)
|
* la liste paginee (ClientProvider) et l'export (ClientExportController)
|
||||||
* partagent strictement la meme logique de selection.
|
* partagent strictement la meme logique de selection.
|
||||||
|
*
|
||||||
|
* @param list<string> $categoryCodes
|
||||||
|
* @param list<int> $siteIds
|
||||||
*/
|
*/
|
||||||
public function createListQueryBuilder(
|
public function createListQueryBuilder(
|
||||||
bool $includeArchived = false,
|
bool $includeArchived = false,
|
||||||
?string $search = null,
|
?string $search = null,
|
||||||
?string $categoryType = null,
|
array $categoryCodes = [],
|
||||||
|
array $siteIds = [],
|
||||||
|
bool $archivedOnly = false,
|
||||||
): QueryBuilder;
|
): QueryBuilder;
|
||||||
}
|
}
|
||||||
|
|||||||
+9
-8
@@ -457,8 +457,9 @@ final class ClientProcessor implements ProcessorInterface
|
|||||||
|
|
||||||
/**
|
/**
|
||||||
* RG-1.03 : distributor et broker mutuellement exclusifs ; un distributor
|
* RG-1.03 : distributor et broker mutuellement exclusifs ; un distributor
|
||||||
* doit referencer un client de categorie DISTRIBUTEUR (idem broker ->
|
* doit referencer un client portant la categorie de code DISTRIBUTEUR (idem
|
||||||
* COURTIER).
|
* broker -> COURTIER). Depuis ERP-78, le filtrage se fait sur le code de la
|
||||||
|
* Category (et non plus sur le type, devenu unique CLIENT).
|
||||||
*/
|
*/
|
||||||
private function validateDistributorBroker(Client $data): void
|
private function validateDistributorBroker(Client $data): void
|
||||||
{
|
{
|
||||||
@@ -473,7 +474,7 @@ final class ClientProcessor implements ProcessorInterface
|
|||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
if (null !== $distributor && !$this->hasCategoryType($distributor, 'DISTRIBUTEUR')) {
|
if (null !== $distributor && !$this->hasCategoryCode($distributor, 'DISTRIBUTEUR')) {
|
||||||
$this->throwViolation(
|
$this->throwViolation(
|
||||||
'distributor',
|
'distributor',
|
||||||
'Le distributeur référencé doit être un client de catégorie DISTRIBUTEUR.',
|
'Le distributeur référencé doit être un client de catégorie DISTRIBUTEUR.',
|
||||||
@@ -481,7 +482,7 @@ final class ClientProcessor implements ProcessorInterface
|
|||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
if (null !== $broker && !$this->hasCategoryType($broker, 'COURTIER')) {
|
if (null !== $broker && !$this->hasCategoryCode($broker, 'COURTIER')) {
|
||||||
$this->throwViolation(
|
$this->throwViolation(
|
||||||
'broker',
|
'broker',
|
||||||
'Le courtier référencé doit être un client de catégorie COURTIER.',
|
'Le courtier référencé doit être un client de catégorie COURTIER.',
|
||||||
@@ -530,13 +531,13 @@ final class ClientProcessor implements ProcessorInterface
|
|||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Vrai si au moins une categorie du client porte le type donne. S'appuie
|
* Vrai si au moins une categorie du client porte le code donne. S'appuie sur
|
||||||
* sur CategoryInterface::getCategoryTypeCode() (pas d'import de Category).
|
* CategoryInterface::getCode() (pas d'import de Category — regle ABSOLUE n°1).
|
||||||
*/
|
*/
|
||||||
private function hasCategoryType(Client $client, string $typeCode): bool
|
private function hasCategoryCode(Client $client, string $code): bool
|
||||||
{
|
{
|
||||||
foreach ($client->getCategories() as $category) {
|
foreach ($client->getCategories() as $category) {
|
||||||
if ($category instanceof CategoryInterface && $category->getCategoryTypeCode() === $typeCode) {
|
if ($category instanceof CategoryInterface && $category->getCode() === $code) {
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -24,7 +24,7 @@ use Symfony\Component\DependencyInjection\Attribute\Autowire;
|
|||||||
* exclus au M1) — RG-1.25 ;
|
* exclus au M1) — RG-1.25 ;
|
||||||
* - tri par defaut companyName ASC — RG-1.26 ;
|
* - tri par defaut companyName ASC — RG-1.26 ;
|
||||||
* - filtres ?search=... (fuzzy companyName + lastName + email) et
|
* - filtres ?search=... (fuzzy companyName + lastName + email) et
|
||||||
* ?categoryType=<code> (clients ayant >= 1 categorie de ce type) ;
|
* ?categoryCode=<code> (clients ayant >= 1 categorie de ce code — ERP-78) ;
|
||||||
* - pagination obligatoire (convention Starseed ERP-72) : Paginator ORM ;
|
* - pagination obligatoire (convention Starseed ERP-72) : Paginator ORM ;
|
||||||
* echappatoire ?pagination=false pour alimenter un <select> sans pagination.
|
* echappatoire ?pagination=false pour alimenter un <select> sans pagination.
|
||||||
*
|
*
|
||||||
@@ -64,14 +64,20 @@ final class ClientProvider implements ProviderInterface
|
|||||||
{
|
{
|
||||||
$filters = $context['filters'] ?? [];
|
$filters = $context['filters'] ?? [];
|
||||||
$includeArchived = $this->readBool($filters['includeArchived'] ?? false);
|
$includeArchived = $this->readBool($filters['includeArchived'] ?? false);
|
||||||
|
$archivedOnly = $this->readBool($filters['archivedOnly'] ?? false);
|
||||||
$search = $filters['search'] ?? null;
|
$search = $filters['search'] ?? null;
|
||||||
$categoryType = $filters['categoryType'] ?? null;
|
// categoryCode accepte un code unique (?categoryCode=DISTRIBUTEUR, selects
|
||||||
|
// RG-1.03) OU une liste (?categoryCode[]=A&categoryCode[]=B, drawer multi).
|
||||||
|
$categoryCodes = $this->readStringList($filters['categoryCode'] ?? []);
|
||||||
|
$siteIds = $this->readIntList($filters['siteId'] ?? []);
|
||||||
|
|
||||||
// Filtrage delegue au repository (logique partagee avec l'export XLSX).
|
// Filtrage delegue au repository (logique partagee avec l'export XLSX).
|
||||||
$qb = $this->repository->createListQueryBuilder(
|
$qb = $this->repository->createListQueryBuilder(
|
||||||
$includeArchived,
|
$includeArchived,
|
||||||
is_string($search) ? $search : null,
|
is_string($search) ? $search : null,
|
||||||
is_string($categoryType) ? $categoryType : null,
|
$categoryCodes,
|
||||||
|
$siteIds,
|
||||||
|
$archivedOnly,
|
||||||
);
|
);
|
||||||
|
|
||||||
// Echappatoire ?pagination=false : collection complete sans Paginator
|
// Echappatoire ?pagination=false : collection complete sans Paginator
|
||||||
@@ -127,4 +133,44 @@ final class ClientProvider implements ProviderInterface
|
|||||||
|
|
||||||
return is_string($raw) && in_array(strtolower($raw), ['true', '1'], true);
|
return is_string($raw) && in_array(strtolower($raw), ['true', '1'], true);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Normalise un filtre en liste de chaines. Tolere un code unique (string)
|
||||||
|
* ou une liste (?key[]=a&key[]=b). Trim + retrait des vides.
|
||||||
|
*
|
||||||
|
* @return list<string>
|
||||||
|
*/
|
||||||
|
private function readStringList(mixed $raw): array
|
||||||
|
{
|
||||||
|
$values = is_array($raw) ? $raw : [$raw];
|
||||||
|
|
||||||
|
$out = [];
|
||||||
|
foreach ($values as $value) {
|
||||||
|
if (is_string($value) && '' !== trim($value)) {
|
||||||
|
$out[] = trim($value);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return $out;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Normalise un filtre en liste d'identifiants entiers positifs. Tolere une
|
||||||
|
* valeur unique ou une liste (?key[]=1&key[]=2).
|
||||||
|
*
|
||||||
|
* @return list<int>
|
||||||
|
*/
|
||||||
|
private function readIntList(mixed $raw): array
|
||||||
|
{
|
||||||
|
$values = is_array($raw) ? $raw : [$raw];
|
||||||
|
|
||||||
|
$out = [];
|
||||||
|
foreach ($values as $value) {
|
||||||
|
if ((is_int($value) || (is_string($value) && ctype_digit($value))) && (int) $value > 0) {
|
||||||
|
$out[] = (int) $value;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return $out;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -52,12 +52,19 @@ final class ClientExportController
|
|||||||
public function __invoke(Request $request): Response
|
public function __invoke(Request $request): Response
|
||||||
{
|
{
|
||||||
$includeArchived = $this->readBool($request->query->get('includeArchived'));
|
$includeArchived = $this->readBool($request->query->get('includeArchived'));
|
||||||
|
$archivedOnly = $this->readBool($request->query->get('archivedOnly'));
|
||||||
$search = $request->query->getString('search') ?: null;
|
$search = $request->query->getString('search') ?: null;
|
||||||
$categoryType = $request->query->getString('categoryType') ?: null;
|
|
||||||
|
// Memes filtres que la vue liste : categoryCode/siteId tolerent une valeur
|
||||||
|
// unique ou une liste (?categoryCode[]=A&siteId[]=1). On lit via all() pour
|
||||||
|
// ne pas lever d'exception sur une valeur scalaire.
|
||||||
|
$query = $request->query->all();
|
||||||
|
$categoryCodes = $this->readStringList($query['categoryCode'] ?? []);
|
||||||
|
$siteIds = $this->readIntList($query['siteId'] ?? []);
|
||||||
|
|
||||||
/** @var list<Client> $clients */
|
/** @var list<Client> $clients */
|
||||||
$clients = $this->repository
|
$clients = $this->repository
|
||||||
->createListQueryBuilder($includeArchived, $search, $categoryType)
|
->createListQueryBuilder($includeArchived, $search, $categoryCodes, $siteIds, $archivedOnly)
|
||||||
->getQuery()
|
->getQuery()
|
||||||
->getResult()
|
->getResult()
|
||||||
;
|
;
|
||||||
@@ -198,4 +205,44 @@ final class ClientExportController
|
|||||||
{
|
{
|
||||||
return is_string($raw) && in_array(strtolower($raw), ['true', '1'], true);
|
return is_string($raw) && in_array(strtolower($raw), ['true', '1'], true);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Normalise un filtre en liste de chaines (valeur unique ou liste).
|
||||||
|
* Aligne sur ClientProvider pour un comportement identique a la liste.
|
||||||
|
*
|
||||||
|
* @return list<string>
|
||||||
|
*/
|
||||||
|
private function readStringList(mixed $raw): array
|
||||||
|
{
|
||||||
|
$values = is_array($raw) ? $raw : [$raw];
|
||||||
|
|
||||||
|
$out = [];
|
||||||
|
foreach ($values as $value) {
|
||||||
|
if (is_string($value) && '' !== trim($value)) {
|
||||||
|
$out[] = trim($value);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return $out;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Normalise un filtre en liste d'identifiants entiers positifs (valeur unique
|
||||||
|
* ou liste). Aligne sur ClientProvider.
|
||||||
|
*
|
||||||
|
* @return list<int>
|
||||||
|
*/
|
||||||
|
private function readIntList(mixed $raw): array
|
||||||
|
{
|
||||||
|
$values = is_array($raw) ? $raw : [$raw];
|
||||||
|
|
||||||
|
$out = [];
|
||||||
|
foreach ($values as $value) {
|
||||||
|
if ((is_int($value) || (is_string($value) && ctype_digit($value))) && (int) $value > 0) {
|
||||||
|
$out[] = (int) $value;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return $out;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -54,9 +54,9 @@ use Symfony\Component\DependencyInjection\Attribute\Autowire;
|
|||||||
*
|
*
|
||||||
* Audit / Blamable : persist hors contexte HTTP -> created_by / updated_by
|
* Audit / Blamable : persist hors contexte HTTP -> created_by / updated_by
|
||||||
* restent null (« Systeme » cote front), c'est attendu. Les donnees respectent
|
* restent null (« Systeme » cote front), c'est attendu. Les donnees respectent
|
||||||
* les CHECK BDD ET les validators applicatifs ERP-76 (exclusivite Prospect,
|
* les CHECK BDD ET les validators applicatifs (exclusivite Prospect, billingEmail
|
||||||
* billingEmail ssi facturation, aucune categorie DISTRIBUTEUR/COURTIER sur une
|
* ssi facturation, aucune categorie de code DISTRIBUTEUR/COURTIER sur une adresse
|
||||||
* adresse).
|
* — RG-1.29, ERP-78).
|
||||||
*
|
*
|
||||||
* Depend de CategoryFixtures (categories), SitesFixtures (sites) et
|
* Depend de CategoryFixtures (categories), SitesFixtures (sites) et
|
||||||
* CommercialReferentialFixtures (referentiels comptables Bank / PaymentType).
|
* CommercialReferentialFixtures (referentiels comptables Bank / PaymentType).
|
||||||
@@ -116,7 +116,7 @@ class ClientFixtures extends Fixture implements DependentFixtureInterface
|
|||||||
lastName: 'Garnier',
|
lastName: 'Garnier',
|
||||||
phonePrimary: '05 56 10 20 30',
|
phonePrimary: '05 56 10 20 30',
|
||||||
email: 'contact@distrib-gso.fr',
|
email: 'contact@distrib-gso.fr',
|
||||||
categoryNames: ['Distributeur Grand Sud-Ouest'],
|
categoryNames: ['Distributeur'],
|
||||||
);
|
);
|
||||||
if ($gsoIsNew) {
|
if ($gsoIsNew) {
|
||||||
$this->addContact($gso, 'Paul', 'Garnier', 'Directeur commercial', '05 56 10 20 30', null, 'paul.garnier@distrib-gso.fr');
|
$this->addContact($gso, 'Paul', 'Garnier', 'Directeur commercial', '05 56 10 20 30', null, 'paul.garnier@distrib-gso.fr');
|
||||||
@@ -131,7 +131,7 @@ class ClientFixtures extends Fixture implements DependentFixtureInterface
|
|||||||
lastName: 'Léonard',
|
lastName: 'Léonard',
|
||||||
phonePrimary: '05 49 11 22 33',
|
phonePrimary: '05 49 11 22 33',
|
||||||
email: 'contact@cabinet-leonard.fr',
|
email: 'contact@cabinet-leonard.fr',
|
||||||
categoryNames: ['Cabinet de courtage Léonard'],
|
categoryNames: ['Courtier'],
|
||||||
);
|
);
|
||||||
if ($leonardIsNew) {
|
if ($leonardIsNew) {
|
||||||
$this->addContact($leonard, 'Sophie', 'Léonard', 'Gérante', '05 49 11 22 33', null, 'sophie.leonard@cabinet-leonard.fr');
|
$this->addContact($leonard, 'Sophie', 'Léonard', 'Gérante', '05 49 11 22 33', null, 'sophie.leonard@cabinet-leonard.fr');
|
||||||
@@ -422,11 +422,11 @@ class ClientFixtures extends Fixture implements DependentFixtureInterface
|
|||||||
|
|
||||||
/**
|
/**
|
||||||
* Ajoute une adresse au client (cascade persist via Client.addresses). Les
|
* Ajoute une adresse au client (cascade persist via Client.addresses). Les
|
||||||
* donnees respectent les validators ERP-76 : exclusivite Prospect,
|
* donnees respectent les validators : exclusivite Prospect, billingEmail ssi
|
||||||
* billingEmail ssi facturation, categories limitees a SECTEUR/AUTRE.
|
* facturation, aucune categorie de code DISTRIBUTEUR/COURTIER (RG-1.29).
|
||||||
*
|
*
|
||||||
* @param list<string> $siteNames au moins un site (RG-1.10)
|
* @param list<string> $siteNames au moins un site (RG-1.10)
|
||||||
* @param list<string> $categoryNames categories SECTEUR/AUTRE uniquement (RG-1.29)
|
* @param list<string> $categoryNames categories hors DISTRIBUTEUR/COURTIER (RG-1.29)
|
||||||
*/
|
*/
|
||||||
private function addAddress(
|
private function addAddress(
|
||||||
Client $client,
|
Client $client,
|
||||||
|
|||||||
@@ -34,19 +34,34 @@ class DoctrineClientRepository extends ServiceEntityRepository implements Client
|
|||||||
public function createListQueryBuilder(
|
public function createListQueryBuilder(
|
||||||
bool $includeArchived = false,
|
bool $includeArchived = false,
|
||||||
?string $search = null,
|
?string $search = null,
|
||||||
?string $categoryType = null,
|
array $categoryCodes = [],
|
||||||
|
array $siteIds = [],
|
||||||
|
bool $archivedOnly = false,
|
||||||
): QueryBuilder {
|
): QueryBuilder {
|
||||||
$qb = $this->createQueryBuilder('c')
|
$qb = $this->createQueryBuilder('c')
|
||||||
|
// Jointures + addSelect pour hydrater en une seule requete les
|
||||||
|
// collections affichees par le Repertoire (colonnes Catégories /
|
||||||
|
// Site(s)) : sans cela, la serialisation declenche un N+1 (une
|
||||||
|
// requete par client, puis par adresse). Le Paginator ORM
|
||||||
|
// (fetchJoinCollection: true, cf. ClientProvider) gere le COUNT
|
||||||
|
// malgre ces jointures to-many.
|
||||||
|
->leftJoin('c.categories', 'cat')->addSelect('cat')
|
||||||
|
->leftJoin('c.addresses', 'addr')->addSelect('addr')
|
||||||
|
->leftJoin('addr.sites', 'site')->addSelect('site')
|
||||||
->andWhere('c.deletedAt IS NULL')
|
->andWhere('c.deletedAt IS NULL')
|
||||||
->orderBy('c.companyName', 'ASC')
|
->orderBy('c.companyName', 'ASC')
|
||||||
;
|
;
|
||||||
|
|
||||||
if (!$includeArchived) {
|
// Perimetre d'archivage : archivedOnly prioritaire sur includeArchived.
|
||||||
|
if ($archivedOnly) {
|
||||||
|
$qb->andWhere('c.isArchived = true');
|
||||||
|
} elseif (!$includeArchived) {
|
||||||
$qb->andWhere('c.isArchived = false');
|
$qb->andWhere('c.isArchived = false');
|
||||||
}
|
}
|
||||||
|
|
||||||
$this->applySearch($qb, $search);
|
$this->applySearch($qb, $search);
|
||||||
$this->applyCategoryType($qb, $categoryType);
|
$this->applyCategoryCodes($qb, $categoryCodes);
|
||||||
|
$this->applySiteIds($qb, $siteIds);
|
||||||
|
|
||||||
return $qb;
|
return $qb;
|
||||||
}
|
}
|
||||||
@@ -73,13 +88,18 @@ class DoctrineClientRepository extends ServiceEntityRepository implements Client
|
|||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Restreint aux clients possedant au moins une categorie du type donne.
|
* Restreint aux clients possedant au moins une categorie dont le code figure
|
||||||
* Sous-requete IN (plutot qu'un JOIN sur la collection M2M) pour ne pas
|
* dans la liste (OR — ERP-78). Alimente le filtre « Catégories » du drawer
|
||||||
* perturber le DISTINCT / ORDER BY de la requete principale.
|
* (multi) ainsi que les selects « distributeur »/« courtier » (un seul code,
|
||||||
|
* RG-1.03). Sous-requete IN (plutot qu'un JOIN sur la collection M2M) pour ne
|
||||||
|
* pas perturber le DISTINCT / ORDER BY principal.
|
||||||
|
*
|
||||||
|
* @param list<string> $categoryCodes
|
||||||
*/
|
*/
|
||||||
private function applyCategoryType(QueryBuilder $qb, ?string $categoryType): void
|
private function applyCategoryCodes(QueryBuilder $qb, array $categoryCodes): void
|
||||||
{
|
{
|
||||||
if (null === $categoryType || '' === trim($categoryType)) {
|
$codes = $this->normalizeStringList($categoryCodes);
|
||||||
|
if ([] === $codes) {
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -87,12 +107,84 @@ class DoctrineClientRepository extends ServiceEntityRepository implements Client
|
|||||||
->select('c2.id')
|
->select('c2.id')
|
||||||
->from(Client::class, 'c2')
|
->from(Client::class, 'c2')
|
||||||
->join('c2.categories', 'cat2')
|
->join('c2.categories', 'cat2')
|
||||||
->join('cat2.categoryType', 'ct2')
|
->where('cat2.code IN (:categoryCodes)')
|
||||||
->where('ct2.code = :categoryType')
|
|
||||||
;
|
;
|
||||||
|
|
||||||
$qb->andWhere($qb->expr()->in('c.id', $sub->getDQL()))
|
$qb->andWhere($qb->expr()->in('c.id', $sub->getDQL()))
|
||||||
->setParameter('categoryType', trim($categoryType))
|
->setParameter('categoryCodes', $codes)
|
||||||
;
|
;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Restreint aux clients ayant au moins une adresse rattachee a l'un des
|
||||||
|
* sites donnes (OR — RG-1.10 : les sites vivent sur les adresses, pas sur le
|
||||||
|
* client). Sous-requete IN pour ne pas perturber le tri/pagination principal.
|
||||||
|
*
|
||||||
|
* @param list<int> $siteIds
|
||||||
|
*/
|
||||||
|
private function applySiteIds(QueryBuilder $qb, array $siteIds): void
|
||||||
|
{
|
||||||
|
$ids = $this->normalizeIntList($siteIds);
|
||||||
|
if ([] === $ids) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
$sub = $this->getEntityManager()->createQueryBuilder()
|
||||||
|
->select('c3.id')
|
||||||
|
->from(Client::class, 'c3')
|
||||||
|
->join('c3.addresses', 'addr3')
|
||||||
|
->join('addr3.sites', 'site3')
|
||||||
|
->where('site3.id IN (:siteIds)')
|
||||||
|
;
|
||||||
|
|
||||||
|
$qb->andWhere($qb->expr()->in('c.id', $sub->getDQL()))
|
||||||
|
->setParameter('siteIds', $ids)
|
||||||
|
;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Nettoie une liste de chaines : trim, retrait des vides, reindexation.
|
||||||
|
* Defensive : tolere des elements scalaires non-string (cast) et ignore le
|
||||||
|
* reste sans lever de TypeError, le contrat etant justement de normaliser une
|
||||||
|
* entree potentiellement brute (query params).
|
||||||
|
*
|
||||||
|
* @param array<mixed> $values
|
||||||
|
*
|
||||||
|
* @return list<string>
|
||||||
|
*/
|
||||||
|
private function normalizeStringList(array $values): array
|
||||||
|
{
|
||||||
|
$out = [];
|
||||||
|
foreach ($values as $value) {
|
||||||
|
if (is_string($value) || is_int($value) || is_float($value)) {
|
||||||
|
$trimmed = trim((string) $value);
|
||||||
|
if ('' !== $trimmed) {
|
||||||
|
$out[] = $trimmed;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return $out;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Nettoie une liste d'identifiants : cast int, retrait des <= 0, reindexation.
|
||||||
|
* Defensive (cf. normalizeStringList) : accepte des entiers ou des chaines
|
||||||
|
* numeriques ('1', '2') sans TypeError, ignore le reste.
|
||||||
|
*
|
||||||
|
* @param array<mixed> $values
|
||||||
|
*
|
||||||
|
* @return list<int>
|
||||||
|
*/
|
||||||
|
private function normalizeIntList(array $values): array
|
||||||
|
{
|
||||||
|
$out = [];
|
||||||
|
foreach ($values as $value) {
|
||||||
|
if (is_numeric($value) && (int) $value > 0) {
|
||||||
|
$out[] = (int) $value;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return $out;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -20,13 +20,25 @@ interface CategoryInterface
|
|||||||
|
|
||||||
public function getName(): ?string;
|
public function getName(): ?string;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Code technique stable de la categorie (Category::code), ou null si non
|
||||||
|
* encore renseigne. Slug MAJUSCULE derive du nom a la creation, fige ensuite.
|
||||||
|
* Expose pour permettre a un module tiers de filtrer/valider par categorie
|
||||||
|
* metier sans dependre du libelle (`name`) ni de l'`id` (non deterministe
|
||||||
|
* entre environnements) ni importer la classe concrete Category (regle
|
||||||
|
* ABSOLUE n°1). Pilote, cote M1 Commercial :
|
||||||
|
* - RG-1.03 : un distributor doit referencer un client portant la categorie
|
||||||
|
* de code DISTRIBUTEUR (resp. COURTIER pour broker) ;
|
||||||
|
* - RG-1.29 : une adresse interdit les categories de code DISTRIBUTEUR /
|
||||||
|
* COURTIER (relations entre clients, pas des attributs d'adresse).
|
||||||
|
*/
|
||||||
|
public function getCode(): ?string;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Code du type de categorie rattache (CategoryType::code), ou null si la
|
* Code du type de categorie rattache (CategoryType::code), ou null si la
|
||||||
* categorie n'a pas de type. Expose pour permettre a un module tiers de
|
* categorie n'a pas de type. Depuis ERP-78, le modele n'a plus qu'un seul
|
||||||
* raisonner sur le type metier (ex: M1 Commercial — RG-1.03 : un distributor
|
* type (CLIENT) : le filtrage metier passe desormais par getCode() ci-dessus.
|
||||||
* doit referencer un client categorise DISTRIBUTEUR ; RG-1.29 : categorie
|
* Conserve pour l'affichage / la retrocompatibilite.
|
||||||
* d'adresse limitee a SECTEUR/AUTRE) sans importer la classe concrete
|
|
||||||
* Category (regle ABSOLUE n°1).
|
|
||||||
*/
|
*/
|
||||||
public function getCategoryTypeCode(): ?string;
|
public function getCategoryTypeCode(): ?string;
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -0,0 +1,60 @@
|
|||||||
|
<?php
|
||||||
|
|
||||||
|
declare(strict_types=1);
|
||||||
|
|
||||||
|
namespace App\Shared\Infrastructure\Database;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Miroir SQL de `CategoryCodeGenerator::slugify()` (module Catalog, ERP-78).
|
||||||
|
*
|
||||||
|
* Le `code` d'une `Category` est un slug MAJUSCULE deterministe du nom. A
|
||||||
|
* l'execution (POST/PATCH API), il est genere en PHP par `CategoryCodeGenerator`
|
||||||
|
* via `AsciiSlugger`. Mais la migration corrective `Version20260602100000` doit
|
||||||
|
* backfiller le `code` des categories pre-existantes en SQL pur (le backfill
|
||||||
|
* tourne dans le plan `addSql`, sans acces aux services applicatifs).
|
||||||
|
*
|
||||||
|
* Deux implementations d'un meme slug = risque de derive : un nom accentue
|
||||||
|
* comme « Independant » doit produire le MEME code (`INDEPENDANT`) quel que soit
|
||||||
|
* le chemin. Cette classe est la SOURCE UNIQUE de l'expression SQL ; son egalite
|
||||||
|
* avec le générateur PHP est verrouillee par `CategoryCodeSqlSlugTest`.
|
||||||
|
*
|
||||||
|
* Domaine couvert : noms francais / Latin-1 (tous les accents, minuscule +
|
||||||
|
* majuscule, translitteres vers l'ASCII comme le fait `AsciiSlugger`). Limite
|
||||||
|
* connue et assumee : les ligatures (`Œ`->`OE`, `ß`->`SS`) ne sont PAS gerees
|
||||||
|
* par `translate()` (mapping 1->1 uniquement) ; elles n'apparaissent pas dans
|
||||||
|
* les noms de categories CLIENT et le backfill ne s'execute de toute facon que
|
||||||
|
* sur des bases dev deja peuplees (en prod la table `category` est vide).
|
||||||
|
*/
|
||||||
|
final class CategoryCodeSql
|
||||||
|
{
|
||||||
|
/** Longueur maximale de la colonne `category.code` (cf. CategoryCodeGenerator). */
|
||||||
|
private const int MAX_LENGTH = 50;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Accents Latin-1 (minuscules puis majuscules) translitteres vers leur
|
||||||
|
* equivalent ASCII minuscule — `UPPER()` repasse tout en majuscule ensuite.
|
||||||
|
* `translate()` mappe caractere a caractere : `ACCENT_FROM` et `ACCENT_TO`
|
||||||
|
* doivent avoir EXACTEMENT le meme nombre de caracteres.
|
||||||
|
*/
|
||||||
|
private const string ACCENT_FROM = 'àâäáãåçéèêëíìîïñóòôöõúùûüýÿÀÂÄÁÃÅÇÉÈÊËÍÌÎÏÑÓÒÔÖÕÚÙÛÜÝŸ';
|
||||||
|
private const string ACCENT_TO = 'aaaaaaceeeeiiiinooooouuuuyyaaaaaaceeeeiiiinooooouuuuyy';
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Expression SQL produisant le slug du `$column` donne (ex: `name`, `c.name`).
|
||||||
|
* Reproduit fidelement `CategoryCodeGenerator::slugify` : translitteration des
|
||||||
|
* accents, separateurs non alphanumeriques reduits a `_`, MAJUSCULE, borne a
|
||||||
|
* 50, `_` de bord retires, fallback `CATEGORY` si vide.
|
||||||
|
*/
|
||||||
|
public static function slugExpression(string $column): string
|
||||||
|
{
|
||||||
|
return sprintf(
|
||||||
|
"COALESCE(NULLIF(TRIM(BOTH '_' FROM "
|
||||||
|
."LEFT(UPPER(REGEXP_REPLACE(translate(%s, '%s', '%s'), '[^A-Za-z0-9]+', '_', 'g')), %d)"
|
||||||
|
."), ''), 'CATEGORY')",
|
||||||
|
$column,
|
||||||
|
self::ACCENT_FROM,
|
||||||
|
self::ACCENT_TO,
|
||||||
|
self::MAX_LENGTH,
|
||||||
|
);
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -53,6 +53,7 @@ final class ColumnCommentsCatalog
|
|||||||
'_table' => 'Categories M0 — referentiel type par category_type, soft-delete via deleted_at, unicite (LOWER(name), category_type_id) parmi les actifs.',
|
'_table' => 'Categories M0 — referentiel type par category_type, soft-delete via deleted_at, unicite (LOWER(name), category_type_id) parmi les actifs.',
|
||||||
'id' => 'Identifiant interne auto-incremente.',
|
'id' => 'Identifiant interne auto-incremente.',
|
||||||
'name' => 'Libelle de la categorie (≤ 120 caracteres) — unique par type parmi les actifs (RG-1.06).',
|
'name' => 'Libelle de la categorie (≤ 120 caracteres) — unique par type parmi les actifs (RG-1.06).',
|
||||||
|
'code' => 'Code technique stable (slug MAJUSCULE du nom, ≤ 50) — unique parmi les actifs (uq_category_code). Fige a la creation. DISTRIBUTEUR/COURTIER pilotent RG-1.03/1.29.',
|
||||||
'category_type_id' => 'Reference au type de la categorie — FK -> category_type.id, ON DELETE RESTRICT (un type ne peut etre supprime tant qu il a des categories).',
|
'category_type_id' => 'Reference au type de la categorie — FK -> category_type.id, ON DELETE RESTRICT (un type ne peut etre supprime tant qu il a des categories).',
|
||||||
'deleted_at' => 'Horodatage UTC du soft-delete (archivage logique) — null si la categorie est active.',
|
'deleted_at' => 'Horodatage UTC du soft-delete (archivage logique) — null si la categorie est active.',
|
||||||
] + self::timestampableBlamableComments(),
|
] + self::timestampableBlamableComments(),
|
||||||
|
|||||||
@@ -6,12 +6,12 @@ namespace App\Shared\Infrastructure\Doctrine;
|
|||||||
|
|
||||||
use App\Shared\Domain\Contract\BlamableInterface;
|
use App\Shared\Domain\Contract\BlamableInterface;
|
||||||
use App\Shared\Domain\Contract\TimestampableInterface;
|
use App\Shared\Domain\Contract\TimestampableInterface;
|
||||||
use DateTimeImmutable;
|
|
||||||
use Doctrine\Bundle\DoctrineBundle\Attribute\AsDoctrineListener;
|
use Doctrine\Bundle\DoctrineBundle\Attribute\AsDoctrineListener;
|
||||||
use Doctrine\ORM\Event\PrePersistEventArgs;
|
use Doctrine\ORM\Event\PrePersistEventArgs;
|
||||||
use Doctrine\ORM\Event\PreUpdateEventArgs;
|
use Doctrine\ORM\Event\PreUpdateEventArgs;
|
||||||
use Doctrine\ORM\Events;
|
use Doctrine\ORM\Events;
|
||||||
use Symfony\Bundle\SecurityBundle\Security;
|
use Symfony\Bundle\SecurityBundle\Security;
|
||||||
|
use Symfony\Component\Clock\ClockInterface;
|
||||||
use Symfony\Component\Security\Core\User\UserInterface;
|
use Symfony\Component\Security\Core\User\UserInterface;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@@ -30,12 +30,19 @@ use Symfony\Component\Security\Core\User\UserInterface;
|
|||||||
#[AsDoctrineListener(event: Events::preUpdate)]
|
#[AsDoctrineListener(event: Events::preUpdate)]
|
||||||
final class TimestampableBlamableSubscriber
|
final class TimestampableBlamableSubscriber
|
||||||
{
|
{
|
||||||
public function __construct(private readonly Security $security) {}
|
// L'horloge est injectee (et non un `new DateTimeImmutable()` direct) pour
|
||||||
|
// que les tests puissent figer/avancer le temps de facon deterministe via
|
||||||
|
// ClockSensitiveTrait (cf. ERP-98). En prod, le service `clock` delegue a
|
||||||
|
// l'horloge systeme reelle.
|
||||||
|
public function __construct(
|
||||||
|
private readonly Security $security,
|
||||||
|
private readonly ClockInterface $clock,
|
||||||
|
) {}
|
||||||
|
|
||||||
public function prePersist(PrePersistEventArgs $args): void
|
public function prePersist(PrePersistEventArgs $args): void
|
||||||
{
|
{
|
||||||
$entity = $args->getObject();
|
$entity = $args->getObject();
|
||||||
$now = new DateTimeImmutable();
|
$now = $this->clock->now();
|
||||||
$user = $this->security->getUser();
|
$user = $this->security->getUser();
|
||||||
|
|
||||||
if ($entity instanceof TimestampableInterface) {
|
if ($entity instanceof TimestampableInterface) {
|
||||||
@@ -55,7 +62,7 @@ final class TimestampableBlamableSubscriber
|
|||||||
$user = $this->security->getUser();
|
$user = $this->security->getUser();
|
||||||
|
|
||||||
if ($entity instanceof TimestampableInterface) {
|
if ($entity instanceof TimestampableInterface) {
|
||||||
$entity->setUpdatedAt(new DateTimeImmutable());
|
$entity->setUpdatedAt($this->clock->now());
|
||||||
}
|
}
|
||||||
|
|
||||||
if ($entity instanceof BlamableInterface && $user instanceof UserInterface) {
|
if ($entity instanceof BlamableInterface && $user instanceof UserInterface) {
|
||||||
|
|||||||
@@ -83,6 +83,9 @@ abstract class AbstractCatalogApiTestCase extends AbstractApiTestCase
|
|||||||
$suffix = substr(bin2hex(random_bytes(4)), 0, 8);
|
$suffix = substr(bin2hex(random_bytes(4)), 0, 8);
|
||||||
$category = new Category();
|
$category = new Category();
|
||||||
$category->setName($name ?? self::TEST_CATEGORY_PREFIX.$suffix);
|
$category->setName($name ?? self::TEST_CATEGORY_PREFIX.$suffix);
|
||||||
|
// ERP-78 : code NOT NULL + unique parmi les actifs (uq_category_code).
|
||||||
|
// Nonce aleatoire -> unicite garantie entre seeds successifs du test.
|
||||||
|
$category->setCode('TEST_'.strtoupper($suffix));
|
||||||
$category->setCategoryType($type);
|
$category->setCategoryType($type);
|
||||||
if (null !== $deletedAt) {
|
if (null !== $deletedAt) {
|
||||||
$category->setDeletedAt($deletedAt);
|
$category->setDeletedAt($deletedAt);
|
||||||
|
|||||||
@@ -0,0 +1,73 @@
|
|||||||
|
<?php
|
||||||
|
|
||||||
|
declare(strict_types=1);
|
||||||
|
|
||||||
|
namespace App\Tests\Module\Catalog\Api;
|
||||||
|
|
||||||
|
use App\Module\Catalog\Application\Service\CategoryCodeGenerator;
|
||||||
|
use App\Shared\Infrastructure\Database\CategoryCodeSql;
|
||||||
|
use Doctrine\DBAL\Connection;
|
||||||
|
use PHPUnit\Framework\Attributes\DataProvider;
|
||||||
|
use Symfony\Bundle\FrameworkBundle\Test\KernelTestCase;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Garde-fou ERP-78 : l'expression SQL de slug (CategoryCodeSql, utilisee par le
|
||||||
|
* backfill de la migration Version20260602100000) doit produire EXACTEMENT le
|
||||||
|
* meme code que le generateur applicatif (CategoryCodeGenerator::slugify), sur
|
||||||
|
* tout le domaine de noms francais / Latin-1.
|
||||||
|
*
|
||||||
|
* Verrouille la cause racine du bug initial : deux implementations d'un meme
|
||||||
|
* slug qui derivent silencieusement (« Independant » -> IND_PENDANT cote SQL
|
||||||
|
* faute de translitteration des accents, vs INDEPENDANT cote PHP). On ne couvre
|
||||||
|
* volontairement PAS les ligatures (`Œ`, `ß`) : `translate()` est 1->1 et ne
|
||||||
|
* peut produire `OE`/`SS` ; elles sont hors du domaine des categories CLIENT.
|
||||||
|
*
|
||||||
|
* @internal
|
||||||
|
*/
|
||||||
|
final class CategoryCodeSqlSlugTest extends KernelTestCase
|
||||||
|
{
|
||||||
|
/**
|
||||||
|
* Noms representatifs du domaine reel : accents, cedille, apostrophe,
|
||||||
|
* separateurs varies, parentheses, majuscules accentuees.
|
||||||
|
*
|
||||||
|
* @return iterable<string, array{string}>
|
||||||
|
*/
|
||||||
|
public static function nameProvider(): iterable
|
||||||
|
{
|
||||||
|
yield 'sans accent' => ['Distributeur'];
|
||||||
|
yield 'tiret' => ['Agro-alimentaire'];
|
||||||
|
yield 'slash' => ['Transport/Logistique'];
|
||||||
|
yield 'accent aigu' => ['Indépendant'];
|
||||||
|
yield 'apostrophe + accent' => ["L'Oréal"];
|
||||||
|
yield 'esperluette' => ['Forêt & Bûcheron'];
|
||||||
|
yield 'cedille majuscule' => ['Ça va'];
|
||||||
|
yield 'accents multiples' => ['Naïve façade'];
|
||||||
|
yield 'circonflexe' => ["Côte d'Azur"];
|
||||||
|
yield 'parentheses' => ['Zone (Sud)'];
|
||||||
|
}
|
||||||
|
|
||||||
|
#[DataProvider('nameProvider')]
|
||||||
|
public function testSqlSlugMatchesPhpSlug(string $name): void
|
||||||
|
{
|
||||||
|
self::bootKernel();
|
||||||
|
$container = self::getContainer();
|
||||||
|
|
||||||
|
/** @var Connection $conn */
|
||||||
|
$conn = $container->get('doctrine')->getConnection();
|
||||||
|
/** @var CategoryCodeGenerator $generator */
|
||||||
|
$generator = $container->get(CategoryCodeGenerator::class);
|
||||||
|
|
||||||
|
// Evaluation pure de l'expression (aucune table requise) : le nom est
|
||||||
|
// passe en parametre lie a la place de la colonne.
|
||||||
|
$sqlSlug = $conn->fetchOne(
|
||||||
|
'SELECT '.CategoryCodeSql::slugExpression(':name'),
|
||||||
|
['name' => $name],
|
||||||
|
);
|
||||||
|
|
||||||
|
self::assertSame(
|
||||||
|
$generator->slugify($name),
|
||||||
|
$sqlSlug,
|
||||||
|
sprintf('SQL et PHP doivent produire le meme slug pour "%s".', $name),
|
||||||
|
);
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -0,0 +1,90 @@
|
|||||||
|
<?php
|
||||||
|
|
||||||
|
declare(strict_types=1);
|
||||||
|
|
||||||
|
namespace App\Tests\Module\Catalog\Api;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Tests ERP-78 : le `code` technique stable de Category.
|
||||||
|
*
|
||||||
|
* Cas couverts :
|
||||||
|
* - POST : le code est auto-genere (slug MAJUSCULE du nom) et expose en lecture ;
|
||||||
|
* - le code est en lecture seule : un `code` envoye dans le payload est ignore
|
||||||
|
* (genere depuis le nom) ;
|
||||||
|
* - deux noms produisant le meme slug recoivent des codes distincts (suffixe).
|
||||||
|
*
|
||||||
|
* @internal
|
||||||
|
*/
|
||||||
|
final class CategoryCodeTest extends AbstractCatalogApiTestCase
|
||||||
|
{
|
||||||
|
public function testPostGeneratesAndExposesCode(): void
|
||||||
|
{
|
||||||
|
$type = $this->createCategoryType();
|
||||||
|
$client = $this->createAdminClient();
|
||||||
|
|
||||||
|
$response = $client->request('POST', '/api/categories', [
|
||||||
|
'headers' => ['Content-Type' => 'application/ld+json'],
|
||||||
|
'json' => [
|
||||||
|
'name' => self::TEST_CATEGORY_PREFIX.'Agro-alimentaire',
|
||||||
|
'categoryType' => '/api/category_types/'.$type->getId(),
|
||||||
|
],
|
||||||
|
]);
|
||||||
|
self::assertResponseStatusCodeSame(201);
|
||||||
|
|
||||||
|
$payload = $response->toArray();
|
||||||
|
// Slug MAJUSCULE du nom, separateurs non alphanumeriques -> `_`.
|
||||||
|
self::assertSame(
|
||||||
|
strtoupper(self::TEST_CATEGORY_PREFIX).'AGRO_ALIMENTAIRE',
|
||||||
|
$payload['code'],
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
public function testCodeIsReadOnlyAndIgnoredFromPayload(): void
|
||||||
|
{
|
||||||
|
$type = $this->createCategoryType();
|
||||||
|
$client = $this->createAdminClient();
|
||||||
|
|
||||||
|
$response = $client->request('POST', '/api/categories', [
|
||||||
|
'headers' => ['Content-Type' => 'application/ld+json'],
|
||||||
|
'json' => [
|
||||||
|
'name' => self::TEST_CATEGORY_PREFIX.'readonly',
|
||||||
|
'categoryType' => '/api/category_types/'.$type->getId(),
|
||||||
|
// Le client tente d'imposer un code : doit etre ignore.
|
||||||
|
'code' => 'CLIENT_FORGED',
|
||||||
|
],
|
||||||
|
]);
|
||||||
|
self::assertResponseStatusCodeSame(201);
|
||||||
|
|
||||||
|
$payload = $response->toArray();
|
||||||
|
self::assertNotSame('CLIENT_FORGED', $payload['code']);
|
||||||
|
self::assertSame(strtoupper(self::TEST_CATEGORY_PREFIX).'READONLY', $payload['code']);
|
||||||
|
}
|
||||||
|
|
||||||
|
public function testCollidingSlugsGetDistinctCodes(): void
|
||||||
|
{
|
||||||
|
$type = $this->createCategoryType();
|
||||||
|
$client = $this->createAdminClient();
|
||||||
|
|
||||||
|
// Deux noms differents (donc autorises par uq_category_name_type_active)
|
||||||
|
// mais qui produisent le meme slug -> codes distincts (suffixe `_2`).
|
||||||
|
$first = $client->request('POST', '/api/categories', [
|
||||||
|
'headers' => ['Content-Type' => 'application/ld+json'],
|
||||||
|
'json' => [
|
||||||
|
'name' => self::TEST_CATEGORY_PREFIX.'Agro Plus',
|
||||||
|
'categoryType' => '/api/category_types/'.$type->getId(),
|
||||||
|
],
|
||||||
|
])->toArray();
|
||||||
|
|
||||||
|
$second = $client->request('POST', '/api/categories', [
|
||||||
|
'headers' => ['Content-Type' => 'application/ld+json'],
|
||||||
|
'json' => [
|
||||||
|
'name' => self::TEST_CATEGORY_PREFIX.'Agro-Plus',
|
||||||
|
'categoryType' => '/api/category_types/'.$type->getId(),
|
||||||
|
],
|
||||||
|
])->toArray();
|
||||||
|
|
||||||
|
self::assertResponseStatusCodeSame(201);
|
||||||
|
self::assertNotSame($first['code'], $second['code']);
|
||||||
|
self::assertStringEndsWith('_2', (string) $second['code']);
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -7,6 +7,8 @@ namespace App\Tests\Module\Catalog\Api;
|
|||||||
use App\Module\Catalog\Domain\Entity\Category;
|
use App\Module\Catalog\Domain\Entity\Category;
|
||||||
use App\Module\Core\Domain\Entity\User;
|
use App\Module\Core\Domain\Entity\User;
|
||||||
use DateTimeImmutable;
|
use DateTimeImmutable;
|
||||||
|
use Symfony\Component\Clock\ClockInterface;
|
||||||
|
use Symfony\Component\Clock\Test\ClockSensitiveTrait;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Tests RG-1.15 / RG-1.16 : le TimestampableBlamableSubscriber doit remplir
|
* Tests RG-1.15 / RG-1.16 : le TimestampableBlamableSubscriber doit remplir
|
||||||
@@ -20,12 +22,39 @@ use DateTimeImmutable;
|
|||||||
* - DELETE : deletedAt rempli ET updatedAt + updatedBy mis a jour (UPDATE
|
* - DELETE : deletedAt rempli ET updatedAt + updatedBy mis a jour (UPDATE
|
||||||
* Doctrine declenche le subscriber)
|
* Doctrine declenche le subscriber)
|
||||||
*
|
*
|
||||||
|
* ERP-98 : ces tests pilotent une horloge mockee (ClockSensitiveTrait) plutot
|
||||||
|
* que de dependre d'un `sleep(1)` reel. Le subscriber lit le service `clock`,
|
||||||
|
* que `self::mockTime()` remplace par un MockClock fige au niveau du process —
|
||||||
|
* ce qui survit aux reboots de kernel entre requetes (POST admin / PATCH bob)
|
||||||
|
* et reste insensible a la derive d'horloge WSL2 a l'origine des flakes.
|
||||||
|
*
|
||||||
* @internal
|
* @internal
|
||||||
*/
|
*/
|
||||||
final class CategoryTimestampableBlamableTest extends AbstractCatalogApiTestCase
|
final class CategoryTimestampableBlamableTest extends AbstractCatalogApiTestCase
|
||||||
{
|
{
|
||||||
|
use ClockSensitiveTrait;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Fige l'horloge globale sur l'instant courant DANS LE FUSEAU PHP par
|
||||||
|
* defaut, et la retourne pour la piloter (`sleep()`).
|
||||||
|
*
|
||||||
|
* Subtilite : `self::mockTime()` cree par defaut un MockClock en UTC, or
|
||||||
|
* les colonnes `TIMESTAMP WITHOUT TIME ZONE` round-trippent via le fuseau
|
||||||
|
* PHP (Europe/Paris). Un MockClock UTC decalerait createdAt de l'offset
|
||||||
|
* (2h) au rechargement. On seede donc avec `new DateTimeImmutable()`
|
||||||
|
* (fuseau par defaut), exactement comme le NativeClock en prod.
|
||||||
|
*/
|
||||||
|
private function freezeClock(): ClockInterface
|
||||||
|
{
|
||||||
|
return self::mockTime(new DateTimeImmutable());
|
||||||
|
}
|
||||||
|
|
||||||
public function testCreatedByAdminOnPost(): void
|
public function testCreatedByAdminOnPost(): void
|
||||||
{
|
{
|
||||||
|
// Horloge figee : le subscriber posera createdAt/updatedAt sur cet
|
||||||
|
// instant exact, insensible a tout decalage d'horloge reel.
|
||||||
|
$clock = $this->freezeClock();
|
||||||
|
|
||||||
$type = $this->createCategoryType();
|
$type = $this->createCategoryType();
|
||||||
|
|
||||||
/** @var User $admin */
|
/** @var User $admin */
|
||||||
@@ -33,9 +62,7 @@ final class CategoryTimestampableBlamableTest extends AbstractCatalogApiTestCase
|
|||||||
self::assertNotNull($admin);
|
self::assertNotNull($admin);
|
||||||
$adminId = $admin->getId();
|
$adminId = $admin->getId();
|
||||||
|
|
||||||
$before = new DateTimeImmutable();
|
$before = $clock->now();
|
||||||
// Petit decalage pour absorber les arrondis a la seconde de Postgres.
|
|
||||||
sleep(1);
|
|
||||||
|
|
||||||
$client = $this->createAdminClient();
|
$client = $this->createAdminClient();
|
||||||
$response = $client->request('POST', '/api/categories', [
|
$response = $client->request('POST', '/api/categories', [
|
||||||
@@ -103,6 +130,8 @@ final class CategoryTimestampableBlamableTest extends AbstractCatalogApiTestCase
|
|||||||
|
|
||||||
public function testPatchUpdatesUpdatedFieldsOnly(): void
|
public function testPatchUpdatesUpdatedFieldsOnly(): void
|
||||||
{
|
{
|
||||||
|
$clock = $this->freezeClock();
|
||||||
|
|
||||||
// Etape 1 : creation par admin pour figer createdBy=admin.
|
// Etape 1 : creation par admin pour figer createdBy=admin.
|
||||||
$type = $this->createCategoryType();
|
$type = $this->createCategoryType();
|
||||||
$adminClient = $this->createAdminClient();
|
$adminClient = $this->createAdminClient();
|
||||||
@@ -127,9 +156,9 @@ final class CategoryTimestampableBlamableTest extends AbstractCatalogApiTestCase
|
|||||||
$initialUpdatedAt = $initial->getUpdatedAt();
|
$initialUpdatedAt = $initial->getUpdatedAt();
|
||||||
$initialCreatedById = $initial->getCreatedBy()->getId();
|
$initialCreatedById = $initial->getCreatedBy()->getId();
|
||||||
|
|
||||||
// Decalage temporel suffisant pour que la precision PG (seconde)
|
// Avance deterministe de l'horloge mockee : garantit un updatedAt
|
||||||
// capte un updatedAt different.
|
// strictement superieur cote PG (precision seconde) sans sleep reel.
|
||||||
sleep(1);
|
$clock->sleep(1);
|
||||||
|
|
||||||
// Etape 2 : PATCH par un autre user (manager non-admin) — simule "bob".
|
// Etape 2 : PATCH par un autre user (manager non-admin) — simule "bob".
|
||||||
$manage = $this->createManageClient();
|
$manage = $this->createManageClient();
|
||||||
@@ -180,6 +209,8 @@ final class CategoryTimestampableBlamableTest extends AbstractCatalogApiTestCase
|
|||||||
|
|
||||||
public function testSoftDeleteAlsoUpdatesUpdatedFields(): void
|
public function testSoftDeleteAlsoUpdatesUpdatedFields(): void
|
||||||
{
|
{
|
||||||
|
$clock = $this->freezeClock();
|
||||||
|
|
||||||
// RG-1.16 : le soft delete est un UPDATE Doctrine, donc le subscriber
|
// RG-1.16 : le soft delete est un UPDATE Doctrine, donc le subscriber
|
||||||
// doit aussi avancer updatedAt et updatedBy en plus de poser deletedAt.
|
// doit aussi avancer updatedAt et updatedBy en plus de poser deletedAt.
|
||||||
$type = $this->createCategoryType();
|
$type = $this->createCategoryType();
|
||||||
@@ -202,7 +233,8 @@ final class CategoryTimestampableBlamableTest extends AbstractCatalogApiTestCase
|
|||||||
$initial = $em->getRepository(Category::class)->find($createdId);
|
$initial = $em->getRepository(Category::class)->find($createdId);
|
||||||
$initialUpdatedAt = $initial->getUpdatedAt();
|
$initialUpdatedAt = $initial->getUpdatedAt();
|
||||||
|
|
||||||
sleep(1);
|
// Avance deterministe de l'horloge mockee (cf. testPatch).
|
||||||
|
$clock->sleep(1);
|
||||||
|
|
||||||
// Soft delete par un manager non-admin.
|
// Soft delete par un manager non-admin.
|
||||||
$manage = $this->createManageClient();
|
$manage = $this->createManageClient();
|
||||||
|
|||||||
@@ -17,13 +17,17 @@ use DateTimeImmutable;
|
|||||||
* Base des tests fonctionnels du module Commercial (M1 — repertoire clients).
|
* Base des tests fonctionnels du module Commercial (M1 — repertoire clients).
|
||||||
*
|
*
|
||||||
* Etend la base Core : ajoute des factories pour seeder vite des categories
|
* Etend la base Core : ajoute des factories pour seeder vite des categories
|
||||||
* typees (DISTRIBUTEUR / COURTIER / SECTEUR) et des clients, plus un helper
|
* codees (DISTRIBUTEUR / COURTIER / SECTEUR...) sous le type unique CLIENT et
|
||||||
* d'authentification admin.
|
* 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
|
* Cleanup : tearDown purge clients, categories `test_cli_cat_*` et users/roles
|
||||||
* `test_*`. Les category_types business sont fetch-or-create (idempotents) et
|
* `test_*`. Le type CLIENT est fetch-or-create (idempotent) et laisse en place.
|
||||||
* laisses en place (pas de DELETE pour ne pas entrer en conflit avec d'autres
|
* Pas de DAMA en local -> purge manuelle obligatoire.
|
||||||
* suites). Pas de DAMA en local -> purge manuelle obligatoire.
|
|
||||||
*
|
*
|
||||||
* @internal
|
* @internal
|
||||||
*/
|
*/
|
||||||
@@ -31,6 +35,14 @@ abstract class AbstractCommercialApiTestCase extends AbstractApiTestCase
|
|||||||
{
|
{
|
||||||
protected const string TEST_CATEGORY_PREFIX = 'test_cli_cat_';
|
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
|
protected function tearDown(): void
|
||||||
{
|
{
|
||||||
$this->cleanupCommercialTestData();
|
$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.
|
* contrainte d'unicite sur category_type.code interdit les doublons.
|
||||||
*/
|
*/
|
||||||
protected function createCategoryType(string $code): CategoryType
|
protected function clientCategoryType(): CategoryType
|
||||||
{
|
{
|
||||||
$em = $this->getEm();
|
$em = $this->getEm();
|
||||||
$existing = $em->getRepository(CategoryType::class)->findOneBy(['code' => $code]);
|
$existing = $em->getRepository(CategoryType::class)->findOneBy(['code' => 'CLIENT']);
|
||||||
if (null !== $existing) {
|
if (null !== $existing) {
|
||||||
return $existing;
|
return $existing;
|
||||||
}
|
}
|
||||||
|
|
||||||
$type = new CategoryType();
|
$type = new CategoryType();
|
||||||
$type->setCode($code);
|
$type->setCode('CLIENT');
|
||||||
$type->setLabel(ucfirst(strtolower($code)));
|
$type->setLabel('Client');
|
||||||
$em->persist($type);
|
$em->persist($type);
|
||||||
$em->flush();
|
$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();
|
$em = $this->getEm();
|
||||||
$suffix = substr(bin2hex(random_bytes(4)), 0, 8);
|
|
||||||
|
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 = new Category();
|
||||||
$category->setName(self::TEST_CATEGORY_PREFIX.$suffix);
|
$category->setName($name);
|
||||||
$category->setCategoryType($this->createCategoryType($typeCode));
|
$category->setCode($effectiveCode);
|
||||||
|
$category->setCategoryType($this->clientCategoryType());
|
||||||
$em->persist($category);
|
$em->persist($category);
|
||||||
$em->flush();
|
$em->flush();
|
||||||
|
|
||||||
@@ -81,9 +116,10 @@ abstract class AbstractCommercialApiTestCase extends AbstractApiTestCase
|
|||||||
|
|
||||||
/**
|
/**
|
||||||
* Seede directement un Client en base (sans passer par l'API), pour les
|
* 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();
|
$em = $this->getEm();
|
||||||
$client = new ClientEntity();
|
$client = new ClientEntity();
|
||||||
@@ -93,7 +129,7 @@ abstract class AbstractCommercialApiTestCase extends AbstractApiTestCase
|
|||||||
$client->setLastName('Seed');
|
$client->setLastName('Seed');
|
||||||
$client->setPhonePrimary('0102030405');
|
$client->setPhonePrimary('0102030405');
|
||||||
$client->setEmail(strtolower(str_replace(' ', '', $companyName)).'@seed.test');
|
$client->setEmail(strtolower(str_replace(' ', '', $companyName)).'@seed.test');
|
||||||
$client->addCategory($this->createCategory($categoryTypeCode));
|
$client->addCategory($this->createCategory($categoryCode));
|
||||||
$client->setIsArchived($isArchived);
|
$client->setIsArchived($isArchived);
|
||||||
if ($isArchived) {
|
if ($isArchived) {
|
||||||
$client->setArchivedAt(new DateTimeImmutable());
|
$client->setArchivedAt(new DateTimeImmutable());
|
||||||
|
|||||||
@@ -15,8 +15,8 @@ use App\Module\Sites\Domain\Entity\Site;
|
|||||||
* - RG-1.06 / RG-1.07 / RG-1.08 : exclusivite is_prospect vs
|
* - RG-1.06 / RG-1.07 / RG-1.08 : exclusivite is_prospect vs
|
||||||
* is_delivery / is_billing ;
|
* is_delivery / is_billing ;
|
||||||
* - RG-1.11 : billing_email obligatoire ssi is_billing ;
|
* - RG-1.11 : billing_email obligatoire ssi is_billing ;
|
||||||
* - RG-1.29 : seules les categories de type SECTEUR / AUTRE sont autorisees sur
|
* - RG-1.29 (ERP-78) : les categories de code DISTRIBUTEUR / COURTIER sont
|
||||||
* une adresse (DISTRIBUTEUR / COURTIER -> 422).
|
* interdites sur une adresse (-> 422) ; toute autre categorie est acceptee.
|
||||||
*
|
*
|
||||||
* Depuis ERP-76, ces regles sont portees par des Assert\Callback sur l'entite
|
* Depuis ERP-76, ces regles sont portees par des Assert\Callback sur l'entite
|
||||||
* ClientAddress (mirror applicatif des CHECK Postgres) : la combinaison invalide
|
* ClientAddress (mirror applicatif des CHECK Postgres) : la combinaison invalide
|
||||||
|
|||||||
@@ -4,6 +4,11 @@ declare(strict_types=1);
|
|||||||
|
|
||||||
namespace App\Tests\Module\Commercial\Api;
|
namespace App\Tests\Module\Commercial\Api;
|
||||||
|
|
||||||
|
use ApiPlatform\Symfony\Bundle\Test\Client;
|
||||||
|
use App\Module\Commercial\Domain\Entity\Client as ClientEntity;
|
||||||
|
use App\Module\Commercial\Domain\Entity\ClientAddress;
|
||||||
|
use App\Module\Sites\Domain\Entity\Site;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Tests fonctionnels de l'API /api/clients (M1) — branche ERP-55.
|
* Tests fonctionnels de l'API /api/clients (M1) — branche ERP-55.
|
||||||
*
|
*
|
||||||
@@ -175,6 +180,49 @@ final class ClientApiTest extends AbstractCommercialApiTestCase
|
|||||||
self::assertResponseStatusCodeSame(201);
|
self::assertResponseStatusCodeSame(201);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public function testPostBrokerReferencingNonBrokerReturns422(): void
|
||||||
|
{
|
||||||
|
$client = $this->createAdminClient();
|
||||||
|
$cat = $this->createCategory('SECTEUR');
|
||||||
|
$notBroker = $this->seedClient('Pas Un Courtier', false, 'SECTEUR');
|
||||||
|
|
||||||
|
$client->request('POST', '/api/clients', [
|
||||||
|
'headers' => ['Content-Type' => self::LD],
|
||||||
|
'json' => [
|
||||||
|
'companyName' => 'Bad Broker Ref',
|
||||||
|
'firstName' => 'A',
|
||||||
|
'phonePrimary' => '0102030405',
|
||||||
|
'email' => 'badbroker@test.fr',
|
||||||
|
'categories' => ['/api/categories/'.$cat->getId()],
|
||||||
|
'broker' => '/api/clients/'.$notBroker->getId(),
|
||||||
|
],
|
||||||
|
]);
|
||||||
|
|
||||||
|
// RG-1.03 (le broker doit porter la categorie de code COURTIER)
|
||||||
|
self::assertResponseStatusCodeSame(422);
|
||||||
|
}
|
||||||
|
|
||||||
|
public function testPostValidBrokerReturns201(): void
|
||||||
|
{
|
||||||
|
$client = $this->createAdminClient();
|
||||||
|
$cat = $this->createCategory('SECTEUR');
|
||||||
|
$broker = $this->seedClient('Vrai Courtier', false, 'COURTIER');
|
||||||
|
|
||||||
|
$client->request('POST', '/api/clients', [
|
||||||
|
'headers' => ['Content-Type' => self::LD],
|
||||||
|
'json' => [
|
||||||
|
'companyName' => 'Client Avec Courtier',
|
||||||
|
'firstName' => 'A',
|
||||||
|
'phonePrimary' => '0102030405',
|
||||||
|
'email' => 'okbroker@test.fr',
|
||||||
|
'categories' => ['/api/categories/'.$cat->getId()],
|
||||||
|
'broker' => '/api/clients/'.$broker->getId(),
|
||||||
|
],
|
||||||
|
]);
|
||||||
|
|
||||||
|
self::assertResponseStatusCodeSame(201);
|
||||||
|
}
|
||||||
|
|
||||||
public function testListSortedByCompanyNameAscAndExcludesArchived(): void
|
public function testListSortedByCompanyNameAscAndExcludesArchived(): void
|
||||||
{
|
{
|
||||||
$client = $this->createAdminClient();
|
$client = $this->createAdminClient();
|
||||||
@@ -282,4 +330,146 @@ final class ClientApiTest extends AbstractCommercialApiTestCase
|
|||||||
self::assertArrayHasKey('addresses', $data);
|
self::assertArrayHasKey('addresses', $data);
|
||||||
self::assertArrayHasKey('ribs', $data);
|
self::assertArrayHasKey('ribs', $data);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* ERP-62 : la LISTE doit alimenter les colonnes « Catégories » (codes) et
|
||||||
|
* « Site(s) » (badges name + color) du Repertoire. On verifie donc que la
|
||||||
|
* collection embarque le `code` de chaque categorie et les sites agreges des
|
||||||
|
* adresses (accessoire Client::getSites()).
|
||||||
|
*/
|
||||||
|
public function testListEmbedsCategoryCodesAndAggregatedSites(): void
|
||||||
|
{
|
||||||
|
$client = $this->createAdminClient();
|
||||||
|
|
||||||
|
// Client seede + une adresse rattachee a un site (fixtures Sites).
|
||||||
|
$seed = $this->seedClient('Embed List Co', false, 'DISTRIBUTEUR');
|
||||||
|
$em = $this->getEm();
|
||||||
|
$site = $em->getRepository(Site::class)->findOneBy([]);
|
||||||
|
self::assertNotNull($site, 'Aucun site seede : impossible de tester la colonne Site(s).');
|
||||||
|
|
||||||
|
$address = new ClientAddress();
|
||||||
|
$address->setClient($seed);
|
||||||
|
$address->setPostalCode('86100');
|
||||||
|
$address->setCity('Châtellerault');
|
||||||
|
$address->setStreet('1 rue du Test');
|
||||||
|
$address->addSite($site);
|
||||||
|
$em->persist($address);
|
||||||
|
$em->flush();
|
||||||
|
|
||||||
|
$member = $client->request('GET', '/api/clients?pagination=false', [
|
||||||
|
'headers' => ['Accept' => self::LD],
|
||||||
|
])->toArray()['member'];
|
||||||
|
|
||||||
|
$row = null;
|
||||||
|
foreach ($member as $candidate) {
|
||||||
|
if ('EMBED LIST CO' === $candidate['companyName']) {
|
||||||
|
$row = $candidate;
|
||||||
|
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
self::assertNotNull($row, 'Le client seede doit figurer dans la liste.');
|
||||||
|
|
||||||
|
// Colonne « Catégories » : chaque categorie embarquee porte son code.
|
||||||
|
self::assertNotEmpty($row['categories']);
|
||||||
|
self::assertArrayHasKey('code', $row['categories'][0]);
|
||||||
|
self::assertSame('DISTRIBUTEUR', $row['categories'][0]['code']);
|
||||||
|
|
||||||
|
// Colonne « Site(s) » : sites agreges des adresses, avec name + color.
|
||||||
|
self::assertArrayHasKey('sites', $row);
|
||||||
|
self::assertNotEmpty($row['sites']);
|
||||||
|
self::assertArrayHasKey('name', $row['sites'][0]);
|
||||||
|
self::assertArrayHasKey('color', $row['sites'][0]);
|
||||||
|
self::assertSame($site->getName(), $row['sites'][0]['name']);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* ERP-62 (drawer) : filtre Catégories multi (?categoryCode[]=A&categoryCode[]=B)
|
||||||
|
* — union des clients possedant l'un OU l'autre code.
|
||||||
|
*/
|
||||||
|
public function testListFilterByMultipleCategoryCodes(): void
|
||||||
|
{
|
||||||
|
$client = $this->createAdminClient();
|
||||||
|
$this->seedClient('Filtre Distrib Co', false, 'DISTRIBUTEUR');
|
||||||
|
$this->seedClient('Filtre Courtier Co', false, 'COURTIER');
|
||||||
|
$this->seedClient('Filtre Secteur Co', false, 'SECTEUR');
|
||||||
|
|
||||||
|
$names = $this->companyNames($client, '/api/clients?pagination=false&categoryCode[]=DISTRIBUTEUR&categoryCode[]=COURTIER');
|
||||||
|
|
||||||
|
self::assertContains('FILTRE DISTRIB CO', $names);
|
||||||
|
self::assertContains('FILTRE COURTIER CO', $names);
|
||||||
|
self::assertNotContains('FILTRE SECTEUR CO', $names);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* ERP-62 (drawer) : filtre Sites (?siteId[]=X) — clients ayant >= 1 adresse
|
||||||
|
* rattachee au site donne.
|
||||||
|
*/
|
||||||
|
public function testListFilterBySite(): void
|
||||||
|
{
|
||||||
|
$client = $this->createAdminClient();
|
||||||
|
$em = $this->getEm();
|
||||||
|
|
||||||
|
$sites = $em->getRepository(Site::class)->findBy([], null, 2);
|
||||||
|
self::assertCount(2, $sites, 'Deux sites seedes requis pour ce test.');
|
||||||
|
[$siteA, $siteB] = $sites;
|
||||||
|
|
||||||
|
$onSiteA = $this->seedClient('Client Sur Site A');
|
||||||
|
$this->attachAddressWithSite($onSiteA, $siteA);
|
||||||
|
|
||||||
|
$onSiteB = $this->seedClient('Client Sur Site B');
|
||||||
|
$this->attachAddressWithSite($onSiteB, $siteB);
|
||||||
|
|
||||||
|
$names = $this->companyNames($client, '/api/clients?pagination=false&siteId[]='.$siteA->getId());
|
||||||
|
|
||||||
|
self::assertContains('CLIENT SUR SITE A', $names);
|
||||||
|
self::assertNotContains('CLIENT SUR SITE B', $names);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* ERP-62 (drawer) : statut « Archivés » (?archivedOnly=true) — uniquement les
|
||||||
|
* archives, contrairement a includeArchived qui ajoute les archives aux actifs.
|
||||||
|
*/
|
||||||
|
public function testListArchivedOnlyReturnsOnlyArchived(): void
|
||||||
|
{
|
||||||
|
$client = $this->createAdminClient();
|
||||||
|
$this->seedClient('Actif Visible Co');
|
||||||
|
$this->seedClient('Archive Visible Co', true);
|
||||||
|
|
||||||
|
$names = $this->companyNames($client, '/api/clients?pagination=false&archivedOnly=true');
|
||||||
|
|
||||||
|
self::assertContains('ARCHIVE VISIBLE CO', $names);
|
||||||
|
self::assertNotContains('ACTIF VISIBLE CO', $names);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Rattache une adresse minimale portant un site au client (les sites vivent
|
||||||
|
* sur les adresses, RG-1.10).
|
||||||
|
*/
|
||||||
|
private function attachAddressWithSite(ClientEntity $client, Site $site): void
|
||||||
|
{
|
||||||
|
$em = $this->getEm();
|
||||||
|
$address = new ClientAddress();
|
||||||
|
$address->setClient($client);
|
||||||
|
$address->setPostalCode('86100');
|
||||||
|
$address->setCity('Châtellerault');
|
||||||
|
$address->setStreet('1 rue du Test');
|
||||||
|
$address->addSite($site);
|
||||||
|
$em->persist($address);
|
||||||
|
$em->flush();
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Helper : recupere les companyName d'une collection /api/clients.
|
||||||
|
*
|
||||||
|
* @return list<string>
|
||||||
|
*/
|
||||||
|
private function companyNames(Client $client, string $url): array
|
||||||
|
{
|
||||||
|
$members = $client->request('GET', $url, [
|
||||||
|
'headers' => ['Accept' => self::LD],
|
||||||
|
])->toArray()['member'];
|
||||||
|
|
||||||
|
return array_map(static fn (array $c): string => $c['companyName'], $members);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -74,14 +74,14 @@ final class ClientExportControllerTest extends AbstractCommercialApiTestCase
|
|||||||
self::assertNotContains('OTHER BETA', $names);
|
self::assertNotContains('OTHER BETA', $names);
|
||||||
}
|
}
|
||||||
|
|
||||||
public function testExportRespectsCategoryTypeFilter(): void
|
public function testExportRespectsCategoryCodeFilter(): void
|
||||||
{
|
{
|
||||||
$client = $this->createAdminClient();
|
$client = $this->createAdminClient();
|
||||||
$this->seedClient('Distrib Co', false, 'DISTRIBUTEUR');
|
$this->seedClient('Distrib Co', false, 'DISTRIBUTEUR');
|
||||||
$this->seedClient('Secteur Co', false, 'SECTEUR');
|
$this->seedClient('Secteur Co', false, 'SECTEUR');
|
||||||
|
|
||||||
$names = $this->companyNames(
|
$names = $this->companyNames(
|
||||||
$client->request('GET', self::EXPORT_URL.'?categoryType=DISTRIBUTEUR')->getContent(),
|
$client->request('GET', self::EXPORT_URL.'?categoryCode=DISTRIBUTEUR')->getContent(),
|
||||||
);
|
);
|
||||||
|
|
||||||
self::assertContains('DISTRIB CO', $names);
|
self::assertContains('DISTRIB CO', $names);
|
||||||
|
|||||||
@@ -0,0 +1,281 @@
|
|||||||
|
<?php
|
||||||
|
|
||||||
|
declare(strict_types=1);
|
||||||
|
|
||||||
|
namespace App\Tests\Module\Commercial\Api;
|
||||||
|
|
||||||
|
use App\Module\Commercial\Domain\Entity\Client as ClientEntity;
|
||||||
|
use App\Module\Commercial\Domain\Entity\ClientAddress;
|
||||||
|
use App\Module\Commercial\Domain\Entity\ClientContact;
|
||||||
|
use App\Module\Commercial\Domain\Entity\ClientRib;
|
||||||
|
use App\Module\Sites\Domain\Entity\Site;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Tests anti-regression du CONTRAT DE SERIALISATION du repertoire clients (M1).
|
||||||
|
*
|
||||||
|
* Captures reelles du 02/06/2026 (cf. docs/specs/M2-suppliers/spec-back.md
|
||||||
|
* § 4.0.ter) ayant revele 4 bugs silencieux du contrat (aucune erreur levee) :
|
||||||
|
* - #81 : booleens d'adresse (isProspect/isDelivery/isBilling) absents du JSON
|
||||||
|
* (Groups sur la propriete `isX`, getter `isX()` derivant l'attribut `x`).
|
||||||
|
* - #80 : fuite RIB (IBAN/BIC) vers un user sans accounting.view.
|
||||||
|
* - #82 : code/libelle de Category et Site non embarques (stub IRI nu).
|
||||||
|
* - enveloppe AP4 : member/totalItems/view sans prefixe `hydra:`, archives exclus.
|
||||||
|
*
|
||||||
|
* REGLE D'OR : ces tests assertent sur le CORPS JSON reel, jamais sur les
|
||||||
|
* annotations. Toute regression de groupe de serialisation casse ici.
|
||||||
|
*
|
||||||
|
* Limite connue (dependance module Sites) : l'entite Site ne porte PAS de champ
|
||||||
|
* `code` (ni SiteInterface) — son libelle est `name`. Les « codes 86/17/82 » de
|
||||||
|
* la spec M2 correspondent en realite au prefixe du code postal des 3 sites
|
||||||
|
* fixtures (86100/17400/82400). On asserte donc le libelle `name` du site
|
||||||
|
* embarque ; l'ajout d'un `Site.code` reste un ticket cote module Sites.
|
||||||
|
*
|
||||||
|
* @internal
|
||||||
|
*/
|
||||||
|
final class ClientSerializationContractTest extends AbstractCommercialApiTestCase
|
||||||
|
{
|
||||||
|
private const string LD = 'application/ld+json';
|
||||||
|
private const string VALID_IBAN = 'FR1420041010050500013M02606';
|
||||||
|
private const string VALID_BIC = 'BNPAFRPPXXX';
|
||||||
|
|
||||||
|
// === #81 — Booleens d'adresse presents dans le JSON ===
|
||||||
|
|
||||||
|
public function testAddressBooleansArePresentInDetail(): void
|
||||||
|
{
|
||||||
|
$this->skipIfSitesModuleDisabled();
|
||||||
|
|
||||||
|
$seed = $this->seedCompleteClient('Bool Addr Co');
|
||||||
|
$id = $seed->getId();
|
||||||
|
|
||||||
|
$http = $this->createAdminClient();
|
||||||
|
$data = $http->request('GET', '/api/clients/'.$id, ['headers' => ['Accept' => self::LD]])->toArray();
|
||||||
|
|
||||||
|
self::assertArrayHasKey('addresses', $data);
|
||||||
|
self::assertNotEmpty($data['addresses']);
|
||||||
|
$address = $data['addresses'][0];
|
||||||
|
|
||||||
|
// Le bug droppait TOTALEMENT ces cles. Apres correctif (Groups +
|
||||||
|
// SerializedName sur le getter), elles sont presentes ET typees bool.
|
||||||
|
self::assertArrayHasKey('isProspect', $address);
|
||||||
|
self::assertArrayHasKey('isDelivery', $address);
|
||||||
|
self::assertArrayHasKey('isBilling', $address);
|
||||||
|
|
||||||
|
// L'adresse seedee est livraison + facturation (prospect exclusif, RG-1.06).
|
||||||
|
// Prouve qu'un booleen `true` est bien serialise (le bug masquait meme les true).
|
||||||
|
self::assertFalse($address['isProspect']);
|
||||||
|
self::assertTrue($address['isDelivery']);
|
||||||
|
self::assertTrue($address['isBilling']);
|
||||||
|
}
|
||||||
|
|
||||||
|
// === #80 — Gating des RIB par accounting.view ===
|
||||||
|
|
||||||
|
public function testRibsPresentForAdminWithAccountingView(): void
|
||||||
|
{
|
||||||
|
$this->skipIfSitesModuleDisabled();
|
||||||
|
|
||||||
|
$seed = $this->seedCompleteClient('Rib Admin Co');
|
||||||
|
$id = $seed->getId();
|
||||||
|
|
||||||
|
$http = $this->createAdminClient();
|
||||||
|
$data = $http->request('GET', '/api/clients/'.$id, ['headers' => ['Accept' => self::LD]])->toArray();
|
||||||
|
|
||||||
|
// Admin bypass RBAC -> accounting.view -> RIB embarques (label/bic/iban).
|
||||||
|
self::assertArrayHasKey('ribs', $data);
|
||||||
|
self::assertNotEmpty($data['ribs']);
|
||||||
|
self::assertSame('Compte principal', $data['ribs'][0]['label']);
|
||||||
|
self::assertSame(self::VALID_IBAN, $data['ribs'][0]['iban']);
|
||||||
|
self::assertSame(self::VALID_BIC, $data['ribs'][0]['bic']);
|
||||||
|
}
|
||||||
|
|
||||||
|
public function testRibsAbsentForUserWithoutAccountingView(): void
|
||||||
|
{
|
||||||
|
$this->skipIfSitesModuleDisabled();
|
||||||
|
|
||||||
|
$seed = $this->seedCompleteClient('Rib Commerciale Co');
|
||||||
|
$id = $seed->getId();
|
||||||
|
|
||||||
|
// Commerciale : commercial.clients.view SANS accounting.view.
|
||||||
|
$creds = $this->createUserWithPermission('commercial.clients.view');
|
||||||
|
$http = $this->authenticatedClient($creds['username'], $creds['password']);
|
||||||
|
|
||||||
|
$data = $http->request('GET', '/api/clients/'.$id, ['headers' => ['Accept' => self::LD]])->toArray();
|
||||||
|
|
||||||
|
// La cle `ribs` est ABSENTE (pas null) : le groupe client:read:accounting
|
||||||
|
// n'est pas ajoute au contexte -> getRibs() jamais serialise. Fin de la
|
||||||
|
// fuite IBAN/BIC.
|
||||||
|
self::assertArrayNotHasKey('ribs', $data);
|
||||||
|
}
|
||||||
|
|
||||||
|
// === #80.bis — Gating par OMISSION des scalaires comptables ===
|
||||||
|
|
||||||
|
public function testAccountingScalarsGatedByOmission(): void
|
||||||
|
{
|
||||||
|
$this->skipIfSitesModuleDisabled();
|
||||||
|
|
||||||
|
$seed = $this->seedCompleteClient('Compta Gating Co');
|
||||||
|
$id = $seed->getId();
|
||||||
|
|
||||||
|
// Admin : scalaires comptables presents.
|
||||||
|
$admin = $this->createAdminClient();
|
||||||
|
$adminData = $admin->request('GET', '/api/clients/'.$id, ['headers' => ['Accept' => self::LD]])->toArray();
|
||||||
|
self::assertArrayHasKey('siren', $adminData);
|
||||||
|
self::assertSame('123456789', $adminData['siren']);
|
||||||
|
self::assertArrayHasKey('accountNumber', $adminData);
|
||||||
|
|
||||||
|
// Commerciale : scalaires comptables ABSENTS (omission, pas null).
|
||||||
|
$creds = $this->createUserWithPermission('commercial.clients.view');
|
||||||
|
$http = $this->authenticatedClient($creds['username'], $creds['password']);
|
||||||
|
$data = $http->request('GET', '/api/clients/'.$id, ['headers' => ['Accept' => self::LD]])->toArray();
|
||||||
|
|
||||||
|
self::assertArrayNotHasKey('siren', $data);
|
||||||
|
self::assertArrayNotHasKey('accountNumber', $data);
|
||||||
|
self::assertArrayNotHasKey('nTva', $data);
|
||||||
|
self::assertArrayNotHasKey('ribs', $data);
|
||||||
|
}
|
||||||
|
|
||||||
|
// === #82 — Embed code/libelle des Category et Site ===
|
||||||
|
|
||||||
|
public function testCategoriesEmbedCodeAndLabel(): void
|
||||||
|
{
|
||||||
|
$this->skipIfSitesModuleDisabled();
|
||||||
|
|
||||||
|
$seed = $this->seedCompleteClient('Embed Cat Co');
|
||||||
|
$id = $seed->getId();
|
||||||
|
|
||||||
|
$http = $this->createAdminClient();
|
||||||
|
$data = $http->request('GET', '/api/clients/'.$id, ['headers' => ['Accept' => self::LD]])->toArray();
|
||||||
|
|
||||||
|
self::assertNotEmpty($data['categories']);
|
||||||
|
$category = $data['categories'][0];
|
||||||
|
// Avant correctif : seuls @id/@type/createdAt/updatedAt (category:read
|
||||||
|
// absent du contexte). Apres : code + name (libelle) embarques.
|
||||||
|
self::assertArrayHasKey('code', $category);
|
||||||
|
self::assertArrayHasKey('name', $category);
|
||||||
|
self::assertNotSame('', $category['code']);
|
||||||
|
}
|
||||||
|
|
||||||
|
public function testAddressSitesEmbedLabel(): void
|
||||||
|
{
|
||||||
|
$this->skipIfSitesModuleDisabled();
|
||||||
|
|
||||||
|
$seed = $this->seedCompleteClient('Embed Site Co');
|
||||||
|
$id = $seed->getId();
|
||||||
|
|
||||||
|
$http = $this->createAdminClient();
|
||||||
|
$data = $http->request('GET', '/api/clients/'.$id, ['headers' => ['Accept' => self::LD]])->toArray();
|
||||||
|
|
||||||
|
$address = $data['addresses'][0];
|
||||||
|
self::assertArrayHasKey('sites', $address);
|
||||||
|
self::assertNotEmpty($address['sites']);
|
||||||
|
// Site embarque : libelle `name` present (avant : stub @id/@type nu).
|
||||||
|
// NB : Site n'a pas de champ `code` (cf. note de classe) -> on asserte name.
|
||||||
|
self::assertArrayHasKey('name', $address['sites'][0]);
|
||||||
|
self::assertNotSame('', $address['sites'][0]['name']);
|
||||||
|
|
||||||
|
// L'adresse seedee est multi-sites : preuve que l'embed parcourt la collection.
|
||||||
|
self::assertGreaterThanOrEqual(2, count($address['sites']));
|
||||||
|
|
||||||
|
// Categories d'adresse : code embarque (category:read dans le contexte).
|
||||||
|
self::assertArrayHasKey('categories', $address);
|
||||||
|
self::assertNotEmpty($address['categories']);
|
||||||
|
self::assertArrayHasKey('code', $address['categories'][0]);
|
||||||
|
}
|
||||||
|
|
||||||
|
// === Enveloppe AP4 (sans prefixe hydra:) + exclusion des archives ===
|
||||||
|
|
||||||
|
public function testCollectionEnvelopeShapeAndArchivedExcluded(): void
|
||||||
|
{
|
||||||
|
$http = $this->createAdminClient();
|
||||||
|
$prefix = 'EnvCheck'.substr(bin2hex(random_bytes(3)), 0, 6);
|
||||||
|
|
||||||
|
$this->seedClient($prefix.' Active');
|
||||||
|
$this->seedClient($prefix.' Archived', true);
|
||||||
|
|
||||||
|
// Liste par defaut filtree sur le prefixe : enveloppe member/totalItems
|
||||||
|
// sans prefixe hydra:, archive EXCLU du totalItems (RG-1.24).
|
||||||
|
$default = $http->request('GET', '/api/clients?search='.$prefix, ['headers' => ['Accept' => self::LD]])->toArray();
|
||||||
|
|
||||||
|
self::assertArrayHasKey('member', $default);
|
||||||
|
self::assertArrayHasKey('totalItems', $default);
|
||||||
|
self::assertArrayNotHasKey('hydra:member', $default);
|
||||||
|
self::assertArrayNotHasKey('hydra:totalItems', $default);
|
||||||
|
self::assertSame(1, $default['totalItems'], 'Archive exclu du totalItems par defaut.');
|
||||||
|
|
||||||
|
// includeArchived : l'archive reintegre le total.
|
||||||
|
$all = $http->request('GET', '/api/clients?search='.$prefix.'&includeArchived=true', ['headers' => ['Accept' => self::LD]])->toArray();
|
||||||
|
self::assertSame(2, $all['totalItems']);
|
||||||
|
|
||||||
|
// `view` (PartialCollectionView) sans prefixe hydra: : force le multi-page
|
||||||
|
// via itemsPerPage=1 sur les 2 resultats archives inclus.
|
||||||
|
$paged = $http->request('GET', '/api/clients?search='.$prefix.'&includeArchived=true&itemsPerPage=1', ['headers' => ['Accept' => self::LD]])->toArray();
|
||||||
|
self::assertArrayHasKey('view', $paged);
|
||||||
|
self::assertArrayNotHasKey('hydra:view', $paged);
|
||||||
|
}
|
||||||
|
|
||||||
|
// === Helper ===
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Seede un client COMPLET (sans passer par l'API, validations applicatives
|
||||||
|
* non rejouees mais CHECK BDD respectes) : bloc comptable non nul, >= 1 RIB,
|
||||||
|
* >= 1 adresse multi-sites avec categories, >= 1 contact, >= 1 categorie.
|
||||||
|
*
|
||||||
|
* L'adresse est livraison + facturation (prospect exclusif, RG-1.06 ; email
|
||||||
|
* de facturation present, RG-1.11) afin de poser des booleens `true`
|
||||||
|
* serialisables tout en respectant les CHECK Postgres.
|
||||||
|
*/
|
||||||
|
private function seedCompleteClient(string $companyName): ClientEntity
|
||||||
|
{
|
||||||
|
$em = $this->getEm();
|
||||||
|
|
||||||
|
// Nom unique parmi les actifs (index partiel uq_client_company_name_active).
|
||||||
|
$suffix = substr(bin2hex(random_bytes(3)), 0, 6);
|
||||||
|
|
||||||
|
$client = new ClientEntity();
|
||||||
|
$client->setCompanyName(mb_strtoupper($companyName.' '.$suffix, 'UTF-8'));
|
||||||
|
$client->setLastName('Complet');
|
||||||
|
$client->setPhonePrimary('0102030405');
|
||||||
|
$client->setEmail('complet'.$suffix.'@seed.test');
|
||||||
|
$client->addCategory($this->createCategory('SECTEUR'));
|
||||||
|
// Bloc comptable non nul (gating par omission cote Commerciale).
|
||||||
|
$client->setSiren('123456789');
|
||||||
|
$client->setAccountNumber('C0001');
|
||||||
|
$client->setNTva('FR00123456789');
|
||||||
|
$em->persist($client);
|
||||||
|
|
||||||
|
// >= 2 sites fixtures pour une adresse multi-sites (RG-1.10).
|
||||||
|
$sites = $em->getRepository(Site::class)->findBy([], null, 2);
|
||||||
|
self::assertGreaterThanOrEqual(2, count($sites), 'Au moins 2 sites fixtures requis (SitesFixtures).');
|
||||||
|
|
||||||
|
$address = new ClientAddress();
|
||||||
|
$address->setClient($client);
|
||||||
|
$address->setIsProspect(false);
|
||||||
|
$address->setIsDelivery(true);
|
||||||
|
$address->setIsBilling(true);
|
||||||
|
$address->setBillingEmail('billing'.$suffix.'@seed.test');
|
||||||
|
$address->setPostalCode('86000');
|
||||||
|
$address->setCity('Poitiers');
|
||||||
|
$address->setStreet('12 rue des Acacias');
|
||||||
|
foreach ($sites as $site) {
|
||||||
|
$address->addSite($site);
|
||||||
|
}
|
||||||
|
$address->addCategory($this->createCategory('SECTEUR'));
|
||||||
|
$em->persist($address);
|
||||||
|
|
||||||
|
$rib = new ClientRib();
|
||||||
|
$rib->setClient($client);
|
||||||
|
$rib->setLabel('Compte principal');
|
||||||
|
$rib->setBic(self::VALID_BIC);
|
||||||
|
$rib->setIban(self::VALID_IBAN);
|
||||||
|
$em->persist($rib);
|
||||||
|
|
||||||
|
$contact = new ClientContact();
|
||||||
|
$contact->setClient($client);
|
||||||
|
$contact->setFirstName('Marie');
|
||||||
|
$contact->setLastName('Martin');
|
||||||
|
$em->persist($contact);
|
||||||
|
|
||||||
|
$em->flush();
|
||||||
|
|
||||||
|
return $client;
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -14,6 +14,7 @@ use Doctrine\ORM\Event\PrePersistEventArgs;
|
|||||||
use Doctrine\ORM\Event\PreUpdateEventArgs;
|
use Doctrine\ORM\Event\PreUpdateEventArgs;
|
||||||
use PHPUnit\Framework\TestCase;
|
use PHPUnit\Framework\TestCase;
|
||||||
use Symfony\Bundle\SecurityBundle\Security;
|
use Symfony\Bundle\SecurityBundle\Security;
|
||||||
|
use Symfony\Component\Clock\MockClock;
|
||||||
use Symfony\Component\Security\Core\User\UserInterface;
|
use Symfony\Component\Security\Core\User\UserInterface;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@@ -30,7 +31,7 @@ final class TimestampableBlamableSubscriberTest extends TestCase
|
|||||||
public function testPrePersistWithUser(): void
|
public function testPrePersistWithUser(): void
|
||||||
{
|
{
|
||||||
$user = $this->createStub(UserInterface::class);
|
$user = $this->createStub(UserInterface::class);
|
||||||
$subscriber = new TimestampableBlamableSubscriber($this->securityReturning($user));
|
$subscriber = new TimestampableBlamableSubscriber($this->securityReturning($user), new MockClock());
|
||||||
$entity = new FullAuditableFixture();
|
$entity = new FullAuditableFixture();
|
||||||
|
|
||||||
$subscriber->prePersist($this->prePersistArgs($entity));
|
$subscriber->prePersist($this->prePersistArgs($entity));
|
||||||
@@ -45,7 +46,7 @@ final class TimestampableBlamableSubscriberTest extends TestCase
|
|||||||
|
|
||||||
public function testPrePersistWithoutUser(): void
|
public function testPrePersistWithoutUser(): void
|
||||||
{
|
{
|
||||||
$subscriber = new TimestampableBlamableSubscriber($this->securityReturning(null));
|
$subscriber = new TimestampableBlamableSubscriber($this->securityReturning(null), new MockClock());
|
||||||
$entity = new FullAuditableFixture();
|
$entity = new FullAuditableFixture();
|
||||||
|
|
||||||
$subscriber->prePersist($this->prePersistArgs($entity));
|
$subscriber->prePersist($this->prePersistArgs($entity));
|
||||||
@@ -59,8 +60,13 @@ final class TimestampableBlamableSubscriberTest extends TestCase
|
|||||||
|
|
||||||
public function testPreUpdate(): void
|
public function testPreUpdate(): void
|
||||||
{
|
{
|
||||||
$user = $this->createStub(UserInterface::class);
|
$user = $this->createStub(UserInterface::class);
|
||||||
$subscriber = new TimestampableBlamableSubscriber($this->securityReturning($user));
|
// Horloge figee 1s apres le createdAt simule : updatedAt doit avancer
|
||||||
|
// de facon deterministe, sans dependre de l'heure reelle.
|
||||||
|
$subscriber = new TimestampableBlamableSubscriber(
|
||||||
|
$this->securityReturning($user),
|
||||||
|
new MockClock(new DateTimeImmutable('2020-01-01 10:00:01')),
|
||||||
|
);
|
||||||
|
|
||||||
// On simule une entite deja persistee : createdAt fige dans le passe,
|
// On simule une entite deja persistee : createdAt fige dans le passe,
|
||||||
// createdBy positionne par une creation anterieure.
|
// createdBy positionne par une creation anterieure.
|
||||||
@@ -80,7 +86,7 @@ final class TimestampableBlamableSubscriberTest extends TestCase
|
|||||||
public function testPartialEntityTimestampableOnly(): void
|
public function testPartialEntityTimestampableOnly(): void
|
||||||
{
|
{
|
||||||
$user = $this->createStub(UserInterface::class);
|
$user = $this->createStub(UserInterface::class);
|
||||||
$subscriber = new TimestampableBlamableSubscriber($this->securityReturning($user));
|
$subscriber = new TimestampableBlamableSubscriber($this->securityReturning($user), new MockClock());
|
||||||
$entity = new TimestampableOnlyFixture();
|
$entity = new TimestampableOnlyFixture();
|
||||||
|
|
||||||
// Entite Timestampable mais NON Blamable : seules les dates sont posees,
|
// Entite Timestampable mais NON Blamable : seules les dates sont posees,
|
||||||
|
|||||||
Reference in New Issue
Block a user