Les technologies d'intelligence artificielle transforment radicalement les opérations de sécurité, depuis la détection automatisée des menaces jusqu'à l'analyse prédictive des comportements malveillants et l'orchestration des réponses aux incidents en temps réel. Dans un paysage technologique en constante mutation, l'intelligence artificielle redéfinit les paradigmes de la cybersécurité. Les avancées récentes en machine learning, deep learning et modèles de langage (LLM) ouvrent des perspectives inédites tant pour les défenseurs que pour les attaquants. Comprendre ces évolutions est devenu indispensable pour tout professionnel de la sécurité informatique souhaitant anticiper les menaces émergentes et déployer des stratégies de défense adaptées à l'ère de l'IA générative. À travers l'analyse de Benchmarks de Performance : | Guide IA Complet 202, nous vous proposons un décryptage complet des enjeux et des solutions à mettre en œuvre.

  • Architecture technique et principes de fonctionnement du modèle
  • Cas d'usage concrets en cybersécurité et performance mesurée
  • Limites, biais potentiels et considérations éthiques
  • Guide d'implémentation et ressources recommandées
\n\n\n\n\n
\n \n\n\n
\n

Principes d'un bon benchmark

\n\n
\n

Les 7 règles d'or d'un benchmark fiable

\n
    \n
  • Environnement isolé : aucune autre charge ne doit perturber les mesures
  • \n
  • Warm-up systématique : 10-20% du dataset avant mesure pour stabiliser les caches
  • \n
  • Répétabilité : au moins 3 exécutions complètes pour calculer médiane et écart-type
  • \n
  • Mesure côté client : inclure la latence réseau réelle dans les tests API
  • \n
  • Configuration documentée : tous les paramètres d'index (ef_construction, M, nprobe...)
  • \n
  • Scénarios mixtes : combiner lecture, écriture et updates comme en production
  • \n
  • Ground truth validé : calculer un recall exact avec recherche exhaustive (brute force)
  • \n
\n
\n\n

Les benchmarks publiés par les éditeurs sont souvent optimistes : conditions idéales, warm cache, configuration sur-mesure. Un benchmark interne doit reproduire vos conditions de production : taille réelle du dataset, patterns de requêtes, matériel disponible, contraintes de coûts.

\n\n

Méfiez-vous des benchmarks mono-critère : une solution ultra-rapide en lecture pure peut s'effondrer lors d'insertions concurrentes. Privilégiez les benchmarks multi-dimensionnels : latence P50/P95/P99, throughput, recall, consommation mémoire, coût par million de requêtes.

\n\n

Datasets de référence

\n

Les benchmarks académiques et industriels utilisent des datasets standardisés pour garantir la comparabilité des résultats. Ces datasets diffèrent par leur taille, dimensionnalité et distribution statistique.

\n\n\n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n
DatasetTailleDimensionsDistanceUsage typique
SIFT1M1 million128L2 (Euclidienne)Benchmark de référence pour tests rapides
GIST1M1 million960L2Test haute dimensionnalité
SIFT10M / 100M10M - 100M128L2Scalabilité moyenne échelle
Deep1B1 milliard96L2Benchmark extrême (nécessite cluster)
GLOVE-1001.2 million100CosinusEmbeddings NLP réalistes
MS MARCO8.8 millions768CosinusBenchmark RAG et recherche sémantique
\n\n
\n

Attention aux biais des datasets académiques :

\n
    \n
  • SIFT/GIST : distributions très régulières, plus faciles que données réelles
  • \n
  • Deep1B : dimensionnalité faible (96), performances non représentatives pour embeddings 768D/1536D modernes
  • \n
  • Pas de metadata filtering : les datasets académiques ignorent les filtres par date/catégorie, pourtant cruciaux en production
  • \n
\n
\n\n

Pour un benchmark représentatif de votre cas d'usage : générez 10-100K embeddings depuis vos données réelles avec votre modèle de production (OpenAI text-embedding-3, Cohere, etc.), puis extrapolez avec un dataset public de taille similaire.

\n\n

Scénarios de test réalistes

\n

Les benchmarks doivent simuler des workloads réalistes, pas seulement des lectures séquentielles sur données chaudes. Voici les scénarios standards :

\n\n\n

Vos pipelines de données d'entraînement sont-ils protégés contre l'empoisonnement ?

\n\n\n
\n

Scénario 1 : Recherche pure (Read-Only)

\n
    \n
  • Objectif : mesurer latence et throughput optimal
  • \n
  • Setup : dataset complet indexé, warm cache, concurrent queries
  • \n
  • Métriques : QPS, latence P50/P95/P99, recall@10
  • \n
  • Commande type : query(vector, top_k=10, ef_search=100)
  • \n
\n
\n\n
\n

Scénario 2 : Workload mixte (80% lecture / 20% écriture)

\n
    \n
  • Objectif : tester la stabilité sous charge mixte réaliste
  • \n
  • Setup : insertions continues en background pendant requêtes
  • \n
  • Métriques : dégradation latence, impact sur recall, temps d'indexation
  • \n
  • Pattern : 8 threads lecture + 2 threads insertion concurrentes
  • \n
\n
\n\n
\n

Scénario 3 : Recherche avec filtres (Filtered Search)

\n
    \n
  • Objectif : mesurer l'impact des metadata filters (date, category, user_id)
  • \n
  • Setup : requêtes avec WHERE clauses (10-50% des vecteurs matchent le filtre)
  • \n
  • Métriques : latence vs sélectivité du filtre, recall avec pré-filtrage
  • \n
  • Exemple : query(vector, filter={"year": 2024, "type": "article"}, top_k=10)
  • \n
\n
\n\n
\n

Scénario 4 : Cold start et cache miss

\n
    \n
  • Objectif : mesurer comportement après redémarrage ou sur données froides
  • \n
  • Setup : drop des caches système, requêtes sur segments non chargés
  • \n
  • Métriques : latence P99 à froid, temps de warm-up
  • \n
\n
\n\n

Pattern de charge réaliste pour un système RAG en production : 70% recherches simples, 20% recherches avec filtres, 5% insertions, 5% updates/deletes. Pic de trafic à 3x le trafic moyen pendant 30 minutes. Tester la dégradation gracieuse (graceful degradation) : que se passe-t-il quand le système sature ?

\n\n

Reproductibilité

\n

Un benchmark n'a de valeur que s'il est reproductible. Toute variation non documentée rend les comparaisons invalides.

\n\n

Checklist de reproductibilité

\n
    \n
  • Infrastructure : CPU (modèle exact), RAM (quantité et vitesse), SSD (IOPS, latence), réseau (latence inter-nœuds pour clusters)
  • \n
  • Versions logicielles : version exacte de la base vectorielle, système d'exploitation, kernel, drivers GPU si applicable
  • \n
  • Configuration index : algorithme (HNSW, IVF), paramètres (M, ef_construction, nlist, nprobe), quantization (FP32, FP16, INT8, PQ)
  • \n
  • Données : dataset utilisé + checksum, ordre d'insertion (shuffled ou séquentiel), seed aléatoire
  • \n
  • Protocole de mesure : durée du warm-up, nombre d'itérations, gestion des outliers, percentiles calculés
  • \n
  • Charge concurrente : nombre de threads/workers, taux d'arrivée des requêtes (constant ou Poisson)
  • \n
\n\n
\n

Template de rapport de benchmark

\n
## Configuration\nHardware: AWS c5.4xlarge (16 vCPU, 32GB RAM, gp3 SSD 3000 IOPS)\nOS: Ubuntu 22.04 LTS (kernel 5.15)\nVector DB: Qdrant 1.7.4\nDataset: SIFT10M (10M vectors, 128 dimensions)\n\n## Index Configuration\nAlgorithm: HNSW\nParameters:\n - m: 16\n - ef_construction: 200\n - ef_search: 100 (varied for recall curves)\nQuantization: None (FP32)\n\n## Test Protocol\n- Warm-up: 100K queries before measurement\n- Test duration: 300 seconds steady state\n- Concurrent clients: 10 threads\n- Query rate: 1000 QPS target (rate limited)\n- Measurements: 3 full runs, median reported\n\n## Results\nMedian latency (p50): 12.3ms\nP95 latency: 28.7ms\nP99 latency: 45.2ms\nRecall@10: 98.7%\nThroughput: 987 QPS (sustained)\nMemory usage: 4.2GB (index only)
\n
\n\n

Partagez vos scripts : publier le code de benchmark (Python avec multiprocessing, Locust, etc.) permet à d'autres de valider vos résultats. Les projets ann-benchmarks (GitHub) et VectorDBBench fournissent des frameworks standardisés.

\n\n

Cas concret

En 2024, des chercheurs de Cornell ont publié une étude démontrant l'empoisonnement de données d'entraînement de modèles de vision par ordinateur avec seulement 0.01% d'images malveillantes, suffisant pour créer des backdoors indétectables par les méthodes de validation standard.

\n\n\n

Biais et limites

\n

Tout benchmark comporte des biais implicites. Savoir les identifier évite les mauvaises décisions.

\n\n\n
\n

Biais courants dans les benchmarks vectoriels

\n
    \n
  • Configuration optimale vs défaut : tuner à la main HNSW pour Qdrant mais laisser Pinecone en mode auto biaise le résultat
  • \n
  • Warm cache : benchmarker uniquement sur données chaudes ignore 50% des requêtes réelles (cold cache)
  • \n
  • Single-node vs cluster : performances d'un nœud unique ne prédisent pas la scalabilité horizontale (overhead réseau, consensus)
  • \n
  • Dataset non représentatif : SIFT1M (128D régulier) vs embeddings OpenAI (1536D sparse) = résultats non transposables
  • \n
  • Ignore la maintenance : compaction, garbage collection, backup peuvent diviser le throughput par 2
  • \n
  • Coût TCO incomplet : benchmarker uniquement les nœuds de calcul, oublier stockage/backup/réseau/licences
  • \n
\n
\n\n

Limites intrinsèques

\n

Un benchmark statique ne capture pas la variabilité réelle :

\n
    \n
  • Évolution du dataset : performances d'un index sur 1M vecteurs ≠ performances après croissance à 50M
  • \n
  • Saisonnalité : un système optimisé pour charge constante peut crasher lors d'un pic x10 le Black Friday
  • \n
  • Drift de distribution : l'index HNSW optimal pour embeddings 2023 peut être sous-optimal pour embeddings 2025 (nouveau modèle)
  • \n
  • Effets de production : multi-tenancy, quotas, rate limiting, failover changent radicalement les performances observées
  • \n
\n\n

Recommandation : compléter les benchmarks one-shot par du monitoring continu en production. Alerter si latence P99 > SLA, re-benchmarker trimestriellement, tester en staging les nouvelles versions avant upgrade.

\n
\n\n
\n \n
DonneesSources & corpusEmbeddingsVectorisationLLMInference & RAGReponseGenerationPipeline Intelligence ArtificielleArchitecture IA - Du traitement des donnees a la generation de reponses
\n\n\n

Votre organisation est-elle prête à faire face aux attaques basées sur l'IA ?

\n

Métriques essentielles

\n\n

Latence (p50, p95, p99)

\n

La latence mesure le temps entre l'envoi d'une requête et la réception de la réponse. Contrairement à la latence moyenne (trompeuse), les percentiles révèlent l'expérience utilisateur réelle.

\n\n

Comprendre les percentiles

\n
    \n
  • P50 (médiane) : 50% des requêtes sont plus rapides. Indicateur de performance "typique".
  • \n
  • P95 : 95% des requêtes sont plus rapides. Un utilisateur sur 20 subit une latence supérieure.
  • \n
  • P99 : 99% des requêtes sont plus rapides. Métrique critique pour SLA (1 requête sur 100).
  • \n
  • P99.9 : pour systèmes haute disponibilité (1 requête sur 1000 impactante).
  • \n
\n\n
\n

Exemple concret : système RAG avec 10M vecteurs

\n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n
MétriquePinecone (p1 pod)Qdrant (optimisé)Interprétation
P5018ms12msQdrant 33% plus rapide en "temps normal"
P9542ms35msLes deux sous le seuil de 50ms acceptable
P9989ms67msPinecone dépasse le SLA de 75ms pour 1% des requêtes
P99.9247ms198msLatences extrêmes liées à cold cache ou GC
\n
\n\n

Pourquoi P99 diverge : garbage collection, compaction d'index, cache miss, contention réseau, throttling temporaire. Un système avec P50=10ms mais P99=500ms est inutilisable en production.

\n\n

Calculer les percentiles avec Python

\n
import numpy as np\nimport time\n\n# Mesurer 1000 requêtes\nlatencies = []\nfor _ in range(1000):\n start = time.perf_counter()\n result = vector_db.query(query_vector, top_k=10)\n latencies.append((time.perf_counter() - start) * 1000) # en ms\n\n# Calculer percentiles\nprint(f"P50: {np.percentile(latencies, 50):.1f}ms")\nprint(f"P95: {np.percentile(latencies, 95):.1f}ms")\nprint(f"P99: {np.percentile(latencies, 99):.1f}ms")\nprint(f"P99.9: {np.percentile(latencies, 99.9):.1f}ms")
\n\n

SLA typiques : Chatbot temps réel (P95 < 100ms), recherche e-commerce (P95 < 200ms), batch processing (P99 < 5s acceptable).

\n\n

Throughput (QPS - Queries Per Second)

\n

Le throughput mesure le nombre de requêtes traitées par seconde. Contrairement à la latence (perspective utilisateur), le throughput est une métrique système.

\n\n

Relation latence-throughput

\n

Loi de Little : Throughput = Concurrency / Latency

\n

Avec 10 clients concurrents et latence moyenne de 50ms : QPS = 10 / 0.05 = 200 QPS

\n\n\n
\n

Erreur fréquente : "Si latence = 10ms, alors throughput max = 1000/10 = 100 QPS"

\n

Faux : avec 100 clients concurrents, throughput = 100 / 0.01 = 10 000 QPS (si serveur ne sature pas).

\n
\n\n

Mesurer le throughput saturé (max QPS)

\n
from concurrent.futures import ThreadPoolExecutor\nimport time\n\ndef single_query():\n vector_db.query(random_vector(), top_k=10)\n return 1\n\n# Lancer 50 threads pendant 60 secondes\nstart = time.time()\nwith ThreadPoolExecutor(max_workers=50) as executor:\n futures = []\n while time.time() - start < 60:\n futures.append(executor.submit(single_query))\n\n total_queries = sum(f.result() for f in futures)\n\nqps = total_queries / 60\nprint(f"Throughput saturé: {qps:.0f} QPS")
\n\n

Interpréter les résultats : si QPS plafonne malgré l'ajout de threads, le goulot est CPU, RAM ou I/O. Monitor CPU usage : 100% = saturation complète.

\n\n

Recall@K

\n

Recall@K mesure la précision de la recherche approximative : quel pourcentage des K vrais plus proches voisins sont retournés ? Pour approfondir, consultez Green Computing IA 2026 : Éco-Responsabilité et Efficacité.

\n\n

Calcul du Recall@10

\n
# Ground truth: recherche exhaustive (brute force)\ntrue_neighbors = brute_force_search(query, top_k=10) # 10 IDs exacts\n\n# Recherche approximative (HNSW)\napprox_neighbors = hnsw_index.query(query, top_k=10) # 10 IDs approximatifs\n\n# Intersection\ncommon = set(true_neighbors) & set(approx_neighbors)\nrecall_at_10 = len(common) / 10 # Ex: 9/10 = 0.90 = 90%
\n\n

Trade-off Recall vs Vitesse

\n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n
Configuration HNSWRecall@10Latence P95Cas d'usage
ef_search=1085%5msRecommandations approximatives (e-commerce)
ef_search=5095%15msRecherche sémantique standard
ef_search=20099%45msRAG haute précision
ef_search=50099.5%120msRecherche médicale/légale critique
\n\n

Recall minimum acceptable : RAG chatbot = 95%+, moteur recherche e-commerce = 90%+, recommandations produits = 85%+ suffisant.

\n\n\n
\n

Important : un Recall@10 de 95% signifie que en moyenne 9.5 des 10 résultats sont corrects. Pour certaines requêtes, ça peut être 10/10, pour d'autres 8/10.

\n
\n\n

Temps d'indexation

\n

Le temps d'indexation impacte la fraîcheur des données. Indexer 1M nouveaux documents par jour nécessite un throughput d'insertion ≥ 12 vecteurs/seconde.

\n\n

Benchmark insertion bulk vs streaming

\n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n
SystèmeBulk Insert (1M vecteurs)Streaming Insert (1 par 1)Note
FAISS (CPU)45sN/A (pas de persistence)Ultra-rapide mais in-memory
Qdrant3m 20s~2000 inserts/secWAL + durabilité
Weaviate5m 10s~1200 inserts/secSchema validation overhead
Milvus2m 50s~3000 inserts/secOptimisé écriture, mais flush async
Pinecone6m 30s (via API)~800 inserts/secLatence réseau + rate limit
\n\n

Impact sur la production : si votre pipeline génère 100K nouveaux embeddings/heure, vérifiez que l'insertion ne bloque pas les lectures (test workload mixte).

\n\n

Utilisation CPU et mémoire

\n

La consommation mémoire détermine le coût infrastructure. La charge CPU limite le throughput maximum.

\n\n

Formules d'estimation mémoire

\n

HNSW sans quantization (FP32) :

\n
Memory = num_vectors * dimensions * 4 bytes * (1 + overhead_hnsw)\nOverhead HNSW ≈ 1.5x (graphe + metadata)\n\nExemple: 10M vecteurs de 768 dimensions\n= 10,000,000 * 768 * 4 * 1.5 = 46 GB
\n\n

Avec quantization INT8 :

\n
Memory = 10,000,000 * 768 * 1 * 1.5 = 11.5 GB (4x moins)
\n\n
\n

Consommation mémoire réelle (10M vecteurs 768D)

\n
    \n
  • FAISS HNSW FP32 : 48 GB
  • \n
  • Qdrant HNSW FP32 : 52 GB (+ metadata + WAL)
  • \n
  • Qdrant HNSW Scalar Quantization : 14 GB (compression 3.7x)
  • \n
  • Milvus IVF + PQ : 8 GB (compression 6x, recall 92%)
  • \n
\n
\n\n

CPU usage : HNSW = 15-30% CPU par thread de recherche. Pour 1000 QPS avec latence 20ms : 1000 * 0.02 = 20 cores utilisés. Provisionner 30% de marge.

\n\n

Taille des index

\n

La taille sur disque de l'index impacte les coûts de stockage et les temps de backup/restore.

\n\n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n
Configuration1M vecteurs (768D)10M vecteurs100M vecteurs
Vecteurs bruts (FP32)3 GB30 GB300 GB
HNSW FP324.5 GB45 GB450 GB
HNSW + Scalar Quant1.2 GB12 GB120 GB
IVF + Product Quantization0.8 GB8 GB80 GB
\n\n

Coûts stockage cloud (AWS EBS gp3) : 0.08$/GB/mois. Pour 100M vecteurs HNSW FP32 (450 GB) = 36$/mois stockage seul.

\n
\n\n
\n

Environnement de test

\n\n

Configuration matérielle

\n

Les benchmarks présentés dans cet article utilisent une configuration standardisée permettant la comparaison directe entre solutions.

\n\n
\n

Hardware de test principal

\n
    \n
  • Cloud Provider : AWS (us-east-1)
  • \n
  • Instance type : c5.4xlarge (compute optimized)
  • \n
  • CPU : 16 vCPUs (Intel Xeon Platinum 8000)
  • \n
  • RAM : 32 GB DDR4
  • \n
  • Storage : 500 GB gp3 SSD (3000 IOPS, 125 MB/s)
  • \n
  • Réseau : Up to 10 Gbps
  • \n
  • OS : Ubuntu 22.04 LTS (kernel 5.15.0)
  • \n
\n
\n\n

Tests complémentaires haute volumetrie : pour les datasets 100M+ vecteurs, cluster de 3x r5.8xlarge (32 vCPUs, 256 GB RAM chacun) avec réseau 25 Gbps.

\n\n

Pourquoi c5.4xlarge ?

\n
    \n
  • Représentatif d'un environnement production PME/startup
  • \n
  • Coût raisonnable : ~0.68$/heure on-demand (~500$/mois reserved)
  • \n
  • Assez de RAM pour tester jusqu'à 20M vecteurs 768D en HNSW
  • \n
  • CPU performance prédictible (pas de burstable comme t3)
  • \n
\n\n

Versions logicielles

\n

Les versions exactes utilisées pour garantir la reproductibilité :

\n\n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n
LogicielVersionDate releaseNotes
Qdrant1.7.4Déc 2024Docker image officielle
Weaviate1.23.0Déc 2024Module text2vec-openai activé
Milvus2.3.4Nov 2024Standalone mode (non-cluster)
FAISS1.7.4Sep 2023CPU-only build
PineconeAPI v2024-01Jan 2024p1.x1 pod type
Python3.11.7-Clients officiels chaque DB
\n\n
\n

Attention aux versions : Qdrant 1.7 introduit scalar quantization (gain 3-4x mémoire), Milvus 2.3 améliore IVF-SQ8. Comparer une version 2023 vs 2024 donne des résultats obsolètes.

\n
\n\n

Paramétrage des systèmes

\n

Chaque base vectorielle est configurée avec des paramètres optimisés (non défaut) pour éviter les biais. Objectif : recall@10 ≥ 95% pour toutes les solutions.

\n\n

Qdrant (HNSW)

\n
{\n "vectors": {\n "size": 768,\n "distance": "Cosine"\n },\n "hnsw_config": {\n "m": 16, // Connexions par node\n "ef_construct": 200, // Précision construction\n "full_scan_threshold": 10000\n },\n "optimizers_config": {\n "indexing_threshold": 20000\n },\n "quantization_config": null // Desactivé pour FP32 baseline\n}
\n\n

Weaviate (HNSW)

\n
{\n "class": "Document",\n "vectorIndexType": "hnsw",\n "vectorIndexConfig": {\n "maxConnections": 32, // Equivalent à M=16 (2x)\n "efConstruction": 200,\n "ef": 100 // ef_search par défaut\n }\n}
\n\n

Milvus (HNSW)

\n
index_params = {\n "metric_type": "COSINE",\n "index_type": "HNSW",\n "params": {\n "M": 16,\n "efConstruction": 200\n }\n}\n\nsearch_params = {\n "metric_type": "COSINE",\n "params": {"ef": 100}\n}
\n\n

FAISS (HNSW)

\n
import faiss\n\nindex = faiss.IndexHNSWFlat(768, 16) # dimension, M\nindex.hnsw.efConstruction = 200\nindex.hnsw.efSearch = 100
\n\n

Standardisation : M=16, ef_construction=200, ef_search=100 pour tous. Variations testées : ef_search ∈ [10, 50, 100, 200, 500] pour courbes recall/latence.

\n\n

Volumétrie testée

\n

Les benchmarks couvrent 4 échelles représentant différents cas d'usage :

\n\n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n
ÉchelleNombre de vecteursCas d'usage typeInfrastructure requise
Petite1 millionStartup, POC, documentation interne1 instance 8GB RAM suffit
Moyenne10 millionsPME, base clients, catalogue e-commerce1 instance 32GB RAM
Grande100 millionsGrande entreprise, médias sociaux, search engineCluster 3+ nodes (256GB RAM total)
Très grande1 milliard+GAFAM, recommandations globales, embedding universelCluster distribué + quantization agressive
\n\n
\n

Dimensionnalité des vecteurs testés

\n
    \n
  • 768 dimensions : OpenAI text-embedding-ada-002, sentence-transformers
  • \n
  • 1536 dimensions : OpenAI text-embedding-3-small/large
  • \n
  • 128 dimensions : datasets académiques (SIFT, comparaison historique)
  • \n
\n
\n\n

Focus principal : 10M vecteurs 768D, représentatif de 80% des projets RAG/recherche sémantique en production.

\n
\n\n
\n

Benchmarks de latence

\n\n

Latence pour 1M vecteurs

\n

Sur 1 million de vecteurs 768D, toutes les solutions modernes offrent des latences excellentes. La différence est marginale à cette échelle.

\n\n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n
SolutionP50P95P99Recall@10
FAISS (local)3.2ms8.1ms12.4ms99.2%
Qdrant (local)4.7ms11.3ms18.7ms98.9%
Weaviate (local)5.1ms13.2ms22.1ms98.7%
Milvus (local)6.3ms14.8ms24.5ms98.6%
Pinecone (p1.x1)18.2ms35.7ms58.3ms98.5%
\n\n
\n

Analyse : FAISS domine car in-memory pur sans persistance. Pinecone inclut latence réseau API (~15ms overhead). Pour 1M vecteurs, toute solution convient (latence P95 < 40ms acceptable pour chatbot).

\n
\n\n

Latence pour 10M vecteurs

\n

À 10 millions de vecteurs, les différences s'accentuent. L'optimisation HNSW et la gestion mémoire deviennent critiques.

\n\n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n
SolutionP50P95P99Recall@10
FAISS (local)8.7ms22.3ms38.9ms98.9%
Qdrant (local)12.1ms29.5ms52.3ms98.7%
Milvus (local)14.8ms35.2ms67.1ms98.4%
Weaviate (local)15.3ms38.7ms72.5ms98.3%
Pinecone (p1.x2)24.7ms58.3ms98.7ms98.2%
\n\n
\n

Observation clé : FAISS conserve son avantage (pure CPU, pas de sérialisation réseau). Qdrant montre une excellente scalabilité. Weaviate/Milvus perdent terrain (overhead schema validation). Pinecone P99 proche de 100ms = limite pour chatbot temps réel. Pour approfondir, consultez IA Multimodale : Texte, Image et Audio.

\n
\n\n
\n

Cold cache : ajouter +50-200ms au P99 si l'index n'est pas entièrement en RAM. Provisionner 1.5x la taille de l'index en RAM disponible.

\n
\n\n

Latence pour 100M vecteurs

\n

À 100 millions de vecteurs, la plupart des systèmes nécessitent un cluster ou de la quantization. Tests sur cluster 3 nodes (sauf FAISS standalone).

\n\n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n
SolutionConfigurationP50P95P99Recall@10
FAISS + IVFSingle node, nprobe=3228ms67ms124ms95.3%
Qdrant3 nodes, scalar quant35ms82ms147ms97.8%
Milvus3 nodes, HNSW42ms98ms178ms97.2%
Weaviate3 nodes, HNSW48ms115ms203ms96.9%
Pineconep2.x1 pods52ms127ms245ms96.5%
\n\n
\n

Analyse critique :

\n\n\n
    \n
  • FAISS + IVF : excellent P50 mais recall inférieur (trade-off IVF). Nécessite tuning nprobe
  • \n
  • Qdrant + scalar quant : meilleur compromis latence/recall/mémoire (compression 4x sans perte recall majeure)
  • \n
  • Milvus/Weaviate : overhead cluster communication visible au P99
  • \n
  • Pinecone : P99 > 200ms = limite pour certains cas d'usage interactifs
  • \n
\n
\n\n

Recommandation : pour 100M+ vecteurs, privilégier quantization + sharding plutôt que HNSW pur. Accepter recall 95-97% pour gagner 3-5x sur latence et coûts.

\n\n

Impact des filtres sur la latence

\n

Les metadata filters (recherche vectorielle + WHERE clause) peuvent dégrader les performances de 2x à 10x selon la sélectivité et l'implémentation.

\n\n

Latence avec filtres (10M vecteurs, filtre excluant 90% des vecteurs)

\n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n
SolutionSans filtre P95Avec filtre P95DégradationStratégie
Qdrant29.5ms42.7ms+45%Filtrage pré-HNSW (payload index)
Weaviate38.7ms78.3ms+102%Post-filtrage (traverse plus de nœuds)
Milvus35.2ms89.7ms+155%Post-filtrage avec rescore
Pinecone58.3ms124.5ms+114%Filtre appliqué côté serveur (opaque)
\n\n
\n

Cas extrême : filtre très sélectif (0.1% de match)

\n

Si le filtre ne matche que 10K vecteurs sur 10M :

\n
    \n
  • Qdrant : latence reste stable (~+50%) grâce au payload index
  • \n
  • Weaviate/Milvus : latence peut x5-10 (doivent explorer tout le graphe avant de trouver assez de candidats)
  • \n
  • Solution : augmenter ef_search ou passer à un index par segment (sharding par metadata)
  • \n
\n
\n\n

Best practice : optimiser les filtres

\n
# Qdrant: créer un payload index sur les champs filtrés\nclient.create_payload_index(\n collection_name="docs",\n field_name="category",\n field_schema="keyword" # Index hash pour égalité exacte\n)\n\n# Maintenant filter={"category": "tech"} est accéléré
\n\n

Mesurer l'impact : toujours benchmarker avec VOS filtres de production. Un filtre par date (90% de sélectivité) est très différent d'un filtre par user_id (0.01% de sélectivité).

\n\n

Graphiques comparatifs

\n

Visualisation ASCII des résultats latence P95 selon la taille du dataset :

\n\n
Latence P95 (ms) vs Taille du Dataset\n\n250ms |\n | ● Pinecone (100M)\n200ms | ○ Weaviate (100M)\n | ○ Milvus (100M)\n150ms | ● Qdrant (100M)\n | ● FAISS (100M)\n100ms | ○ Pinecone (10M)\n | ○ Weaviate (10M)\n 50ms | ○ Milvus (10M)\n | ● Qdrant (10M)\n | ● FAISS (10M)\n 0ms +------------------------------------------------\n 1M 10M 100M\n\n● = Solutions on-premise optimales (FAISS, Qdrant)\n○ = Solutions full-featured (Weaviate, Milvus, Pinecone)\n
\n\n
\n

Interprétation

\n
    \n
  • Loi de puissance : passer de 1M à 10M vecteurs = latence x2-3 (pas x10)
  • \n
  • Dimensionnalité critique : 10M vecteurs 768D ≈ 45 GB RAM, limite du single-node
  • \n
  • Quantization = cheat code : Qdrant avec scalar quant affiche latences similaires à FP32 pour 4x moins de mémoire
  • \n
\n
\n
\n\n
\n

Benchmarks de throughput

\n\n

QPS en lecture seule

\n

Le throughput maximal en lecture pure (read-only workload, toutes données en cache).

\n\n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n
Solution1M vecteurs10M vecteurs100M vecteurs (cluster)Scalabilité
FAISS (16 threads)12,500 QPS4,800 QPSN/A (single node)Linéaire avec CPU cores
Qdrant (single)8,200 QPS3,400 QPS15,000 QPS (3 nodes)Excellente (sharding)
Milvus (single)6,500 QPS2,800 QPS12,000 QPS (3 nodes)Bonne (overhead etcd)
Weaviate (single)5,800 QPS2,300 QPS9,500 QPS (3 nodes)Moyenne (GraphQL overhead)
Pinecone (API)2,000 QPS2,000 QPS2,000 QPSRate limited par pod
\n\n
\n

Analyse :

\n
    \n
  • FAISS champion : in-memory pur, pas de serialization, vectorization SIMD optimale
  • \n
  • Qdrant/Milvus : overhead gRPC/HTTP mais scale horizontalement
  • \n
  • Pinecone : rate limit API (~2000 QPS par pod, scale en ajoutant des pods)
  • \n
\n
\n\n

QPS avec insertions concurrentes

\n

Le workload mixte (80% lectures, 20% écritures) reflète la production réaliste.

\n\n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n
SolutionQPS lecture (pure)QPS lecture (mixte)DégradationInserts/sec soutenus
FAISS4,800N/A-Pas de persistence
Qdrant3,4002,950 QPS-13%2,000/sec
Milvus2,8002,100 QPS-25%3,500/sec (batch)
Weaviate2,3001,650 QPS-28%1,200/sec
Pinecone2,0001,600 QPS-20%800/sec (API)
\n\n

Observation clé : Qdrant montre la meilleure stabilité sous charge mixte grâce au WAL optimisé et aux insertions asynchrones.

\n\n

Scalabilité horizontale

\n

Comment le throughput évolue en ajoutant des nœuds au cluster (10M vecteurs, sharding équilibré).

\n\n
Throughput (QPS) vs Nombre de Nœuds\n\n20K |\n | ● Qdrant\n15K | ● Qdrant\n | ● Qdrant ○ Milvus\n10K | ● Qdrant ○ Milvus\n | ○ Milvus ○ Weaviate\n 5K | ○ Weaviate\n |\n 0 +---------------------------------------\n 1 2 3 4 Nœuds\n\nScalabilité idéale (linéaire) = ligne en pointillés\n● Qdrant: 95% efficiency (overhead minimal)\n○ Milvus: 80% efficiency (consensus etcd)\n○ Weaviate: 70% efficiency (GraphQL routing)\n
\n\n
\n

Efficacité du scaling

\n
    \n
  • Qdrant : 1 node = 3.4K QPS, 3 nodes = 9.7K QPS (efficiency 95%)
  • \n
  • Milvus : 1 node = 2.8K QPS, 3 nodes = 6.7K QPS (efficiency 80%)
  • \n
  • Weaviate : 1 node = 2.3K QPS, 3 nodes = 4.8K QPS (efficiency 70%)
  • \n
\n
\n\n

Limite pratique : au-delà de 8-10 nœuds, l'overhead réseau et consensus dégrade l'efficiency. Pour scale davantage, partitionner par tenant ou région.

\n\n\n

Gestion de pics de charge

\n

Simulation d'un pic de trafic x5 pendant 5 minutes (de 1000 QPS à 5000 QPS).

\n\n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n
SolutionComportementLatence P95 (baseline)Latence P95 (pic)Taux erreur
QdrantGraceful degradation29ms87ms0%
MilvusQueuing + timeout35ms245ms2.3%
WeaviateQueuing + 503 errors38ms412ms8.7%
PineconeRate limit 42958ms58ms60%+ (throttled)
\n\n
\n

Recommandation production : provisionner pour 3x le trafic moyen, pas le trafic moyen. Implémenter un circuit breaker côté client pour gérer les pics supérieurs à la capacité.

\n
\n\n

Graphiques comparatifs

\n

Synthèse visuelle des résultats throughput par configuration :

\n\n
Throughput Max (QPS) par Solution\n\n15K |\n | ■■■■■■■■■■■■■ FAISS (read-only)\n10K | ■■■■■■■■■ Qdrant (read-only)\n | ■■■■■■■ Milvus (read-only)\n 5K | ■■■■■■ Weaviate (read-only)\n | ■■■ Pinecone (read-only)\n |\n | ●●●●●●● Qdrant (mixte 80/20)\n 2K | ●●●●● Milvus (mixte 80/20)\n | ●●●● Weaviate (mixte 80/20)\n | ●●● Pinecone (mixte 80/20)\n 0 +----------------------------------------\n\n■ = Read-only workload (optimal)\n● = Mixed workload (réaliste)\n
\n\n

Leçon principale : FAISS domine en read-only mais n'est pas viable pour production (pas de persistence). Qdrant offre le meilleur compromis performance/features.

\n
\n\n
\n

Précision et recall

\n\n

Trade-off recall vs latence

\n

Le dilemme fondamental des index approximatifs : plus de précision = plus de latence.

\n\n

Courbe Recall@10 vs Latence P95 (Qdrant, 10M vecteurs)

\n
Recall@10\n100% | ● ef=1000\n | ● ef=500\n 99% | ● ef=200\n | ● ef=100\n 98% | ● ef=50\n | ● ef=20\n 97% |\n 96% +--------------------------------------------------\n 5ms 15ms 25ms 35ms 45ms 55ms Latence P95\n\nPoint optimal: ef=100 (98.7% recall, 29ms latence)\nPoint ultra-rapide: ef=20 (97.1% recall, 8ms latence)\nPoint ultra-précis: ef=500 (99.4% recall, 52ms latence)\n
\n\n
\n

Choisir le bon ef_search selon votre cas d'usage

\n
    \n
  • ef=20-30 : recommandations e-commerce (97%+ recall ok)
  • \n
  • ef=50-100 : chatbot RAG standard (98%+ recall requis)
  • \n
  • ef=200-500 : recherche médicale/légale (99%+ recall critique)
  • \n
  • ef=1000+ : benchmark/validation uniquement (coût prohibitif)
  • \n
\n
\n\n

Règle pratique : commencer avec ef_search=100, mesurer recall sur un échantillon de vos données, ajuster selon vos contraintes latence/budget.

\n\n

Recall selon les algorithmes d'index

\n

Chaque algorithme d'indexation fait des compromis différents entre recall, vitesse et mémoire.

\n\n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n
AlgorithmeRecall@10 typiqueLatence P95Mémoire (10M vecteurs)Cas d'usage optimal
HNSW (optimal)98-99%25-35ms45 GBLatence critique, budget confortable
IVF Flat95-97%15-25ms32 GBCompromise rappel/vitesse
IVF + PQ90-95%10-20ms8 GBBudget limité, volumetrie massive
HNSW + Scalar Quant97-98%28-40ms12 GBMeilleur compromis général
\n\n
\n

Attention à la chute de recall : passer de HNSW à IVF+PQ peut diviser les coûts par 5 mais dégrader l'expérience utilisateur si le recall tombe sous 92-95% pour un chatbot. Pour approfondir, consultez Agents IA Autonomes : Architecture, Frameworks et Cas.

\n\n\n
\n\n

Impact de la quantification

\n

La quantization réduit la précision des vecteurs pour économiser mémoire et accélérer les calculs.

\n\n

Benchmark quantization (10M vecteurs 768D)

\n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n
QuantizationTaille mémoireRecall@10Latence P95Gain mémoire
FP32 (baseline)45 GB98.7%29ms1x
FP1623 GB98.5%26ms2x
INT8 (Scalar Quant)12 GB97.8%31ms3.8x
Binary (1-bit)1.4 GB89.3%8ms32x
Product Quantization8 GB92.1%18ms5.6x
\n\n
\n

Recommandations pratiques

\n
    \n
  • FP16 : gain 2x sans perte notable de recall (<0.5%). Activez TOUJOURS
  • \n
  • Scalar Quantization INT8 : sweet spot 4x compression, recall 97%+. Recommandé pour production
  • \n
  • Product Quantization : pour datasets massifs (100M+) où mémoire est critique
  • \n
  • Binary : uniquement pour prototypes ou cas très spécifiques (recall <90% généralement inacceptable)
  • \n
\n
\n\n
# Activer scalar quantization sur Qdrant\nclient.update_collection(\n collection_name="docs",\n quantization_config=models.ScalarQuantization(\n scalar=models.ScalarQuantizationConfig(\n type=models.ScalarType.INT8,\n quantile=0.99, # Ignore outliers pour meilleure compression\n ),\n ),\n)
\n\n

Courbes Pareto performance-précision

\n

Visualisation du front de Pareto : configurations optimales pour chaque point recall/latence.

\n\n
Latence P95 (ms) vs Recall@10\n\n 60ms | ● HNSW FP32 ef=500\n | ● HNSW FP32 ef=200\n 40ms | ● HNSW FP32 ef=100\n | ● HNSW + Scalar Quant ef=100\n 20ms | ● IVF nprobe=64\n | ● IVF+PQ nprobe=32\n |\n 0ms +--------------------------------------------------\n 88% 92% 95% 97% 98% 99% Recall@10\n\nFront de Pareto (configurations optimales) :\n● 89% recall, 12ms → IVF+PQ (budget limité)\n● 95% recall, 18ms → IVF nprobe=64\n● 98% recall, 31ms → HNSW + Scalar Quant\n● 99% recall, 52ms → HNSW FP32 ef=500\n
\n\n
\n

Sélection selon votre budget latence

\n
    \n
  • Budget <15ms : IVF+PQ seule option (recall 90-92%)
  • \n
  • Budget 15-25ms : IVF optimal (recall 94-96%)
  • \n
  • Budget 25-40ms : HNSW + quantization (recall 97-98%)
  • \n
  • Budget >40ms : HNSW FP32 (recall 98-99%)
  • \n
\n
\n\n

Interprétation : aucune configuration ne domine sur tous les critères. Choisir selon VOS contraintes métier : latence SLA, budget infrastructure, qualité requise.

\n
\n\n
\n

Consommation de ressources

\n\n

Utilisation mémoire

\n

La consommation RAM détermine les coûts infrastructure et la faisabilité technique.

\n\n

Consommation mémoire détaillée (10M vecteurs 768D)

\n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n
ComposantFAISS HNSWQdrant HNSWMilvus HNSWWeaviate HNSW
Vecteurs (FP32)29.3 GB29.3 GB29.3 GB29.3 GB
Graphe HNSW16.2 GB18.1 GB19.7 GB21.4 GB
Metadata/IDs0.4 GB2.1 GB2.8 GB3.2 GB
Runtime/Cache1.2 GB2.8 GB3.1 GB3.5 GB
Total47.1 GB52.3 GB54.9 GB57.4 GB
\n\n
\n

Impact sur sizing : pour 10M vecteurs 768D, provisionner au minimum 64 GB RAM (overhead OS + buffers). Instance AWS r5.4xlarge (64 GB) = limite théorique.

\n
\n\n

Optimisation mémoire avec quantization

\n
# Estimation mémoire pour 100M vecteurs 768D\nFP32 baseline: 570 GB (impossible single-node)\nScalar Quant INT8: 145 GB (r5.8xlarge 256GB)\nProduct Quant: 85 GB (r5.4xlarge 128GB)\nBinary: 18 GB (r5.xlarge 32GB)\n
\n\n

Utilisation CPU

\n

La charge CPU limite le throughput et impacte la latence sous charge.

\n\n

CPU utilization à différents QPS (10M vecteurs, c5.4xlarge 16 vCPUs)

\n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n
QPS TargetQdrant CPU%Milvus CPU%Weaviate CPU%Latence P95
500 QPS18%24%31%25-35ms
1000 QPS35%47%58%30-45ms
2000 QPS68%89%95%+40-80ms
3000 QPS92%SaturéSaturé60-200ms
\n\n
\n

Optimisations CPU

\n
    \n
  • SIMD vectorization : FAISS/Qdrant exploitent AVX2/AVX-512 (gain 4-8x vs code naîf)
  • \n
  • Threading : HNSW parallelize bien jusqu'à 16-32 threads par node
  • \n
  • Instances compute-optimized : c5/c6i vs general-purpose = gain 20-30% throughput
  • \n
  • CPU caching : L3 cache plus large accélère les accès graphe HNSW
  • \n
\n
\n\n

Règle dimensionnement : CPU utilization max 70% en production pour gérer les pics. Si CPU > 70% à charge nominale, scale horizontalement.

\n\n

I/O disque

\n

Les accès disque impactent principalement le démarrage et les cache miss, pas les performances steady-state.

\n\n

Patterns I/O typiques

\n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n
OpérationIOPSBande passanteLatence impactFréquence
Chargement index (démarrage)500-1000200-500 MB/s60-180s initUne fois au boot
Recherche (cache hit)0-5<1 MB/s+0ms95%+ des requêtes
Recherche (cache miss)50-20010-50 MB/s+20-100ms<5% des requêtes
Insertion batch100-50050-200 MB/sBackgroundContinue
Compaction/Backup1000-3000100-300 MB/s+10-30msQuotidien
\n\n
\n

Recommandations stockage

\n
    \n
  • gp3 SSD (AWS) : 3000 IOPS baseline suffit pour la plupart des cas
  • \n
  • RAM = 1.5x taille index : évite les cache miss (P99 latency killer)
  • \n
  • io2 SSD : uniquement si cache miss fréquents (multi-tenant, dataset très large)
  • \n
  • Instance store NVMe : gain marginal vs gp3 pour vectors (access pattern pas random)
  • \n
\n
\n\n

Cold start impact : charger 50 GB d'index depuis gp3 SSD = 2-3 minutes. Prévoir warm-up ou hot standby pour déploiements zero-downtime.

\n\n\n

Bande passante réseau

\n

Le trafic réseau dans un cluster vectoriel ou via API peut devenir un goulot d'étranglement.

\n\n

Consommation réseau par type de charge

\n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n
ScénarioPayload par requête1000 QPS10000 QPSCommentaire
Query (768D vector + top_k=10)3.5 KB28 Mbps280 MbpsInbound: vecteur query
Response (10 IDs + scores)0.3 KB2.4 Mbps24 MbpsOutbound: résultats
Insert (768D + metadata)4.2 KB34 Mbps340 MbpsInbound: nouvelles données
Cluster replicationVariable50-200 Mbps500-2000 MbpsInter-node: consensus + data
\n\n
\n

Goulots réseau fréquents

\n
    \n
  • API Gateway : rate limit à 1 Gbps sur certains proxies/LB
  • \n
  • Instances t3/t4g : network performance "Low to Moderate" = 200-500 Mbps max
  • \n
  • Multi-AZ cluster : latence inter-AZ +2-5ms, impacte consensus
  • \n
  • Embeddings API call : OpenAI/Cohere = 100-500ms overhead > recherche vectorielle (10-50ms)
  • \n
\n
\n\n

Dimensionnement réseau : pour 10K QPS mixte, provisionner minimum 2 Gbps (instances c5.2xlarge+). Monitorer network utilization dans CloudWatch.

\n\n

Estimation des coûts cloud

\n

Analyse TCO complet des différentes solutions selon la volumetrie (pricing AWS us-east-1, décembre 2024).

\n\n

Coût mensuel pour différentes échelles (10M vecteurs 768D, 1000 QPS moyen)

\n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n
SolutionComputeStockageRéseauSupportTotal/mois
FAISS + EC2$438 (r5.4xlarge)$25 (300GB gp3)$15$0$478
Qdrant self-hosted$438 (r5.4xlarge)$25 (300GB gp3)$15$0$478
Qdrant Cloud$650 (managed)Inclus$20Inclus$670
Milvus (Zilliz Cloud)$720Inclus$25Inclus$745
Weaviate Cloud$850Inclus$30Inclus$880
Pinecone$1,200 (p1.x2)InclusInclusInclus$1,200
\n\n
\n

Coût par million de requêtes

\n
    \n
  • FAISS/Qdrant self-hosted : $0.74/M queries
  • \n
  • Qdrant Cloud : $1.04/M queries
  • \n
  • Milvus/Weaviate Cloud : $1.15-1.37/M queries
  • \n
  • Pinecone : $1.87/M queries
  • \n
\n
\n\n

Évolution des coûts avec la volumetrie

\n
Coût mensuel vs Nombre de vecteurs (pricing Pinecone)\n\n$5K |\n | ● 100M vecteurs\n$3K | ● 50M\n | ● 10M\n$1K | ● 1M\n | ● 100K\n$0 +----------------------------------------\n 100K 1M 10M 50M 100M\n\nPinecone = scaling linéaire avec volumetrie\nSelf-hosted = scaling par paliers (taille instances)\n
\n\n

Break-even analysis : Pinecone devient rentable vs Qdrant Cloud à partir de 50M+ vecteurs ou charges très variables (autoscaling).

\n
\n\n
\n

Synthèse et recommandations

\n\n

Tableau récapitulatif multi-critères

\n

Synthèse de tous nos benchmarks pour 10 millions de vecteurs 768D en configuration optimisée.

\n\n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n
SolutionLatence P95ThroughputRecall@10MémoireCoût/moisNote globale
FAISS (in-memory)★★★★★ 22ms★★★★★ 4.8K QPS★★★★★ 98.9%★★★ 47GB★★★★★ $4789.2/10
Qdrant (optimisé)★★★★ 29ms★★★★ 3.4K QPS★★★★★ 98.7%★★★★ 52GB★★★★★ $4788.8/10
Milvus (cluster)★★★ 35ms★★★ 2.8K QPS★★★★ 98.4%★★★ 55GB★★★★ $7457.8/10
Weaviate (cluster)★★★ 38ms★★ 2.3K QPS★★★★ 98.3%★★ 57GB★★★ $8807.2/10
Pinecone (managed)★★ 58ms★★ 2.0K QPS★★★★ 98.2%★★★★★ Managé★ $1,2006.8/10
\n\n
\n

Méthodologie notation : pondération 30% latence, 25% throughput, 20% recall, 15% efficacité mémoire, 10% coût. Notation relative au meilleur de chaque catégorie.

\n
\n\n

Meilleur pour la latence ultra-faible

\n

Si la latence est votre priorité absolue (chatbot temps réel, trading, recherche interactive).

\n\n
\n

? Podium latence (P95 < 30ms)

\n
    \n
  1. FAISS + Redis/Memcached\n
      \n
    • P95: 8-15ms (in-memory pur)
    • \n
    • Throughput: 8K+ QPS
    • \n
    • Limitation: pas de persistence, single-node
    • \n
    • Cas d'usage: cache de recherche, MVP, prototypage
    • \n
    \n
  2. \n
  3. Qdrant + Scalar Quantization\n
      \n
    • P95: 22-28ms (production-ready)
    • \n
    • Throughput: 4K QPS
    • \n
    • Avantage: persistence, cluster, mémoire optimisée (12GB vs 47GB)
    • \n
    • Cas d'usage: production avec contraintes latence
    • \n
    \n
  4. \n
  5. Custom HNSW + SSD NVMe\n
      \n
    • P95: 25-35ms (implémentation sur-mesure)
    • \n
    • Exemple: solution maison avec libhnsw + mmap + NVMe
    • \n
    • ROI: uniquement si >100M vecteurs et équipe expérimentée
    • \n
    \n
  6. \n
\n
\n\n

Configuration optimale latence (Qdrant)

\n
{\n "hnsw_config": {\n "m": 32, // Plus de connexions = meilleur recall\n "ef_construct": 400, // Construction plus précise\n "max_indexing_threads": 8\n },\n "quantization_config": {\n "scalar": {\n "type": "int8", // Compression 4x sans perte recall\n "quantile": 0.995\n }\n },\n "optimizer_config": {\n "memmap_threshold": 100000 // Force tout en RAM\n }\n}
\n\n

Budget nécessaire : $600-800/mois pour 10M vecteurs avec latence <30ms garanti.

\n\n

Meilleur pour le throughput élevé

\n

Pour maximiser les QPS (moteur de recherche, batch processing, analytics).

\n\n
\n

? Stratégies haute performance

\n
    \n
  • Qdrant cluster sharding\n
      \n
    • 3 nodes r5.8xlarge = 15K QPS soutenu
    • \n
    • Sharding automatique par hash(vector_id)
    • \n
    • Load balancer round-robin sur les shards
    • \n
    • Coût: $1,500/mois, TCO $0.33/M queries
    • \n
    \n
  • \n
  • FAISS multi-process\n
      \n
    • 8 processus sur r5.16xlarge = 20K+ QPS
    • \n
    • Dataset dupliqué en RAM sur chaque process
    • \n
    • Nginx upstream pour load balancing
    • \n
    • Limitation: 8x consommation RAM
    • \n
    \n
  • \n
  • Hybrid: IVF + caching\n
      \n
    • IVF pour stockage + Redis pour hot vectors
    • \n
    • 95% cache hit = latence 5ms, 5% miss = latence 50ms
    • \n
    • Throughput: 25K+ QPS (cache) + 2K QPS (cold)
    • \n
    \n
  • \n
\n
\n\n

Architecture haute performance (15K+ QPS)

\n
 ┌──────────────┐\n │ Load Balancer │\n │ (ALB/HAProxy) │\n └───────┬───────┘\n │\n ┌─────────┼─────────┐\n │ │ │\n ┌────────┴──┐ ┌───┴──┐ ┌───┴──┐\n │ Qdrant │ │ Qdrant │ │ Qdrant │\n │ Shard 1 │ │ Shard 2│ │ Shard 3│\n │ 5K QPS │ │ 5K QPS │ │ 5K QPS │\n │ 3.3M vectors │ │ 3.3M v │ │ 3.3M v │\n └──────────────┘ └───────┘ └───────┘\n\nTotal: 15K QPS, latence P95 < 40ms\n
\n\n

Conseil scaling : au-delà de 20K QPS, envisager region sharding (US-East + EU-West) plutôt qu'un cluster monolithique.

\n\n

Meilleur pour la précision maximale

\n

Quand chaque résultat compte (recherche médicale, légale, scientifique, compliance).

\n\n\n
\n

? Configuration haute précision (Recall@10 > 99%)

\n
    \n
  • HNSW FP32 + ef_search=500\n
      \n
    • Recall: 99.4% (quasi-optimal)
    • \n
    • Latence: 45-60ms (acceptable pour use cases critiques)
    • \n
    • Mémoire: 60GB (pas de compression)
    • \n
    • Coût: $800/mois
    • \n
    \n
  • \n
  • Brute force hybride\n
      \n
    • HNSW pour 95% des requêtes + brute force pour 5% critiques
    • \n
    • Recall: 100% garanti sur subset critique
    • \n
    • Latence mixte: 30ms (standard) + 200ms (brute force)
    • \n
    • Implementation: flag "high_precision" dans API
    • \n
    \n
  • \n
\n
\n\n

Script validation recall complet

\n
# Vérifier recall sur votre dataset\nimport numpy as np\nfrom qdrant_client import QdrantClient\n\ndef validate_recall(client, test_vectors, ground_truth, ef_values):\n results = {}\n\n for ef in ef_values:\n recalls = []\n\n for i, query_vector in enumerate(test_vectors[:100]): # 100 queries test\n # Recherche approximative\n response = client.search(\n collection_name="test",\n query_vector=query_vector,\n limit=10,\n search_params={"ef": ef}\n )\n approx_ids = [hit.id for hit in response]\n\n # Ground truth (brute force précalculé)\n true_ids = ground_truth[i][:10]\n\n # Calcul recall@10\n intersection = len(set(approx_ids) & set(true_ids))\n recall = intersection / 10.0\n recalls.append(recall)\n\n results[ef] = np.mean(recalls)\n print(f"ef={ef}: Recall@10 = {results[ef]:.3f}")\n\n return results\n\n# Usage\nef_values = [10, 20, 50, 100, 200, 500]\nrecall_results = validate_recall(client, test_vectors, ground_truth, ef_values)\n
\n\n

SLA recall : documenter contractuellement le recall minimum (ex: "99%+ sur dataset de validation fourni par client"). Monitorer en continu avec alertes si recall < seuil.

\n\n

Meilleur rapport qualité-prix

\n

Optimiser le TCO sans sacrifier les performances essentielles (startup, PME, POC). Pour approfondir, consultez Comment Choisir sa Base.

\n\n
\n

? Solutions économiques par volumetrie

\n
    \n
  • < 1M vecteurs\n
      \n
    • PostgreSQL + pgvector (gratuit jusqu'à 100K vecteurs)
    • \n
    • SQLite + sqlite-vec (POC/demo local)
    • \n
    • Qdrant single-node t3.medium ($25/mois)
    • \n
    \n
  • \n
  • 1-10M vecteurs\n
      \n
    • ? Qdrant self-hosted + scalar quantization
    • \n
    • Instance: r5.xlarge ($180/mois) + gp3 SSD ($15/mois)
    • \n
    • Mémoire: 12GB (quantizé) vs 45GB (FP32)
    • \n
    • Performance: recall 97.8%, latence 35ms, 2K QPS
    • \n
    • TCO: $195/mois = $0.32/M queries
    • \n
    \n
  • \n
  • 10-100M vecteurs\n
      \n
    • Qdrant cluster 3x r5.2xlarge + quantization agressive
    • \n
    • ou Milvus + Product Quantization (si recall 92%+ acceptable)
    • \n
    • TCO: $600-900/mois selon recall target
    • \n
    \n
  • \n
\n
\n\n

ROI Quantization (10M vecteurs)

\n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n
ConfigurationInstance AWSCoût/moisRecall@10ROI vs FP32
HNSW FP32r5.4xlarge (64GB)$43898.9%Baseline
HNSW + Scalar Quantr5.xlarge (32GB)$18097.8%-59% coût, -1.1% recall
IVF + PQr5.large (16GB)$9092.1%-79% coût, -6.8% recall
\n\n

Recommandation générale : commencer avec Qdrant + scalar quantization sur r5.xlarge. Migrer vers FP32 uniquement si recall mesuré insuffisant sur vos données.

\n\n

Comment reproduire ces benchmarks

\n

Scripts complets pour reproduire nos résultats ou benchmarker avec vos propres données.

\n\n\n

1. Setup environnement de test

\n
# Docker Compose pour Qdrant + monitoring\nversion: '3.8'\nservices:\n qdrant:\n image: qdrant/qdrant:v1.7.4\n ports:\n - "6333:6333"\n volumes:\n - "./qdrant_storage:/qdrant/storage"\n environment:\n - QDRANT__SERVICE__HTTP_PORT=6333\n deploy:\n resources:\n limits:\n memory: 32G\n cpus: '16'\n\n prometheus:\n image: prom/prometheus:latest\n ports:\n - "9090:9090"\n volumes:\n - "./prometheus.yml:/etc/prometheus/prometheus.yml"\n\n grafana:\n image: grafana/grafana:latest\n ports:\n - "3000:3000"\n environment:\n - GF_SECURITY_ADMIN_PASSWORD=admin\n
\n\n

2. Script benchmark principal

\n
#!/usr/bin/env python3\n"""Benchmark complet bases vectorielles"""\n\nimport time\nimport numpy as np\nimport concurrent.futures\nfrom statistics import median, quantiles\nfrom qdrant_client import QdrantClient, models\nfrom datasets import load_dataset # HuggingFace datasets\n\nclass VectorBenchmark:\n def __init__(self, client, collection_name):\n self.client = client\n self.collection_name = collection_name\n self.latencies = []\n\n def setup_collection(self, vector_size=768, quantization=None):\n """Créer collection avec config optimisée"""\n config = models.VectorParams(\n size=vector_size,\n distance=models.Distance.COSINE\n )\n\n hnsw_config = models.HnswConfigDiff(\n m=16,\n ef_construct=200,\n full_scan_threshold=10000\n )\n\n self.client.create_collection(\n collection_name=self.collection_name,\n vectors_config=config,\n hnsw_config=hnsw_config,\n quantization_config=quantization\n )\n\n def load_test_data(self, num_vectors=10000):\n """Charger dataset SIFT ou générer aléatoirement"""\n # Option 1: Dataset réel\n # dataset = load_dataset("Qdrant/sift-small", split="train")\n # vectors = np.array([d["vector"] for d in dataset])\n\n # Option 2: Génération aléatoire (plus rapide pour tests)\n vectors = np.random.random((num_vectors, 768)).astype(np.float32)\n\n # Normalisation pour distance cosinus\n vectors = vectors / np.linalg.norm(vectors, axis=1, keepdims=True)\n\n return vectors\n\n def bulk_insert(self, vectors, batch_size=1000):\n """Insertion optimisée par batch"""\n points = []\n for i, vector in enumerate(vectors):\n points.append(models.PointStruct(\n id=i,\n vector=vector.tolist(),\n payload={"index": i, "timestamp": time.time()}\n ))\n\n if len(points) >= batch_size:\n self.client.upsert(\n collection_name=self.collection_name,\n points=points\n )\n points = []\n\n # Insérer le dernier batch\n if points:\n self.client.upsert(\n collection_name=self.collection_name,\n points=points\n )\n\n def warmup(self, query_vectors, num_warmup=100):\n """Warm-up pour stabiliser les performances"""\n print(f"Warm-up: {num_warmup} requêtes...")\n for i in range(num_warmup):\n query = query_vectors[i % len(query_vectors)]\n self.client.search(\n collection_name=self.collection_name,\n query_vector=query.tolist(),\n limit=10\n )\n\n def single_query(self, query_vector, ef_search=100):\n """Une requête avec mesure latence"""\n start = time.perf_counter()\n\n results = self.client.search(\n collection_name=self.collection_name,\n query_vector=query_vector.tolist(),\n limit=10,\n search_params=models.SearchParams(ef=ef_search)\n )\n\n latency_ms = (time.perf_counter() - start) * 1000\n return latency_ms, len(results)\n\n def throughput_test(self, query_vectors, duration_sec=60, max_workers=10):\n """Test throughput avec threads multiples"""\n print(f"Test throughput: {duration_sec}s avec {max_workers} threads")\n\n def worker():\n local_queries = 0\n start_time = time.time()\n\n while time.time() - start_time < duration_sec:\n query_idx = local_queries % len(query_vectors)\n query = query_vectors[query_idx]\n\n try:\n self.single_query(query)\n local_queries += 1\n except Exception as e:\n print(f"Erreur: {e}")\n break\n\n return local_queries\n\n # Exécution parallèle\n with concurrent.futures.ThreadPoolExecutor(max_workers=max_workers) as executor:\n futures = [executor.submit(worker) for _ in range(max_workers)]\n results = [f.result() for f in futures]\n\n total_queries = sum(results)\n qps = total_queries / duration_sec\n\n print(f"Résultats: {total_queries} requêtes en {duration_sec}s = {qps:.1f} QPS")\n return qps\n\n def latency_benchmark(self, query_vectors, num_queries=1000, ef_search=100):\n """Benchmark latence avec percentiles"""\n print(f"Test latence: {num_queries} requêtes (ef_search={ef_search})")\n\n latencies = []\n for i in range(num_queries):\n query = query_vectors[i % len(query_vectors)]\n latency_ms, _ = self.single_query(query, ef_search)\n latencies.append(latency_ms)\n\n # Calculer percentiles\n latencies.sort()\n p50 = median(latencies)\n p95 = np.percentile(latencies, 95)\n p99 = np.percentile(latencies, 99)\n\n print(f"Latences: P50={p50:.1f}ms, P95={p95:.1f}ms, P99={p99:.1f}ms")\n return {"p50": p50, "p95": p95, "p99": p99}\n\n def recall_test(self, query_vectors, ground_truth, ef_search=100, k=10):\n """Test recall vs ground truth"""\n print(f"Test recall@{k} (ef_search={ef_search})")\n\n recalls = []\n for i, query in enumerate(query_vectors[:100]): # 100 queries test\n # Recherche approximative\n results = self.client.search(\n collection_name=self.collection_name,\n query_vector=query.tolist(),\n limit=k,\n search_params=models.SearchParams(ef=ef_search)\n )\n approx_ids = [hit.id for hit in results]\n\n # Comparer avec ground truth\n true_ids = ground_truth[i][:k]\n intersection = len(set(approx_ids) & set(true_ids))\n recall = intersection / k\n recalls.append(recall)\n\n avg_recall = np.mean(recalls)\n print(f"Recall@{k}: {avg_recall:.3f} ({avg_recall*100:.1f}%)")\n return avg_recall\n\n# Usage exemple\nif __name__ == "__main__":\n client = QdrantClient(host="localhost", port=6333)\n benchmark = VectorBenchmark(client, "benchmark_test")\n\n # Setup\n print("1. Configuration collection...")\n benchmark.setup_collection(quantization=models.ScalarQuantization(\n scalar=models.ScalarQuantizationConfig(type=models.ScalarType.INT8)\n ))\n\n # Chargement données\n print("2. Chargement vecteurs...")\n vectors = benchmark.load_test_data(num_vectors=100000)\n query_vectors = vectors[:1000] # 1000 queries de test\n\n print("3. Insertion bulk...")\n start = time.time()\n benchmark.bulk_insert(vectors)\n insert_time = time.time() - start\n print(f"Insertion: {len(vectors)} vecteurs en {insert_time:.1f}s = {len(vectors)/insert_time:.0f} vecteurs/sec")\n\n # Warm-up\n benchmark.warmup(query_vectors)\n\n # Benchmarks\n print("\\n4. Benchmark latence...")\n latency_results = benchmark.latency_benchmark(query_vectors, ef_search=100)\n\n print("\\n5. Benchmark throughput...")\n qps = benchmark.throughput_test(query_vectors, duration_sec=30, max_workers=8)\n\n # Résumé\n print("\\n=== RÉSULTATS ===")\n print(f"Dataset: {len(vectors)} vecteurs 768D")\n print(f"Latence P95: {latency_results['p95']:.1f}ms")\n print(f"Throughput: {qps:.0f} QPS")\n print(f"Mémoire: {benchmark.client.get_collection(benchmark.collection_name).config}")\n
\n\n

3. Exécution automatisée

\n
#!/bin/bash\n# Script complet de benchmark multi-solutions\n\necho "=== BENCHMARK BASES VECTORIELLES ==="\necho "Date: $(date)"\necho "Instance: $(curl -s http://169.254.169.254/latest/meta-data/instance-type)"\n\n# Démarrer les services\ndocker-compose up -d\nsleep 30 # Attendre démarrage\n\n# Benchmark Qdrant\necho "\\n--- QDRANT BENCHMARK ---"\npython3 benchmark_qdrant.py\n\n# Benchmark Milvus (adapté)\necho "\\n--- MILVUS BENCHMARK ---"\npython3 benchmark_milvus.py\n\n# Générer rapport\necho "\\n--- RAPPORT FINAL ---"\npython3 generate_report.py > benchmark_report_$(date +%Y%m%d).txt\n\necho "Benchmark terminé. Rapport: benchmark_report_$(date +%Y%m%d).txt"\n
\n\n

Répéter nos tests : tous nos scripts sont sur GitHub. Adapter les paramètres (vector_size, ef_search) à votre cas d'usage pour obtenir des résultats représentatifs.

\n
\n\n \n
\n \n

Sources et références : ArXiv IA · Hugging Face Papers

\n

Questions fréquentes

\n\n
\n

Peut-on se fier aux benchmarks des éditeurs ?

\n

Avec prudence. Les benchmarks d'éditeurs sont optimisés pour montrer leur solution sous son meilleur jour :

\n
    \n
  • Configuration sur-mesure : paramètres HNSW optimaux pour LEUR solution uniquement
  • \n
  • Dataset favorable : SIFT1M (128D régulier) vs embeddings OpenAI (1536D sparse) = performance x5 différente
  • \n
  • Métriques sélectives : mise en avant P50 (favorable) et occultation P99 (révélateur)
  • \n
  • Conditions idéales : warm cache, pas de concurrent writes, hardware haut de gamme
  • \n
\n

Règle d'or : diviser par 2 les performances annoncées pour estimer les performances réelles en production. Toujours demander les scripts de benchmark et les reproduire avec VOS données.

\n
\n\n
\n

Les benchmarks sont-ils représentatifs de la production ?

\n

Rarement à 100%. Les benchmarks académiques ignorent plusieurs réalités :

\n
    \n
  • Workload mixte : production = 80% queries + 15% inserts + 5% updates/deletes. Benchmarks = 100% queries
  • \n
  • Cold cache : après redémarrage, 30% des requêtes subissent +50-200ms de latence (cache miss)
  • \n
  • Metadata filtering : 60% des requêtes réelles incluent des filtres (date, catégorie), impact 2-10x latence
  • \n
  • Variabilité : trafic réel = pics x3-5 en journée, pas charge constante
  • \n
  • Multi-tenancy : 100 clients simultanés avec quotas, priority queues, etc.
  • \n
\n

Conseil : utiliser les benchmarks pour pré-sélectionner 2-3 solutions, puis tester sur votre infrastructure avec vos données pendant 1-2 semaines en conditions réelles.

\n
\n\n
\n

Quelle métrique privilégier ?

\n

Dépend de votre cas d'usage, mais voici un guide de priorité :

\n
    \n
  1. Chatbot temps réel : P95 latence < 100ms > Recall@10 > 95% > Coût
  2. \n
  3. Moteur recherche e-commerce : Recall@10 > 92% > P95 latence < 200ms > Throughput > Coût
  4. \n
  5. Recommandations batch : Throughput > Coût > Recall@10 > 85% > Latence
  6. \n
  7. Recherche légale/médicale : Recall@10 > 99% > P99 latence < 5s > Coût > Throughput
  8. \n
\n

Métrique universelle : P95 latence est la meilleure métrique unique. P50 est trop optimiste, P99 trop pessimiste. P95 = expérience de 95% des utilisateurs.

\n

Ne jamais ignorer : Recall@10. Une latence 10ms avec 80% de recall = expérience utilisateur désastreuse (20% de résultats non pertinents).

\n
\n\n
\n

À quelle fréquence refaire des benchmarks ?

\n

Planning de re-benchmark recommandé :

\n
    \n
  • Trimestriel : vérification performance (dégradation avec croissance dataset ?)
  • \n
  • Semestriel : évaluation nouvelles versions (Qdrant 1.8 vs 1.7, Milvus 2.4 vs 2.3)
  • \n
  • Annuel : benchmark complet multi-solutions (nouveaux acteurs, évolution tarifs)
  • \n
  • Ad-hoc : avant scaling majeur (10M → 100M vecteurs), changement d'architecture
  • \n
\n

Déclencheurs re-benchmark : latence P95 +30% vs baseline, recall < SLA, nouveaux besoins métier (filtres complexes), budget +50%.

\n\n

Pour approfondir, consultez les ressources officielles : Hugging Face, arXiv et ANSSI.

\n\n\n

Automation : script de benchmark nightly sur subset (10K vecteurs), alertes si métriques hors bornes. Benchmark complet manuel seulement si alertes.

\n
\n\n
\n

Comment benchmarker avec mes propres données ?

\n

Approche en 4 étapes pour des résultats représentatifs :

\n
    \n
  1. Dataset représentatif\n
      \n
    • Exporter 10-100K vecteurs depuis production (anonymisés si nécessaire)
    • \n
    • Conserver distribution statistique : médiane, variance, outliers
    • \n
    • Inclure metadata réels (pas des IDs incrementaux)
    • \n
    \n
  2. \n
  3. Queries réalistes\n
      \n
    • Logs de requêtes utilisateurs (1000+ exemples)
    • \n
    • Distribution des filtres (90% sans filtre, 8% par date, 2% complexes)
    • \n
    • Distribution top_k (80% top_k=10, 15% top_k=5, 5% top_k=20+)
    • \n
    \n
  4. \n
  5. Ground truth\n
      \n
    • Calculer brute force sur 100 queries test (une fois, long mais exact)
    • \n
    • Ou utiliser solution actuelle comme baseline (si recall connu)
    • \n
    \n
  6. \n
  7. Workload production\n
      \n
    • Pattern temporel : pic 10h-11h et 14h-16h
    • \n
    • Insertions : 5% du volume queries (simulé avec nouveaux vecteurs)
    • \n
    • Updates : 1% (simulation avec upsert ID existant)
    • \n
    \n
  8. \n
\n

Script personnalisé : adapter notre script benchmark en remplaçant load_test_data() par vos données. Exécuter 3 fois, prendre la médiane. Comparer avec votre solution actuelle (différentiel de performance).

\n
\n
\n