422). Les * deux colonnes scalaires restent en lecture seule (embarquees au detail). * * Audite (#[Auditable]) + Timestampable / Blamable. */ #[ApiResource( operations: [ new Get( security: "is_granted('transport.carriers.view')", normalizationContext: ['groups' => ['carrier:item:read', 'default:read']], ), new Post( uriTemplate: '/carriers/{carrierId}/contacts', uriVariables: [ 'carrierId' => new Link(fromClass: Carrier::class, toProperty: 'carrier'), ], // read:false : pas de stade lecture du parent. Le Link toProperty // resoudrait l'enfant (SELECT CarrierContact ... WHERE carrier = :id) // et casse en NonUniqueResult des >= 2 enfants. Le parent est rattache // manuellement par CarrierContactProcessor::linkParent (404 si absent). read: false, security: "is_granted('transport.carriers.manage')", normalizationContext: ['groups' => ['carrier:item:read', 'default:read']], denormalizationContext: ['groups' => ['carrier:write:contacts']], processor: CarrierContactProcessor::class, ), new Patch( security: "is_granted('transport.carriers.manage')", normalizationContext: ['groups' => ['carrier:item:read', 'default:read']], denormalizationContext: ['groups' => ['carrier:write:contacts']], processor: CarrierContactProcessor::class, ), new Delete( security: "is_granted('transport.carriers.manage')", processor: CarrierContactProcessor::class, ), ], )] #[ORM\Entity] #[ORM\Table(name: 'carrier_contact')] #[ORM\Index(name: 'idx_carrier_contact_carrier', columns: ['carrier_id'])] #[ORM\Index(name: 'idx_carrier_contact_created_by', columns: ['created_by'])] #[ORM\Index(name: 'idx_carrier_contact_updated_by', columns: ['updated_by'])] #[Auditable] class CarrierContact implements TimestampableInterface, BlamableInterface { use TimestampableBlamableTrait; #[ORM\Id] #[ORM\GeneratedValue] #[ORM\Column] #[Groups(['carrier:item:read'])] private ?int $id = null; #[ORM\ManyToOne(targetEntity: Carrier::class, inversedBy: 'contacts')] #[ORM\JoinColumn(name: 'carrier_id', referencedColumnName: 'id', nullable: false, onDelete: 'CASCADE')] private ?Carrier $carrier = null; // RG-4.08 : aucun champ obligatoire isolement (≥ 1 champ rempli, garde // Processor + CHECK BDD). Les colonnes restent nullable au niveau ORM. #[ORM\Column(name: 'first_name', length: 120, nullable: true)] #[Assert\Length(max: 120, maxMessage: 'Le prénom ne peut dépasser {{ limit }} caractères.', normalizer: 'trim')] #[Groups(['carrier:item:read', 'carrier:write:contacts'])] private ?string $firstName = null; #[ORM\Column(name: 'last_name', length: 120, nullable: true)] #[Assert\Length(max: 120, maxMessage: 'Le nom ne peut dépasser {{ limit }} caractères.', normalizer: 'trim')] #[Groups(['carrier:item:read', 'carrier:write:contacts'])] private ?string $lastName = null; #[ORM\Column(name: 'job_title', length: 120, nullable: true)] #[Assert\Length(max: 120, maxMessage: 'La fonction ne peut dépasser {{ limit }} caractères.', normalizer: 'trim')] #[Groups(['carrier:item:read', 'carrier:write:contacts'])] private ?string $jobTitle = null; // Telephones en LECTURE seule : alimentes en ecriture via le tableau virtuel // `phones` (mappe par le CarrierContactProcessor). Pas de groupe write -> pas // de saisie directe (et donc exemptes du miroir Assert\Length, le Processor // borne deja la longueur). #[ORM\Column(name: 'phone_primary', length: 20, nullable: true)] #[Groups(['carrier:item:read'])] private ?string $phonePrimary = null; #[ORM\Column(name: 'phone_secondary', length: 20, nullable: true)] #[Groups(['carrier:item:read'])] private ?string $phoneSecondary = null; #[ORM\Column(length: 180, nullable: true)] #[Assert\Email(message: 'L\'adresse email n\'est pas valide.')] #[Assert\Length(max: 180, maxMessage: 'L\'email ne peut dépasser {{ limit }} caractères.', normalizer: 'trim')] #[Groups(['carrier:item:read', 'carrier:write:contacts'])] private ?string $email = null; /** * Telephones en ecriture (RG-4.08, max 2), NON persiste : le * CarrierContactProcessor normalise chaque numero (RG-4.13) puis le mappe vers * phonePrimary / phoneSecondary. null = non fourni (PATCH partiel : on ne * touche pas aux telephones existants). Un 3e numero -> 422 sur `phones`. * * @var null|list */ #[Groups(['carrier:write:contacts'])] private ?array $phones = null; #[ORM\Column(options: ['default' => 0])] private int $position = 0; public function getId(): ?int { return $this->id; } public function getCarrier(): ?Carrier { return $this->carrier; } public function setCarrier(?Carrier $carrier): static { $this->carrier = $carrier; return $this; } public function getFirstName(): ?string { return $this->firstName; } public function setFirstName(?string $firstName): static { $this->firstName = $firstName; return $this; } public function getLastName(): ?string { return $this->lastName; } public function setLastName(?string $lastName): static { $this->lastName = $lastName; return $this; } public function getJobTitle(): ?string { return $this->jobTitle; } public function setJobTitle(?string $jobTitle): static { $this->jobTitle = $jobTitle; return $this; } public function getPhonePrimary(): ?string { return $this->phonePrimary; } public function setPhonePrimary(?string $phonePrimary): static { $this->phonePrimary = $phonePrimary; return $this; } public function getPhoneSecondary(): ?string { return $this->phoneSecondary; } public function setPhoneSecondary(?string $phoneSecondary): static { $this->phoneSecondary = $phoneSecondary; return $this; } public function getEmail(): ?string { return $this->email; } public function setEmail(?string $email): static { $this->email = $email; return $this; } /** * @return null|list */ public function getPhones(): ?array { return $this->phones; } /** * @param null|list $phones */ public function setPhones(?array $phones): static { $this->phones = $phones; return $this; } public function getPosition(): int { return $this->position; } public function setPosition(int $position): static { $this->position = $position; return $this; } }