Saltar al contenido principal
El cliente oficial de C# para conectarse a ClickHouse. El código fuente del cliente está disponible en el repositorio de GitHub. Desarrollado originalmente por Oleg V. Kozlyuk. La biblioteca ofrece dos API principales:
  • ClickHouseClient (recomendado): un cliente de alto nivel, seguro para subprocesos, diseñado para usarse como singleton. Ofrece una API asíncrona sencilla para consultas e inserciones masivas. Es la mejor opción para la mayoría de las aplicaciones.
  • ADO.NET (ClickHouseDataSource, ClickHouseConnection, ClickHouseCommand): abstracciones estándar de base de datos de .NET. Son necesarias para la integración con ORM (Dapper, Linq2db) y cuando necesita compatibilidad con ADO.NET. ClickHouseBulkCopy es una clase auxiliar para insertar datos de forma eficiente mediante una conexión ADO.NET. ClickHouseBulkCopy está obsoleto y se eliminará en una versión futura; en su lugar, use ClickHouseClient.InsertBinaryAsync.
Ambas API comparten el mismo pool de conexiones HTTP y pueden usarse juntas en la misma aplicación.

Guía de migración

  1. Actualiza tu archivo .csproj con el nuevo nombre del paquete ClickHouse.Driver y la versión más reciente en NuGet.
  2. Actualiza en tu código todas las referencias de ClickHouse.Client a ClickHouse.Driver.

Versiones de .NET compatibles

ClickHouse.Driver es compatible con las siguientes versiones de .NET:
  • .NET 6.0
  • .NET 8.0
  • .NET 9.0
  • .NET 10.0

Instalación

Instale el paquete desde NuGet:
dotnet add package ClickHouse.Driver
O bien, usa el Administrador de paquetes NuGet:
Install-Package ClickHouse.Driver

Inicio rápido

using ClickHouse.Driver;

// Crear un cliente (normalmente como singleton)
using var client = new ClickHouseClient("Host=my.clickhouse;Protocol=https;Port=8443;Username=user");

// Ejecutar una consulta
var version = await client.ExecuteScalarAsync("SELECT version()");
Console.WriteLine(version);

Configuración

Hay dos formas de configurar su conexión a ClickHouse:
  • Cadena de conexión: pares clave/valor separados por punto y coma que especifican el host, las credenciales de autenticación y otras opciones de conexión.
  • Objeto ClickHouseClientSettings: objeto de configuración fuertemente tipado que puede cargarse desde archivos de configuración o establecerse en el código.
A continuación se muestra una lista completa de todas las opciones de configuración, sus valores predeterminados y sus efectos.

Configuración de la conexión

PropiedadTipoPredeterminadoClave de la cadena de conexiónDescripción
Hoststring"localhost"HostNombre de host o dirección IP del servidor de ClickHouse
Portushort8123 (HTTP) / 8443 (HTTPS)PortNúmero de puerto; los valores predeterminados dependen del protocolo
Usernamestring"default"UsernameNombre de usuario para la autenticación
Passwordstring""PasswordContraseña de autenticación
Databasestring""DatabaseBase de datos predeterminada; si está vacía, se usa la predeterminada del servidor o del usuario
Protocolstring"http"ProtocolProtocolo de conexión: "http" o "https"
PathstringnullPathRuta de la URL para entornos con proxy inverso (p. ej., /clickhouse)
TimeoutTimeSpan2 minutosTimeoutTiempo de espera de la operación (se almacena como segundos en la cadena de conexión)

Formato de datos y serialización

PropiedadTipoPredeterminadoClave de la cadena de conexiónDescripción
UseCompressionbooltrueCompressionHabilita la compresión gzip para la transferencia de datos
UseCustomDecimalsbooltrueUseCustomDecimalsUsa ClickHouseDecimal para precisión arbitraria; si es false, usa decimal de .NET (límite de 128 bits)
ReadStringsAsByteArraysboolfalseReadStringsAsByteArraysLee las columnas String y FixedString como byte[] en lugar de string; útil para datos binarios
UseFormDataParametersboolfalseUseFormDataParametersEnvía los parámetros como datos de formulario en lugar de la cadena de consulta de la URL
ParameterTypeResolverIParameterTypeResolvernullResolver personalizado para la correspondencia de tipos de parámetros con estilo @; consulta Correspondencia personalizada de tipos de parámetros
JsonReadModeJsonReadModeBinaryJsonReadModeCómo se devuelven los datos JSON: Binary (devuelve JsonObject) o String (devuelve la cadena JSON sin procesar)
JsonWriteModeJsonWriteModeStringJsonWriteModeCómo se envían los datos JSON: String (serializa mediante JsonSerializer, acepta cualquier entrada) o Binary (solo POCO registrados con indicaciones de tipo)

Gestión de sesiones

PropiedadTipoPredeterminadoClave de la cadena de conexiónDescripción
UseSessionboolfalseUseSessionHabilita sesiones con estado; serializa las solicitudes
SessionIdstringnullSessionIdID de sesión; genera automáticamente un GUID si es null y UseSession es true
El indicador UseSession habilita la persistencia de la sesión del servidor, lo que permite usar sentencias SET y tablas temporales. Las sesiones se reinician tras 60 segundos de inactividad (timeout predeterminado). La duración de la sesión puede ampliarse configurando ajustes de sesión mediante sentencias de ClickHouse o la configuración del servidor.La clase ClickHouseConnection normalmente permite operaciones en paralelo (varios hilos pueden ejecutar consultas de forma concurrente). Sin embargo, habilitar el indicador UseSession lo limita a una sola consulta activa por conexión en un momento dado (esta es una limitación del lado del servidor).

Seguridad

PropiedadTipoPredeterminadoClave de la cadena de conexiónDescripción
SkipServerCertificateValidationboolfalseOmite la validación del certificado HTTPS; no debe usarse en producción

Configuración del cliente HTTP

PropiedadTipoValor predeterminadoClave de la cadena de conexiónDescripción
HttpClientHttpClientnullInstancia personalizada de HttpClient ya configurada
HttpClientFactoryIHttpClientFactorynullFábrica personalizada para crear instancias de HttpClient
HttpClientNamestringnullNombre para que HttpClientFactory cree un cliente concreto

Registro y depuración

PropiedadTipoPredeterminadoClave de cadena de conexiónDescripción
LoggerFactoryILoggerFactorynullFábrica de registradores para el registro de diagnósticos
EnableDebugModeboolfalseActiva las trazas de red de .NET (requiere LoggerFactory con el nivel configurado en Trace); impacto significativo en el rendimiento

Ajustes personalizados y roles

PropiedadTipoPredeterminadoClave de cadena de conexiónDescripción
CustomSettingsIDictionary<string, object>Vacíoprefijo set_*ajustes del servidor de ClickHouse; consulta la nota a continuación
RolesIReadOnlyList<string>VacíoRolesroles de ClickHouse separados por comas (p. ej., Roles=admin,reader)
Al usar una cadena de conexión para establecer ajustes personalizados, usa el prefijo set_, p. ej., “set_max_threads=4”. Al usar un objeto ClickHouseClientSettings, no uses el prefijo set_.Para ver la lista completa de ajustes disponibles, consulta aquí.

Ejemplos de cadenas de conexión

Conexión básica

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

Con ajustes personalizados de ClickHouse

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

QueryOptions

QueryOptions permite anular la configuración del cliente para una consulta concreta. Todas las propiedades son opcionales y solo anulan los valores predeterminados del cliente cuando se especifican.
PropiedadTipoDescripción
QueryIdstringIdentificador de consulta personalizado para el seguimiento en system.query_log o para cancelarla
DatabasestringAnula la base de datos predeterminada para esta consulta
RolesIReadOnlyList<string>Anula los roles del cliente para esta consulta
CustomSettingsIDictionary<string, object>Configuración del servidor de ClickHouse para esta consulta (por ejemplo, max_threads)
CustomHeadersIDictionary<string, string>Encabezados HTTP adicionales para esta consulta
UseSessionbool?Anula el comportamiento de la sesión para esta consulta
SessionIdstringID de sesión para esta consulta (requiere UseSession = true)
BearerTokenstringAnula el token de autenticación para esta consulta
ParameterTypeResolverIParameterTypeResolverAnula el resolver a nivel de cliente para la correspondencia de tipos de parámetros con estilo @; consulte Correspondencia personalizada de tipos de parámetros
MaxExecutionTimeTimeSpan?Timeout de la consulta en el servidor (se pasa como la configuración max_execution_time); el servidor cancela la consulta si se supera
Ejemplo:
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 amplía QueryOptions con opciones específicas para operaciones de inserción masiva mediante InsertBinaryAsync.
PropiedadTypePredeterminadoDescripción
BatchSizeint100,000Número de filas por lote
MaxDegreeOfParallelismint1Número de cargas de lotes en paralelo
FormatRowBinaryFormatRowBinaryFormato binario: RowBinary o RowBinaryWithDefaults
ColumnTypesIReadOnlyDictionary<string, string>nullNombre de columna → cadena de tipo de ClickHouse. Omite la consulta de sondeo del esquema cuando se especifica.
UseSchemaCacheboolfalseAlmacena en caché el esquema completo de la tabla para cada par (base de datos, tabla) durante la vida útil del cliente.
Todas las propiedades de QueryOptions también están disponibles en InsertOptions. Ejemplo:
var insertOptions = new InsertOptions
{
    BatchSize = 50_000,
    MaxDegreeOfParallelism = 4,
    QueryId = "bulk-import-001"
};

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

Omitir la consulta de sondeo del esquema

De forma predeterminada, InsertBinaryAsync envía una consulta SELECT ... WHERE 1=0 antes de cada inserción para detectar los tipos de columna. Para escenarios de alto rendimiento, puedes eliminar esta sobrecarga con dos opciones: Opción 1: Proporcionar los tipos de columna explícitamente Cuando conoces el esquema de la tabla en tiempo de compilación, pásalo directamente mediante ColumnTypes. No se envía ninguna consulta de esquema en absoluto:
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);
Opción 2: Almacenar en caché el esquema Cuando realices inserciones repetidas en la misma tabla, establece UseSchemaCache = true para consultar el esquema una sola vez y reutilizarlo en las inserciones posteriores de la misma instancia de ClickHouseClient:
var options = new InsertOptions { UseSchemaCache = true };

// La primera llamada obtiene el esquema del servidor
await client.InsertBinaryAsync("my_table", columns, batch1, options);

// La segunda llamada reutiliza el esquema en caché — sin ida y vuelta adicional
await client.InsertBinaryAsync("my_table", columns, batch2, options);
  • ColumnTypes tiene prioridad sobre UseSchemaCache. Si se configuran ambos, se usan los tipos explícitos.
  • La caché de esquema no detecta cambios realizados con ALTER TABLE. Si modifica el esquema de la tabla, cree un nuevo ClickHouseClient o evite UseSchemaCache para esa tabla.
  • La caché se limita a la instancia de ClickHouseClient y se indexa por (database, table). Los distintos subconjuntos de columnas de una misma tabla comparten un único esquema en caché.

ClickHouseClient

ClickHouseClient es la API recomendada para interactuar con ClickHouse. Es seguro para subprocesos, está diseñado para usarse como singleton y gestiona internamente el pool de conexiones HTTP.

Crear un cliente

Cree un ClickHouseClient con una cadena de conexión o un objeto ClickHouseClientSettings. Consulte la sección Configuración para ver las opciones disponibles. Los detalles de su servicio de ClickHouse Cloud están disponibles en la consola de ClickHouse Cloud. Seleccione un servicio y haga clic en Connect: Elija C#. Los detalles de la conexión se muestran a continuación. Si utiliza ClickHouse autogestionado, los detalles de la conexión los establece el administrador de ClickHouse. Usar una cadena de conexión:
using ClickHouse.Driver;

using var client = new ClickHouseClient("Host=localhost;Username=default;Password=secret");
O bien, usando ClickHouseClientSettings:
using ClickHouse.Driver;

var settings = new ClickHouseClientSettings
{
    Host = "localhost",
    Username = "default",
    Password = "secret"
};
using var client = new ClickHouseClient(settings);
Para escenarios con inyección de dependencias, use IHttpClientFactory:
// En tu configuración de inyección de dependencias (DI)
services.AddHttpClient("ClickHouse", client =>
{
    client.Timeout = TimeSpan.FromMinutes(5);
}).ConfigurePrimaryHttpMessageHandler(() => new HttpClientHandler
{
    AutomaticDecompression = DecompressionMethods.GZip | DecompressionMethods.Deflate
});

// Crear el cliente con factory
var factory = serviceProvider.GetRequiredService<IHttpClientFactory>();
var client = new ClickHouseClient("Host=localhost", factory, "ClickHouse");
ClickHouseClient está diseñado para ser de larga duración y para compartirse en toda la aplicación. Créelo una sola vez (normalmente como un singleton) y reutilícelo para todas las operaciones de base de datos. El cliente administra internamente el pool de conexiones HTTP.

Ejecutar consultas

Use ExecuteNonQueryAsync para las sentencias que no devuelven resultados:
// Crear una tabla
await client.ExecuteNonQueryAsync(
    "CREATE TABLE IF NOT EXISTS default.my_table (id Int64, name String) ENGINE = Memory"
);

// Eliminar una tabla
await client.ExecuteNonQueryAsync("DROP TABLE IF EXISTS default.my_table");
Usa ExecuteScalarAsync para obtener un único valor:
var count = await client.ExecuteScalarAsync("SELECT count() FROM default.my_table");
Console.WriteLine($"Número de filas: {count}");

var version = await client.ExecuteScalarAsync("SELECT version()");
Console.WriteLine($"Versión del servidor: {version}");

Insertar datos

Inserciones parametrizadas

Inserta datos mediante consultas parametrizadas con ExecuteNonQueryAsync. Los tipos de los parámetros deben especificarse en el SQL usando la sintaxis {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
);

Inserciones masivas

Use InsertBinaryAsync para insertar un gran número de filas de forma eficiente. Transmite los datos mediante el formato binario nativo de filas de ClickHouse, admite envíos por lotes en paralelo y evita los errores de “URL too long” que pueden producirse con consultas parametrizadas.
// Preparar datos como IEnumerable<object[]>
var rows = Enumerable.Range(0, 1_000_000)
    .Select(i => new object[] { (long)i, $"value{i}" });

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

// Inserción básica
long rowsInserted = await client.InsertBinaryAsync("default.my_table", columns, rows);
Console.WriteLine($"Rows inserted: {rowsInserted}");
Para grandes volúmenes de datos, configure el procesamiento por lotes y el paralelismo con InsertOptions:
var options = new InsertOptions
{
    BatchSize = 100_000,           // Filas por lote (predeterminado: 100,000)
    MaxDegreeOfParallelism = 4     // Cargas de lotes en paralelo (predeterminado: 1)
};
  • El cliente obtiene automáticamente la estructura de la tabla mediante SELECT * FROM <table> WHERE 1=0 antes de insertar. Los valores proporcionados deben coincidir con los tipos de las columnas de destino. Para omitir esta consulta, use InsertOptions.ColumnTypes o InsertOptions.UseSchemaCache.
  • Cuando MaxDegreeOfParallelism > 1, los batches se cargan en paralelo. Las sesiones no son compatibles con la inserción en paralelo; desactive las sesiones o establezca MaxDegreeOfParallelism = 1.
  • Use RowBinaryFormat.RowBinaryWithDefaults en InsertOptions.Format si desea que el servidor aplique valores DEFAULT a las columnas no proporcionadas.

Inserciones con POCO

En lugar de construir arrays object[], puede insertar directamente objetos POCO fuertemente tipados. Registre el tipo una sola vez y luego pase IEnumerable<T>:
// Defina un POCO que se corresponda con las columnas de su tabla
public class SensorReading
{
    public ulong Id { get; set; }
    public string SensorName { get; set; }
    public double Value { get; set; }
    public DateTime Timestamp { get; set; }
}

// Registre el tipo (una vez durante la vida útil del cliente)
client.RegisterBinaryInsertType<SensorReading>();

// Inserte directamente: los nombres de las columnas se obtienen de los nombres de las propiedades
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);
De forma predeterminada, todas las propiedades públicas de lectura se asignan a columnas mediante una coincidencia estricta de nombres que distingue entre mayúsculas y minúsculas. Puede personalizar la asignación con atributos:
public class Event
{
    [ClickHouseColumn(Name = "event_id")]     // Mapear a una columna con nombre distinto
    public ulong Id { get; set; }

    [ClickHouseColumn(Type = "LowCardinality(String)")]  // Tipo ClickHouse explícito
    public string Category { get; set; }

    public string Payload { get; set; }

    [ClickHouseNotMapped]                     // Excluir del insert
    public string InternalTag { get; set; }
}
AtributoPropósito
[ClickHouseColumn(Name = "...")]Sobrescribe el nombre de la columna de destino
[ClickHouseColumn(Type = "...")]Declara explícitamente el tipo de ClickHouse
[ClickHouseNotMapped]Excluye la propiedad de la inserción
Cuando todas las propiedades mapeadas especifican un Type explícito, la consulta de sondeo del esquema se omite por completo. Cuando solo algunas propiedades tienen tipos explícitos, el driver recurre a la consulta de sondeo del esquema para el conjunto completo de columnas. InsertBinaryAsync<T> admite las mismas InsertOptions (agrupación en lotes, paralelismo, almacenamiento en caché del esquema) que la sobrecarga object[].
A diferencia de la sobrecarga object[], InsertBinaryAsync<T> no acepta una lista explícita de columnas. Las columnas se determinan a partir de las propiedades mapeadas del tipo registrado. Para controlar qué columnas se insertan, use [ClickHouseNotMapped] para excluir propiedades o [ClickHouseColumn(Name = "...")] para cambiarles el nombre.Si se establece ColumnTypes en InsertOptions, sobrescribirá los atributos del POCO.

Evolución del esquema

Las inserciones de POCO funcionan sin problemas cuando se añaden columnas a la tabla de destino después de registrar el tipo. Como el driver solo inserta las columnas asignadas por el POCO, cualquier columna nueva con DEFAULT (u otras expresiones predeterminadas) la rellena automáticamente el servidor. No se requieren cambios en el código ni volver a registrar nada.

Lectura de datos

Use ExecuteReaderAsync para ejecutar consultas SELECT. El ClickHouseDataReader devuelto proporciona acceso tipado a las columnas del resultado mediante métodos como GetInt64(), GetString() y GetFieldValue<T>(). Llame a Read() para avanzar a la fila siguiente. Devuelve false cuando ya no quedan más filas. Acceda a las columnas por índice (empezando en 0) o por nombre de columna.
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)}");
}

Parámetros SQL

En ClickHouse, el formato estándar de los parámetros en las consultas SQL es {parameter_name:DataType}. Ejemplos:
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)})
Los parámetros SQL ‘bind’ se pasan como parámetros de consulta en la URI HTTP, por lo que usar demasiados puede provocar una excepción de “URL demasiado larga”. Use InsertBinaryAsync para la inserción masiva de datos y así evitar esta limitación.

ID de consulta

A cada consulta se le asigna un query_id único que puede usarse para obtener datos de la tabla system.query_log o cancelar consultas de larga duración. Puedes especificar un ID de consulta personalizado mediante QueryOptions:
var options = new QueryOptions
{
    QueryId = $"report-{Guid.NewGuid()}"
};

var reader = await client.ExecuteReaderAsync(
    "SELECT * FROM large_table",
    parameters: null,
    options: options
);
Si especificas un QueryId personalizado, asegúrate de que sea único en cada llamada. Un GUID aleatorio es una buena opción.

Correspondencia personalizada de tipos de parámetros

Al usar parámetros con el estilo @ (por ejemplo, WHERE id = @id), el driver infiere automáticamente el tipo de ClickHouse a partir del tipo de valor de .NET. Por ejemplo, int se corresponde con Int32, y DateTime con DateTime. Para anular estos valores predeterminados, configure ParameterTypeResolver en ClickHouseClientSettings. Esto resulta útil cuando desea que todos los parámetros DateTime usen DateTime64(3) con precisión de milisegundos, o que todos los decimales usen una escala específica, sin tener que establecer ClickHouseType en cada parámetro individual. Uso de DictionaryParameterTypeResolver para correspondencias de tipos 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);     // Mapeado a DateTime64(3)
parameters.AddParameter("amount", 99.1234m);         // Mapeado a Decimal64(4)

await client.ExecuteReaderAsync("SELECT @dt, @amount", parameters);
IParameterTypeResolver personalizado para casos avanzados: Para una resolución basada en el valor o en el nombre, implemente directamente la interfaz IParameterTypeResolver. Devuelva null para que se aplique la inferencia predeterminada:
public class SmartDecimalResolver : IParameterTypeResolver
{
    public string ResolveType(Type clrType, object value, string parameterName)
    {
        if (clrType != typeof(decimal))
            return null; // Pasar a la inferencia predeterminada

        var scale = (decimal.GetBits((decimal)value)[3] >> 16) & 0x7F;
        return scale <= 4 ? $"Decimal64({scale})" : $"Decimal128({scale})";
    }
}
También puede configurar un resolver para una sola consulta mediante QueryOptions.ParameterTypeResolver. Cuando se establece, tiene prioridad sobre el resolver a nivel de cliente. Precedencia de la resolución de tipos: El resolver es un paso dentro de una cadena de precedencia. De mayor a menor prioridad:
  1. ClickHouseType explícito establecido en el parámetro
  2. Indicación de tipo SQL de la sintaxis {name:Type} en la consulta
  3. IParameterTypeResolver (de QueryOptions.ParameterTypeResolver, con fallback a ClickHouseClientSettings.ParameterTypeResolver)
  4. Inferencia de tipos integrada (TypeConverter.ToClickHouseType)
El resolver también funciona con la vía ClickHouseConnection de ADO.NET: las conexiones creadas desde el cliente heredan la configuración.

Transmisión sin procesar

Use ExecuteRawResultAsync para transmitir directamente los resultados de una consulta en un formato específico, omitiendo el lector de datos. Esto resulta útil para exportar datos a archivos o enviarlos a otros sistemas:
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();
Formatos comunes: JSONEachRow, CSV, TSV, Parquet, Native. Consulta la documentación sobre formatos para conocer todas las opciones.

Inserción desde flujo sin procesar

Use InsertRawStreamAsync para insertar datos directamente desde flujos de archivo o de memoria en formatos como CSV, JSON, Parquet o cualquier formato compatible con ClickHouse. Insertar desde un archivo 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"] // Opcional: especifica las columnas
);
Consulte la documentación sobre la configuración de formatos para conocer las opciones que permiten controlar el comportamiento de la ingestión de datos.

Más ejemplos

Para ver más ejemplos prácticos de uso, consulta el directorio de ejemplos en el repositorio de GitHub.

ADO.NET

La biblioteca ofrece compatibilidad completa con ADO.NET mediante ClickHouseConnection, ClickHouseCommand y ClickHouseDataReader. Esta API es necesaria para la integración con ORM (Dapper, Linq2db) y cuando necesita las abstracciones estándar de bases de datos de .NET.

Gestión del ciclo de vida con ClickHouseDataSource

Cree siempre conexiones desde un ClickHouseDataSource para garantizar una gestión correcta del ciclo de vida y del pool de conexiones. El DataSource administra internamente un único ClickHouseClient, y todas las conexiones comparten su pool de conexiones HTTP.
using ClickHouse.Driver.ADO;

// Crear DataSource una vez (registrar como singleton en DI)
var dataSource = new ClickHouseDataSource("Host=localhost;Username=default;Password=secret");

// Crear conexiones ligeras según sea necesario
await using var connection = await dataSource.OpenConnectionAsync();

// Usar la conexión
await using var command = connection.CreateCommand("SELECT version()");
var version = await command.ExecuteScalarAsync();
Para la inyección de dependencias:
// En Startup.cs o Program.cs
services.AddSingleton(sp =>
{
    var factory = sp.GetRequiredService<IHttpClientFactory>();
    return new ClickHouseDataSource("Host=localhost", factory, "ClickHouse");
});

// En el servicio
public class MyService
{
    private readonly ClickHouseDataSource _dataSource;

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

    public async Task DoWorkAsync()
    {
        await using var connection = await _dataSource.OpenConnectionAsync();
        // Usa la conexión...
    }
}
No cree ClickHouseConnection directamente en producción. Cada instanciación directa crea un nuevo cliente HTTP y un nuevo pool de conexiones, lo que puede provocar agotamiento de sockets con carga elevada:
// NO HAGA ESTO: crea un pool de conexiones nuevo cada vez
using var conn = new ClickHouseConnection("Host=localhost");
await conn.OpenAsync();
En su lugar, use siempre ClickHouseDataSource o comparta una sola instancia de ClickHouseClient.

Uso de ClickHouseCommand

Cree comandos a partir de una conexión para ejecutar SQL:
await using var connection = await dataSource.OpenConnectionAsync();

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

// Ejecutar y leer resultados
await using var reader = await command.ExecuteReaderAsync();
while (reader.Read())
{
    Console.WriteLine($"Name: {reader.GetString("name")}");
}
Métodos del comando:
  • ExecuteNonQueryAsync() - Para INSERT, UPDATE, DELETE y sentencias DDL
  • ExecuteScalarAsync() - Devuelve la primera columna de la primera fila
  • ExecuteReaderAsync() - Devuelve un ClickHouseDataReader para recorrer los resultados

Uso de ClickHouseDataReader

ClickHouseDataReader proporciona acceso tipado a los resultados de la consulta:
await using var reader = await command.ExecuteReaderAsync();

while (reader.Read())
{
    // Acceso por índice de columna
    var id = reader.GetInt64(0);
    var name = reader.GetString(1);

    // Acceso por nombre de columna
    var email = reader.GetString("email");

    // Acceso genérico
    var timestamp = reader.GetFieldValue<DateTime>("created_at");

    // Comprobar si es NULL
    if (!reader.IsDBNull("optional_field"))
    {
        var value = reader.GetString("optional_field");
    }
}

Buenas prácticas

Tiempo de vida de las conexiones y pool de conexiones

ClickHouse.Driver usa System.Net.Http.HttpClient internamente. HttpClient tiene un pool de conexiones por endpoint. Como consecuencia:
  • Las sesiones de la base de datos se multiplexan a través de conexiones HTTP administradas por el pool de conexiones.
  • El pool recicla automáticamente las conexiones HTTP.
  • Las conexiones pueden permanecer activas después de desechar los objetos ClickHouseClient o ClickHouseConnection.
Patrones recomendados:
EscenarioEnfoque recomendado
Uso generalUse un ClickHouseClient singleton
ADO.NET / ORMsUse ClickHouseDataSource (crea conexiones que comparten el mismo pool)
Entornos de DIRegistre ClickHouseClient o ClickHouseDataSource como singleton con IHttpClientFactory
Al usar un HttpClient o HttpClientFactory personalizado, asegúrese de que PooledConnectionIdleTimeout esté configurado con un valor menor que el keep_alive_timeout del servidor para evitar errores debidos a conexiones semicerradas. El valor predeterminado de keep_alive_timeout en implementaciones de Cloud es de 10 segundos.
Evite crear varias instancias de ClickHouseClient o instancias independientes de ClickHouseConnection sin un HttpClient compartido. Cada instancia crea su propio pool de conexiones.

Gestión de DateTime

  1. Usa UTC siempre que sea posible. Almacena las marcas de tiempo como columnas DateTime('UTC') y usa DateTimeKind.Utc en tu código. Esto elimina la ambigüedad de la zona horaria.
  2. Usa DateTimeOffset para gestionar explícitamente la zona horaria. Siempre representa un instante específico e incluye la información de desplazamiento.
  3. Especifica la zona horaria en las indicaciones de tipo de SQL. Al usar parámetros con valores DateTime Unspecified destinados a columnas que no son UTC, incluye la zona horaria en el SQL:
    var parameters = new ClickHouseParameterCollection();
    parameters.AddParameter("dt", myDateTime);
    
    await client.ExecuteNonQueryAsync(
        "INSERT INTO table (dt) VALUES ({dt:DateTime('Europe/Amsterdam')})",
        parameters
    );
    

Inserciones asíncronas

Las inserciones asíncronas trasladan la responsabilidad de agrupar en lotes del cliente al servidor. En lugar de requerir el agrupamiento en lotes del lado del cliente, el servidor guarda en un búfer los datos entrantes y los vuelca al almacenamiento en función de umbrales configurables. Esto resulta útil en escenarios de alta concurrencia, como las cargas de trabajo de observabilidad, donde muchos agentes envían payloads pequeños. Habilite las inserciones asíncronas mediante CustomSettings o la cadena de conexión:
// Usando CustomSettings
var settings = new ClickHouseClientSettings("Host=localhost");
settings.CustomSettings["async_insert"] = 1;
settings.CustomSettings["wait_for_async_insert"] = 1; // Recomendado: esperar confirmación de flush

// O mediante connection string
// "Host=localhost;set_async_insert=1;set_wait_for_async_insert=1"
Dos modos (controlados por wait_for_async_insert):
ModoComportamientoCaso de uso
wait_for_async_insert=1La inserción devuelve después de que los datos se escriben en disco. Los errores se devuelven al cliente.Recomendado para la mayoría de las cargas de trabajo
wait_for_async_insert=0La inserción devuelve inmediatamente cuando los datos se almacenan en el búfer. No se garantiza que los datos se persistan.Solo cuando la pérdida de datos sea aceptable
Con wait_for_async_insert=0, los errores solo aparecen durante el vaciado y no pueden rastrearse hasta la inserción original. El cliente tampoco proporciona contrapresión, lo que puede sobrecargar el servidor.
Configuraciones clave:
ConfiguraciónDescripción
async_insert_max_data_sizeVacía el búfer cuando alcanza este tamaño (bytes)
async_insert_busy_timeout_msVacía el búfer cuando se alcanza este tiempo de espera (milisegundos)
async_insert_max_query_numberVacía el búfer cuando se acumula esta cantidad de consultas

Sesiones

Habilita las sesiones solo cuando necesites funcionalidades con estado en el servidor, por ejemplo:
  • Tablas temporales (CREATE TEMPORARY TABLE)
  • Mantener el contexto de la consulta entre varias sentencias
  • Ajustes a nivel de sesión (SET max_threads = 4)
Cuando las sesiones están habilitadas, las solicitudes se serializan para evitar el uso concurrente de la misma sesión. Esto añade sobrecarga a las cargas de trabajo que no requieren estado de sesión.
var settings = new ClickHouseClientSettings
{
    Host = "localhost",
    UseSession = true,
    SessionId = "my-session", // Opcional -- se generará automáticamente si no se especifica
};

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)"
);
Uso de ADO.NET (para compatibilidad con 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();

Tipos de datos compatibles

ClickHouse.Driver admite todos los tipos de datos de ClickHouse. Las tablas siguientes muestran la correspondencia entre los tipos de ClickHouse y los tipos nativos de .NET al leer datos de la base de datos.

Correspondencia de tipos: lectura desde ClickHouse

Tipos enteros

Tipo de ClickHouseTipo de .NET
Int8sbyte
UInt8byte
Int16short
UInt16ushort
Int32int
UInt32uint
Int64long
UInt64ulong
Int128BigInteger
UInt128BigInteger
Int256BigInteger
UInt256BigInteger

Tipos de coma flotante

Tipo de ClickHouseTipo de .NET
Float32float
Float64double
BFloat16float

Tipos decimales

Tipo de ClickHouseTipo de .NET
Decimal(P, S)decimal / ClickHouseDecimal
Decimal32(S)decimal / ClickHouseDecimal
Decimal64(S)decimal / ClickHouseDecimal
Decimal128(S)decimal / ClickHouseDecimal
Decimal256(S)decimal / ClickHouseDecimal
La conversión de tipos decimales se controla con la configuración UseCustomDecimals.

Tipo booleano

Tipo de ClickHouseTipo de .NET
Boolbool

Tipos String

Tipo de ClickHouseTipo de .NET
Stringstring
FixedString(N)string
De forma predeterminada, las columnas String y FixedString(N) se devuelven como string. Establezca ReadStringsAsByteArrays=true en la cadena de conexión para leerlas como byte[] en su lugar. Esto es útil cuando se almacenan datos binarios que podrían no ser UTF-8 válidos.

Tipos de fecha y hora

ClickHouse Type.NET Type
DateDateTime
Date32DateTime
DateTimeDateTime
DateTime32DateTime
DateTime64DateTime
TimeTimeSpan
Time64TimeSpan
ClickHouse almacena internamente los valores DateTime y DateTime64 como marcas de tiempo Unix (segundos o fracciones de segundo desde la época Unix). Aunque el almacenamiento siempre está en UTC, las columnas pueden tener una zona horaria asociada que afecta a cómo se muestran e interpretan los valores. Al leer valores DateTime, la propiedad DateTime.Kind se establece en función de la zona horaria de la columna:
Definición de columnaDateTime.Kind devueltoNotas
DateTime('UTC')UtcZona horaria UTC explícita
DateTime('Europe/Amsterdam')UnspecifiedSe aplica el desplazamiento
DateTimeUnspecifiedSe conserva la hora tal cual
Para las columnas que no son UTC, el DateTime devuelto representa la hora local en esa zona horaria. Usa ClickHouseDataReader.GetDateTimeOffset() para obtener un DateTimeOffset con el desplazamiento correcto para esa zona horaria:
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)
Para las columnas sin una zona horaria explícita (es decir, DateTime en lugar de DateTime('Europe/Amsterdam')), el driver devuelve un DateTime con Kind=Unspecified. Esto conserva la hora local exactamente tal como está almacenada, sin hacer suposiciones sobre la zona horaria. Si necesita un comportamiento con reconocimiento de zona horaria para columnas sin una zona horaria explícita, haga una de estas dos cosas:
  1. Use zonas horarias explícitas en las definiciones de sus columnas: DateTime('UTC') o DateTime('Europe/Amsterdam')
  2. Aplique usted mismo la zona horaria después de leer el valor.

Tipo JSON

Tipo de ClickHouseTipo de .NETNotas
JsonJsonObjectPredeterminado (JsonReadMode=Binary)
JsonstringCuando JsonReadMode=String
El tipo de retorno de las columnas JSON está determinado por la configuración JsonReadMode:
  • Binary (predeterminado): Devuelve System.Text.Json.Nodes.JsonObject. Proporciona acceso estructurado a los datos JSON, pero los tipos especializados de ClickHouse (como direcciones IP, UUIDs y decimales grandes) se convierten a su representación en cadena dentro de la estructura JSON.
  • String: Devuelve el JSON sin procesar como string. Conserva la representación exacta del JSON de ClickHouse, lo que resulta útil cuando necesitas pasar el JSON sin analizarlo o cuando quieres encargarte tú mismo de la deserialización.
// Configurar el modo string mediante configuración
var settings = new ClickHouseClientSettings("Host=localhost")
{
    JsonReadMode = JsonReadMode.String
};

// O mediante cadena de conexión
// "Host=localhost;JsonReadMode=String"

Otros tipos

Tipo de ClickHouseTipo de .NET
UUIDGuid
IPv4IPAddress
IPv6IPAddress
NothingDBNull
DynamicVea la nota
Array(T)T[]
Tuple(T1, T2, …)Tuple<T1, T2, ...> / LargeTuple
Map(K, V)Dictionary<K, V>
Nullable(T)T?
Enum8string
Enum16string
LowCardinality(T)Igual que T
SimpleAggregateFunctionIgual que el tipo subyacente
Nested(…)Tuple[]
Variant(T1, T2, …)Vea la nota
QBit(T, dimension)T[]
Los tipos Dynamic y Variant se convertirán al tipo correspondiente según el tipo subyacente real de cada fila.

Tipos de geometría

Tipo de ClickHouseTipo de .NET
PointTuple<double, double>
RingTuple<double, double>[]
LineStringTuple<double, double>[]
PolygonRing[]
MultiLineStringLineString[]
MultiPolygonPolygon[]
GeometryConsulte la nota
El tipo Geometry es un tipo Variant que puede contener cualquiera de los tipos de geometría. Se convertirá al tipo correspondiente.

Correspondencia de tipos: escritura en ClickHouse

Al insertar datos, el driver convierte los tipos de .NET en sus correspondientes tipos de ClickHouse. Las tablas siguientes muestran qué tipos de .NET se admiten para cada tipo de columna de ClickHouse.

Tipos enteros

Tipo de ClickHouseTipos .NET aceptadosNotas
Int8sbyte, cualquier tipo compatible con Convert.ToSByte()
UInt8byte, cualquier tipo compatible con Convert.ToByte()
Int16short, cualquier tipo compatible con Convert.ToInt16()
UInt16ushort, cualquier tipo compatible con Convert.ToUInt16()
Int32int, cualquier tipo compatible con Convert.ToInt32()
UInt32uint, cualquier tipo compatible con Convert.ToUInt32()
Int64long, cualquier tipo compatible con Convert.ToInt64()
UInt64ulong, cualquier tipo compatible con Convert.ToUInt64()
Int128BigInteger, decimal, double, float, int, uint, long, ulong, cualquier tipo compatible con Convert.ToInt64()
UInt128BigInteger, decimal, double, float, int, uint, long, ulong, cualquier tipo compatible con Convert.ToInt64()
Int256BigInteger, decimal, double, float, int, uint, long, ulong, cualquier tipo compatible con Convert.ToInt64()
UInt256BigInteger, decimal, double, float, int, uint, long, ulong, cualquier tipo compatible con Convert.ToInt64()

Tipos de coma flotante

Tipo de ClickHouseTipos de .NET aceptadosNotas
Float32float, cualquier tipo compatible con Convert.ToSingle()
Float64double, cualquier tipo compatible con Convert.ToDouble()
BFloat16float, cualquier tipo compatible con Convert.ToSingle()Se trunca al formato brain float de 16 bits

Tipo booleano

Tipo de ClickHouseTipos de .NET aceptadosNotas
Boolbool

Tipos de cadena

Tipo de ClickHouseTipos de .NET aceptadosNotas
Stringstring, byte[], ReadOnlyMemory<byte>, StreamLos tipos binarios se escriben directamente; los streams pueden ser posicionables o no
FixedString(N)string, byte[], ReadOnlyMemory<byte>, StreamString se codifica en UTF-8 y se rellena; los tipos binarios deben tener exactamente N bytes

Tipos de fecha y hora

Tipo de ClickHouseTipos de .NET aceptadosNotas
DateDateTime, DateTimeOffset, DateOnly, tipos de NodaTimeConvertido a días Unix como UInt16
Date32DateTime, DateTimeOffset, DateOnly, tipos de NodaTimeConvertido a días Unix como Int32
DateTimeDateTime, DateTimeOffset, DateOnly, tipos de NodaTimeConsulta más abajo para obtener detalles
DateTime32DateTime, DateTimeOffset, DateOnly, tipos de NodaTimeIgual que DateTime
DateTime64DateTime, DateTimeOffset, DateOnly, tipos de NodaTimePrecisión basada en el parámetro Scale
TimeTimeSpan, intLimitado a ±999:59:59; int se trata como segundos
Time64TimeSpan, decimal, double, float, int, long, stringLa cadena se interpreta como [-]HHH:MM:SS[.fraction]; limitado a ±999:59:59.999999999
El driver respeta DateTime.Kind al escribir valores:
DateTime.KindParámetros HTTPCopia masiva
UtcSe conserva el instanteSe conserva el instante
LocalSe conserva el instanteSe conserva el instante
UnspecifiedSe trata como hora de pared en la zona horaria del tipo de parámetro (UTC de forma predeterminada)Se trata como hora de pared en la zona horaria de la columna
Los valores DateTimeOffset siempre conservan el instante exacto. Ejemplo: DateTime UTC (se conserva el instante)
var utcTime = new DateTime(2024, 1, 15, 12, 0, 0, DateTimeKind.Utc);
// Almacenado como 12:00 UTC
// Leído desde la columna DateTime('Europe/Amsterdam'): 13:00 (UTC+1)
// Leído desde la columna DateTime('UTC'): 12:00 UTC
Ejemplo: DateTime sin especificar (hora local)
var wallClock = new DateTime(2024, 1, 15, 14, 30, 0, DateTimeKind.Unspecified);
// Escrito en la columna DateTime('Europe/Amsterdam'): almacenado como 14:30 hora de Ámsterdam
// Leído de la columna DateTime('Europe/Amsterdam'): 14:30
Recomendación: para obtener el comportamiento más simple y predecible, use DateTimeKind.Utc o DateTimeOffset para todas las operaciones con DateTime. Esto garantiza que su código funcione de forma coherente independientemente de la zona horaria del servidor, la zona horaria del cliente o la zona horaria de la columna.

Parámetros HTTP vs Bulk Copy

Hay una diferencia importante entre la vinculación de parámetros HTTP y Bulk Copy al escribir valores DateTime Unspecified: Bulk Copy conoce la zona horaria de la columna de destino e interpreta correctamente los valores Unspecified en esa zona horaria. HTTP Parameters no conocen automáticamente la zona horaria de la columna. Debe especificarla en la indicación de tipo de SQL:
// CORRECTO: Zona horaria en la indicación de tipo SQL - el tipo se extrae automáticamente
command.CommandText = "INSERT INTO table (dt_amsterdam) VALUES ({dt:DateTime('Europe/Amsterdam')})";
command.AddParameter("dt", myDateTime);

// INCORRECTO: Sin indicación de zona horaria, se interpreta como UTC
command.CommandText = "INSERT INTO table (dt_amsterdam) VALUES ({dt:DateTime})";
command.AddParameter("dt", myDateTime);
// ¡El valor de cadena "2024-01-15 14:30:00" se interpreta como UTC, no como hora de Ámsterdam!
DateTime.KindColumna de destinoParámetro HTTP (con indicación de zona horaria)Parámetro HTTP (sin indicación de zona horaria)Copia masiva
UtcUTCSe conserva el instanteSe conserva el instanteSe conserva el instante
UtcEurope/AmsterdamSe conserva el instanteSe conserva el instanteSe conserva el instante
LocalCualquieraSe conserva el instanteSe conserva el instanteSe conserva el instante
UnspecifiedUTCSe trata como UTCSe trata como UTCSe trata como UTC
UnspecifiedEurope/AmsterdamSe trata como hora de ÁmsterdamSe trata como UTCSe trata como hora de Ámsterdam

Tipos Decimal

Tipo de ClickHouseTipos de .NET aceptadosNotas
Decimal(P,S)decimal, ClickHouseDecimal, cualquier tipo compatible con Convert.ToDecimal()Lanza OverflowException si excede la precisión
Decimal32decimal, ClickHouseDecimal, cualquier tipo compatible con Convert.ToDecimal()Precisión máxima: 9
Decimal64decimal, ClickHouseDecimal, cualquier tipo compatible con Convert.ToDecimal()Precisión máxima: 18
Decimal128decimal, ClickHouseDecimal, cualquier tipo compatible con Convert.ToDecimal()Precisión máxima: 38
Decimal256decimal, ClickHouseDecimal, cualquier tipo compatible con Convert.ToDecimal()Precisión máxima: 76

Tipo JSON

Tipo de ClickHouseTipos de .NET aceptadosNotas
Jsonstring, JsonObject, JsonNode, cualquier objetoEl comportamiento depende del ajuste JsonWriteMode
El comportamiento al escribir JSON está controlado por el ajuste JsonWriteMode:
Tipo de entradaJsonWriteMode.String (predeterminado)JsonWriteMode.Binary
stringSe pasa directamenteLanza ArgumentException
JsonObjectSe serializa con ToJsonString()Lanza ArgumentException
JsonNodeSe serializa con ToJsonString()Lanza ArgumentException
POCO registradoSe serializa con JsonSerializer.Serialize()Codificación binaria con indicaciones de tipo; se admiten atributos de ruta personalizados
POCO no registrado / objeto anónimoSe serializa con JsonSerializer.Serialize()Lanza ClickHouseJsonSerializationException
  • String (predeterminado): Acepta string, JsonObject, JsonNode o cualquier objeto. Todas las entradas se serializan mediante System.Text.Json.JsonSerializer y se envían como cadenas JSON para que el servidor las procese. Este es el modo más flexible y funciona sin registrar tipos.
  • Binary: Solo acepta tipos POCO registrados. Los datos se convierten en el cliente al formato JSON binario de ClickHouse, con compatibilidad completa con indicaciones de tipo. Requiere llamar a connection.RegisterJsonSerializationType<T>() antes de usarlo. Escribir valores string o JsonNode en este modo lanza ArgumentException.
// El modo String predeterminado funciona con cualquier entrada
await client.InsertBinaryAsync(
    "my_table",
    new[] { "id", "data" },
    new[] { new object[] { 1u, new { name = "test", value = 42 } } }
);

// El modo Binary requiere habilitación explícita y registro de tipos
var settings = new ClickHouseClientSettings("Host=localhost")
{
    JsonWriteMode = JsonWriteMode.Binary
};
using var client = new ClickHouseClient(settings);
client.RegisterJsonSerializationType<MyPocoType>();
Columnas JSON tipadas
Cuando una columna JSON tiene indicaciones de tipo (p. ej., JSON(id UInt64, price Decimal128(2))), el driver usa estas indicaciones para serializar los valores respetando plenamente sus tipos. Esto preserva la precisión de tipos como UInt64, Decimal, UUID y DateTime64, que de otro modo la perderían al serializarse como JSON genérico.
Serialización de POCO
Los POCO se pueden escribir en columnas JSON de dos formas, según JsonWriteMode: Modo String (predeterminado): los POCO se serializan mediante System.Text.Json.JsonSerializer. No es necesario registrar tipos. Es el enfoque más sencillo y funciona con objetos anónimos. Modo binario: los POCO se serializan usando el formato JSON binario del driver, con compatibilidad completa con indicaciones de tipo. Los tipos deben registrarse con connection.RegisterJsonSerializationType<T>() antes de usarlos. Este modo admite asignaciones de rutas personalizadas mediante atributos:
  • [ClickHouseJsonPath("path")]: Asigna una propiedad a una ruta JSON personalizada. Es útil para estructuras anidadas o cuando el nombre de la propiedad difiere de la clave JSON deseada. Solo funciona en modo binario.
  • [ClickHouseJsonIgnore]: Excluye una propiedad de la serialización. Solo funciona en modo binario.
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; }  // No se serializa
}

// Para el modo Binary: registre el tipo y habilítelo
var settings = new ClickHouseClientSettings("Host=localhost") { JsonWriteMode = JsonWriteMode.Binary };
using var client = new ClickHouseClient(settings);
client.RegisterJsonSerializationType<UserEvent>();

// Insertar un POCO: se serializa a JSON con una estructura anidada mediante atributos de ruta personalizados
await client.InsertBinaryAsync(
    "events",
    new[] { "id", "data" },
    new[] { new object[] { 1u, new UserEvent { UserId = 123, UserName = "Alice", Timestamp = DateTime.UtcNow } } }
);
// JSON resultante: {"user": {"id": 123, "name": "Alice"}, "Timestamp": "2024-01-15T..."}
La coincidencia entre el nombre de la propiedad y las indicaciones de tipo de la columna es sensible a mayúsculas y minúsculas. Una propiedad UserId solo coincidirá con una indicación definida como UserId, no como userid. Esto sigue el comportamiento de ClickHouse, que permite que rutas como userName y UserName coexistan como campos independientes. Limitaciones (solo en modo Binary):
  • Los tipos POCO deben registrarse en la conexión con connection.RegisterJsonSerializationType<T>() antes de serializarse. Si se intenta serializar un tipo no registrado, se lanza ClickHouseJsonSerializationException.
  • Las propiedades de diccionario y array/lista requieren indicaciones de tipo en la definición de la columna para serializarse correctamente. Sin esas indicaciones, use el modo String.
  • Los valores NULL en las propiedades POCO solo se escriben cuando la ruta tiene una indicación de tipo Nullable(T) en la definición de la columna. ClickHouse no permite tipos Nullable dentro de rutas JSON dinámicas, por lo que las propiedades con valor NULL sin indicación se omiten.
  • Los atributos ClickHouseJsonPath y ClickHouseJsonIgnore se ignoran en modo String (solo funcionan en modo Binary).

Otros tipos

Tipo de ClickHouseTipos de .NET aceptadosNotas
UUIDGuid, stringLa cadena se interpreta como Guid
IPv4IPAddress, stringDebe ser IPv4; la cadena se analiza con IPAddress.Parse()
IPv6IPAddress, stringDebe ser IPv6; la cadena se analiza con IPAddress.Parse()
NothingAnyNo escribe nada (no-op)
DynamicNo admitido (lanza NotImplementedException)
Array(T)IList, nullnull escribe un array vacío
Tuple(T1, T2, …)ITuple, IListEl número de elementos debe coincidir con la aridad de la tupla
Map(K, V)IDictionary
Nullable(T)null, DBNull, o tipos aceptados por TEscribe un byte indicador de null antes del valor
Enum8string, sbyte, tipos numéricosLa cadena se busca en el diccionario del enum
Enum16string, short, tipos numéricosLa cadena se busca en el diccionario del enum
LowCardinality(T)Tipos aceptados por TSe delega en el tipo subyacente
SimpleAggregateFunctionTipos aceptados por el tipo subyacenteSe delega en el tipo subyacente
Nested(…)IList de tuplasEl número de elementos debe coincidir con el número de campos
Variant(T1, T2, …)Valor que coincida con uno de T1, T2, …Lanza ArgumentException si no coincide ningún tipo
QBit(T, dim)IListSe delega en Array; la dimensión es solo metadatos

Tipos de geometría

Tipo de ClickHouseTipos de .NET aceptadosNotas
PointSystem.Drawing.Point, ITuple, IList (2 elementos)
RingIList de Point
LineStringIList de Point
PolygonIList de Ring
MultiLineStringIList de LineString
MultiPolygonIList de Polygon
GeometryCualquier tipo de geometría anteriorVariante de todos los tipos de geometría

No admitido para escritura

Tipo de ClickHouseNotas
DynamicLanza NotImplementedException
AggregateFunctionLanza AggregateFunctionException

Manejo de tipos anidados

Los tipos anidados de ClickHouse (Nested(...)) se pueden leer y escribir usando la semántica de arrays.
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 }
);

Registro y diagnósticos

El cliente .NET de ClickHouse se integra con las abstracciones de Microsoft.Extensions.Logging para ofrecer un registro ligero y opcional. Cuando está habilitado, el driver emite mensajes estructurados sobre eventos del ciclo de vida de la conexión, la ejecución de comandos, las operaciones de transporte y las operaciones de inserción masiva. El registro es totalmente opcional: las aplicaciones que no configuran un logger siguen ejecutándose sin sobrecarga adicional.

Primeros pasos

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);

Uso de appsettings.json

Puede configurar los niveles de registro mediante la configuración estándar de .NET:
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);

Uso de la configuración en memoria

También puede configurar en el código la verbosidad del registro por categoría:
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);

Categorías y emisores

El driver usa categorías específicas para que puedas ajustar con precisión los niveles de registro de cada componente:
CategoríaOrigenAspectos destacados
ClickHouse.Driver.ConnectionClickHouseConnectionCiclo de vida de la conexión, selección de la factoría de clientes HTTP, apertura y cierre de conexiones, gestión de sesiones.
ClickHouse.Driver.CommandClickHouseCommandInicio y finalización de la ejecución de consultas, tiempos, ID de consulta, estadísticas del servidor y detalles de errores.
ClickHouse.Driver.TransportClickHouseConnectionSolicitudes HTTP streaming de bajo nivel, indicadores de compresión, códigos de estado de la respuesta y errores de transporte.
ClickHouse.Driver.ClientClickHouseClientInserción binaria, consultas y otras operaciones
ClickHouse.Driver.NetTraceTraceHelperTrazado de red, solo cuando el modo de depuración está habilitado

Ejemplo: Cómo diagnosticar problemas de conexión

{
    "Logging": {
        "LogLevel": {
            "ClickHouse.Driver.Connection": "Trace",
            "ClickHouse.Driver.Transport": "Trace"
        }
    }
}
Esto registrará:
  • Selección de la fábrica de clientes HTTP (pool predeterminado frente a conexión única)
  • Configuración del controlador HTTP (SocketsHttpHandler o HttpClientHandler)
  • Configuración del pool de conexiones (MaxConnectionsPerServer, PooledConnectionLifetime, etc.)
  • Configuración de timeout (ConnectTimeout, Expect100ContinueTimeout, etc.)
  • Configuración de SSL/TLS
  • Eventos de apertura/cierre de conexiones
  • Seguimiento del ID de sesión

Modo de depuración: tracing de red y diagnóstico

Para ayudar a diagnosticar problemas de red, la biblioteca del driver incluye un asistente que habilita el tracing de bajo nivel de los componentes internos de red de .NET. Para habilitarlo, debe pasar una LoggerFactory con el nivel establecido en Trace y establecer EnableDebugMode en true (o habilitarlo manualmente mediante la clase ClickHouse.Driver.Diagnostic.TraceHelper). Los eventos se registrarán en la categoría ClickHouse.Driver.NetTrace. Advertencia: esto generará logs extremadamente verbosos y afectará al rendimiento. No se recomienda habilitar el modo de depuración en producción.
var loggerFactory = LoggerFactory.Create(builder =>
{
    builder
        .AddConsole()
        .SetMinimumLevel(LogLevel.Trace); // Debe estar en el nivel Trace para ver los eventos de red
});

var settings = new ClickHouseClientSettings()
{
    LoggerFactory = loggerFactory,
    EnableDebugMode = true,  // Habilita el tracing de red de bajo nivel
};

OpenTelemetry

El driver ofrece compatibilidad integrada con el tracing distribuido de OpenTelemetry mediante la API de .NET System.Diagnostics.Activity. Cuando está habilitado, el driver emite spans para las operaciones de base de datos que pueden exportarse a backends de observabilidad como Jaeger o al propio ClickHouse (mediante el OpenTelemetry Collector).

Habilitar el tracing

En las aplicaciones ASP.NET Core, agregue el ActivitySource del driver de ClickHouse a su configuración de OpenTelemetry:
builder.Services.AddOpenTelemetry()
    .WithTracing(tracing => tracing
        .AddSource(ClickHouseDiagnosticsOptions.ActivitySourceName)  // Suscribirse a los spans del driver de ClickHouse
        .AddAspNetCoreInstrumentation()
        .AddOtlpExporter());             // O bien AddJaegerExporter(), etc.
Para aplicaciones de consola, pruebas o configuración manual:
using OpenTelemetry;
using OpenTelemetry.Trace;

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

Atributos del span

Cada span incluye atributos de base de datos estándar de OpenTelemetry, además de estadísticas de consulta específicas de ClickHouse que pueden usarse para depuración.
AtributoDescripción
db.systemSiempre "clickhouse"
db.nameNombre de la base de datos
db.userNombre de usuario
db.statementConsulta SQL (si está habilitada)
db.clickhouse.read_rowsFilas leídas por la consulta
db.clickhouse.read_bytesBytes leídos por la consulta
db.clickhouse.written_rowsFilas escritas por la consulta
db.clickhouse.written_bytesBytes escritos por la consulta
db.clickhouse.elapsed_nsTiempo de ejecución del lado del servidor en nanosegundos

Opciones de configuración

Controle el comportamiento del tracing con ClickHouseDiagnosticsOptions:
using ClickHouse.Driver.Diagnostic;

// Incluir sentencias SQL en spans (valor predeterminado: false por seguridad)
ClickHouseDiagnosticsOptions.IncludeSqlInActivityTags = true;

// Truncar sentencias SQL largas (valor predeterminado: 1000 caracteres)
ClickHouseDiagnosticsOptions.StatementMaxLength = 500;
Habilitar IncludeSqlInActivityTags puede exponer datos confidenciales en las trazas. Úselo con precaución en entornos de producción.

Configuración de TLS

Al conectarse a ClickHouse a través de HTTPS, puede configurar el comportamiento de TLS/SSL de varias formas.

Validación personalizada de certificados

Para entornos de producción que requieran una lógica personalizada de validación de certificados, proporcione su propio HttpClient con un controlador ServerCertificateCustomValidationCallback configurado:
using System.Net;
using System.Net.Security;
using ClickHouse.Driver;

var handler = new HttpClientHandler
{
    // Obligatorio cuando la compresión está activada (valor predeterminado)
    AutomaticDecompression = DecompressionMethods.GZip | DecompressionMethods.Deflate,

    ServerCertificateCustomValidationCallback = (message, cert, chain, sslPolicyErrors) =>
    {
        // Ejemplo: Aceptar la huella digital de un certificado concreto
        if (cert?.Thumbprint == "YOUR_EXPECTED_THUMBPRINT")
            return true;

        // Ejemplo: Aceptar certificados de un emisor concreto
        if (cert?.Issuer.Contains("YourOrganization") == true)
            return true;

        // Valor predeterminado: usar la validación estándar
        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);
Consideraciones importantes al proporcionar un HttpClient personalizado
  • Descompresión automática: Debes habilitar AutomaticDecompression si la compresión no está desactivada (la compresión está activada de forma predeterminada).
  • Tiempo de espera de inactividad: Configura PooledConnectionIdleTimeout con un valor inferior al keep_alive_timeout del servidor (10 segundos en ClickHouse Cloud) para evitar errores de conexión causados por conexiones semiabiertas.

Compatibilidad con los ORM

Los ORM requieren la API de ADO.NET (ClickHouseConnection). Para gestionar correctamente el ciclo de vida de la conexión, cree las conexiones desde un ClickHouseDataSource:
// Registrar DataSource como singleton
var dataSource = new ClickHouseDataSource("Host=localhost;Username=default");

// Crear conexiones para usarlas con el ORM
await using var connection = await dataSource.OpenConnectionAsync();
// Pasar la conexión al ORM...

Dapper

ClickHouse.Driver funciona con Dapper. El driver convierte automáticamente la sintaxis @parameter de Dapper a la sintaxis nativa {parameter:Type} de ClickHouse, e infiere los tipos a partir de los valores de .NET. Usa ClickHouseDataSource para gestionar correctamente el ciclo de vida de la conexión:
var dataSource = new ClickHouseDataSource("Host=localhost");
services.AddSingleton(dataSource); // Registrar como singleton en la DI

using var connection = dataSource.CreateConnection();

Estilos para pasar parámetros

Se admiten todos los estilos estándar de parámetros de Dapper: Objetos anónimos:
await connection.ExecuteAsync(
    "INSERT INTO users (id, name, balance) VALUES (@Id, @Name, @Balance)",
    new { Id = 1, Name = "alice", Balance = 3.14 });
Clases 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);
Diccionario:
var parameters = new Dictionary<string, object> { { "Id", 2 } };
var rows = await connection.QueryAsync<User>(
    "SELECT id, name FROM users WHERE id = @Id", parameters);
DynamicParameters (de un diccionario o de un objeto anónimo):
var dynParams = new DynamicParameters(new { Id = 1 });
// o bien: new DynamicParameters(new Dictionary<string, object> { { "Id", 1 } });

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

Consultas con POCOs

Dapper asigna columnas a propiedades por nombre (sin distinguir entre mayúsculas y minúsculas):
class User
{
    public int Id { get; set; }
    public string Name { get; set; }
    public double Balance { get; set; }
}

// De una tabla
var users = (await connection.QueryAsync<User>("SELECT id, name, balance FROM users")).ToList();

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

Sintaxis de parámetros nativa de ClickHouse

Cuando necesites un control explícito de los tipos, usa directamente en el SQL la sintaxis {param:Type} de ClickHouse con un Dictionary<string, object> para los valores de los parámetros. No combines la sintaxis @param con la sintaxis {param:Type} para el mismo parámetro.
var parameters = new Dictionary<string, object> { { "value", 42 } };
var result = await connection.QueryAsync<int>("SELECT {value:Int32}", parameters);

WHERE IN

La expansión nativa de IN de Dapper funciona:
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 reescribe esto como WHERE id IN (@Ids1, @Ids2, @Ids3), y el driver convierte cada parámetro expandido. La función has() de ClickHouse con un parámetro Array también funciona:
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);

Manejadores de tipos personalizados

Algunos tipos de ClickHouse, p. ej., ITuple, BigInteger y ClickHouseDecimal, requieren registrar manejadores al inicio:
// ClickHouseDecimal (para columnas Decimal64/128/256)
SqlMapper.AddTypeHandler(new ClickHouseDecimalHandler());

// BigInteger (para columnas Int128/Int256/UInt128/UInt256)
SqlMapper.AddTypeHandler(new BigIntegerHandler());

// IPAddress (para columnas IPv4/IPv6)
SqlMapper.AddTypeHandler(new IpAddressHandler());
Consulte el ejemplo de Dapper como ejemplo de implementación de un manejador de tipos.

Dapper.Contrib

GetAll<T>() y Get<T>(id) funcionan. Insert<T>() no: genera sintaxis de SQL Server (SCOPE_IDENTITY, []). En su lugar, se recomienda usar el método nativo 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);
Los nombres de las propiedades deben coincidir exactamente con los nombres de columna de ClickHouse (la coincidencia es sensible a mayúsculas y minúsculas).

Limitaciones

QuéEstadoDetalles
Tuple como resultadoFuncionaRequiere registrar SqlMapper.TypeHandler<ITuple>
Tuple como parámetroNo compatibleDapper no puede serializar ITuple/Tuple<> como valor de DbParameter
Tipos anidados como parámetroNo compatibleMismo motivo: Dapper rechaza los tipos complejos como valores de parámetro
Tipos Geo como parámetroNo compatiblePoint, Ring, Polygon, LineString, MultiLineString, MultiPolygon
Dapper.Contrib.Insert<T>()No compatibleGenera sintaxis específica de SQL Server
Tipo NothingNo compatibleNo tiene una representación significativa en .NET

Linq2db

Este driver es compatible con linq2db, un ORM ligero y un proveedor de LINQ para .NET. Consulta el sitio web del proyecto para obtener documentación detallada. Ejemplo de uso: Crea una DataConnection con el proveedor de 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);
Los mapeos de tablas pueden definirse mediante atributos o la API fluida. Si los nombres de la clase y de las propiedades coinciden exactamente con los nombres de la tabla y de las columnas, no se necesita ninguna configuración:
public class Product
{
    public int Id { get; set; }
    public string Name { get; set; }
    public decimal Price { get; set; }
}
Consultas:
await using var db = new DataConnection(options);

var products = await db.GetTable<Product>()
    .Where(p => p.Price > 100)
    .OrderByDescending(p => p.Name)
    .ToListAsync();
Copia masiva: Utilice BulkCopyAsync para realizar inserciones masivas de forma eficiente.
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

El proveedor oficial de Entity Framework Core para ClickHouse. Permite asignar clases de C# a tablas de ClickHouse, realizar consultas con LINQ e insertar datos mediante SaveChanges, todo ello con los patrones habituales de EF Core.
Este proveedor está en desarrollo activo. La versión actual admite consultas LINQ (incluidos JOIN, subconsultas y operaciones de conjuntos), INSERT mediante SaveChanges / BulkInsertAsync, migraciones con DDL completo (CREATE / ALTER / DROP) y la configuración del motor de tabla específica de ClickHouse. UPDATE / DELETE no son compatibles.

Instalación

dotnet add package ClickHouse.EntityFrameworkCore
Requiere .NET 10.0 y EF Core 10.

Inicio rápido

Define la entidad y DbContext, y luego haz consultas con 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");
}

// Consulta
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();

Tipos compatibles

CategoríaTipos de ClickHouseTipos de CLR
EnterosInt8Int64, UInt8UInt64sbyte, short, int, long, byte, ushort, uint, ulong
Enteros grandesInt128, Int256, UInt128, UInt256BigInteger
Punto flotanteFloat32, Float64, BFloat16float, double
DecimalesDecimal(P,S), Decimal32(S), Decimal64(S), Decimal128(S)decimal o ClickHouseDecimal
BoolBoolbool
CadenasString, FixedString(N)string
EnumeracionesEnum8(...), Enum16(...)string o enum de C#
Fecha/horaDate, Date32, DateTime, DateTime64(P, 'TZ')DateOnly, DateTime
HoraTime, Time64(N)TimeSpan
UUIDUUIDGuid
RedIPv4, IPv6IPAddress
ArraysArray(T)T[], List<T>, IList<T>, ICollection<T>, IReadOnlyList<T>, IReadOnlyCollection<T>, IEnumerable<T>
MapasMap(K, V)Dictionary<K,V>
TuplesTuple(T1, ...)Tuple<...> o ValueTuple<...>
VariantVariant(T1, T2, ...)object
DinámicoDynamicobject
JSONJsonJsonNode o string
GeográficosPoint, Ring, LineString, Polygon, MultiLineString, MultiPolygon, GeometryTuple<double,double> y arrays de estos; object para Geometry
EnvoltoriosNullable(T), LowCardinality(T)Se desempaquetan automáticamente
Usa ClickHouseDecimal (de ClickHouse.Driver.Numerics) en lugar de decimal cuando necesites toda la precisión de las columnas Decimal128/Decimal256: decimal de .NET está limitado a 28–29 dígitos significativos.

Operaciones LINQ compatibles

Consultas: Where, OrderBy, Take, Skip, Select, First, Single, Any, All, Count, Distinct, AsNoTracking GROUP BY y agregaciones: GroupBy con Count, LongCount, Sum, Average, Min, Max — incluido HAVING (.Where() después de .GroupBy()), varias agregaciones en una sola proyección y OrderBy sobre resultados agregados. JOINs: Join (INNER) y patrones GroupJoin/SelectMany (LEFT y CROSS). LEFT JOIN devuelve null real para las filas sin coincidencia (consulta la semántica de null de LEFT JOIN más abajo). Subconsultas: Contains / IN correlacionados, Any / EXISTS, All y subconsultas escalares en proyecciones. Operaciones de conjuntos: Concat (→ UNION ALL), Union (→ UNION DISTINCT), Intersect, Except. Colecciones locales insertadas en línea: los joins y Contains con colecciones en memoria (int[], List<T>, etc.) se traducen en una serie de UNION. Métodos de cadena: Contains, StartsWith, EndsWith, IndexOf, Replace, Substring, Trim/TrimStart/TrimEnd, ToLower, ToUpper, Length, IsNullOrEmpty, Concat (y el operador +). Funciones matemáticas: los métodos estándar de Math y MathF se traducen a sus equivalentes en ClickHouse — funciones aritméticas, logarítmicas, trigonométricas y auxiliares.
Semántica de NULL en LEFT JOIN
El proveedor inserta automáticamente set_join_use_nulls=1 en cada connection path para ajustarse a las expectativas de Entity Framework sobre el comportamiento de JOIN. Si su servidor ClickHouse o profile impide cambiar esta configuración (por ejemplo, un profile readonly=1), desactívelo con:
optionsBuilder.UseClickHouse(connectionString, o => o.DisableJoinNullSemantics());
Con la exclusión activada, LEFT JOIN devuelve los valores predeterminados de las columnas de ClickHouse y la detección de navegación de EF basada en valores nulos deja de funcionar como se espera. Use comparaciones explícitas con 0 / "" en lugar de == null.

Inserción de datos

SaveChanges usa la API nativa InsertBinaryAsync del driver: codificación RowBinary con compresión GZip, mucho más eficiente que SQL con parámetros:
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();
Las entidades pasan de Added a Unchanged tras guardar, igual que con cualquier otro proveedor de EF Core. El tamaño del lote es configurable (valor predeterminado: 1000):
optionsBuilder.UseClickHouse("Host=localhost", o => o.MaxBatchSize(5000));

Inserción masiva

Para cargas de alto rendimiento, use BulkInsertAsync en lugar de SaveChanges. Es un método de extensión de DbContext que omite por completo el seguimiento de cambios, la resolución de identidad y la administración del estado de EF Core; llama directamente al método InsertBinaryAsync del driver con codificación RowBinary y compresión GZip. Esto lo hace adecuado para cargar grandes volúmenes de datos cuando no necesita el seguimiento de entidades después de la inserción:
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);
La entrada puede ser cualquier IEnumerable<T> — recorre las entidades en streaming sin cargarlas todas en memoria. El valor devuelto es el número de filas insertadas. Las entidades no se adjuntan al DbContext después de la inserción, por lo que no hay transición de estado AddedUnchanged.

Enumeraciones

Las columnas Enum8/Enum16 de ClickHouse se pueden asignar a propiedades string o a tipos enum de C#. Al usar enumeraciones de C#, el proveedor convierte automáticamente entre la enumeración y su representación textual:
public enum Status { Active, Inactive, Pending }

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

// Consulta con valores de enum
var active = await ctx.Users
    .Where(u => u.Status == Status.Active)
    .ToListAsync();

Conversiones de tipos personalizadas

El sistema ValueConverter de EF Core te permite mapear tipos personalizados a tipos que el proveedor ya admite. El proveedor nunca ve tu tipo personalizado: EF Core realiza la conversión en ese punto. Conversión por propiedad:
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; }
}

// En el método 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");
Clase de convertidor reutilizable:
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] };
    }
}

// Aplicar a una propiedad individual:
.HasConversion<MoneyConverter>()

// O aplicarlo a todas las propiedades de un tipo mediante convenciones:
protected override void ConfigureConventions(ModelConfigurationBuilder configurationBuilder)
{
    configurationBuilder.Properties<Money>()
        .HaveConversion<MoneyConverter>();
}

Anotaciones de tipo de columna

Para tipos escalares como string, int, DateTime, etc., el proveedor infiere automáticamente el tipo de ClickHouse. Para los tipos parametrizados y los envoltorios, debe especificar explícitamente el tipo de ClickHouse. Uso de anotaciones de datos (atributos):
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; }
}
Uso de la API fluida en 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)");
});
Se admiten envoltorios anidados como Array(Nullable(Int32)) y LowCardinality(Nullable(String)) — el proveedor elimina automáticamente Nullable y LowCardinality en cada nivel de anidamiento.

Columnas Variant y Dynamic

Las columnas Variant(T1, T2, ...) y Dynamic de ClickHouse se corresponden con object en .NET. Como object es demasiado genérico para la inferencia automática de tipos, debe declarar explícitamente el tipo de almacenamiento mediante .HasColumnType():
public class Event
{
    public long Id { get; set; }
    public object? Payload { get; set; }
}

// En OnModelCreating:
entity.Property(e => e.Payload).HasColumnType("Variant(String, UInt64, Array(UInt64))");
// o bien:
entity.Property(e => e.Payload).HasColumnType("Dynamic");
Al leer, el valor se deserializa automáticamente al tipo .NET correspondiente según el discriminador almacenado (p. ej., string, ulong, ulong[]).

Columnas JSON

El proveedor admite el tipo de columna Json de ClickHouse, que se corresponde con System.Text.Json.Nodes.JsonNode (principal) o string (mediante ValueConverter automático):
using System.Text.Json.Nodes;

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

// En OnModelCreating:
entity.Property(e => e.Data).HasColumnType("Json");
La lectura y escritura de JSON funcionan tanto con SaveChanges como con 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 prefiere cadenas JSON sin procesar, asigne a la propiedad el tipo string con un tipo de columna Json; el proveedor aplica automáticamente un ValueConverter:
public class Event
{
    public long Id { get; set; }
    public string? Data { get; set; }  // cadena JSON sin procesar
}

entity.Property(e => e.Data).HasColumnType("Json");
  • Sin traducción de rutas JSONentity.Data["name"] en LINQ no se corresponde con la sintaxis SQL data.name de ClickHouse. Filtre por columnas no JSON e inspeccione el JSON en memoria.
  • Semántica de NULL — El tipo JSON de ClickHouse devuelve {} (objeto vacío) para los valores NULL en lugar de SQL NULL.
  • Precisión de enteros — El JSON de ClickHouse almacena todos los enteros como Int64. Al leerlo mediante JsonNode, use GetValue<long>() en lugar de GetValue<int>().

Motores de tablas

Configure los motores de tablas de ClickHouse y las cláusulas específicas de cada motor mediante la API fluida ToTable(name, t => ...). Si no se configura ningún motor, el proveedor usa MergeTree de forma predeterminada, con ORDER BY derivado de la clave primaria de la entidad.
modelBuilder.Entity<Event>(e =>
{
    e.ToTable("events", t => t
        .HasMergeTreeEngine()
        .WithOrderBy("UserId", "Timestamp")
        .WithPartitionBy("toYYYYMM(Timestamp)")
        .WithPrimaryKey("UserId")
        .WithSettings("index_granularity = 8192"));
});
Familias de motores compatibles:
EngineMétodo fluidoNotas
MergeTreeHasMergeTreeEngine()Predeterminado si no se configura ninguno
ReplacingMergeTreeHasReplacingMergeTreeEngine("Version", "IsDeleted") o HasReplacingMergeTreeEngine<T>(e => e.Version)Las columnas Version / IsDeleted son opcionales
SummingMergeTreeHasSummingMergeTreeEngine(…) o HasSummingMergeTreeEngine<T>(e => new { … })Columnas a sumar opcionales
AggregatingMergeTreeHasAggregatingMergeTreeEngine()
CollapsingMergeTreeHasCollapsingMergeTreeEngine("Sign") o HasCollapsingMergeTreeEngine<T>(e => e.Sign)La columna Sign debe ser Int8
VersionedCollapsingMergeTreeHasVersionedCollapsingMergeTreeEngine("Sign", "Version") o <T>(e => e.Sign, e => e.Version)
GraphiteMergeTreeHasGraphiteMergeTreeEngine("config_section")
Log, TinyLog, StripeLog, MemoryHasLogEngine(), HasTinyLogEngine(), HasStripeLogEngine(), HasMemoryEngine()Sin ORDER BY / PARTITION BY
Cláusulas del motor: WithOrderBy, WithPartitionBy, WithPrimaryKey, WithSampleBy, WithTtl, WithSettings. Todas se aplican al generador de motores devuelto por HasXxxEngine(). Características a nivel de columna: HasCodec, HasTtl, HasComment, HasDefault — todas forman parte de las migraciones. Índices de omisión de datos — mediante HasIndex(...).HasSkippingIndexType(...):
modelBuilder.Entity<Event>()
    .HasIndex(e => e.UserId)
    .HasSkippingIndexType("minmax")
    .HasGranularity(4);

// Índice con parámetros (p. ej., bloom_filter, tokenbf_v1):
modelBuilder.Entity<Event>()
    .HasIndex(e => e.Tag)
    .HasSkippingIndexType("bloom_filter")
    .HasSkippingIndexParams("0.01")
    .HasGranularity(1);
Los índices estándar (sin omitir datos) se ignoran sin avisar, ya que ClickHouse no tiene ningún equivalente. Los índices únicos provocan un error, ya que ClickHouse no garantiza la unicidad.

Migraciones

Flujo de trabajo estándar para las migraciones de EF Core:
dotnet ef migrations add InitialCreate
dotnet ef database update
Operaciones admitidas:
OperaciónGenera
CREATE TABLEIncluye la cláusula ENGINE, ORDER BY, PARTITION BY, SETTINGS, códecs/TTL/comentarios/valores predeterminados de columnas
ALTER TABLE ADD COLUMN
ALTER TABLE DROP COLUMN
ALTER TABLE MODIFY COLUMNGestiona el cambio de tipo, además de la adición/eliminación de anotaciones (CODEC, TTL, COMMENT, DEFAULT)
ALTER TABLE RENAME COLUMN
RENAME TABLE
ALTER TABLE ADD INDEX / DROP INDEXSolo índices de omisión de datos
CREATE DATABASE / DROP DATABASEMediante EnsureCreated / EnsureDeleted y migraciones

Limitaciones de las migraciones

FuncionalidadMotivo
Claves foráneasClickHouse no hace cumplir las claves foráneas. Las migraciones rechazan AddForeignKey; el validador del modelo emite una advertencia al compilar el modelo.
Restricciones de unicidad / índices únicosClickHouse no hace cumplir la unicidad. Los índices únicos generan un error durante la migración.
Valores generados por el servidor (incremento automático / IDENTITY)ClickHouse no tiene un equivalente.
columnas Nested(…)Aún no son compatibles como tipo CLR asignado.
Entidades owned como JSON (.ToJson())La correspondencia estructural de JSON para entidades owned aún no está implementada. Use JsonNode / string en una columna Json en su lugar (consulte columnas JSON).
Además de las migraciones, el proveedor tampoco admite aún:
  • UPDATE / DELETE
  • Transacciones: BeginTransaction es una operación sin efecto. ClickHouse no admite transacciones ACID.
  • Traducción de consultas con rutas JSON: entity.Data["key"] en LINQ no se traduce a la sintaxis SQL data.key de ClickHouse. Filtre por columnas que no sean JSON e inspeccione el JSON en memoria.

Limitaciones

Columnas de AggregateFunction

Las columnas de tipo AggregateFunction(...) no se pueden consultar ni insertar directamente. Para insertar:
INSERT INTO t VALUES (uniqState(1));
Para consultar:
SELECT uniqMerge(c) FROM t;

Última modificación el 10 de junio de 2026