ligne audit_log entity_type='commercial.Supplier' * avec l'action et le diff attendus ; * - RIB : `#[Auditable]` SANS `#[AuditIgnore]` sur iban/bic -> ces champs sensibles * DOIVENT apparaitre dans le diff audite (decision § 2.7, miroir M1). * * @internal */ final class SupplierAuditTest extends AbstractSupplierApiTestCase { private const string SUPPLIER_TYPE = 'commercial.Supplier'; private const string RIB_TYPE = 'commercial.SupplierRib'; private ?Connection $auditConnection = null; protected function setUp(): void { parent::setUp(); self::bootKernel(); /** @var Connection $conn */ $conn = self::getContainer()->get('doctrine.dbal.audit_connection'); $this->auditConnection = $conn; } protected function tearDown(): void { if (null !== $this->auditConnection) { $this->auditConnection->close(); } parent::tearDown(); } public function testPostSupplierIsAudited(): void { $admin = $this->createAdminClient(); $cat = $this->supplierCategory('NEGOCIANT'); $created = $admin->request('POST', '/api/suppliers', [ 'headers' => ['Content-Type' => self::LD], 'json' => [ 'companyName' => 'Audit Created Co', 'categories' => ['/api/categories/'.$cat->getId()], ], ])->toArray(); self::assertResponseStatusCodeSame(201); self::assertGreaterThanOrEqual( 1, $this->countAudit(self::SUPPLIER_TYPE, (string) $created['id'], 'create'), 'Un audit_log "create" doit etre genere pour le fournisseur.', ); } public function testPatchSupplierIsAudited(): void { $admin = $this->createAdminClient(); $seed = $this->seedSupplier('Audit Patch Co'); $admin->request('PATCH', '/api/suppliers/'.$seed->getId(), [ 'headers' => ['Content-Type' => self::MERGE], 'json' => ['companyName' => 'Audit Patch Renamed'], ]); self::assertResponseStatusCodeSame(200); self::assertGreaterThanOrEqual( 1, $this->countAudit(self::SUPPLIER_TYPE, (string) $seed->getId(), 'update'), 'Un audit_log "update" doit etre genere pour le PATCH.', ); } public function testArchiveSupplierIsAudited(): void { $admin = $this->createAdminClient(); $seed = $this->seedSupplier('Audit Archive Co'); $admin->request('PATCH', '/api/suppliers/'.$seed->getId(), [ 'headers' => ['Content-Type' => self::MERGE], 'json' => ['isArchived' => true], ]); self::assertResponseStatusCodeSame(200); $rows = $this->auditConnection->fetchAllAssociative( 'SELECT changes FROM audit_log WHERE entity_type = :type AND entity_id = :id AND action = :action ORDER BY performed_at DESC', ['type' => self::SUPPLIER_TYPE, 'id' => (string) $seed->getId(), 'action' => 'update'], ); self::assertGreaterThanOrEqual(1, count($rows)); /** @var array $changes */ $changes = json_decode((string) $rows[0]['changes'], true, flags: JSON_THROW_ON_ERROR); self::assertArrayHasKey('isArchived', $changes, 'Le diff d\'archivage doit tracer isArchived.'); } public function testRibCreateAuditIncludesIbanAndBic(): void { $admin = $this->createAdminClient(); $seed = $this->seedSupplier('Rib Audit Host'); $rib = $admin->request('POST', '/api/suppliers/'.$seed->getId().'/ribs', [ 'headers' => ['Content-Type' => self::LD], 'json' => [ 'label' => 'Compte audite', 'bic' => self::VALID_BIC, 'iban' => self::VALID_IBAN, ], ])->toArray(); self::assertResponseStatusCodeSame(201); $rows = $this->auditConnection->fetchAllAssociative( 'SELECT changes FROM audit_log WHERE entity_type = :type AND entity_id = :id AND action = :action ORDER BY performed_at DESC', ['type' => self::RIB_TYPE, 'id' => (string) $rib['id'], 'action' => 'create'], ); self::assertGreaterThanOrEqual(1, count($rows), 'Un audit_log "create" doit etre genere pour le RIB.'); /** @var array $changes */ $changes = json_decode((string) $rows[0]['changes'], true, flags: JSON_THROW_ON_ERROR); self::assertArrayHasKey('iban', $changes, 'iban doit figurer dans le diff audite (pas d\'AuditIgnore).'); self::assertArrayHasKey('bic', $changes, 'bic doit figurer dans le diff audite (pas d\'AuditIgnore).'); self::assertSame(self::VALID_IBAN, $changes['iban']); self::assertSame(self::VALID_BIC, $changes['bic']); } private function countAudit(string $type, string $id, string $action): int { return (int) $this->auditConnection->fetchOne( 'SELECT COUNT(*) FROM audit_log WHERE entity_type = :type AND entity_id = :id AND action = :action', ['type' => $type, 'id' => $id, 'action' => $action], ); } }