Passer au contenu principal
Le client C# officiel pour se connecter à ClickHouse. Le code source du client est disponible dans le dépôt GitHub. Développé à l’origine par Oleg V. Kozlyuk. La bibliothèque propose deux API principales :
  • ClickHouseClient (recommandé) : un client de haut niveau, thread-safe, conçu pour être utilisé comme singleton. Il fournit une API asynchrone simple pour les requêtes et les insertions en masse. C’est le meilleur choix pour la plupart des applications.
  • ADO.NET (ClickHouseDataSource, ClickHouseConnection, ClickHouseCommand) : les abstractions standard de base de données de .NET. Indispensable pour l’intégration avec les ORM (Dapper, Linq2db) et lorsque vous avez besoin de la compatibilité ADO.NET. ClickHouseBulkCopy est une classe utilitaire qui permet d’insérer efficacement des données à l’aide d’une connexion ADO.NET. ClickHouseBulkCopy est obsolète et sera supprimé dans une prochaine version ; utilisez plutôt ClickHouseClient.InsertBinaryAsync.
Les deux API partagent le même pool de connexions HTTP sous-jacent et peuvent être utilisées ensemble dans la même application.

Guide de migration

  1. Mettez à jour votre fichier .csproj avec le nouveau nom du paquet ClickHouse.Driver et la dernière version disponible sur NuGet.
  2. Remplacez dans votre code toutes les références à ClickHouse.Client par ClickHouse.Driver.

Versions de .NET prises en charge

ClickHouse.Driver prend en charge les versions de .NET suivantes :
  • .NET 6.0
  • .NET 8.0
  • .NET 9.0
  • .NET 10.0

Installation

Installez le paquet à partir de NuGet :
dotnet add package ClickHouse.Driver
Ou avec le gestionnaire de packages NuGet :
Install-Package ClickHouse.Driver

Démarrage rapide

using ClickHouse.Driver;

// Create a client (typically as a singleton)
using var client = new ClickHouseClient("Host=my.clickhouse;Protocol=https;Port=8443;Username=user");

// Execute a query
var version = await client.ExecuteScalarAsync("SELECT version()");
Console.WriteLine(version);

Configuration

Il existe deux façons de configurer votre connexion à ClickHouse :
  • Chaîne de connexion : paires clé-valeur séparées par des points-virgules, qui indiquent l’hôte, les informations d’authentification et d’autres options de connexion.
  • Objet ClickHouseClientSettings : objet de configuration fortement typé, qui peut être chargé depuis des fichiers de configuration ou défini dans le code.
Vous trouverez ci-dessous la liste complète de tous les paramètres, de leurs valeurs par défaut et de leurs effets.

Paramètres de connexion

PropriétéTypePar défautClé de chaîne de connexionDescription
Hôtestring"localhost"HostNom d’hôte ou adresse IP du serveur ClickHouse
Portushort8123 (HTTP) / 8443 (HTTPS)PortNuméro de port ; valeur par défaut selon le protocole
Nom d’utilisateurstring"default"UsernameNom d’utilisateur pour l’authentification
Mot de passestring""PasswordMot de passe pour l’authentification
Base de donnéesstring""DatabaseBase de données par défaut ; si vide, utilise celle par défaut du serveur ou de l’utilisateur
Protocolestring"http"ProtocolProtocole de connexion : "http" ou "https"
CheminstringnullPathChemin URL pour les scénarios avec reverse proxy (p. ex., /clickhouse)
Délai d’expirationTimeSpan2 minutesTimeoutDélai d’expiration de l’opération (stocké en secondes dans la chaîne de connexion)

Format des données et sérialisation

PropriétéTypeValeur par défautClé de chaîne de connexionDescription
UseCompressionbooltrueCompressionActiver la compression gzip pour le transfert de données
UseCustomDecimalsbooltrueUseCustomDecimalsUtiliser ClickHouseDecimal pour une précision arbitraire ; si false, utilise decimal de .NET (limite de 128 bits)
ReadStringsAsByteArraysboolfalseReadStringsAsByteArraysLire les colonnes String et FixedString sous forme de byte[] au lieu de string ; utile pour les données binaires
UseFormDataParametersboolfalseUseFormDataParametersEnvoyer les paramètres sous forme de form data au lieu de la query string de l’URL
ParameterTypeResolverIParameterTypeResolvernullRésolveur personnalisé pour la mise en correspondance des types de paramètres de style @ ; voir Mise en correspondance personnalisée des types de paramètres
JsonReadModeJsonReadModeBinaryJsonReadModeMode de renvoi des données JSON : Binary (renvoie JsonObject) ou String (renvoie la chaîne JSON brute)
JsonWriteModeJsonWriteModeStringJsonWriteModeMode d’envoi des données JSON : String (sérialise via JsonSerializer, accepte toutes les entrées) ou Binary (uniquement les POCO enregistrés avec des type hints)

Gestion des sessions

PropriétéTypePar défautClé de chaîne de connexionDescription
UseSessionboolfalseUseSessionActive les sessions avec état ; sérialise les requêtes
SessionIdstringnullSessionIdID de session ; génère automatiquement un GUID si null et si UseSession vaut true
L’option UseSession active la persistance de la session du serveur, ce qui permet d’utiliser des instructions SET et des tables temporaires. Les sessions sont réinitialisées après 60 secondes d’inactivité (délai d’expiration par défaut). La durée de vie de la session peut être prolongée en définissant des paramètres de session via des instructions ClickHouse ou la configuration du serveur.La classe ClickHouseConnection permet normalement un fonctionnement en parallèle (plusieurs threads peuvent exécuter des requêtes simultanément). Toutefois, l’activation de l’option UseSession limite cela à une seule requête active par connexion à un instant donné (il s’agit d’une limitation côté serveur).

Sécurité

PropriétéTypePar défautClé de la chaîne de connexionDescription
SkipServerCertificateValidationboolfalseIgnorer la validation du certificat HTTPS ; à ne pas utiliser en production

Configuration du client HTTP

PropriétéTypePar défautClé de chaîne de connexionDescription
HttpClientHttpClientnullInstance HttpClient personnalisée et préconfigurée
HttpClientFactoryIHttpClientFactorynullFabrique personnalisée pour créer des instances HttpClient
HttpClientNamestringnullNom utilisé par HttpClientFactory pour créer un client spécifique

Journalisation et débogage

PropriétéTypePar défautClé de chaîne de connexionDescription
LoggerFactoryILoggerFactorynullFabrique de loggers pour la journalisation de diagnostic
EnableDebugModeboolfalseActive le traçage réseau .NET (nécessite LoggerFactory avec le niveau défini sur Trace) ; impact significatif sur les performances

Paramètres personnalisés et rôles

PropriétéTypePar défautClé de chaîne de connexionDescription
CustomSettingsIDictionary<string, object>Videpréfixe set_*Paramètres du serveur ClickHouse, voir la note ci-dessous
RolesIReadOnlyList<string>VideRolesRôles ClickHouse séparés par des virgules (par ex. : Roles=admin,reader)
Lorsque vous utilisez une chaîne de connexion pour définir des paramètres personnalisés, utilisez le préfixe set_, par ex. : “set_max_threads=4”. Si vous utilisez un objet ClickHouseClientSettings, n’utilisez pas le préfixe set_.Pour obtenir la liste complète des paramètres disponibles, consultez cette page.

Exemples de chaînes de connexion

Connexion de base

Host=localhost;Port=8123;Username=default;Password=secret;Database=mydb

Avec des paramètres ClickHouse personnalisés

Host=localhost;set_max_threads=4;set_readonly=1;set_max_memory_usage=10000000000

QueryOptions

QueryOptions vous permet de surcharger les paramètres définis au niveau du client pour chaque requête. Toutes les propriétés sont facultatives et ne remplacent les valeurs par défaut du client que lorsqu’elles sont spécifiées.
PropriétéTypeDescription
QueryIdstringIdentifiant de requête personnalisé pour le suivi dans system.query_log ou pour l’annulation
DatabasestringRemplace la base de données par défaut pour cette requête
RolesIReadOnlyList<string>Remplace les rôles du client pour cette requête
CustomSettingsIDictionary<string, object>Paramètres du serveur ClickHouse pour cette requête (par ex. max_threads)
CustomHeadersIDictionary<string, string>En-têtes HTTP supplémentaires pour cette requête
UseSessionbool?Remplace le comportement de la session pour cette requête
SessionIdstringID de session pour cette requête (nécessite UseSession = true)
BearerTokenstringRemplace le jeton d’authentification pour cette requête
ParameterTypeResolverIParameterTypeResolverRemplace le résolveur défini au niveau du client pour la mise en correspondance des types de paramètres de style @ ; voir Mise en correspondance personnalisée des types de paramètres
MaxExecutionTimeTimeSpan?Délai d’expiration côté serveur pour la requête (transmis comme paramètre max_execution_time) ; le serveur annule la requête si cette durée est dépassée
Exemple :
var options = new QueryOptions
{
    QueryId = "report-2024-001",
    Database = "analytics",
    CustomSettings = new Dictionary<string, object>
    {
        { "max_threads", 4 },
        { "max_memory_usage", 10_000_000_000 }
    },
    MaxExecutionTime = TimeSpan.FromMinutes(5)
};

var reader = await client.ExecuteReaderAsync(
    "SELECT * FROM large_table",
    parameters: null,
    options: options
);

InsertOptions

InsertOptions étend QueryOptions avec des paramètres spécifiques aux opérations d’insertion en masse via InsertBinaryAsync.
PropriétéTypePar défautDescription
BatchSizeint100,000Nombre de lignes par lot
MaxDegreeOfParallelismint1Nombre d’envois de lots en parallèle
FormatRowBinaryFormatRowBinaryFormat binaire : RowBinary ou RowBinaryWithDefaults
ColumnTypesIReadOnlyDictionary<string, string>nullNom de colonne → chaîne de type ClickHouse. Ignore la requête de sondage du schéma lorsqu’elle est définie.
UseSchemaCacheboolfalseMet en cache le schéma complet de la table pour chaque paire (base de données, table) pendant la durée de vie du client.
Toutes les propriétés de QueryOptions sont également disponibles dans InsertOptions. Exemple :
var insertOptions = new InsertOptions
{
    BatchSize = 50_000,
    MaxDegreeOfParallelism = 4,
    QueryId = "bulk-import-001"
};

long rowsInserted = await client.InsertBinaryAsync(
    "my_table",
    columns,
    rows,
    insertOptions
);

Ignorer la requête de sondage du schéma

Par défaut, InsertBinaryAsync envoie une requête SELECT ... WHERE 1=0 avant chaque insertion afin d’identifier les types de colonnes. Pour les scénarios à haut débit, vous pouvez supprimer cette surcharge de deux façons : Option 1 : fournir explicitement les types de colonnes Lorsque vous connaissez le schéma de la table à la compilation, transmettez-le directement via ColumnTypes. Aucune requête de schéma n’est alors envoyée :
var options = new InsertOptions
{
    ColumnTypes = new Dictionary<string, string>
    {
        ["id"] = "UInt64",
        ["name"] = "Nullable(String)",
        ["score"] = "Float32",
    },
};

await client.InsertBinaryAsync("my_table", ["id", "name", "score"], rows, options);
Option 2 : Mettre en cache le schéma Lorsque vous effectuez plusieurs insertions dans la même table, définissez UseSchemaCache = true pour n’interroger le schéma qu’une seule fois et le réutiliser lors des insertions suivantes sur la même instance ClickHouseClient :
var options = new InsertOptions { UseSchemaCache = true };

// First call fetches schema from the server
await client.InsertBinaryAsync("my_table", columns, batch1, options);

// Second call reuses cached schema — no extra round-trip
await client.InsertBinaryAsync("my_table", columns, batch2, options);
  • ColumnTypes est prioritaire sur UseSchemaCache. Si les deux sont définis, les types explicites sont utilisés.
  • Le cache de schéma ne détecte pas les modifications apportées via ALTER TABLE. Si vous modifiez le schéma de la table, créez un nouveau ClickHouseClient ou évitez UseSchemaCache pour cette table.
  • Le cache est propre à l’instance ClickHouseClient et indexé par (database, table). Différents sous-ensembles de colonnes d’une même table partagent un schéma mis en cache unique.

ClickHouseClient

ClickHouseClient est l’API recommandée pour interagir avec ClickHouse. Il est thread-safe, conçu pour être utilisé comme singleton et gère en interne le pool de connexions HTTP.

Créer un client

Créez un ClickHouseClient à l’aide d’une chaîne de connexion ou d’un objet ClickHouseClientSettings. Consultez la section Configuration pour connaître les options disponibles. Les informations de votre service ClickHouse Cloud sont disponibles dans la ClickHouse Cloud console. Sélectionnez un service, puis cliquez sur Connect : Choisissez C#. Les détails de connexion s’affichent ci-dessous. Si vous utilisez ClickHouse autogéré, les détails de connexion sont définis par votre administrateur ClickHouse. Avec une chaîne de connexion :
using ClickHouse.Driver;

using var client = new ClickHouseClient("Host=localhost;Username=default;Password=secret");
Ou avec ClickHouseClientSettings :
using ClickHouse.Driver;

var settings = new ClickHouseClientSettings
{
    Host = "localhost",
    Username = "default",
    Password = "secret"
};
using var client = new ClickHouseClient(settings);
Pour les cas d’injection de dépendances, utilisez IHttpClientFactory :
// In your DI configuration
services.AddHttpClient("ClickHouse", client =>
{
    client.Timeout = TimeSpan.FromMinutes(5);
}).ConfigurePrimaryHttpMessageHandler(() => new HttpClientHandler
{
    AutomaticDecompression = DecompressionMethods.GZip | DecompressionMethods.Deflate
});

// Create client with factory
var factory = serviceProvider.GetRequiredService<IHttpClientFactory>();
var client = new ClickHouseClient("Host=localhost", factory, "ClickHouse");
ClickHouseClient est conçu pour être conservé sur la durée et partagé dans toute votre application. Créez-le une seule fois (généralement sous forme de singleton) et réutilisez-le pour toutes les opérations sur la base de données. Le client gère en interne le pool de connexions HTTP.

Exécution des requêtes

Utilisez ExecuteNonQueryAsync pour les instructions qui ne renvoient pas de résultats :
// Create a table
await client.ExecuteNonQueryAsync(
    "CREATE TABLE IF NOT EXISTS default.my_table (id Int64, name String) ENGINE = Memory"
);

// Drop a table
await client.ExecuteNonQueryAsync("DROP TABLE IF EXISTS default.my_table");
Utilisez ExecuteScalarAsync pour récupérer une seule valeur :
var count = await client.ExecuteScalarAsync("SELECT count() FROM default.my_table");
Console.WriteLine($"Row count: {count}");

var version = await client.ExecuteScalarAsync("SELECT version()");
Console.WriteLine($"Server version: {version}");

Insérer des données

Insertions paramétrées

Insérez des données à l’aide de requêtes paramétrées avec ExecuteNonQueryAsync. Les types des paramètres doivent être spécifiés dans le SQL à l’aide de la syntaxe {name:Type} :
using ClickHouse.Driver;
using ClickHouse.Driver.ADO.Parameters;

var parameters = new ClickHouseParameterCollection();
parameters.AddParameter("id", 1L);
parameters.AddParameter("name", "Alice");

await client.ExecuteNonQueryAsync(
    "INSERT INTO default.my_table (id, name) VALUES ({id:Int64}, {name:String})",
    parameters
);

Insertions en masse

Utilisez InsertBinaryAsync pour insérer efficacement un grand nombre de lignes. Il transmet les données en flux à l’aide du format binaire natif de lignes de ClickHouse, prend en charge les envois parallèles par lots et évite les erreurs “URL trop longue” qui peuvent survenir avec les requêtes paramétrées.
// Prepare data as IEnumerable<object[]>
var rows = Enumerable.Range(0, 1_000_000)
    .Select(i => new object[] { (long)i, $"value{i}" });

var columns = new[] { "id", "name" };

// Basic insert
long rowsInserted = await client.InsertBinaryAsync("default.my_table", columns, rows);
Console.WriteLine($"Rows inserted: {rowsInserted}");
Pour les jeux de données volumineux, configurez l’envoi par lots et le parallélisme avec InsertOptions :
var options = new InsertOptions
{
    BatchSize = 100_000,           // Rows per batch (default: 100,000)
    MaxDegreeOfParallelism = 4     // Parallel batch uploads (default: 1)
};
  • Le client récupère automatiquement la structure de la table via SELECT * FROM <table> WHERE 1=0 avant d’insérer les données. Les valeurs fournies doivent correspondre aux types des colonnes cibles. Pour ignorer cette requête, utilisez InsertOptions.ColumnTypes ou InsertOptions.UseSchemaCache.
  • Lorsque MaxDegreeOfParallelism > 1, les batches sont envoyés en parallèle. Les sessions ne sont pas compatibles avec l’insertion en parallèle ; désactivez-les ou définissez MaxDegreeOfParallelism = 1.
  • Utilisez RowBinaryFormat.RowBinaryWithDefaults dans InsertOptions.Format si vous souhaitez que le serveur applique les valeurs DEFAULT aux colonnes non fournies.

Insertion de POCO

Au lieu de construire des tableaux object[], vous pouvez insérer directement des objets POCO fortement typés. Enregistrez le type une seule fois, puis passez IEnumerable<T> :
// Define a POCO matching your table columns
public class SensorReading
{
    public ulong Id { get; set; }
    public string SensorName { get; set; }
    public double Value { get; set; }
    public DateTime Timestamp { get; set; }
}

// Register the type (once per client lifetime)
client.RegisterBinaryInsertType<SensorReading>();

// Insert directly — column names are derived from property names
var readings = Enumerable.Range(0, 100_000)
    .Select(i => new SensorReading
    {
        Id = (ulong)i,
        SensorName = $"sensor_{i % 10}",
        Value = Random.Shared.NextDouble() * 100,
        Timestamp = DateTime.UtcNow,
    });

long rowsInserted = await client.InsertBinaryAsync("sensors", readings);
Par défaut, toutes les propriétés publiques accessibles en lecture sont associées à des colonnes selon une correspondance stricte des noms, sensible à la casse. Vous pouvez personnaliser ce mappage à l’aide d’attributs :
public class Event
{
    [ClickHouseColumn(Name = "event_id")]     // Map to a differently-named column
    public ulong Id { get; set; }

    [ClickHouseColumn(Type = "LowCardinality(String)")]  // Explicit ClickHouse type
    public string Category { get; set; }

    public string Payload { get; set; }

    [ClickHouseNotMapped]                     // Exclude from insert
    public string InternalTag { get; set; }
}
AttributObjectif
[ClickHouseColumn(Name = "...")]Redéfinir le nom de la colonne cible
[ClickHouseColumn(Type = "...")]Déclarer explicitement le type ClickHouse
[ClickHouseNotMapped]Exclure la propriété de l’insertion
Lorsque toutes les propriétés mappées spécifient un Type explicite, la requête de sondage du schéma est entièrement omise. Lorsque seules certaines propriétés ont des types explicites, le pilote revient à la requête de sondage du schéma pour l’ensemble des colonnes. InsertBinaryAsync<T> prend en charge les mêmes InsertOptions (batching, parallélisme, mise en cache du schéma) que la surcharge object[].
Contrairement à la surcharge object[], InsertBinaryAsync<T> n’accepte pas de liste explicite de colonnes. Les colonnes sont déterminées par les propriétés mappées du type enregistré. Pour contrôler les colonnes insérées, utilisez [ClickHouseNotMapped] pour exclure des propriétés ou [ClickHouseColumn(Name = "...")] pour les renommer.Si ColumnTypes est défini dans InsertOptions, elles remplaceront les attributs POCO.

Évolution du schéma

Les insertions de POCO fonctionnent de façon transparente lorsque des colonnes sont ajoutées à la table cible après l’enregistrement du type. Comme le pilote n’insère que les colonnes associées au POCO, toute nouvelle colonne avec DEFAULT (ou d’autres expressions par défaut) est automatiquement remplie par le serveur. Aucune modification du code ni aucun nouvel enregistrement ne sont nécessaires.

Lecture des données

Utilisez ExecuteReaderAsync pour exécuter des requêtes SELECT. Le ClickHouseDataReader renvoyé fournit un accès typé aux colonnes de résultat via des méthodes comme GetInt64(), GetString() et GetFieldValue<T>(). Appelez Read() pour passer à la ligne suivante. Cette méthode renvoie false lorsqu’il n’y a plus de lignes. Accédez aux colonnes par index (à partir de 0) ou par nom de colonne.
using ClickHouse.Driver.ADO.Parameters;

var parameters = new ClickHouseParameterCollection();
parameters.AddParameter("max_id", 100L);

var reader = await client.ExecuteReaderAsync(
    "SELECT * FROM default.my_table WHERE id < {max_id:Int64}",
    parameters
);

while (reader.Read())
{
    Console.WriteLine($"Id: {reader.GetInt64(0)}, Name: {reader.GetString(1)}");
}

Paramètres SQL

Dans ClickHouse, le format standard des paramètres dans les requêtes SQL est {parameter_name:DataType}. Exemples :
SELECT {value:Array(UInt16)} as a
SELECT * FROM table WHERE val = {tuple_in_tuple:Tuple(UInt8, Tuple(String, UInt8))}
INSERT INTO table VALUES ({val1:Int32}, {val2:Array(UInt8)})
Les paramètres SQL ‘bind’ sont transmis sous forme de paramètres de requête dans l’URI HTTP. En utiliser un trop grand nombre peut donc entraîner une exception “URL too long”. Utilisez InsertBinaryAsync pour l’insertion en masse de données afin d’éviter cette limitation.

ID de requête

Chaque requête se voit attribuer un query_id unique, qui peut être utilisé pour extraire des données de la table system.query_log ou annuler des requêtes de longue durée. Vous pouvez spécifier un ID de requête personnalisé via QueryOptions:
var options = new QueryOptions
{
    QueryId = $"report-{Guid.NewGuid()}"
};

var reader = await client.ExecuteReaderAsync(
    "SELECT * FROM large_table",
    parameters: null,
    options: options
);
Si vous définissez un QueryId personnalisé, assurez-vous qu’il soit unique à chaque appel. Un GUID aléatoire est un bon choix.

Correspondance personnalisée des types de paramètres

Lorsque vous utilisez des paramètres de style @ (par exemple, WHERE id = @id), le driver déduit automatiquement le type ClickHouse à partir du type de valeur .NET. Par exemple, int correspond à Int32 et DateTime à DateTime. Pour remplacer ces valeurs par défaut, définissez ParameterTypeResolver dans ClickHouseClientSettings. C’est utile si vous voulez que tous les paramètres DateTime utilisent DateTime64(3) pour une précision à la milliseconde, ou que toutes les valeurs décimales utilisent une échelle spécifique, sans avoir à définir ClickHouseType sur chaque paramètre individuellement. Utilisation de DictionaryParameterTypeResolver pour des correspondances de types simples :
using ClickHouse.Driver.ADO.Parameters;

var settings = new ClickHouseClientSettings("Host=localhost")
{
    ParameterTypeResolver = new DictionaryParameterTypeResolver(new Dictionary<Type, string>
    {
        [typeof(DateTime)] = "DateTime64(3)",
        [typeof(decimal)] = "Decimal64(4)",
    }),
};
using var client = new ClickHouseClient(settings);

var parameters = new ClickHouseParameterCollection();
parameters.AddParameter("dt", DateTime.UtcNow);     // Mapped to DateTime64(3)
parameters.AddParameter("amount", 99.1234m);         // Mapped to Decimal64(4)

await client.ExecuteReaderAsync("SELECT @dt, @amount", parameters);
IParameterTypeResolver personnalisé pour les cas d’usage avancés : Pour une résolution basée sur le nom ou tenant compte de la valeur, implémentez directement l’interface IParameterTypeResolver. Renvoyez null pour utiliser l’inférence par défaut :
public class SmartDecimalResolver : IParameterTypeResolver
{
    public string ResolveType(Type clrType, object value, string parameterName)
    {
        if (clrType != typeof(decimal))
            return null; // Fall through to default

        var scale = (decimal.GetBits((decimal)value)[3] >> 16) & 0x7F;
        return scale <= 4 ? $"Decimal64({scale})" : $"Decimal128({scale})";
    }
}
Vous pouvez également définir un résolveur pour une seule requête via QueryOptions.ParameterTypeResolver. Lorsqu’il est défini, il prévaut sur le résolveur défini au niveau du client. Ordre de priorité pour la résolution des types : Le résolveur s’inscrit lui aussi dans une chaîne de priorité. De la priorité la plus élevée à la plus faible :
  1. ClickHouseType explicite défini sur le paramètre
  2. Indice de type SQL issu de la syntaxe {name:Type} dans la requête
  3. IParameterTypeResolver (via QueryOptions.ParameterTypeResolver, avec repli sur ClickHouseClientSettings.ParameterTypeResolver)
  4. Inférence de type intégrée (TypeConverter.ToClickHouseType)
Le résolveur fonctionne également avec le chemin ClickHouseConnection d’ADO.NET : les paramètres de configuration sont hérités par les connexions créées à partir du client.

Flux brut

Utilisez ExecuteRawResultAsync pour transmettre directement le résultat de la requête dans un format spécifique, sans passer par le lecteur de données. Cela est utile pour exporter des données vers des fichiers ou les acheminer vers d’autres systèmes :
using var result = await client.ExecuteRawResultAsync(
    "SELECT * FROM default.my_table LIMIT 100 FORMAT JSONEachRow"
);

await using var stream = await result.ReadAsStreamAsync();
using var reader = new StreamReader(stream);
var json = await reader.ReadToEndAsync();
Formats courants : JSONEachRow, CSV, TSV, Parquet, Native. Consultez la documentation des formats pour voir toutes les options.

Insertion de flux bruts

Utilisez InsertRawStreamAsync pour insérer des données directement à partir de flux de fichier ou de mémoire, dans des formats comme CSV, JSON, Parquet ou tout format ClickHouse pris en charge. Insérer à partir d’un fichier CSV :
await using var fileStream = File.OpenRead("data.csv");

using var response = await client.InsertRawStreamAsync(
    table: "my_table",
    stream: fileStream,
    format: "CSV",
    columns: ["id", "product", "price"] // Optional: specify columns
);
Consultez la documentation des paramètres de format pour découvrir les options qui permettent de contrôler le comportement de l’ingestion de données.

Autres exemples

Pour d’autres exemples pratiques d’utilisation, consultez le répertoire examples du dépôt GitHub.

ADO.NET

La bibliothèque offre une prise en charge complète d’ADO.NET via ClickHouseConnection, ClickHouseCommand et ClickHouseDataReader. Cette API est nécessaire pour l’intégration avec les ORM (Dapper, Linq2db) et lorsque vous avez besoin des abstractions .NET standard pour les bases de données.

Gestion du cycle de vie avec ClickHouseDataSource

Créez toujours des connexions à partir d’un ClickHouseDataSource afin de garantir une gestion correcte du cycle de vie ainsi qu’un pool de connexions. Le ClickHouseDataSource gère en interne une unique instance de ClickHouseClient, et toutes les connexions partagent son pool de connexions HTTP.
using ClickHouse.Driver.ADO;

// Create DataSource once (register as singleton in DI)
var dataSource = new ClickHouseDataSource("Host=localhost;Username=default;Password=secret");

// Create lightweight connections as needed
await using var connection = await dataSource.OpenConnectionAsync();

// Use the connection
await using var command = connection.CreateCommand("SELECT version()");
var version = await command.ExecuteScalarAsync();
Avec l’injection de dépendances :
// In Startup.cs or Program.cs
services.AddSingleton(sp =>
{
    var factory = sp.GetRequiredService<IHttpClientFactory>();
    return new ClickHouseDataSource("Host=localhost", factory, "ClickHouse");
});

// In your service
public class MyService
{
    private readonly ClickHouseDataSource _dataSource;

    public MyService(ClickHouseDataSource dataSource)
    {
        _dataSource = dataSource;
    }

    public async Task DoWorkAsync()
    {
        await using var connection = await _dataSource.OpenConnectionAsync();
        // Use connection...
    }
}
Ne créez pas ClickHouseConnection directement dans le code de production. Chaque instanciation directe crée un nouveau client HTTP et un nouveau pool de connexions, ce qui peut entraîner un épuisement des sockets en cas de charge :
// NE FAITES PAS CECI - crée un nouveau pool de connexions à chaque fois
using var conn = new ClickHouseConnection("Host=localhost");
await conn.OpenAsync();
Utilisez toujours ClickHouseDataSource à la place, ou partagez une unique instance de ClickHouseClient.

Utilisation de ClickHouseCommand

Créez des commandes à partir d’une connexion pour exécuter des requêtes SQL :
await using var connection = await dataSource.OpenConnectionAsync();

// Create command with SQL
await using var command = connection.CreateCommand("SELECT * FROM my_table WHERE id = {id:Int64}");
command.AddParameter("id", 42L);

// Execute and read results
await using var reader = await command.ExecuteReaderAsync();
while (reader.Read())
{
    Console.WriteLine($"Name: {reader.GetString("name")}");
}
Méthodes de commande :
  • ExecuteNonQueryAsync() - Pour les instructions INSERT, UPDATE, DELETE et DDL
  • ExecuteScalarAsync() - Renvoie la première colonne de la première ligne
  • ExecuteReaderAsync() - Renvoie un ClickHouseDataReader permettant de parcourir les résultats

Utilisation de ClickHouseDataReader

Le ClickHouseDataReader permet un accès typé au résultat de la requête :
await using var reader = await command.ExecuteReaderAsync();

while (reader.Read())
{
    // Access by column index
    var id = reader.GetInt64(0);
    var name = reader.GetString(1);

    // Access by column name
    var email = reader.GetString("email");

    // Generic access
    var timestamp = reader.GetFieldValue<DateTime>("created_at");

    // Check for null
    if (!reader.IsDBNull("optional_field"))
    {
        var value = reader.GetString("optional_field");
    }
}

Bonnes pratiques

Durée de vie des connexions et pool de connexions

ClickHouse.Driver utilise System.Net.Http.HttpClient en interne. HttpClient dispose d’un pool de connexions par endpoint. Par conséquent :
  • Les sessions de base de données sont multiplexées via des connexions HTTP gérées par le pool de connexions.
  • Les connexions HTTP sont automatiquement recyclées par le pool.
  • Les connexions peuvent rester actives après la libération des objets ClickHouseClient ou ClickHouseConnection.
Approches recommandées :
ScénarioApproche recommandée
Usage généralUtilisez un ClickHouseClient singleton
ADO.NET / ORMsUtilisez ClickHouseDataSource (crée des connexions partageant le même pool)
Environnements DIEnregistrez ClickHouseClient ou ClickHouseDataSource comme singleton avec IHttpClientFactory
Si vous utilisez un HttpClient ou un HttpClientFactory personnalisé, assurez-vous que PooledConnectionIdleTimeout est défini sur une valeur inférieure au keep_alive_timeout du serveur, afin d’éviter les erreurs dues à des connexions semi-fermées. La valeur par défaut de keep_alive_timeout pour les déploiements Cloud est de 10 secondes.
Évitez de créer plusieurs instances de ClickHouseClient ou des instances autonomes de ClickHouseConnection sans HttpClient partagé. Chaque instance crée son propre pool de connexions.

Gestion des valeurs DateTime

  1. Utilisez UTC chaque fois que possible. Stockez les horodatages dans des colonnes DateTime('UTC') et utilisez DateTimeKind.Utc dans votre code. Cela élimine toute ambiguïté liée au fuseau horaire.
  2. Utilisez DateTimeOffset pour gérer explicitement le fuseau horaire. Il représente toujours un instant précis et inclut les informations de décalage.
  3. Spécifiez le fuseau horaire dans les annotations de type SQL. Lorsque vous utilisez des paramètres avec des valeurs DateTime Unspecified pour des colonnes non UTC, incluez le fuseau horaire dans le SQL :
    var parameters = new ClickHouseParameterCollection();
    parameters.AddParameter("dt", myDateTime);
    
    await client.ExecuteNonQueryAsync(
        "INSERT INTO table (dt) VALUES ({dt:DateTime('Europe/Amsterdam')})",
        parameters
    );
    

Insertions asynchrones

Les insertions asynchrones transfèrent la responsabilité du batching du client vers le serveur. Au lieu d’exiger un batching côté client, le serveur met les données entrantes en mémoire tampon, puis les écrit dans le stockage en fonction de seuils configurables. Cela est utile dans les scénarios à forte concurrence, comme les workloads d’observability où de nombreux agents envoient de petites charges utiles. Activez les insertions asynchrones via CustomSettings ou la chaîne de connexion :
// Using CustomSettings
var settings = new ClickHouseClientSettings("Host=localhost");
settings.CustomSettings["async_insert"] = 1;
settings.CustomSettings["wait_for_async_insert"] = 1; // Recommended: wait for flush acknowledgment

// Or via connection string
// "Host=localhost;set_async_insert=1;set_wait_for_async_insert=1"
Deux modes (contrôlés par wait_for_async_insert) :
ModeComportementCas d’usage
wait_for_async_insert=1L’insertion renvoie une réponse une fois les données écrites sur le disque. Les erreurs sont renvoyées au client.Recommandé pour la plupart des charges de travail
wait_for_async_insert=0L’insertion renvoie immédiatement lorsque les données sont placées en tampon. Aucune garantie que les données seront persistées.Uniquement si la perte de données est acceptable
Avec wait_for_async_insert=0, les erreurs n’apparaissent qu’au moment de l’écriture sur disque et ne peuvent pas être rattachées à l’insertion d’origine. Le client ne fournit pas non plus de mécanisme de régulation, ce qui risque de surcharger le serveur.
Paramètres clés :
SettingDescription
async_insert_max_data_sizeÉcrit sur disque lorsque le tampon atteint cette taille (octets)
async_insert_busy_timeout_msÉcrit sur disque après ce délai d’expiration (millisecondes)
async_insert_max_query_numberÉcrit sur disque après l’accumulation de ce nombre de requêtes

Sessions

N’activez les sessions que si vous avez besoin de fonctionnalités côté serveur avec état, par exemple :
  • Tables temporaires (CREATE TEMPORARY TABLE)
  • Conservation du contexte de requête d’une instruction à l’autre
  • Paramètres de session (SET max_threads = 4)
Lorsque les sessions sont activées, les requêtes sont sérialisées afin d’éviter l’utilisation concurrente d’une même session. Cela ajoute un surcoût pour les charges de travail qui ne nécessitent pas d’état de session.
var settings = new ClickHouseClientSettings
{
    Host = "localhost",
    UseSession = true,
    SessionId = "my-session", // Optional -- will be auto-generated if not provided
};

using var client = new ClickHouseClient(settings);

await client.ExecuteNonQueryAsync("CREATE TEMPORARY TABLE temp_ids (id UInt64)");
await client.ExecuteNonQueryAsync("INSERT INTO temp_ids VALUES (1), (2), (3)");

var reader = await client.ExecuteReaderAsync(
    "SELECT * FROM users WHERE id IN (SELECT id FROM temp_ids)"
);
Utilisation d’ADO.NET (pour assurer la compatibilité avec les ORM) :
var settings = new ClickHouseClientSettings
{
    Host = "localhost",
    UseSession = true,
    SessionId = "my-session",
};

var dataSource = new ClickHouseDataSource(settings);
await using var connection = await dataSource.OpenConnectionAsync();

await using var cmd1 = connection.CreateCommand("CREATE TEMPORARY TABLE temp_ids (id UInt64)");
await cmd1.ExecuteNonQueryAsync();

await using var cmd2 = connection.CreateCommand("INSERT INTO temp_ids VALUES (1), (2), (3)");
await cmd2.ExecuteNonQueryAsync();

await using var cmd3 = connection.CreateCommand("SELECT * FROM users WHERE id IN (SELECT id FROM temp_ids)");
await using var reader = await cmd3.ExecuteReaderAsync();

Types de données pris en charge

ClickHouse.Driver prend en charge tous les types de données de ClickHouse. Les tableaux ci-dessous présentent les correspondances entre les types ClickHouse et les types .NET natifs lors de la lecture des données depuis la base de données.

Correspondance de types : lecture à partir de ClickHouse

Types d’entiers

Type ClickHouseType .NET
Int8sbyte
UInt8byte
Int16short
UInt16ushort
Int32int
UInt32uint
Int64long
UInt64ulong
Int128BigInteger
UInt128BigInteger
Int256BigInteger
UInt256BigInteger

Types à virgule flottante

Type ClickHouseType .NET
Float32float
Float64double
BFloat16float

Types décimaux

Type ClickHouseType .NET
Decimal(P, S)decimal / ClickHouseDecimal
Decimal32(S)decimal / ClickHouseDecimal
Decimal64(S)decimal / ClickHouseDecimal
Decimal128(S)decimal / ClickHouseDecimal
Decimal256(S)decimal / ClickHouseDecimal
La conversion du type Decimal est gérée par le paramètre UseCustomDecimals.

Type booléen

Type ClickHouseType .NET
Boolbool

Types de chaînes

Type ClickHouseType .NET
Stringstring
FixedString(N)string
Par défaut, les colonnes String et FixedString(N) sont toutes deux renvoyées sous forme de string. Définissez ReadStringsAsByteArrays=true dans votre chaîne de connexion pour les lire sous forme de byte[]. Cela est utile lorsque vous stockez des données binaires qui peuvent ne pas être en UTF-8 valide.

Types de date et d’heure

Type ClickHouseType .NET
DateDateTime
Date32DateTime
DateTimeDateTime
DateTime32DateTime
DateTime64DateTime
TimeTimeSpan
Time64TimeSpan
ClickHouse stocke les valeurs DateTime et DateTime64 en interne sous forme de timestamps Unix (secondes ou fractions de seconde écoulées depuis l’époque). Bien que le stockage soit toujours en UTC, les colonnes peuvent avoir un fuseau horaire associé, qui influe sur la manière dont les valeurs sont affichées et interprétées. Lors de la lecture de valeurs DateTime, la propriété DateTime.Kind est définie en fonction du fuseau horaire de la colonne :
Définition de la colonneDateTime.Kind renvoyéRemarques
DateTime('UTC')UtcFuseau horaire UTC explicite
DateTime('Europe/Amsterdam')UnspecifiedDécalage appliqué
DateTimeUnspecifiedHeure d’horloge conservée telle quelle
Pour les colonnes non UTC, le DateTime renvoyé représente l’heure d’horloge dans ce fuseau horaire. Utilisez ClickHouseDataReader.GetDateTimeOffset() pour obtenir un DateTimeOffset avec le décalage correct pour ce fuseau horaire :
var reader = (ClickHouseDataReader)await connection.ExecuteReaderAsync(
    "SELECT toDateTime('2024-06-15 14:30:00', 'Europe/Amsterdam')");
reader.Read();

var dt = reader.GetDateTime(0);    // 2024-06-15 14:30:00, Kind=Unspecified
var dto = reader.GetDateTimeOffset(0); // 2024-06-15 14:30:00 +02:00 (CEST)
Pour les colonnes sans fuseau horaire explicite (c.-à-d. DateTime au lieu de DateTime('Europe/Amsterdam')), le pilote renvoie un DateTime avec Kind=Unspecified. Cela préserve exactement l’heure telle qu’elle est stockée, sans supposer de fuseau horaire. Si vous avez besoin d’un comportement tenant compte du fuseau horaire pour des colonnes sans fuseau horaire explicite, vous pouvez :
  1. Utiliser des fuseaux horaires explicites dans vos définitions de colonnes : DateTime('UTC') ou DateTime('Europe/Amsterdam')
  2. Appliquer vous-même le fuseau horaire après la lecture.

Type JSON

Type ClickHouseType .NETRemarques
JsonJsonObjectPar défaut (JsonReadMode=Binary)
JsonstringLorsque JsonReadMode=String
Le type de retour des colonnes JSON dépend du paramètre JsonReadMode :
  • Binary (par défaut) : renvoie System.Text.Json.Nodes.JsonObject. Fournit un accès structuré aux données JSON, mais les types ClickHouse spécialisés (comme les adresses IP, les UUID ou les grands nombres décimaux) sont convertis en représentations sous forme de chaîne dans la structure JSON.
  • String : renvoie le JSON brut sous forme de string. Préserve la représentation JSON exacte de ClickHouse, ce qui est utile lorsque vous devez transmettre le JSON sans l’analyser, ou lorsque vous souhaitez gérer vous-même la désérialisation.
// Configure string mode via settings
var settings = new ClickHouseClientSettings("Host=localhost")
{
    JsonReadMode = JsonReadMode.String
};

// Or via connection string
// "Host=localhost;JsonReadMode=String"

Autres types

Type ClickHouseType .NET
UUIDGuid
IPv4IPAddress
IPv6IPAddress
NothingDBNull
DynamicVoir la note
Array(T)T[]
Tuple(T1, T2, …)Tuple<T1, T2, ...> / LargeTuple
Map(K, V)Dictionary<K, V>
Nullable(T)T?
Enum8string
Enum16string
LowCardinality(T)Identique à T
SimpleAggregateFunctionIdentique au type sous-jacent
Nested(…)Tuple[]
Variant(T1, T2, …)Voir la note
QBit(T, dimension)T[]
Les types Dynamic et Variant sont convertis dans le type correspondant au type sous-jacent réel de chaque ligne.

Types de géométrie

Type ClickHouseType .NET
PointTuple<double, double>
RingTuple<double, double>[]
LineStringTuple<double, double>[]
PolygonRing[]
MultiLineStringLineString[]
MultiPolygonPolygon[]
GeometryVoir la note
Le type Geometry est un type Variant qui peut contenir n’importe quel type de géométrie. Il sera converti en type correspondant.

Correspondance des types : écriture dans ClickHouse

Lors de l’insertion de données, le pilote convertit les types .NET vers leurs types ClickHouse correspondants. Les tableaux ci-dessous indiquent quels types .NET sont pris en charge pour chaque type de colonne ClickHouse.

Types entiers

Type ClickHouseTypes .NET acceptésRemarques
Int8sbyte, tout type compatible avec Convert.ToSByte()
UInt8byte, tout type compatible avec Convert.ToByte()
Int16short, tout type compatible avec Convert.ToInt16()
UInt16ushort, tout type compatible avec Convert.ToUInt16()
Int32int, tout type compatible avec Convert.ToInt32()
UInt32uint, tout type compatible avec Convert.ToUInt32()
Int64long, tout type compatible avec Convert.ToInt64()
UInt64ulong, tout type compatible avec Convert.ToUInt64()
Int128BigInteger, decimal, double, float, int, uint, long, ulong, tout type compatible avec Convert.ToInt64()
UInt128BigInteger, decimal, double, float, int, uint, long, ulong, tout type compatible avec Convert.ToInt64()
Int256BigInteger, decimal, double, float, int, uint, long, ulong, tout type compatible avec Convert.ToInt64()
UInt256BigInteger, decimal, double, float, int, uint, long, ulong, tout type compatible avec Convert.ToInt64()

Types en virgule flottante

Type ClickHouseTypes .NET acceptésRemarques
Float32float, tout type compatible avec Convert.ToSingle()
Float64double, tout type compatible avec Convert.ToDouble()
BFloat16float, tout type compatible avec Convert.ToSingle()Tronqué au format bfloat16 sur 16 bits

Type Boolean

Type ClickHouseTypes .NET acceptésRemarques
Boolbool

Types de chaînes

Type ClickHouseTypes .NET acceptésNotes
Stringstring, byte[], ReadOnlyMemory<byte>, StreamLes types binaires sont écrits directement ; les flux peuvent être repositionnables ou non
FixedString(N)string, byte[], ReadOnlyMemory<byte>, StreamLa chaîne est encodée en UTF-8 et complétée ; les types binaires doivent comporter exactement N octets

Types de date et d’heure

Type ClickHouseTypes .NET acceptésRemarques
DateDateTime, DateTimeOffset, DateOnly, types NodaTimeConverti en jours Unix sous forme d’UInt16
Date32DateTime, DateTimeOffset, DateOnly, types NodaTimeConverti en jours Unix sous forme d’Int32
DateTimeDateTime, DateTimeOffset, DateOnly, types NodaTimeVoir ci-dessous pour plus de détails
DateTime32DateTime, DateTimeOffset, DateOnly, types NodaTimeIdentique à DateTime
DateTime64DateTime, DateTimeOffset, DateOnly, types NodaTimePrécision basée sur le paramètre Scale
TimeTimeSpan, intBorné à ±999:59:59 ; int interprété comme un nombre de secondes
Time64TimeSpan, decimal, double, float, int, long, stringChaîne analysée comme [-]HHH:MM:SS[.fraction] ; borné à ±999:59:59.999999999
Le driver respecte DateTime.Kind lors de l’écriture des valeurs :
DateTime.KindParamètres HTTPInsertion en bloc
UtcInstant préservéInstant préservé
LocalInstant préservéInstant préservé
UnspecifiedTraité comme une heure locale dans le fuseau horaire du type du paramètre (UTC par défaut)Traité comme une heure locale dans le fuseau horaire de la colonne
Les valeurs DateTimeOffset préservent toujours l’instant exact. Exemple : DateTime UTC (instant préservé)
var utcTime = new DateTime(2024, 1, 15, 12, 0, 0, DateTimeKind.Utc);
// Stored as 12:00 UTC
// Read from DateTime('Europe/Amsterdam') column: 13:00 (UTC+1)
// Read from DateTime('UTC') column: 12:00 UTC
Exemple : DateTime non spécifié (heure locale)
var wallClock = new DateTime(2024, 1, 15, 14, 30, 0, DateTimeKind.Unspecified);
// Written to DateTime('Europe/Amsterdam') column: stored as 14:30 Amsterdam time
// Read back from DateTime('Europe/Amsterdam') column: 14:30
Recommandation : pour un comportement aussi simple et prévisible que possible, utilisez DateTimeKind.Utc ou DateTimeOffset pour toutes les opérations sur les types DateTime. Cela garantit que votre code fonctionne de manière cohérente, quel que soit le fuseau horaire du serveur, du client ou de la colonne.

Paramètres HTTP vs copie en masse

Il existe une différence importante entre la liaison de paramètres HTTP et la copie en masse lors de l’écriture de valeurs DateTime Unspecified : Copie en masse connaît le fuseau horaire de la colonne cible et interprète correctement les valeurs Unspecified dans ce fuseau horaire. Paramètres HTTP ne connaissent pas automatiquement le fuseau horaire de la colonne. Vous devez le spécifier dans l’indication de type SQL :
// CORRECT: Timezone in SQL type hint - type is extracted automatically
command.CommandText = "INSERT INTO table (dt_amsterdam) VALUES ({dt:DateTime('Europe/Amsterdam')})";
command.AddParameter("dt", myDateTime);

// INCORRECT: Without timezone hint, interpreted as UTC
command.CommandText = "INSERT INTO table (dt_amsterdam) VALUES ({dt:DateTime})";
command.AddParameter("dt", myDateTime);
// String value "2024-01-15 14:30:00" interpreted as UTC, not Amsterdam time!
DateTime.KindColonne cibleParamètre HTTP (avec indication du fuseau horaire)Paramètre HTTP (sans indication du fuseau horaire)Insertion en bloc
UtcUTCInstant préservéInstant préservéInstant préservé
UtcEurope/AmsterdamInstant préservéInstant préservéInstant préservé
LocalQuelconqueInstant préservéInstant préservéInstant préservé
UnspecifiedUTCInterprété comme UTCInterprété comme UTCInterprété comme UTC
UnspecifiedEurope/AmsterdamInterprété comme l’heure d’AmsterdamInterprété comme UTCInterprété comme l’heure d’Amsterdam

Types décimaux

Type ClickHouseTypes .NET acceptésNotes
Decimal(P,S)decimal, ClickHouseDecimal, tout type compatible avec Convert.ToDecimal()Lève une OverflowException si la précision maximale est dépassée
Decimal32decimal, ClickHouseDecimal, tout type compatible avec Convert.ToDecimal()Précision maximale : 9
Decimal64decimal, ClickHouseDecimal, tout type compatible avec Convert.ToDecimal()Précision maximale : 18
Decimal128decimal, ClickHouseDecimal, tout type compatible avec Convert.ToDecimal()Précision maximale : 38
Decimal256decimal, ClickHouseDecimal, tout type compatible avec Convert.ToDecimal()Précision maximale : 76

Type JSON

Type ClickHouseTypes .NET acceptésRemarques
Jsonstring, JsonObject, JsonNode, tout objetLe comportement dépend du paramètre JsonWriteMode
Le comportement lors de l’écriture de JSON est contrôlé par le paramètre JsonWriteMode :
Type d’entréeJsonWriteMode.String (par défaut)JsonWriteMode.Binary
stringTransmis tel quelLève ArgumentException
JsonObjectSérialisé via ToJsonString()Lève ArgumentException
JsonNodeSérialisé via ToJsonString()Lève ArgumentException
POCO enregistréSérialisé via JsonSerializer.Serialize()Encodage binaire avec indications de type, prise en charge des attributs de chemin personnalisés
POCO non enregistré / objet anonymeSérialisé via JsonSerializer.Serialize()Lève ClickHouseJsonSerializationException
  • String (par défaut) : Accepte string, JsonObject, JsonNode ou tout objet. Toutes les entrées sont sérialisées via System.Text.Json.JsonSerializer et envoyées sous forme de chaînes JSON pour un traitement côté serveur. C’est le mode le plus flexible et il fonctionne sans enregistrement préalable de type.
  • Binary : Accepte uniquement les types POCO enregistrés. Les données sont converties côté client au format JSON binaire de ClickHouse avec une prise en charge complète des indications de type. Nécessite d’appeler connection.RegisterJsonSerializationType<T>() avant utilisation. L’écriture de valeurs string ou JsonNode dans ce mode lève ArgumentException.
// Default String mode works with any input
await client.InsertBinaryAsync(
    "my_table",
    new[] { "id", "data" },
    new[] { new object[] { 1u, new { name = "test", value = 42 } } }
);

// Binary mode requires explicit opt-in and type registration
var settings = new ClickHouseClientSettings("Host=localhost")
{
    JsonWriteMode = JsonWriteMode.Binary
};
using var client = new ClickHouseClient(settings);
client.RegisterJsonSerializationType<MyPocoType>();
Colonnes JSON typées
Lorsqu’une colonne JSON inclut des indications de type (par ex. JSON(id UInt64, price Decimal128(2))), le driver utilise ces indications pour sérialiser les valeurs en respectant pleinement les types. Cela préserve la précision de types comme UInt64, Decimal, UUID et DateTime64, qui perdraient sinon en précision s’ils étaient sérialisés sous forme de JSON générique.
Sérialisation des POCO
Les POCO peuvent être écrits dans des colonnes JSON de deux manières, selon le JsonWriteMode : Mode String (par défaut) : les POCO sont sérialisés via System.Text.Json.JsonSerializer. Aucun enregistrement de type n’est nécessaire. C’est l’approche la plus simple, et elle fonctionne avec les objets anonymes. Mode binaire : les POCO sont sérialisés à l’aide du format JSON binaire du driver, avec prise en charge complète des indications de type. Les types doivent être enregistrés avec connection.RegisterJsonSerializationType<T>() avant utilisation. Ce mode prend en charge des mappages de chemins personnalisés via des attributs :
  • [ClickHouseJsonPath("path")] : associe une propriété à un chemin JSON personnalisé. Utile pour les structures imbriquées ou lorsque le nom de la propriété diffère de la clé JSON souhaitée. Fonctionne uniquement en mode binaire.
  • [ClickHouseJsonIgnore] : exclut une propriété de la sérialisation. Fonctionne uniquement en mode binaire.
CREATE TABLE events (
    id UInt32,
    data JSON(`user.id` Int64, `user.name` String, Timestamp DateTime64(3))
) ENGINE = MergeTree() ORDER BY id
using ClickHouse.Driver.Json;

public class UserEvent
{
    [ClickHouseJsonPath("user.id")]
    public long UserId { get; set; }

    [ClickHouseJsonPath("user.name")]
    public string UserName { get; set; }

    public DateTime Timestamp { get; set; }

    [ClickHouseJsonIgnore]
    public string InternalData { get; set; }  // Not serialized
}

// For Binary mode: Register the type and enable Binary mode
var settings = new ClickHouseClientSettings("Host=localhost") { JsonWriteMode = JsonWriteMode.Binary };
using var client = new ClickHouseClient(settings);
client.RegisterJsonSerializationType<UserEvent>();

// Insert POCO - serialized to JSON with nested structure via custom path attributes
await client.InsertBinaryAsync(
    "events",
    new[] { "id", "data" },
    new[] { new object[] { 1u, new UserEvent { UserId = 123, UserName = "Alice", Timestamp = DateTime.UtcNow } } }
);
// Resulting JSON: {"user": {"id": 123, "name": "Alice"}, "Timestamp": "2024-01-15T..."}
La correspondance entre les noms de propriété et les indications de type de colonne est sensible à la casse. Une propriété UserId ne correspondra qu’à une indication définie comme UserId, et non userid. Cela correspond au comportement de ClickHouse, qui permet à des chemins comme userName et UserName de coexister en tant que champs distincts. Limitations (mode binaire uniquement) :
  • Les types POCO doivent être enregistrés sur la connexion avec connection.RegisterJsonSerializationType<T>() avant la sérialisation. Toute tentative de sérialiser un type non enregistré lève une ClickHouseJsonSerializationException.
  • Les propriétés de type Dictionary et array/list nécessitent des indications de type dans la définition de la colonne pour être sérialisées correctement. Sans ces indications, utilisez plutôt String mode.
  • Les valeurs nulles des propriétés POCO ne sont écrites que lorsque le chemin possède une indication de type Nullable(T) dans la définition de la colonne. ClickHouse n’autorise pas les types Nullable dans les chemins JSON dynamiques ; les propriétés nulles sans indication sont donc ignorées.
  • Les attributs ClickHouseJsonPath et ClickHouseJsonIgnore sont ignorés en String mode (ils ne fonctionnent qu’en mode binaire).

Autres types

Type ClickHouseTypes .NET acceptésRemarques
UUIDGuid, stringChaîne analysée comme un Guid
IPv4IPAddress, stringDoit être une adresse IPv4 ; chaîne analysée via IPAddress.Parse()
IPv6IPAddress, stringDoit être une adresse IPv6 ; chaîne analysée via IPAddress.Parse()
NothingN’importe quel typeN’écrit rien (no-op)
DynamicNon pris en charge (lève NotImplementedException)
Array(T)IList, nullnull écrit un tableau vide
Tuple(T1, T2, …)ITuple, IListLe nombre d’éléments doit correspondre à l’arité du tuple
Map(K, V)IDictionary
Nullable(T)null, DBNull ou types acceptés par TÉcrit l’octet indicateur de null avant la valeur
Enum8string, sbyte, types numériquesLa chaîne est recherchée dans le dictionnaire de l’enum
Enum16string, short, types numériquesLa chaîne est recherchée dans le dictionnaire de l’enum
LowCardinality(T)Types acceptés par TDélègue au type sous-jacent
SimpleAggregateFunctionTypes acceptés par le type sous-jacentDélègue au type sous-jacent
Nested(…)IList de tuplesLe nombre d’éléments doit correspondre au nombre de champs
Variant(T1, T2, …)Valeur correspondant à l’un de T1, T2, …Lève ArgumentException si aucun type ne correspond
QBit(T, dim)IListDélègue à Array ; la dimension n’est qu’une métadonnée

Types de géométrie

Type ClickHouseTypes .NET acceptésRemarques
PointSystem.Drawing.Point, ITuple, IList (2 éléments)
RingIList de Point
LineStringIList de Point
PolygonIList de Ring
MultiLineStringIList de LineString
MultiPolygonIList de Polygon
GeometryN’importe quel type de géométrie ci-dessusVariante de tous les types de géométrie

Non pris en charge en écriture

ClickHouse TypeRemarques
DynamicLève NotImplementedException
AggregateFunctionLève AggregateFunctionException

Gestion du type Nested

Les types Nested de ClickHouse (Nested(...)) peuvent être lus et écrits avec la sémantique des tableaux.
CREATE TABLE test.nested (
    id UInt32,
    params Nested (param_id UInt8, param_val String)
) ENGINE = Memory
var row1 = new object[] { 1, new[] { 1, 2, 3 }, new[] { "v1", "v2", "v3" } };
var row2 = new object[] { 2, new[] { 4, 5, 6 }, new[] { "v4", "v5", "v6" } };

await client.InsertBinaryAsync(
    "test.nested",
    new[] { "id", "params.param_id", "params.param_val" },
    new[] { row1, row2 }
);

Journalisation et diagnostics

Le client .NET ClickHouse s’intègre aux abstractions Microsoft.Extensions.Logging afin d’offrir une journalisation légère, activée sur demande. Lorsqu’elle est activée, le driver émet des messages structurés pour les événements du cycle de vie de la connexion, l’exécution des commandes, les opérations de transport et les opérations d’insertion en masse. La journalisation est entièrement facultative : les applications qui ne configurent pas de logger continuent de s’exécuter sans surcharge supplémentaire.

Démarrage rapide

using ClickHouse.Driver;
using Microsoft.Extensions.Logging;

var loggerFactory = LoggerFactory.Create(builder =>
{
    builder
        .AddConsole()
        .SetMinimumLevel(LogLevel.Information);
});

var settings = new ClickHouseClientSettings("Host=localhost;Port=8123")
{
    LoggerFactory = loggerFactory
};

using var client = new ClickHouseClient(settings);

Utilisation du fichier appsettings.json

Vous pouvez configurer les niveaux de journalisation à l’aide de la configuration .NET standard :
using ClickHouse.Driver;
using Microsoft.Extensions.Configuration;
using Microsoft.Extensions.Logging;

var configuration = new ConfigurationBuilder()
    .SetBasePath(Directory.GetCurrentDirectory())
    .AddJsonFile("appsettings.json")
    .Build();

var loggerFactory = LoggerFactory.Create(builder =>
{
    builder
        .AddConfiguration(configuration.GetSection("Logging"))
        .AddConsole();
});

var settings = new ClickHouseClientSettings("Host=localhost;Port=8123")
{
    LoggerFactory = loggerFactory
};

using var client = new ClickHouseClient(settings);

Utilisation d’une configuration en mémoire

Vous pouvez également configurer le niveau de verbosité de la journalisation par catégorie dans le code :
using ClickHouse.Driver;
using Microsoft.Extensions.Configuration;
using Microsoft.Extensions.Logging;

var categoriesConfiguration = new Dictionary<string, string>
{
    { "LogLevel:Default", "Warning" },
    { "LogLevel:ClickHouse.Driver.Connection", "Information" },
    { "LogLevel:ClickHouse.Driver.Command", "Debug" }
};

var config = new ConfigurationBuilder()
    .AddInMemoryCollection(categoriesConfiguration)
    .Build();

using var loggerFactory = LoggerFactory.Create(builder =>
{
    builder
        .AddConfiguration(config)
        .AddSimpleConsole();
});

var settings = new ClickHouseClientSettings("Host=localhost;Port=8123")
{
    LoggerFactory = loggerFactory
};

using var client = new ClickHouseClient(settings);

Catégories et émetteurs

Le driver utilise des catégories dédiées afin de vous permettre d’ajuster finement les niveaux de journalisation par composant :
CatégorieSourcePoints clés
ClickHouse.Driver.ConnectionClickHouseConnectionCycle de vie de la connexion, sélection de la fabrique de clients HTTP, ouverture/fermeture de la connexion, gestion des sessions.
ClickHouse.Driver.CommandClickHouseCommandDébut/fin d’exécution des requêtes, durée d’exécution, ID de requête, statistiques du serveur et détails des erreurs.
ClickHouse.Driver.TransportClickHouseConnectionRequêtes HTTP streaming de bas niveau, indicateurs de compression, codes d’état des réponses et échecs de transport.
ClickHouse.Driver.ClientClickHouseClientInsertions binaires, requêtes et autres opérations
ClickHouse.Driver.NetTraceTraceHelperTraçage réseau, uniquement lorsque le mode débogage est activé

Exemple : diagnostic des problèmes de connexion

{
    "Logging": {
        "LogLevel": {
            "ClickHouse.Driver.Connection": "Trace",
            "ClickHouse.Driver.Transport": "Trace"
        }
    }
}
Les éléments suivants seront consignés :
  • Sélection de la fabrique de clients HTTP (pool par défaut ou connexion unique)
  • Configuration du gestionnaire HTTP (SocketsHttpHandler ou HttpClientHandler)
  • Paramètres du pool de connexions (MaxConnectionsPerServer, PooledConnectionLifetime, etc.)
  • Paramètres de délai d’expiration (ConnectTimeout, Expect100ContinueTimeout, etc.)
  • Configuration SSL/TLS
  • Événements d’ouverture/fermeture des connexions
  • Suivi des ID de session

Mode Débogage : tracing réseau et diagnostics

Pour faciliter le diagnostic des problèmes réseau, la bibliothèque du driver inclut un utilitaire qui active le tracing de bas niveau des mécanismes réseau internes de .NET. Pour l’activer, vous devez fournir une instance de LoggerFactory avec le niveau défini sur Trace, et définir EnableDebugMode sur true (ou l’activer manuellement via la classe ClickHouse.Driver.Diagnostic.TraceHelper). Les événements seront consignés dans la catégorie ClickHouse.Driver.NetTrace. Avertissement : cela générera des logs extrêmement verbeux et aura un impact sur les performances. Il n’est pas recommandé d’activer le mode Débogage en production.
var loggerFactory = LoggerFactory.Create(builder =>
{
    builder
        .AddConsole()
        .SetMinimumLevel(LogLevel.Trace); // Must be Trace level to see network events
});

var settings = new ClickHouseClientSettings()
{
    LoggerFactory = loggerFactory,
    EnableDebugMode = true,  // Enable low-level network tracing
};

OpenTelemetry

Le driver intègre une prise en charge native du tracing distribué avec OpenTelemetry via l’API .NET System.Diagnostics.Activity. Lorsqu’il est activé, le driver émet des spans pour les opérations sur la base de données, qui peuvent être exportés vers des backends d’observabilité comme Jaeger ou ClickHouse lui-même (via l’OpenTelemetry Collector).

Activer le tracing

Dans les applications ASP.NET Core, ajoutez l’ActivitySource du driver ClickHouse à votre configuration OpenTelemetry :
builder.Services.AddOpenTelemetry()
    .WithTracing(tracing => tracing
        .AddSource(ClickHouseDiagnosticsOptions.ActivitySourceName)  // Subscribe to ClickHouse driver spans
        .AddAspNetCoreInstrumentation()
        .AddOtlpExporter());             // Or AddJaegerExporter(), etc.
Pour les applications en ligne de commande, les tests ou la configuration manuelle :
using OpenTelemetry;
using OpenTelemetry.Trace;

var tracerProvider = Sdk.CreateTracerProviderBuilder()
    .AddSource(ClickHouseDiagnosticsOptions.ActivitySourceName)
    .AddConsoleExporter()
    .Build();

Attributs des spans

Chaque span inclut les attributs de base de données standard d’OpenTelemetry, ainsi que des statistiques de requête propres à ClickHouse utiles pour le débogage.
AttributDescription
db.systemToujours "clickhouse"
db.nameNom de la base de données
db.userNom d’utilisateur
db.statementRequête SQL (si activée)
db.clickhouse.read_rowsLignes lues par la requête
db.clickhouse.read_bytesOctets lus par la requête
db.clickhouse.written_rowsLignes écrites par la requête
db.clickhouse.written_bytesOctets écrits par la requête
db.clickhouse.elapsed_nsTemps d’exécution côté serveur en nanosecondes

Options de configuration

Contrôlez le comportement du tracing à l’aide de ClickHouseDiagnosticsOptions :
using ClickHouse.Driver.Diagnostic;

// Include SQL statements in spans (default: false for security)
ClickHouseDiagnosticsOptions.IncludeSqlInActivityTags = true;

// Truncate long SQL statements (default: 1000 characters)
ClickHouseDiagnosticsOptions.StatementMaxLength = 500;
L’activation de IncludeSqlInActivityTags peut exposer des données sensibles dans vos traces. À utiliser avec prudence dans les environnements de production.

Configuration TLS

Lorsque vous vous connectez à ClickHouse via HTTPS, vous pouvez configurer le comportement de TLS/SSL de plusieurs manières.

Validation personnalisée des certificats

Pour les environnements de production nécessitant une logique de validation des certificats personnalisée, fournissez votre propre HttpClient avec un gestionnaire ServerCertificateCustomValidationCallback configuré :
using System.Net;
using System.Net.Security;
using ClickHouse.Driver;

var handler = new HttpClientHandler
{
    // Required when compression is enabled (default)
    AutomaticDecompression = DecompressionMethods.GZip | DecompressionMethods.Deflate,

    ServerCertificateCustomValidationCallback = (message, cert, chain, sslPolicyErrors) =>
    {
        // Example: Accept a specific certificate thumbprint
        if (cert?.Thumbprint == "YOUR_EXPECTED_THUMBPRINT")
            return true;

        // Example: Accept certificates from a specific issuer
        if (cert?.Issuer.Contains("YourOrganization") == true)
            return true;

        // Default: Use standard validation
        return sslPolicyErrors == SslPolicyErrors.None;
    },
};

var httpClient = new HttpClient(handler) { Timeout = TimeSpan.FromMinutes(5) };

var settings = new ClickHouseClientSettings
{
    Host = "my.clickhouse.server",
    Protocol = "https",
    HttpClient = httpClient,
};

using var client = new ClickHouseClient(settings);
Points importants à prendre en compte lors de la fourniture d’un HttpClient personnalisé
  • Décompression automatique : vous devez activer AutomaticDecompression si la compression n’est pas désactivée (elle est activée par défaut).
  • Délai d’inactivité : définissez PooledConnectionIdleTimeout sur une valeur inférieure au keep_alive_timeout du serveur (10 secondes pour ClickHouse Cloud) afin d’éviter les erreurs de connexion dues à des connexions semi-ouvertes.

Prise en charge des ORM

Les ORM nécessitent l’API ADO.NET (ClickHouseConnection). Pour gérer correctement le cycle de vie des connexions, créez-les à partir d’un ClickHouseDataSource :
// Register DataSource as singleton
var dataSource = new ClickHouseDataSource("Host=localhost;Username=default");

// Create connections for ORM use
await using var connection = await dataSource.OpenConnectionAsync();
// Pass connection to your ORM...

Dapper

ClickHouse.Driver est compatible avec Dapper. Le pilote convertit automatiquement la syntaxe @parameter de Dapper en syntaxe native {parameter:Type} de ClickHouse, en déduisant les types à partir des valeurs .NET. Utilisez ClickHouseDataSource pour gérer correctement le cycle de vie de la connexion :
var dataSource = new ClickHouseDataSource("Host=localhost");
services.AddSingleton(dataSource); // Register as singleton in DI

using var connection = dataSource.CreateConnection();

Modes de passage des paramètres

Tous les modes standard de passage des paramètres de Dapper sont pris en charge : Objets anonymes :
await connection.ExecuteAsync(
    "INSERT INTO users (id, name, balance) VALUES (@Id, @Name, @Balance)",
    new { Id = 1, Name = "alice", Balance = 3.14 });
Classes POCO :
class InsertParams
{
    public int Id { get; set; }
    public string Name { get; set; }
    public double Balance { get; set; }
}

var param = new InsertParams { Id = 42, Name = "bob", Balance = 99.9 };
await connection.ExecuteAsync(
    "INSERT INTO users (id, name, balance) VALUES (@Id, @Name, @Balance)", param);
Dictionnaire :
var parameters = new Dictionary<string, object> { { "Id", 2 } };
var rows = await connection.QueryAsync<User>(
    "SELECT id, name FROM users WHERE id = @Id", parameters);
DynamicParameters (à partir d’un dictionnaire ou d’un objet anonyme) :
var dynParams = new DynamicParameters(new { Id = 1 });
// or: new DynamicParameters(new Dictionary<string, object> { { "Id", 1 } });

var rows = await connection.QueryAsync<User>(
    "SELECT id, name FROM users WHERE id = @Id", dynParams);

Requêtes vers des POCO

Dapper associe les colonnes aux propriétés par leur nom (sans tenir compte de la casse) :
class User
{
    public int Id { get; set; }
    public string Name { get; set; }
    public double Balance { get; set; }
}

// From a table
var users = (await connection.QueryAsync<User>("SELECT id, name, balance FROM users")).ToList();

// From a literal
var row = (await connection.QueryAsync<User>("SELECT 1 as id, 'hello' as name, 2.5 as balance")).Single();

Syntaxe native des paramètres ClickHouse

Lorsque vous avez besoin d’un contrôle explicite des types, utilisez directement dans le SQL la syntaxe {param:Type} de ClickHouse avec un Dictionary<string, object> pour les valeurs de paramètre. N’utilisez pas à la fois la syntaxe @param et la syntaxe {param:Type} pour un même paramètre.
var parameters = new Dictionary<string, object> { { "value", 42 } };
var result = await connection.QueryAsync<int>("SELECT {value:Int32}", parameters);

WHERE IN

L’expansion native de IN dans Dapper fonctionne :
var rows = await connection.QueryAsync<User>(
    "SELECT id, name FROM users WHERE id IN @Ids ORDER BY id",
    new { Ids = new[] { 1, 3, 5 } });
Dapper réécrit cela en WHERE id IN (@Ids1, @Ids2, @Ids3), et le driver convertit chaque paramètre étendu. La fonction has() de ClickHouse avec un paramètre Array fonctionne également :
var parameters = new Dictionary<string, object> { { "ids", new[] { 1, 3, 5 } } };
var rows = await connection.QueryAsync<User>(
    "SELECT id, name FROM users WHERE has({ids:Array(Int32)}, id) ORDER BY id",
    parameters);

Gestionnaires de types personnalisés

Certains types ClickHouse, par exemple ITuple, BigInteger et ClickHouseDecimal, nécessitent l’enregistrement de gestionnaires au démarrage :
// ClickHouseDecimal (for Decimal64/128/256 columns)
SqlMapper.AddTypeHandler(new ClickHouseDecimalHandler());

// BigInteger (for Int128/Int256/UInt128/UInt256 columns)
SqlMapper.AddTypeHandler(new BigIntegerHandler());

// IPAddress (for IPv4/IPv6 columns)
SqlMapper.AddTypeHandler(new IpAddressHandler());
Voir l’exemple Dapper pour un exemple d’implémentation d’un type handler.

Dapper.Contrib

GetAll<T>() et Get<T>(id) fonctionnent. En revanche, Insert<T>() ne fonctionne pas : il génère une syntaxe SQL Server (SCOPE_IDENTITY, []). Il est recommandé d’utiliser à la place la méthode native InsertBinaryAsync de ClickHouseClient.
[Table("test.users")]
record class UserRecord(int Id, string Name, DateTime Timestamp);

var all = await connection.GetAllAsync<UserRecord>();
var one = await connection.GetAsync<UserRecord>(1);
Les noms des propriétés doivent correspondre exactement aux noms de colonnes de ClickHouse (respect de la casse).

Limitations

ÉlémentStatutDétails
Tuple comme résultatFonctionneNécessite l’enregistrement de SqlMapper.TypeHandler<ITuple>
Tuple comme paramètreNon pris en chargeDapper ne peut pas sérialiser ITuple/Tuple<> en tant que valeur de DbParameter
Types Nested comme paramètreNon pris en chargeMême raison — Dapper rejette les types complexes comme valeurs de paramètre
Types Geo comme paramètreNon pris en chargePoint, Ring, Polygon, LineString, MultiLineString, MultiPolygon
Dapper.Contrib.Insert<T>()Non pris en chargeGénère une syntaxe spécifique à SQL Server
Type NothingNon pris en chargeAucune représentation .NET pertinente

Linq2db

Ce pilote est compatible avec linq2db, un ORM léger et un fournisseur LINQ pour .NET. Consultez le site du projet pour une documentation détaillée. Exemple d’utilisation : Créez une DataConnection à l’aide du fournisseur ClickHouse :
using LinqToDB;
using LinqToDB.Data;
using LinqToDB.DataProvider.ClickHouse;

var connectionString = "Host=localhost;Port=8123;Database=default";
var options = new DataOptions()
    .UseClickHouse(connectionString, ClickHouseProvider.ClickHouseDriver);

await using var db = new DataConnection(options);
Les mappages de tables peuvent être définis à l’aide d’attributs ou de l’API fluide. Si les noms de votre classe et de vos propriétés correspondent exactement aux noms de la table et des colonnes, aucune configuration n’est nécessaire :
public class Product
{
    public int Id { get; set; }
    public string Name { get; set; }
    public decimal Price { get; set; }
}
Requêtes :
await using var db = new DataConnection(options);

var products = await db.GetTable<Product>()
    .Where(p => p.Price > 100)
    .OrderByDescending(p => p.Name)
    .ToListAsync();
Bulk Copy : Utilisez BulkCopyAsync pour effectuer efficacement des insertions en bloc.
await using var db = new DataConnection(options);
var table = db.GetTable<Product>();

var options = new BulkCopyOptions
{
    MaxBatchSize = 100000,
    MaxDegreeOfParallelism = 1,
    WithoutSession = true
};

await table.BulkCopyAsync(options, products);

Entity Framework Core

Le fournisseur officiel Entity Framework Core pour ClickHouse. Associez des classes C# à des tables ClickHouse, effectuez des requêtes avec LINQ et insérez des données via SaveChanges — le tout avec les conventions EF Core habituelles.
Ce fournisseur est activement développé. La version actuelle prend en charge les requêtes LINQ (y compris les JOIN, les sous-requêtes et les opérations ensemblistes), INSERT via SaveChanges / BulkInsertAsync, les migrations avec DDL complet (CREATE / ALTER / DROP), ainsi que la configuration du moteur de table spécifique à ClickHouse. UPDATE / DELETE ne sont pas pris en charge.

Installation

dotnet add package ClickHouse.EntityFrameworkCore
Nécessite .NET 10.0 et EF Core 10.

Démarrage rapide

Définissez votre entité et le DbContext, puis effectuez des requêtes avec LINQ :
using Microsoft.EntityFrameworkCore;

public class PageView
{
    public long Id { get; set; }
    public string Path { get; set; }
    public DateOnly Date { get; set; }
    public string UserAgent { get; set; }
}

public class AnalyticsContext : DbContext
{
    public DbSet<PageView> PageViews { get; set; }

    protected override void OnConfiguring(DbContextOptionsBuilder optionsBuilder)
        => optionsBuilder.UseClickHouse("Host=localhost;Database=analytics");
}

// Query
await using var ctx = new AnalyticsContext();

var topPages = await ctx.PageViews
    .Where(v => v.Date >= new DateOnly(2024, 1, 1))
    .GroupBy(v => v.Path)
    .Select(g => new { Path = g.Key, Views = g.Count() })
    .OrderByDescending(x => x.Views)
    .Take(10)
    .ToListAsync();

Types pris en charge

CatégorieTypes ClickHouseTypes CLR
EntiersInt8Int64, UInt8UInt64sbyte, short, int, long, byte, ushort, uint, ulong
Grands entiersInt128, Int256, UInt128, UInt256BigInteger
FlottantsFloat32, Float64, BFloat16float, double
DécimauxDecimal(P,S), Decimal32(S), Decimal64(S), Decimal128(S)decimal ou ClickHouseDecimal
BoolBoolbool
ChaînesString, FixedString(N)string
ÉnumérationsEnum8(...), Enum16(...)string ou enum C#
Date/heureDate, Date32, DateTime, DateTime64(P, 'TZ')DateOnly, DateTime
HeureTime, Time64(N)TimeSpan
UUIDUUIDGuid
RéseauIPv4, IPv6IPAddress
TableauxArray(T)T[], List<T>, IList<T>, ICollection<T>, IReadOnlyList<T>, IReadOnlyCollection<T>, IEnumerable<T>
MapsMap(K, V)Dictionary<K,V>
TuplesTuple(T1, ...)Tuple<...> ou ValueTuple<...>
VariantVariant(T1, T2, ...)object
DynamicDynamicobject
JSONJsonJsonNode ou string
GéographiquesPoint, Ring, LineString, Polygon, MultiLineString, MultiPolygon, GeometryTuple<double,double> et les tableaux correspondants ; object pour Geometry
WrappersNullable(T), LowCardinality(T)Déballés automatiquement
Utilisez ClickHouseDecimal (de ClickHouse.Driver.Numerics) au lieu de decimal lorsque vous avez besoin de toute la précision des colonnes Decimal128/Decimal256decimal en .NET est limité à 28–29 chiffres significatifs.

Opérations LINQ prises en charge

Requêtes : Where, OrderBy, Take, Skip, Select, First, Single, Any, All, Count, Distinct, AsNoTracking GROUP BY et agrégats : GroupBy avec Count, LongCount, Sum, Average, Min, Max — y compris HAVING (.Where() après .GroupBy()), plusieurs agrégats dans une même projection et OrderBy sur les résultats agrégés. JOINs : Join (INNER), schémas GroupJoin/SelectMany (LEFT et CROSS). LEFT JOIN renvoie de vraies valeurs null pour les lignes sans correspondance (voir la sémantique des valeurs nulles de LEFT JOIN ci-dessous). Sous-requêtes : Contains / IN corrélés, Any / EXISTS, All, et sous-requêtes scalaires dans les projections. Opérations ensemblistes : Concat (→ UNION ALL), Union (→ UNION DISTINCT), Intersect, Except. Collections locales en mémoire : les joins et Contains sur des collections en mémoire (int[], List<T>, etc.) sont traduits en une série de UNION. Méthodes de chaîne : Contains, StartsWith, EndsWith, IndexOf, Replace, Substring, Trim/TrimStart/TrimEnd, ToLower, ToUpper, Length, IsNullOrEmpty, Concat (et l’opérateur +). Fonctions mathématiques : les méthodes standard de Math et MathF sont traduites en leurs équivalents ClickHouse — fonctions arithmétiques, logarithmiques, trigonométriques et utilitaires.
Sémantique des valeurs nulles de LEFT JOIN
Le fournisseur injecte automatiquement set_join_use_nulls=1 dans chaque chemin de connexion afin de correspondre aux attentes d’Entity Framework concernant le comportement des JOIN. Si votre serveur ClickHouse ou votre profil interdit la modification de ce paramètre (par exemple, un profil readonly=1), désactivez ce comportement avec :
optionsBuilder.UseClickHouse(connectionString, o => o.DisableJoinNullSemantics());
Lorsque l’opt-out est activé, LEFT JOIN renvoie les valeurs par défaut des colonnes ClickHouse, et la détection par EF des propriétés de navigation basée sur les valeurs nulles ne fonctionne plus comme prévu. Utilisez des comparaisons explicites avec 0 / "" plutôt que == null.

Insertion de données

SaveChanges utilise l’API native InsertBinaryAsync du pilote — l’encodage RowBinary avec compression GZip est bien plus efficace que le SQL paramétré :
await using var ctx = new AnalyticsContext();

ctx.PageViews.Add(new PageView
{
    Id = 1,
    Path = "/home",
    Date = new DateOnly(2024, 6, 15),
    UserAgent = "Mozilla/5.0"
});

await ctx.SaveChangesAsync();
Les entités passent de Added à Unchanged après la sauvegarde, comme avec tout autre fournisseur EF Core. La taille du lot est configurable (1000 par défaut) :
optionsBuilder.UseClickHouse("Host=localhost", o => o.MaxBatchSize(5000));

Insertion en masse

Pour les chargements à haut débit, utilisez BulkInsertAsync au lieu de SaveChanges. Il s’agit d’une méthode d’extension sur DbContext qui contourne entièrement le suivi des modifications d’EF Core, la résolution des identités et la gestion d’état — elle appelle directement InsertBinaryAsync du pilote avec l’encodage RowBinary et la compression GZip. Cette méthode convient donc au chargement de grands ensembles de données lorsque vous n’avez pas besoin du suivi des entités après l’insertion :
var events = Enumerable.Range(0, 100_000)
    .Select(i => new PageView
    {
        Id = i,
        Path = $"/page/{i}",
        Date = DateOnly.FromDateTime(DateTime.Today)
    });

long rowsInserted = await ctx.BulkInsertAsync(events);
L’entrée peut être n’importe quel IEnumerable<T> — les entités sont traitées en flux, sans être toutes chargées en mémoire. La valeur de retour est le nombre de lignes insérées. Les entités ne sont pas rattachées au DbContext après l’insertion, il n’y a donc pas de transition d’état AddedUnchanged.

Énumérations

Les colonnes ClickHouse Enum8/Enum16 peuvent être mappées sur des propriétés string ou sur des types C# enum. Lorsqu’on utilise des énumérations C#, le fournisseur convertit automatiquement l’énumération depuis et vers sa représentation sous forme de chaîne :
public enum Status { Active, Inactive, Pending }

public class User
{
    public long Id { get; set; }
    public Status Status { get; set; }
}

// Query with enum values
var active = await ctx.Users
    .Where(u => u.Status == Status.Active)
    .ToListAsync();

Conversions de type personnalisées

Le système ValueConverter d’EF Core vous permet d’associer des types personnalisés à des types déjà pris en charge par le fournisseur. Le fournisseur ne voit jamais votre type personnalisé — EF Core effectue la conversion à la limite entre les deux. Conversion par propriété :
public class Money
{
    public decimal Amount { get; set; }
    public string Currency { get; set; }
}

public class Order
{
    public long Id { get; set; }
    public Money Price { get; set; }
}

// In OnModelCreating:
modelBuilder.Entity<Order>()
    .Property(o => o.Price)
    .HasConversion(
        m => $"{m.Amount}|{m.Currency}",
        s => new Money
        {
            Amount = decimal.Parse(s.Split('|')[0]),
            Currency = s.Split('|')[1]
        })
    .HasColumnType("String");
Classe de convertisseur réutilisable :
public class MoneyConverter : ValueConverter<Money, string>
{
    public MoneyConverter() : base(
        m => $"{m.Amount}|{m.Currency}",
        s => Parse(s)) { }

    private static Money Parse(string s)
    {
        var parts = s.Split('|');
        return new Money { Amount = decimal.Parse(parts[0]), Currency = parts[1] };
    }
}

// Apply to a single property:
.HasConversion<MoneyConverter>()

// Or apply to all properties of a type via conventions:
protected override void ConfigureConventions(ModelConfigurationBuilder configurationBuilder)
{
    configurationBuilder.Properties<Money>()
        .HaveConversion<MoneyConverter>();
}

Annotations de type de colonne

Pour les types scalaires comme string, int, DateTime, etc., le fournisseur détermine automatiquement le type ClickHouse. Pour les types paramétrés et les wrappers, vous devez spécifier explicitement le type ClickHouse. Utilisation des annotations de données (attributs) :
using System.ComponentModel.DataAnnotations.Schema;
using Microsoft.EntityFrameworkCore;

[Table("sensor_readings")]
public class SensorReading
{
    public long Id { get; set; }

    [Column(TypeName = "Array(String)")]
    public string[] Tags { get; set; }

    [Column(TypeName = "Map(String, String)")]
    public Dictionary<string, string> Metadata { get; set; }

    [Column(TypeName = "Nullable(Float64)")]
    public double? Value { get; set; }

    [Column(TypeName = "Decimal128(18)")]
    public decimal HighPrecision { get; set; }
}
Utilisation de l’API fluide dans OnModelCreating :
modelBuilder.Entity<SensorReading>(e =>
{
    e.ToTable("sensor_readings");
    e.Property(x => x.Tags).HasColumnType("Array(String)");
    e.Property(x => x.Metadata).HasColumnType("Map(String, String)");
    e.Property(x => x.Value).HasColumnType("Nullable(Float64)");
    e.Property(x => x.Category).HasColumnType("LowCardinality(String)");
    e.Property(x => x.HighPrecision).HasColumnType("Decimal128(18)");
});
Les wrappers imbriqués comme Array(Nullable(Int32)) et LowCardinality(Nullable(String)) sont pris en charge — le fournisseur retire automatiquement les wrappers Nullable et LowCardinality à chaque niveau d’imbrication.

Colonnes Variant et Dynamic

Dans .NET, les colonnes ClickHouse Variant(T1, T2, ...) et Dynamic sont mappées à object. Comme object est trop générique pour une inférence de type automatique, vous devez déclarer explicitement le type de stockage via .HasColumnType() :
public class Event
{
    public long Id { get; set; }
    public object? Payload { get; set; }
}

// In OnModelCreating:
entity.Property(e => e.Payload).HasColumnType("Variant(String, UInt64, Array(UInt64))");
// or:
entity.Property(e => e.Payload).HasColumnType("Dynamic");
Lors de la lecture, la valeur est automatiquement désérialisée dans le type .NET correspondant au discriminateur stocké (par ex. string, ulong, ulong[]).

Colonnes JSON

Le fournisseur prend en charge le type de colonne Json de ClickHouse, qu’il associe à System.Text.Json.Nodes.JsonNode (par défaut) ou à string (via un ValueConverter automatique) :
using System.Text.Json.Nodes;

public class Event
{
    public long Id { get; set; }
    public JsonNode? Data { get; set; }
}

// In OnModelCreating:
entity.Property(e => e.Data).HasColumnType("Json");
La lecture et l’écriture du JSON s’effectuent via SaveChanges et BulkInsertAsync :
ctx.Events.Add(new Event
{
    Id = 1,
    Data = JsonNode.Parse("""{"action": "click", "x": 100, "y": 200}""")
});
await ctx.SaveChangesAsync();

var ev = await ctx.Events.Where(e => e.Id == 1).SingleAsync();
string action = ev.Data!["action"]!.GetValue<string>(); // "click"
Si vous préférez des chaînes JSON brutes, mappez la propriété en string avec un type de colonne Json — le fournisseur applique automatiquement un ValueConverter :
public class Event
{
    public long Id { get; set; }
    public string? Data { get; set; }  // raw JSON string
}

entity.Property(e => e.Data).HasColumnType("Json");
  • Pas de traduction des chemins JSONentity.Data["name"] dans LINQ n’est pas converti en syntaxe SQL ClickHouse data.name. Filtrez sur des colonnes non JSON et examinez le JSON en mémoire.
  • Sémantique de NULL — le type JSON de ClickHouse renvoie {} (objet vide) pour les valeurs NULL au lieu de SQL NULL.
  • Précision des entiers — le JSON de ClickHouse stocke tous les entiers en Int64. Lors de la lecture via JsonNode, utilisez GetValue<long>() plutôt que GetValue<int>().

Moteurs de table

Configurez les moteurs de table ClickHouse et les clauses propres à chaque moteur à l’aide de l’API fluide ToTable(name, t => ...). Si aucun moteur n’est configuré, le fournisseur utilise par défaut MergeTree, avec ORDER BY déduit de la clé primaire de l’entité.
modelBuilder.Entity<Event>(e =>
{
    e.ToTable("events", t => t
        .HasMergeTreeEngine()
        .WithOrderBy("UserId", "Timestamp")
        .WithPartitionBy("toYYYYMM(Timestamp)")
        .WithPrimaryKey("UserId")
        .WithSettings("index_granularity = 8192"));
});
Familles de moteurs prises en charge :
MoteurMéthode FluentRemarques
MergeTreeHasMergeTreeEngine()Par défaut si rien n’est configuré
ReplacingMergeTreeHasReplacingMergeTreeEngine("Version", "IsDeleted") ou HasReplacingMergeTreeEngine<T>(e => e.Version)Colonnes Version / IsDeleted facultatives
SummingMergeTreeHasSummingMergeTreeEngine(…) ou HasSummingMergeTreeEngine<T>(e => new { … })Colonnes à additionner facultatives
AggregatingMergeTreeHasAggregatingMergeTreeEngine()
CollapsingMergeTreeHasCollapsingMergeTreeEngine("Sign") ou HasCollapsingMergeTreeEngine<T>(e => e.Sign)La colonne Sign doit être de type Int8
VersionedCollapsingMergeTreeHasVersionedCollapsingMergeTreeEngine("Sign", "Version") ou <T>(e => e.Sign, e => e.Version)
GraphiteMergeTreeHasGraphiteMergeTreeEngine("config_section")
Log, TinyLog, StripeLog, MemoryHasLogEngine(), HasTinyLogEngine(), HasStripeLogEngine(), HasMemoryEngine()Pas de ORDER BY / PARTITION BY
Clauses du moteur : WithOrderBy, WithPartitionBy, WithPrimaryKey, WithSampleBy, WithTtl, WithSettings. Elles s’appliquent toutes au builder de moteur renvoyé par HasXxxEngine(). Fonctionnalités au niveau des colonnes : HasCodec, HasTtl, HasComment, HasDefault — toutes sont prises en compte dans les migrations. Index de saut de données — via HasIndex(...).HasSkippingIndexType(...):
modelBuilder.Entity<Event>()
    .HasIndex(e => e.UserId)
    .HasSkippingIndexType("minmax")
    .HasGranularity(4);

// Index with parameters (e.g. bloom_filter, tokenbf_v1):
modelBuilder.Entity<Event>()
    .HasIndex(e => e.Tag)
    .HasSkippingIndexType("bloom_filter")
    .HasSkippingIndexParams("0.01")
    .HasGranularity(1);
Les index standard (non-skipping) sont ignorés sans avertissement, car ClickHouse n’a pas d’équivalent. Les index uniques provoquent une exception, car ClickHouse ne garantit pas l’unicité.

Migrations

Processus standard des migrations EF Core :
dotnet ef migrations add InitialCreate
dotnet ef database update
Opérations prises en charge :
OpérationGénère
CREATE TABLEInclut la clause moteur, ORDER BY, PARTITION BY, SETTINGS, ainsi que les codecs/TTL/commentaires/valeurs par défaut des colonnes
ALTER TABLE ADD COLUMN
ALTER TABLE DROP COLUMN
ALTER TABLE MODIFY COLUMNGère les changements de type ainsi que l’ajout/la suppression d’annotations (CODEC, TTL, COMMENT, DEFAULT)
ALTER TABLE RENAME COLUMN
RENAME TABLE
ALTER TABLE ADD INDEX / DROP INDEXUniquement les index de saut de données
CREATE DATABASE / DROP DATABASEVia EnsureCreated / EnsureDeleted et les migrations

Limitations des migrations

FonctionnalitéRaison
Clés étrangèresClickHouse n’applique pas les clés étrangères. Les migrations rejettent AddForeignKey ; le validateur du modèle émet un avertissement lors de la génération du modèle.
Contraintes uniques / index uniquesClickHouse n’applique pas l’unicité. Les index uniques génèrent une exception lors de la migration.
Valeurs générées par le serveur (auto-incrémentation / IDENTITY)ClickHouse n’a pas d’équivalent.
Colonnes Nested(…)Pas encore prises en charge comme type CLR mappé.
Entités possédées en JSON (.ToJson())Le mappage JSON structurel des entités possédées n’est pas encore implémenté. Utilisez plutôt JsonNode / string sur une colonne Json (voir colonnes JSON).
Au-delà des migrations, le fournisseur ne prend pas encore en charge :
  • UPDATE / DELETE
  • Transactions : BeginTransaction est sans effet. Les transactions ACID ne sont pas prises en charge dans ClickHouse.
  • Traduction des requêtes avec chemin JSON : entity.Data["key"] dans LINQ ne se traduit pas en syntaxe SQL ClickHouse data.key. Filtrez sur des colonnes non JSON et inspectez le JSON en mémoire.

Limites

Colonnes de type AggregateFunction

Les colonnes de type AggregateFunction(...) ne peuvent pas être interrogées ni faire l’objet d’une insertion directe. Pour insérer :
INSERT INTO t VALUES (uniqState(1));
Pour sélectionner :
SELECT uniqMerge(c) FROM t;

Dernière modification le 25 juin 2026