> ## Documentation Index
> Fetch the complete documentation index at: https://private-7c7dfe99-mintlify-8a08bda2.mintlify.site/llms.txt
> Use this file to discover all available pages before exploring further.

> The official C# client for connecting to ClickHouse.

# ClickHouse C# client

export const Image = ({img, alt, size}) => {
  return <Frame>
      <img src={img} alt={alt} />
    </Frame>;
};

The official C# client for connecting to ClickHouse.
The client source code is available in the [GitHub repository](https://github.com/ClickHouse/clickhouse-cs).
Originally developed by [Oleg V. Kozlyuk](https://github.com/DarkWanderer).

The library provides two main APIs:

* **`ClickHouseClient`** (recommended): A high-level, thread-safe client designed for singleton use. Provides a simple async API for queries and bulk inserts. Best for most applications.

* **ADO.NET** (`ClickHouseDataSource`, `ClickHouseConnection`, `ClickHouseCommand`): Standard .NET database abstractions. Required for ORM integration (Dapper, Linq2db) and when you need ADO.NET compatibility. `ClickHouseBulkCopy` is a helper class for efficiently inserting data using an ADO.NET connection. `ClickHouseBulkCopy` is deprecated and will be removed in a future release; use `ClickHouseClient.InsertBinaryAsync` instead.

Both APIs share the same underlying HTTP connection pool and can be used together in the same application.

<h2 id="migration-guide">
  Migration guide
</h2>

1. Update your `.csproj` file with the new package name `ClickHouse.Driver` and [the latest version on NuGet](https://www.nuget.org/packages/ClickHouse.Driver).
2. Update all `ClickHouse.Client` references to `ClickHouse.Driver` in your codebase.

***

<h2 id="supported-net-versions">
  Supported .NET versions
</h2>

`ClickHouse.Driver` supports the following .NET versions:

* .NET 6.0
* .NET 8.0
* .NET 9.0
* .NET 10.0

<h2 id="installation">
  Installation
</h2>

Install the package from NuGet:

```bash theme={null}
dotnet add package ClickHouse.Driver
```

Or using the NuGet Package Manager:

```bash theme={null}
Install-Package ClickHouse.Driver
```

<h2 id="quick-start">
  Quick start
</h2>

```csharp theme={null}
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);
```

<h2 id="configuration">
  Configuration
</h2>

There are two ways of configuring your connection to ClickHouse:

* **Connection string:** Semicolon-separated key/value pairs that specify the host, authentication credentials, and other connection options.
* **`ClickHouseClientSettings` object:** A strongly typed configuration object that can be loaded from configuration files or set in code.

Below is a full list of all the settings, their default values, and their effects.

<h3 id="connection-settings">
  Connection settings
</h3>

| Property | Type       | Default                    | Connection String Key | Description                                                |
| -------- | ---------- | -------------------------- | --------------------- | ---------------------------------------------------------- |
| Host     | `string`   | `"localhost"`              | `Host`                | Hostname or IP address of the ClickHouse server            |
| Port     | `ushort`   | 8123 (HTTP) / 8443 (HTTPS) | `Port`                | Port number; defaults based on protocol                    |
| Username | `string`   | `"default"`                | `Username`            | Authentication username                                    |
| Password | `string`   | `""`                       | `Password`            | Authentication password                                    |
| Database | `string`   | `""`                       | `Database`            | Default database; empty uses server/user default           |
| Protocol | `string`   | `"http"`                   | `Protocol`            | Connection protocol: `"http"` or `"https"`                 |
| Path     | `string`   | `null`                     | `Path`                | URL path for reverse proxy scenarios (e.g., `/clickhouse`) |
| Timeout  | `TimeSpan` | 2 minutes                  | `Timeout`             | Operation timeout (stored as seconds in connection string) |

<h3 id="data-format-serialization">
  Data format & serialization
</h3>

| Property                | Type                     | Default  | Connection String Key     | Description                                                                                                                               |
| ----------------------- | ------------------------ | -------- | ------------------------- | ----------------------------------------------------------------------------------------------------------------------------------------- |
| UseCompression          | `bool`                   | `true`   | `Compression`             | Enable gzip compression for data transfer                                                                                                 |
| UseCustomDecimals       | `bool`                   | `true`   | `UseCustomDecimals`       | Use `ClickHouseDecimal` for arbitrary precision; if false, uses .NET `decimal` (128-bit limit)                                            |
| ReadStringsAsByteArrays | `bool`                   | `false`  | `ReadStringsAsByteArrays` | Read `String` and `FixedString` columns as `byte[]` instead of `string`; useful for binary data                                           |
| UseFormDataParameters   | `bool`                   | `false`  | `UseFormDataParameters`   | Send parameters as form data instead of URL query string                                                                                  |
| ParameterTypeResolver   | `IParameterTypeResolver` | `null`   | —                         | Custom resolver for `@`-style parameter type mapping; see [Custom parameter type mapping](#parameter-type-mapping)                        |
| JsonReadMode            | `JsonReadMode`           | `Binary` | `JsonReadMode`            | How JSON data is returned: `Binary` (returns `JsonObject`) or `String` (returns raw JSON string)                                          |
| JsonWriteMode           | `JsonWriteMode`          | `String` | `JsonWriteMode`           | How JSON data is sent: `String` (serializes via `JsonSerializer`, accepts all inputs) or `Binary` (registered POCOs only with type hints) |

<h3 id="session-management">
  Session management
</h3>

| Property   | Type     | Default | Connection String Key | Description                                                    |
| ---------- | -------- | ------- | --------------------- | -------------------------------------------------------------- |
| UseSession | `bool`   | `false` | `UseSession`          | Enable stateful sessions; serializes requests                  |
| SessionId  | `string` | `null`  | `SessionId`           | Session ID; auto-generates GUID if null and UseSession is true |

<Note>
  The `UseSession` flag enables persistence of the server session, allowing use of `SET` statements and temporary tables. Sessions will be reset after 60 seconds of inactivity (default timeout). Session lifetime can be extended by setting session settings via ClickHouse statements or the server configuration.

  The `ClickHouseConnection` class normally allows for parallel operation (multiple threads can run queries concurrently). However, enabling `UseSession` flag will limit that to one active query per connection at any moment of time (this is a server-side limitation).
</Note>

<h3 id="security">
  Security
</h3>

| Property                        | Type   | Default | Connection String Key | Description                                                   |
| ------------------------------- | ------ | ------- | --------------------- | ------------------------------------------------------------- |
| SkipServerCertificateValidation | `bool` | `false` | —                     | Skip HTTPS certificate validation; **not for production use** |

<h3 id="http-client-configuration">
  HTTP client configuration
</h3>

| Property          | Type                 | Default | Connection String Key | Description                                          |
| ----------------- | -------------------- | ------- | --------------------- | ---------------------------------------------------- |
| HttpClient        | `HttpClient`         | `null`  | —                     | Custom pre-configured HttpClient instance            |
| HttpClientFactory | `IHttpClientFactory` | `null`  | —                     | Custom factory for creating HttpClient instances     |
| HttpClientName    | `string`             | `null`  | —                     | Name for HttpClientFactory to create specific client |

<h3 id="logging-debugging">
  Logging & debugging
</h3>

| Property        | Type             | Default | Connection String Key | Description                                                                                                      |
| --------------- | ---------------- | ------- | --------------------- | ---------------------------------------------------------------------------------------------------------------- |
| LoggerFactory   | `ILoggerFactory` | `null`  | —                     | Logger factory for diagnostic logging                                                                            |
| EnableDebugMode | `bool`           | `false` | —                     | Enable .NET network tracing (requires LoggerFactory with level set to Trace); **significant performance impact** |

<h3 id="custom-settings-roles">
  Custom settings & roles
</h3>

| Property       | Type                          | Default | Connection String Key | Description                                                   |
| -------------- | ----------------------------- | ------- | --------------------- | ------------------------------------------------------------- |
| CustomSettings | `IDictionary<string, object>` | Empty   | `set_*` prefix        | ClickHouse server settings, see note below                    |
| Roles          | `IReadOnlyList<string>`       | Empty   | `Roles`               | Comma-separated ClickHouse roles (e.g., `Roles=admin,reader`) |

<Note>
  When using a connection string to set custom settings, use the `set_` prefix, e.g. "set\_max\_threads=4". When using a ClickHouseClientSettings object, don't use the `set_` prefix.

  For a full list of available settings, see [here](/reference/settings/session-settings).
</Note>

***

<h3 id="connection-string-examples">
  Connection string examples
</h3>

<h4 id="basic-connection">
  Basic connection
</h4>

```text theme={null}
Host=localhost;Port=8123;Username=default;Password=secret;Database=mydb
```

<h4 id="with-custom-clickhouse-settings">
  With custom ClickHouse settings
</h4>

```text theme={null}
Host=localhost;set_max_threads=4;set_readonly=1;set_max_memory_usage=10000000000
```

***

<h3 id="query-options">
  QueryOptions
</h3>

`QueryOptions` allows you to override client-level settings on a per-query basis. All properties are optional and only override the client defaults when specified.

| Property              | Type                          | Description                                                                                                                       |
| --------------------- | ----------------------------- | --------------------------------------------------------------------------------------------------------------------------------- |
| QueryId               | `string`                      | Custom query identifier for tracking in `system.query_log` or cancellation                                                        |
| Database              | `string`                      | Override the default database for this query                                                                                      |
| Roles                 | `IReadOnlyList<string>`       | Override client roles for this query                                                                                              |
| CustomSettings        | `IDictionary<string, object>` | ClickHouse server settings for this query (e.g., `max_threads`)                                                                   |
| CustomHeaders         | `IDictionary<string, string>` | Additional HTTP headers for this query                                                                                            |
| UseSession            | `bool?`                       | Override session behavior for this query                                                                                          |
| SessionId             | `string`                      | Session ID for this query (requires `UseSession = true`)                                                                          |
| BearerToken           | `string`                      | Override authentication token for this query                                                                                      |
| ParameterTypeResolver | `IParameterTypeResolver`      | Override client-level resolver for `@`-style parameter type mapping; see [Custom parameter type mapping](#parameter-type-mapping) |
| MaxExecutionTime      | `TimeSpan?`                   | Server-side query timeout (passed as `max_execution_time` setting); server cancels query if exceeded                              |

**Example:**

```csharp theme={null}
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
);
```

***

<h3 id="insert-options">
  InsertOptions
</h3>

`InsertOptions` extends `QueryOptions` with settings specific to bulk insert operations via `InsertBinaryAsync`.

| Property               | Type                                  | Default     | Description                                                                  |
| ---------------------- | ------------------------------------- | ----------- | ---------------------------------------------------------------------------- |
| BatchSize              | `int`                                 | 100,000     | Number of rows per batch                                                     |
| MaxDegreeOfParallelism | `int`                                 | 1           | Number of parallel batch uploads                                             |
| Format                 | `RowBinaryFormat`                     | `RowBinary` | Binary format: `RowBinary` or `RowBinaryWithDefaults`                        |
| ColumnTypes            | `IReadOnlyDictionary<string, string>` | `null`      | Column name → ClickHouse type string. Skips the schema probe query when set. |
| UseSchemaCache         | `bool`                                | `false`     | Cache full table schema per (database, table) for the client's lifetime.     |

All `QueryOptions` properties are also available on `InsertOptions`.

**Example:**

```csharp theme={null}
var insertOptions = new InsertOptions
{
    BatchSize = 50_000,
    MaxDegreeOfParallelism = 4,
    QueryId = "bulk-import-001"
};

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

<h4 id="skip-schema-query">
  Skipping the schema probe query
</h4>

By default, `InsertBinaryAsync` sends a `SELECT ... WHERE 1=0` query before each insert to discover column types. For high-throughput scenarios, you can eliminate this overhead with two options:

**Option 1: Provide column types explicitly**

When you know the table schema at compile time, pass it directly via `ColumnTypes`. No schema query is sent at all:

```csharp theme={null}
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: Cache the schema**

When you insert into the same table repeatedly, set `UseSchemaCache = true` to query the schema once and reuse it for subsequent inserts on the same `ClickHouseClient` instance:

```csharp theme={null}
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);
```

<Note>
  * `ColumnTypes` takes priority over `UseSchemaCache`. If both are set, the explicit types are used.
  * The schema cache does not detect `ALTER TABLE` changes. If you modify the table schema, create a new `ClickHouseClient` or avoid `UseSchemaCache` for that table.
  * The cache is scoped to the `ClickHouseClient` instance and keyed by (database, table). Different column subsets on the same table share a single cached schema.
</Note>

<h2 id="clickhouse-client">
  ClickHouseClient
</h2>

`ClickHouseClient` is the recommended API for interacting with ClickHouse. It is thread-safe, designed for singleton use, and manages HTTP connection pooling internally.

<h3 id="creating-a-client">
  Creating a client
</h3>

Create a `ClickHouseClient` with a connection string or a `ClickHouseClientSettings` object. See the [Configuration](#configuration) section for available options.

The details for your ClickHouse Cloud service are available in the ClickHouse Cloud console.

Select a service and click **Connect**:

<Image img="https://mintcdn.com/private-7c7dfe99-mintlify-8a08bda2/wqlFf5fgND6FRlHN/images/_snippets/cloud-connect-button.png?fit=max&auto=format&n=wqlFf5fgND6FRlHN&q=85&s=ae1aa98f2d03607ef24cd1e8f582d1c9" size="md" alt="ClickHouse Cloud service connect button" border width="998" height="932" data-path="images/_snippets/cloud-connect-button.png" />

Choose **C#**. Connection details are displayed below.

<Image img="https://mintcdn.com/private-7c7dfe99-mintlify-8a08bda2/wqlFf5fgND6FRlHN/images/_snippets/connection-details-csharp.png?fit=max&auto=format&n=wqlFf5fgND6FRlHN&q=85&s=3a6aa341f6490c2d5d233dd5f337751e" size="md" alt="ClickHouse Cloud C# connection details" border width="851" height="805" data-path="images/_snippets/connection-details-csharp.png" />

If you're using self-managed ClickHouse, the connection details are set by your ClickHouse administrator.

Using a connection string:

```csharp theme={null}
using ClickHouse.Driver;

using var client = new ClickHouseClient("Host=localhost;Username=default;Password=secret");
```

Or using `ClickHouseClientSettings`:

```csharp theme={null}
using ClickHouse.Driver;

var settings = new ClickHouseClientSettings
{
    Host = "localhost",
    Username = "default",
    Password = "secret"
};
using var client = new ClickHouseClient(settings);
```

For dependency injection scenarios, use `IHttpClientFactory`:

```csharp theme={null}
// 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");
```

<Note>
  `ClickHouseClient` is designed to be long-lived and shared across your application. Create it once (typically as a singleton) and reuse it for all database operations. The client manages HTTP connection pooling internally.
</Note>

***

<h3 id="executing-queries">
  Executing queries
</h3>

Use `ExecuteNonQueryAsync` for statements that don't return results:

```csharp theme={null}
// 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");
```

Use `ExecuteScalarAsync` to retrieve a single value:

```csharp theme={null}
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}");
```

***

<h3 id="inserting-data">
  Inserting data
</h3>

<h4 id="parameterized-inserts">
  Parameterized inserts
</h4>

Insert data using parameterized queries with `ExecuteNonQueryAsync`. Parameter types must be specified in the SQL using `{name:Type}` syntax:

```csharp theme={null}
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
);
```

***

<h4 id="bulk-insert">
  Bulk inserts
</h4>

Use `InsertBinaryAsync` for inserting large numbers of rows efficiently. It streams data using ClickHouse's native row binary format, supports parallel batch uploads, and avoids "URL too long" errors that can occur with parameterized queries.

```csharp theme={null}
// 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}");
```

For large datasets, configure batching and parallelism with `InsertOptions`:

```csharp theme={null}
var options = new InsertOptions
{
    BatchSize = 100_000,           // Rows per batch (default: 100,000)
    MaxDegreeOfParallelism = 4     // Parallel batch uploads (default: 1)
};
```

<Note>
  * The client automatically fetches table structure via `SELECT * FROM <table> WHERE 1=0` before inserting. Provided values must match the target column types. To skip this query, use [`InsertOptions.ColumnTypes` or `InsertOptions.UseSchemaCache`](#skip-schema-query).
  * When `MaxDegreeOfParallelism > 1`, batches are uploaded in parallel. Sessions are not compatible with parallel insertion; either disable sessions or set `MaxDegreeOfParallelism = 1`.
  * Use `RowBinaryFormat.RowBinaryWithDefaults` in `InsertOptions.Format` if you want the server to apply DEFAULT values for columns not provided.
</Note>

<h4 id="poco-insert">
  POCO inserts
</h4>

Instead of constructing `object[]` arrays, you can insert strongly-typed POCO objects directly. Register the type once, then pass `IEnumerable<T>`:

```csharp theme={null}
// 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);
```

By default, all public readable properties are mapped to columns using strict case-sensitive name matching. You can customize the mapping with attributes:

```csharp theme={null}
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; }
}
```

| Attribute                          | Purpose                                |
| ---------------------------------- | -------------------------------------- |
| `[ClickHouseColumn(Name = "...")]` | Override the target column name        |
| `[ClickHouseColumn(Type = "...")]` | Declare the ClickHouse type explicitly |
| `[ClickHouseNotMapped]`            | Exclude the property from the insert   |

When **all** mapped properties specify an explicit `Type`, the schema probe query is skipped entirely. When only some properties have explicit types, the driver falls back to the schema probe for the full column set.

`InsertBinaryAsync<T>` supports the same `InsertOptions` (batching, parallelism, schema caching) as the `object[]` overload.

<Note>
  Unlike the `object[]` overload, `InsertBinaryAsync<T>` does not accept an explicit column list. Columns are determined by the registered type's mapped properties. To control which columns are inserted, use `[ClickHouseNotMapped]` to exclude properties or `[ClickHouseColumn(Name = "...")]` to rename them.

  If `ColumnTypes` is set in `InsertOptions`, they will override the POCO attributes.
</Note>

<h4 id="poco-insert-schema-evolution">
  Schema evolution
</h4>

POCO inserts work seamlessly when columns are added to the target table after the type is registered. Because the driver only inserts the columns mapped by the POCO, any new columns with `DEFAULT` (or other default expressions) are filled in by the server automatically. No code changes or re-registration are needed.

***

<h3 id="reading-data">
  Reading data
</h3>

Use `ExecuteReaderAsync` to execute SELECT queries. The returned `ClickHouseDataReader` provides typed access to result columns via methods like `GetInt64()`, `GetString()`, and `GetFieldValue<T>()`.

Call `Read()` to advance to the next row. It returns `false` when there are no more rows. Access columns by index (0-based) or by column name.

```csharp theme={null}
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)}");
}
```

***

<h3 id="sql-parameters">
  SQL parameters
</h3>

In ClickHouse, the standard format for query parameters in SQL queries is `{parameter_name:DataType}`.

**Examples:**

```sql theme={null}
SELECT {value:Array(UInt16)} as a
```

```sql theme={null}
SELECT * FROM table WHERE val = {tuple_in_tuple:Tuple(UInt8, Tuple(String, UInt8))}
```

```sql theme={null}
INSERT INTO table VALUES ({val1:Int32}, {val2:Array(UInt8)})
```

<Note>
  SQL 'bind' parameters are passed as HTTP URI query parameters, so using too many of them may result in a "URL too long" exception. Use `InsertBinaryAsync` for bulk data insertion to avoid this limitation.
</Note>

***

<h3 id="query-id">
  Query ID
</h3>

Every query is assigned a unique `query_id` that can be used to fetch data from the `system.query_log` table or cancel long-running queries. You can specify a custom query ID via `QueryOptions`:

```csharp theme={null}
var options = new QueryOptions
{
    QueryId = $"report-{Guid.NewGuid()}"
};

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

<Tip>
  If you're specifying a custom `QueryId`, ensure it is unique for every call. A random GUID is a good choice.
</Tip>

***

<h3 id="parameter-type-mapping">
  Custom parameter type mapping
</h3>

When using `@`-style parameters (e.g., `WHERE id = @id`), the driver automatically infers the ClickHouse type from the .NET value type. For example, `int` maps to `Int32`, and `DateTime` maps to `DateTime`.

To override these defaults, set `ParameterTypeResolver` on `ClickHouseClientSettings`. This is useful when you want all `DateTime` parameters to use `DateTime64(3)` for millisecond precision, or all decimals to use a specific scale, without setting `ClickHouseType` on every individual parameter.

**Using `DictionaryParameterTypeResolver` for simple type mappings:**

```csharp theme={null}
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);
```

**Custom `IParameterTypeResolver` for advanced scenarios:**

For value-aware or name-based resolution, implement the `IParameterTypeResolver` interface directly. Return `null` to fall through to the default inference:

```csharp theme={null}
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})";
    }
}
```

You can also set a resolver for a single query via `QueryOptions.ParameterTypeResolver`. When set, it takes precedence over the client-level resolver.

**Type resolution precedence:**

The resolver is one step in a precedence chain. From highest to lowest priority:

1. Explicit `ClickHouseType` set on the parameter
2. SQL type hint from `{name:Type}` syntax in the query
3. `IParameterTypeResolver` (from `QueryOptions.ParameterTypeResolver`, falling back to `ClickHouseClientSettings.ParameterTypeResolver`)
4. Built-in type inference (`TypeConverter.ToClickHouseType`)

The resolver also works with the ADO.NET `ClickHouseConnection` path — the settings are inherited by connections created from the client.

***

<h3 id="raw-streaming">
  Raw streaming
</h3>

Use `ExecuteRawResultAsync` to stream query results in a specific format directly, bypassing the data reader. This is useful for exporting data to files or passing through to other systems:

```csharp theme={null}
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();
```

Common formats: `JSONEachRow`, `CSV`, `TSV`, `Parquet`, `Native`. See the [formats documentation](/reference/formats) for all options.

***

<h3 id="raw-stream-insert">
  Raw stream insert
</h3>

Use `InsertRawStreamAsync` to insert data directly from file or memory streams in formats like CSV, JSON, Parquet, or any [supported ClickHouse format](/reference/formats).

**Insert from a CSV file:**

```csharp theme={null}
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
);
```

<Note>
  See the [format settings documentation](/reference/settings/formats) for options to control data ingestion behavior.
</Note>

***

<h3 id="more-examples">
  More examples
</h3>

For additional practical usage examples, see the [examples directory](https://github.com/ClickHouse/clickhouse-cs/tree/main/examples) in the GitHub repository.

<h2 id="ado-net">
  ADO.NET
</h2>

The library provides full ADO.NET support through `ClickHouseConnection`, `ClickHouseCommand`, and `ClickHouseDataReader`. This API is required for ORM integration (Dapper, Linq2db) and when you need standard .NET database abstractions.

<h3 id="ado-net-datasource">
  Lifetime management with ClickHouseDataSource
</h3>

**Always create connections from a `ClickHouseDataSource`** to ensure proper lifetime management and connection pooling. The DataSource manages a single `ClickHouseClient` internally, and all connections share its HTTP connection pool.

```csharp theme={null}
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();
```

For dependency injection:

```csharp theme={null}
// 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...
    }
}
```

<Warning>
  **Do not create `ClickHouseConnection` directly** in production code. Each direct instantiation creates a new HTTP client and connection pool, which can lead to socket exhaustion under load:

  ```csharp theme={null}
  // DON'T DO THIS - creates new connection pool each time
  using var conn = new ClickHouseConnection("Host=localhost");
  await conn.OpenAsync();
  ```

  Instead, always use `ClickHouseDataSource` or share a single `ClickHouseClient` instance.
</Warning>

***

<h3 id="ado-net-command">
  Using ClickHouseCommand
</h3>

Create commands from a connection to execute SQL:

```csharp theme={null}
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")}");
}
```

Command methods:

* `ExecuteNonQueryAsync()` - For INSERT, UPDATE, DELETE, DDL statements
* `ExecuteScalarAsync()` - Returns first column of first row
* `ExecuteReaderAsync()` - Returns a `ClickHouseDataReader` for iterating results

***

<h3 id="ado-net-reader">
  Using ClickHouseDataReader
</h3>

The `ClickHouseDataReader` provides typed access to query results:

```csharp theme={null}
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");
    }
}
```

<h2 id="best-practices">
  Best practices
</h2>

<h3 id="best-practices-connection-lifetime">
  Connection lifetime and pooling
</h3>

`ClickHouse.Driver` uses `System.Net.Http.HttpClient` under the hood. `HttpClient` has a per-endpoint connection pool. As a consequence:

* Database sessions are multiplexed through HTTP connections managed by the connection pool.
* HTTP connections are recycled automatically by the pool.
* Connections can stay alive after `ClickHouseClient` or `ClickHouseConnection` objects are disposed.

**Recommended patterns:**

| Scenario        | Recommended Approach                                                                         |
| --------------- | -------------------------------------------------------------------------------------------- |
| General use     | Use a singleton `ClickHouseClient`                                                           |
| ADO.NET / ORMs  | Use `ClickHouseDataSource` (creates connections that share the same pool)                    |
| DI environments | Register `ClickHouseClient` or `ClickHouseDataSource` as singleton with `IHttpClientFactory` |

<Warning>
  When using a custom `HttpClient` or `HttpClientFactory`, ensure that the `PooledConnectionIdleTimeout` is set to a value smaller than the server's `keep_alive_timeout`, in order to avoid errors due to half-closed connections. The default `keep_alive_timeout` for Cloud deployments is 10 seconds.
</Warning>

<Warning>
  Avoid creating multiple `ClickHouseClient` or standalone `ClickHouseConnection` instances without a shared `HttpClient`. Each instance creates its own connection pool.
</Warning>

***

<h3 id="best-practice-datetime">
  DateTime handling
</h3>

1. **Use UTC whenever possible.** Store timestamps as `DateTime('UTC')` columns and use `DateTimeKind.Utc` in your code. This eliminates timezone ambiguity.

2. **Use `DateTimeOffset` for explicit timezone handling.** It always represents a specific instant and includes the offset information.

3. **Specify timezone in SQL type hints.** When using parameters with `Unspecified` DateTime values targeting non-UTC columns, include the timezone in the SQL:
   ```csharp theme={null}
   var parameters = new ClickHouseParameterCollection();
   parameters.AddParameter("dt", myDateTime);

   await client.ExecuteNonQueryAsync(
       "INSERT INTO table (dt) VALUES ({dt:DateTime('Europe/Amsterdam')})",
       parameters
   );
   ```

***

<h3 id="async-inserts">
  Async inserts
</h3>

[Async inserts](/concepts/features/operations/insert/asyncinserts) shift batching responsibility from the client to the server. Instead of requiring client-side batching, the server buffers incoming data and flushes it to storage based on configurable thresholds. This is useful for high-concurrency scenarios like observability workloads where many agents send small payloads.

Enable async inserts via `CustomSettings` or the connection string:

```csharp theme={null}
// 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"
```

**Two modes** (controlled by `wait_for_async_insert`):

| Mode                      | Behavior                                                                               | Use case                           |
| ------------------------- | -------------------------------------------------------------------------------------- | ---------------------------------- |
| `wait_for_async_insert=1` | Insert returns after data is flushed to disk. Errors are returned to the client.       | **Recommended** for most workloads |
| `wait_for_async_insert=0` | Insert returns immediately when data is buffered. No guarantee data will be persisted. | Only when data loss is acceptable  |

<Warning>
  With `wait_for_async_insert=0`, errors only surface during flush and can't be traced back to the original insert. The client also provides no backpressure, risking server overload.
</Warning>

**Key settings:**

| Setting                         | Description                                 |
| ------------------------------- | ------------------------------------------- |
| `async_insert_max_data_size`    | Flush when buffer reaches this size (bytes) |
| `async_insert_busy_timeout_ms`  | Flush after this timeout (milliseconds)     |
| `async_insert_max_query_number` | Flush after this many queries accumulate    |

***

<h3 id="best-practices-sessions">
  Sessions
</h3>

Only enable sessions when you need stateful server-side features, e.g.:

* Temporary tables (`CREATE TEMPORARY TABLE`)
* Maintaining query context across multiple statements
* Session-level settings (`SET max_threads = 4`)

When sessions are enabled, requests are serialized to prevent concurrent use of the same session. This adds overhead for workloads that don't require session state.

```csharp theme={null}
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)"
);
```

**Using ADO.NET (for ORM compatibility):**

```csharp theme={null}
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();
```

<h2 id="supported-data-types">
  Supported data types
</h2>

`ClickHouse.Driver` supports all ClickHouse data types. The tables below show the mappings between ClickHouse types and native .NET types when reading data from the database.

<h3 id="clickhouse-native-type-map-reading">
  Type mapping: reading from ClickHouse
</h3>

<h4 id="type-map-reading-integer">
  Integer types
</h4>

| ClickHouse Type | .NET Type    |
| --------------- | ------------ |
| Int8            | `sbyte`      |
| UInt8           | `byte`       |
| Int16           | `short`      |
| UInt16          | `ushort`     |
| Int32           | `int`        |
| UInt32          | `uint`       |
| Int64           | `long`       |
| UInt64          | `ulong`      |
| Int128          | `BigInteger` |
| UInt128         | `BigInteger` |
| Int256          | `BigInteger` |
| UInt256         | `BigInteger` |

***

<h4 id="type-map-reading-floating-points">
  Floating point types
</h4>

| ClickHouse Type | .NET Type |
| --------------- | --------- |
| Float32         | `float`   |
| Float64         | `double`  |
| BFloat16        | `float`   |

***

<h4 id="type-map-reading-decimal">
  Decimal types
</h4>

| ClickHouse Type | .NET Type                       |
| --------------- | ------------------------------- |
| Decimal(P, S)   | `decimal` / `ClickHouseDecimal` |
| Decimal32(S)    | `decimal` / `ClickHouseDecimal` |
| Decimal64(S)    | `decimal` / `ClickHouseDecimal` |
| Decimal128(S)   | `decimal` / `ClickHouseDecimal` |
| Decimal256(S)   | `decimal` / `ClickHouseDecimal` |

<Note>
  Decimal type conversion is controlled via the UseCustomDecimals setting.
</Note>

***

<h4 id="type-map-reading-boolean">
  Boolean type
</h4>

| ClickHouse Type | .NET Type |
| --------------- | --------- |
| Bool            | `bool`    |

***

<h4 id="type-map-reading-strings">
  String types
</h4>

| ClickHouse Type | .NET Type |
| --------------- | --------- |
| String          | `string`  |
| FixedString(N)  | `string`  |

<Note>
  By default, both `String` and `FixedString(N)` columns are returned as `string`. Set `ReadStringsAsByteArrays=true` in your connection string to read them as `byte[]` instead. This is useful when storing binary data that may not be valid UTF-8.
</Note>

***

<h4 id="type-map-reading-datetime">
  Date and time types
</h4>

| ClickHouse Type | .NET Type  |
| --------------- | ---------- |
| Date            | `DateTime` |
| Date32          | `DateTime` |
| DateTime        | `DateTime` |
| DateTime32      | `DateTime` |
| DateTime64      | `DateTime` |
| Time            | `TimeSpan` |
| Time64          | `TimeSpan` |

ClickHouse stores `DateTime` and `DateTime64` values internally as Unix timestamps (seconds or sub-second units since epoch). While the storage is always in UTC, columns can have an associated timezone that affects how values are displayed and interpreted.

When reading `DateTime` values, the `DateTime.Kind` property is set based on the column's timezone:

| Column Definition              | Returned DateTime.Kind | Notes                           |
| ------------------------------ | ---------------------- | ------------------------------- |
| `DateTime('UTC')`              | `Utc`                  | Explicit UTC timezone           |
| `DateTime('Europe/Amsterdam')` | `Unspecified`          | Offset applied                  |
| `DateTime`                     | `Unspecified`          | Wall-clock time preserved as-is |

For non-UTC columns, the returned `DateTime` represents the wall-clock time in that timezone. Use `ClickHouseDataReader.GetDateTimeOffset()` to get a `DateTimeOffset` with the correct offset for that timezone:

```csharp theme={null}
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)
```

For columns **without** an explicit timezone (i.e., `DateTime` instead of `DateTime('Europe/Amsterdam')`), the driver returns a `DateTime` with `Kind=Unspecified`. This preserves the wall-clock time exactly as stored without making assumptions about timezone.

If you need timezone-aware behavior for columns without explicit timezones, either:

1. Use explicit timezones in your column definitions: `DateTime('UTC')` or `DateTime('Europe/Amsterdam')`
2. Apply the timezone yourself after reading.

***

<h4 id="type-map-reading-json">
  JSON type
</h4>

| ClickHouse Type | .NET Type    | Notes                           |
| --------------- | ------------ | ------------------------------- |
| Json            | `JsonObject` | Default (`JsonReadMode=Binary`) |
| Json            | `string`     | When `JsonReadMode=String`      |

The return type for JSON columns is controlled by the `JsonReadMode` setting:

* **`Binary` (default)**: Returns `System.Text.Json.Nodes.JsonObject`. Provides structured access to JSON data, but specialized ClickHouse types (like IP addresses, UUIDs, large decimals) are converted to their string representations within the JSON structure.

* **`String`**: Returns the raw JSON as a `string`. Preserves the exact JSON representation from ClickHouse, which is useful when you need to pass the JSON through without parsing, or when you want to handle deserialization yourself.

```csharp theme={null}
// Configure string mode via settings
var settings = new ClickHouseClientSettings("Host=localhost")
{
    JsonReadMode = JsonReadMode.String
};

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

***

<h4 id="type-map-reading-other">
  Other types
</h4>

| ClickHouse Type         | .NET Type                           |
| ----------------------- | ----------------------------------- |
| UUID                    | `Guid`                              |
| IPv4                    | `IPAddress`                         |
| IPv6                    | `IPAddress`                         |
| Nothing                 | `DBNull`                            |
| Dynamic                 | See note                            |
| Array(T)                | `T[]`                               |
| Tuple(T1, T2, ...)      | `Tuple<T1, T2, ...>` / `LargeTuple` |
| Map(K, V)               | `Dictionary<K, V>`                  |
| Nullable(T)             | `T?`                                |
| Enum8                   | `string`                            |
| Enum16                  | `string`                            |
| LowCardinality(T)       | Same as T                           |
| SimpleAggregateFunction | Same as underlying type             |
| Nested(...)             | `Tuple[]`                           |
| Variant(T1, T2, ...)    | See note                            |
| QBit(T, dimension)      | `T[]`                               |

<Note>
  The Dynamic and Variant types will be converted to the corresponding type for the actual underlying type in each row.
</Note>

***

<h4 id="type-map-reading-geometry">
  Geometry types
</h4>

| ClickHouse Type | .NET Type                 |
| --------------- | ------------------------- |
| Point           | `Tuple<double, double>`   |
| Ring            | `Tuple<double, double>[]` |
| LineString      | `Tuple<double, double>[]` |
| Polygon         | `Ring[]`                  |
| MultiLineString | `LineString[]`            |
| MultiPolygon    | `Polygon[]`               |
| Geometry        | See note                  |

<Note>
  The Geometry type is a Variant type that can hold any of the geometry types. It will be converted to the corresponding type.
</Note>

***

<h3 id="clickhouse-native-type-map-writing">
  Type mapping: writing to ClickHouse
</h3>

When inserting data, the driver converts .NET types to their corresponding ClickHouse types. The tables below show which .NET types are accepted for each ClickHouse column type.

<h4 id="type-map-writing-integer">
  Integer types
</h4>

| ClickHouse Type | Accepted .NET Types                                                                                            | Notes |
| --------------- | -------------------------------------------------------------------------------------------------------------- | ----- |
| Int8            | `sbyte`, any `Convert.ToSByte()` compatible                                                                    |       |
| UInt8           | `byte`, any `Convert.ToByte()` compatible                                                                      |       |
| Int16           | `short`, any `Convert.ToInt16()` compatible                                                                    |       |
| UInt16          | `ushort`, any `Convert.ToUInt16()` compatible                                                                  |       |
| Int32           | `int`, any `Convert.ToInt32()` compatible                                                                      |       |
| UInt32          | `uint`, any `Convert.ToUInt32()` compatible                                                                    |       |
| Int64           | `long`, any `Convert.ToInt64()` compatible                                                                     |       |
| UInt64          | `ulong`, any `Convert.ToUInt64()` compatible                                                                   |       |
| Int128          | `BigInteger`, `decimal`, `double`, `float`, `int`, `uint`, `long`, `ulong`, any `Convert.ToInt64()` compatible |       |
| UInt128         | `BigInteger`, `decimal`, `double`, `float`, `int`, `uint`, `long`, `ulong`, any `Convert.ToInt64()` compatible |       |
| Int256          | `BigInteger`, `decimal`, `double`, `float`, `int`, `uint`, `long`, `ulong`, any `Convert.ToInt64()` compatible |       |
| UInt256         | `BigInteger`, `decimal`, `double`, `float`, `int`, `uint`, `long`, `ulong`, any `Convert.ToInt64()` compatible |       |

***

<h4 id="type-map-writing-floating-point">
  Floating point types
</h4>

| ClickHouse Type | Accepted .NET Types                           | Notes                                  |
| --------------- | --------------------------------------------- | -------------------------------------- |
| Float32         | `float`, any `Convert.ToSingle()` compatible  |                                        |
| Float64         | `double`, any `Convert.ToDouble()` compatible |                                        |
| BFloat16        | `float`, any `Convert.ToSingle()` compatible  | Truncates to 16-bit brain float format |

***

<h4 id="type-map-writing-boolean">
  Boolean type
</h4>

| ClickHouse Type | Accepted .NET Types | Notes |
| --------------- | ------------------- | ----- |
| Bool            | `bool`              |       |

***

<h4 id="type-map-writing-strings">
  String types
</h4>

| ClickHouse Type | Accepted .NET Types                                  | Notes                                                                    |
| --------------- | ---------------------------------------------------- | ------------------------------------------------------------------------ |
| String          | `string`, `byte[]`, `ReadOnlyMemory<byte>`, `Stream` | Binary types written directly; streams can be seekable or non-seekable   |
| FixedString(N)  | `string`, `byte[]`, `ReadOnlyMemory<byte>`, `Stream` | String is UTF-8 encoded and padded; binary types must be exactly N bytes |

***

<h4 id="type-map-writing-datetime">
  Date and time types
</h4>

| ClickHouse Type | Accepted .NET Types                                               | Notes                                                                       |
| --------------- | ----------------------------------------------------------------- | --------------------------------------------------------------------------- |
| Date            | `DateTime`, `DateTimeOffset`, `DateOnly`, NodaTime types          | Converted to Unix days as UInt16                                            |
| Date32          | `DateTime`, `DateTimeOffset`, `DateOnly`, NodaTime types          | Converted to Unix days as Int32                                             |
| DateTime        | `DateTime`, `DateTimeOffset`, `DateOnly`, NodaTime types          | See below for details                                                       |
| DateTime32      | `DateTime`, `DateTimeOffset`, `DateOnly`, NodaTime types          | Same as DateTime                                                            |
| DateTime64      | `DateTime`, `DateTimeOffset`, `DateOnly`, NodaTime types          | Precision based on Scale parameter                                          |
| Time            | `TimeSpan`, `int`                                                 | Clamped to ±999:59:59; int treated as seconds                               |
| Time64          | `TimeSpan`, `decimal`, `double`, `float`, `int`, `long`, `string` | String parsed as `[-]HHH:MM:SS[.fraction]`; clamped to ±999:59:59.999999999 |

The driver respects `DateTime.Kind` when writing values:

| DateTime.Kind | HTTP Parameters                                                      | Bulk                                       |
| ------------- | -------------------------------------------------------------------- | ------------------------------------------ |
| Utc           | Instant preserved                                                    | Instant preserved                          |
| Local         | Instant preserved                                                    | Instant preserved                          |
| Unspecified   | Treated as wall-clock in parameter type's timezone (defaults to UTC) | Treated as wall-clock in column's timezone |

`DateTimeOffset` values always preserve the exact instant.

**Example: UTC DateTime (instant preserved)**

```csharp theme={null}
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
```

**Example: unspecified DateTime (wall-clock time)**

```csharp theme={null}
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
```

**Recommendation:** for simplest and most predictable behavior, use `DateTimeKind.Utc` or `DateTimeOffset` for all DateTime operations. This ensures your code works consistently regardless of server timezone, client timezone, or column timezone.

<h4 id="datetime-http-param-vs-bulkcopy">
  HTTP parameters vs bulk copy
</h4>

There is an important difference between HTTP parameter binding and bulk copy when writing `Unspecified` DateTime values:

**Bulk Copy** knows the target column's timezone and correctly interprets `Unspecified` values in that timezone.

**HTTP Parameters** do not automatically know the column timezone. You must specify it in the SQL type hint:

```csharp theme={null}
// 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.Kind` | Target Column    | HTTP Param (with tz hint) | HTTP Param (no tz hint) | Bulk Copy                 |
| --------------- | ---------------- | ------------------------- | ----------------------- | ------------------------- |
| `Utc`           | UTC              | Instant preserved         | Instant preserved       | Instant preserved         |
| `Utc`           | Europe/Amsterdam | Instant preserved         | Instant preserved       | Instant preserved         |
| `Local`         | Any              | Instant preserved         | Instant preserved       | Instant preserved         |
| `Unspecified`   | UTC              | Treated as UTC            | Treated as UTC          | Treated as UTC            |
| `Unspecified`   | Europe/Amsterdam | Treated as Amsterdam time | **Treated as UTC**      | Treated as Amsterdam time |

***

<h4 id="type-map-writing-decimal">
  Decimal types
</h4>

| ClickHouse Type | Accepted .NET Types                                                  | Notes                                           |
| --------------- | -------------------------------------------------------------------- | ----------------------------------------------- |
| Decimal(P,S)    | `decimal`, `ClickHouseDecimal`, any `Convert.ToDecimal()` compatible | Throws `OverflowException` if exceeds precision |
| Decimal32       | `decimal`, `ClickHouseDecimal`, any `Convert.ToDecimal()` compatible | Max precision 9                                 |
| Decimal64       | `decimal`, `ClickHouseDecimal`, any `Convert.ToDecimal()` compatible | Max precision 18                                |
| Decimal128      | `decimal`, `ClickHouseDecimal`, any `Convert.ToDecimal()` compatible | Max precision 38                                |
| Decimal256      | `decimal`, `ClickHouseDecimal`, any `Convert.ToDecimal()` compatible | Max precision 76                                |

***

<h4 id="type-map-writing-json">
  JSON type
</h4>

| ClickHouse Type | Accepted .NET Types                            | Notes                                       |
| --------------- | ---------------------------------------------- | ------------------------------------------- |
| Json            | `string`, `JsonObject`, `JsonNode`, any object | Behavior depends on `JsonWriteMode` setting |

The behavior when writing JSON is controlled by the `JsonWriteMode` setting:

| Input Type                           | `JsonWriteMode.String` (default)            | `JsonWriteMode.Binary`                                            |
| ------------------------------------ | ------------------------------------------- | ----------------------------------------------------------------- |
| `string`                             | Passed through directly                     | Throws `ArgumentException`                                        |
| `JsonObject`                         | Serialized via `ToJsonString()`             | Throws `ArgumentException`                                        |
| `JsonNode`                           | Serialized via `ToJsonString()`             | Throws `ArgumentException`                                        |
| Registered POCO                      | Serialized via `JsonSerializer.Serialize()` | Binary encoding with type hints, custom path attributes supported |
| Unregistered POCO / Anonymous object | Serialized via `JsonSerializer.Serialize()` | Throws `ClickHouseJsonSerializationException`                     |

* **`String` (default)**: Accepts `string`, `JsonObject`, `JsonNode`, or any object. All inputs are serialized via `System.Text.Json.JsonSerializer` and sent as JSON strings for server-side parsing. This is the most flexible mode and works without type registration.

* **`Binary`**: Only accepts registered POCO types. Data is converted to ClickHouse's binary JSON format client-side with full type hint support. Requires calling `connection.RegisterJsonSerializationType<T>()` before use. Writing `string` or `JsonNode` values in this mode throws `ArgumentException`.

```csharp theme={null}
// 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>();
```

<h5 id="json-typed-columns">
  Typed JSON columns
</h5>

When a JSON column has type hints (e.g., `JSON(id UInt64, price Decimal128(2))`), the driver uses these hints to serialize values with full type fidelity. This preserves precision for types like `UInt64`, `Decimal`, `UUID`, and `DateTime64` that would otherwise lose precision when serialized as generic JSON.

<h5 id="json-poco-serialization">
  POCO serialization
</h5>

POCOs can be written to JSON columns in two ways depending on the `JsonWriteMode`:

**String mode (default)**: POCOs are serialized via `System.Text.Json.JsonSerializer`. No type registration is required. This is the simplest approach and works with anonymous objects.

**Binary mode**: POCOs are serialized using the driver's binary JSON format with full type hint support. Types must be registered with `connection.RegisterJsonSerializationType<T>()` before use. This mode supports custom path mappings via attributes:

* **`[ClickHouseJsonPath("path")]`**: Maps a property to a custom JSON path. Useful for nested structures or when the property name differs from the desired JSON key. **Only works in Binary mode.**

* **`[ClickHouseJsonIgnore]`**: Excludes a property from serialization. **Only works in Binary mode.**

```sql theme={null}
CREATE TABLE events (
    id UInt32,
    data JSON(`user.id` Int64, `user.name` String, Timestamp DateTime64(3))
) ENGINE = MergeTree() ORDER BY id
```

```csharp theme={null}
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..."}
```

Property name matching with column type hints is case-sensitive. A property `UserId` will only match a hint defined as `UserId`, not `userid`. This matches ClickHouse behavior which allows paths like `userName` and `UserName` to coexist as separate fields.

**Limitations (Binary mode only):**

* POCO types must be registered on the connection with `connection.RegisterJsonSerializationType<T>()` before serialization. Attempting to serialize an unregistered type throws `ClickHouseJsonSerializationException`.
* Dictionary and array/list properties require type hints in the column definition to be serialized correctly. Without hints, use String mode instead.
* Null values in POCO properties are only written when the path has a `Nullable(T)` type hint in the column definition. ClickHouse doesn't allow `Nullable` types inside dynamic JSON paths, so un-hinted null properties are skipped.
* `ClickHouseJsonPath` and `ClickHouseJsonIgnore` attributes are ignored in String mode (they only work in Binary mode).

***

<h4 id="type-map-writing-other">
  Other types
</h4>

| ClickHouse Type         | Accepted .NET Types                      | Notes                                                |
| ----------------------- | ---------------------------------------- | ---------------------------------------------------- |
| UUID                    | `Guid`, `string`                         | String parsed as Guid                                |
| IPv4                    | `IPAddress`, `string`                    | Must be IPv4; string parsed via `IPAddress.Parse()`  |
| IPv6                    | `IPAddress`, `string`                    | Must be IPv6; string parsed via `IPAddress.Parse()`  |
| Nothing                 | Any                                      | Writes nothing (no-op)                               |
| Dynamic                 | —                                        | **Not supported** (throws `NotImplementedException`) |
| Array(T)                | `IList`, `null`                          | Null writes empty array                              |
| Tuple(T1, T2, ...)      | `ITuple`, `IList`                        | Element count must match tuple arity                 |
| Map(K, V)               | `IDictionary`                            |                                                      |
| Nullable(T)             | `null`, `DBNull`, or types accepted by T | Writes null flag byte before value                   |
| Enum8                   | `string`, `sbyte`, numeric types         | String looked up in enum dictionary                  |
| Enum16                  | `string`, `short`, numeric types         | String looked up in enum dictionary                  |
| LowCardinality(T)       | Types accepted by T                      | Delegates to underlying type                         |
| SimpleAggregateFunction | Types accepted by underlying type        | Delegates to underlying type                         |
| Nested(...)             | `IList` of tuples                        | Element count must match field count                 |
| Variant(T1, T2, ...)    | Value matching one of T1, T2, ...        | Throws `ArgumentException` if no type match          |
| QBit(T, dim)            | `IList`                                  | Delegates to Array; dimension is metadata only       |

***

<h4 id="type-map-writing-geometry">
  Geometry types
</h4>

| ClickHouse Type | Accepted .NET Types                                    | Notes                         |
| --------------- | ------------------------------------------------------ | ----------------------------- |
| Point           | `System.Drawing.Point`, `ITuple`, `IList` (2 elements) |                               |
| Ring            | `IList` of Points                                      |                               |
| LineString      | `IList` of Points                                      |                               |
| Polygon         | `IList` of Rings                                       |                               |
| MultiLineString | `IList` of LineStrings                                 |                               |
| MultiPolygon    | `IList` of Polygons                                    |                               |
| Geometry        | Any geometry type above                                | Variant of all geometry types |

***

<h4 id="type-map-writing-not-supported">
  Not supported for writing
</h4>

| ClickHouse Type   | Notes                               |
| ----------------- | ----------------------------------- |
| Dynamic           | Throws `NotImplementedException`    |
| AggregateFunction | Throws `AggregateFunctionException` |

***

<h3 id="nested-type-handling">
  Nested type handling
</h3>

ClickHouse nested types (`Nested(...)`) can be read and written using array semantics.

```sql theme={null}
CREATE TABLE test.nested (
    id UInt32,
    params Nested (param_id UInt8, param_val String)
) ENGINE = Memory
```

```csharp theme={null}
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 }
);
```

<h2 id="logging-and-diagnostics">
  Logging and diagnostics
</h2>

The ClickHouse .NET client integrates with the `Microsoft.Extensions.Logging` abstractions to offer lightweight, opt-in logging. When enabled, the driver emits structured messages for connection lifecycle events, command execution, transport operations, and bulk insert operations. Logging is entirely optional—applications that do not configure a logger continue to run without additional overhead.

<h3 id="logging-quick-start">
  Quick start
</h3>

```csharp theme={null}
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);
```

<h4 id="logging-appsettings-config">
  Using appsettings.json
</h4>

You can configure logging levels using standard .NET configuration:

```csharp theme={null}
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);
```

<h4 id="logging-inmemory-config">
  Using in-memory configuration
</h4>

You can also configure logging verbosity by category in code:

```csharp theme={null}
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);
```

<h3 id="logging-categories">
  Categories and emitters
</h3>

The driver uses dedicated categories so that you can fine-tune log levels per component:

| Category                       | Source                 | Highlights                                                                                           |
| ------------------------------ | ---------------------- | ---------------------------------------------------------------------------------------------------- |
| `ClickHouse.Driver.Connection` | `ClickHouseConnection` | Connection lifecycle, HTTP client factory selection, connection opening/closing, session management. |
| `ClickHouse.Driver.Command`    | `ClickHouseCommand`    | Query execution start/completion, timing, query IDs, server statistics, and error details.           |
| `ClickHouse.Driver.Transport`  | `ClickHouseConnection` | Low-level HTTP streaming requests, compression flags, response status codes, and transport failures. |
| `ClickHouse.Driver.Client`     | `ClickHouseClient`     | Binary insert, queries, and other operations                                                         |
| `ClickHouse.Driver.NetTrace`   | `TraceHelper`          | Network tracing, only when debug mode is enabled                                                     |

<h4 id="logging-config-example">
  Example: Diagnosing connection issues
</h4>

```json theme={null}
{
    "Logging": {
        "LogLevel": {
            "ClickHouse.Driver.Connection": "Trace",
            "ClickHouse.Driver.Transport": "Trace"
        }
    }
}
```

This will log:

* HTTP client factory selection (default pool vs single connection)
* HTTP handler configuration (SocketsHttpHandler or HttpClientHandler)
* Connection pool settings (MaxConnectionsPerServer, PooledConnectionLifetime, etc.)
* Timeout settings (ConnectTimeout, Expect100ContinueTimeout, etc.)
* SSL/TLS configuration
* Connection open/close events
* Session ID tracking

<h3 id="logging-debugmode">
  Debug mode: network tracing and diagnostics
</h3>

To help with diagnosing networking issues, the driver library includes a helper that enables low-level tracing of .NET networking internals. To enable it you must pass a LoggerFactory with the level set to Trace, and set EnableDebugMode to true (or manually enable it via the `ClickHouse.Driver.Diagnostic.TraceHelper` class). Events will be logged to the `ClickHouse.Driver.NetTrace` category. Warning: this will generate extremely verbose logs, and impact performance. It isn't recommended to enable debug mode in production.

```csharp theme={null}
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
};
```

<h2 id="opentelemetry">
  OpenTelemetry
</h2>

The driver provides built-in support for OpenTelemetry distributed tracing via the .NET [`System.Diagnostics.Activity`](https://learn.microsoft.com/en-us/dotnet/core/diagnostics/distributed-tracing) API. When enabled, the driver emits spans for database operations that can be exported to observability backends like Jaeger or ClickHouse itself (via the [OpenTelemetry Collector](/guides/use-cases/observability/build-your-own/integrating-opentelemetry)).

<h3 id="opentelemetry-enabling">
  Enabling tracing
</h3>

In ASP.NET Core applications, add the ClickHouse driver's `ActivitySource` to your OpenTelemetry configuration:

```csharp theme={null}
builder.Services.AddOpenTelemetry()
    .WithTracing(tracing => tracing
        .AddSource(ClickHouseDiagnosticsOptions.ActivitySourceName)  // Subscribe to ClickHouse driver spans
        .AddAspNetCoreInstrumentation()
        .AddOtlpExporter());             // Or AddJaegerExporter(), etc.
```

For console applications, testing, or manual setup:

```csharp theme={null}
using OpenTelemetry;
using OpenTelemetry.Trace;

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

<h3 id="opentelemetry-attributes">
  Span attributes
</h3>

Each span includes standard OpenTelemetry database attributes plus ClickHouse-specific query statistics that can be used for debugging.

| Attribute                     | Description                               |
| ----------------------------- | ----------------------------------------- |
| `db.system`                   | Always `"clickhouse"`                     |
| `db.name`                     | Database name                             |
| `db.user`                     | Username                                  |
| `db.statement`                | SQL query (if enabled)                    |
| `db.clickhouse.read_rows`     | Rows read by the query                    |
| `db.clickhouse.read_bytes`    | Bytes read by the query                   |
| `db.clickhouse.written_rows`  | Rows written by the query                 |
| `db.clickhouse.written_bytes` | Bytes written by the query                |
| `db.clickhouse.elapsed_ns`    | Server-side execution time in nanoseconds |

<h3 id="opentelemetry-configuration">
  Configuration options
</h3>

Control tracing behavior via `ClickHouseDiagnosticsOptions`:

```csharp theme={null}
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;
```

<Warning>
  Enabling `IncludeSqlInActivityTags` may expose sensitive data in your traces. Use with caution in production environments.
</Warning>

<h2 id="tls-configuration">
  TLS configuration
</h2>

When connecting to ClickHouse over HTTPS, you can configure TLS/SSL behavior in several ways.

<h3 id="custom-certificate-validation">
  Custom certificate validation
</h3>

For production environments requiring custom certificate validation logic, provide your own `HttpClient` with a configured `ServerCertificateCustomValidationCallback` handler:

```csharp theme={null}
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);
```

<Note>
  Important considerations when providing a custom HttpClient

  * **Automatic decompression**: You must enable `AutomaticDecompression` if compression isn't disabled (compression is enabled by default).
  * **Idle timeout**: Set `PooledConnectionIdleTimeout` smaller than the server's `keep_alive_timeout` (10 seconds for ClickHouse Cloud) to avoid connection errors from half-open connections.
</Note>

<h2 id="orm-support">
  ORM support
</h2>

ORMs require the ADO.NET API (`ClickHouseConnection`). For proper connection lifetime management, create connections from a `ClickHouseDataSource`:

```csharp theme={null}
// 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...
```

<h3 id="orm-support-dapper">
  Dapper
</h3>

`ClickHouse.Driver` works with Dapper. The driver automatically converts Dapper's `@parameter` syntax to ClickHouse's native `{parameter:Type}` syntax, with types inferred from .NET values.

Use `ClickHouseDataSource` for proper connection lifetime management:

```csharp theme={null}
var dataSource = new ClickHouseDataSource("Host=localhost");
services.AddSingleton(dataSource); // Register as singleton in DI

using var connection = dataSource.CreateConnection();
```

<h4 id="dapper-parameter-passing">
  Parameter passing styles
</h4>

All standard Dapper parameter styles are supported:

**Anonymous objects:**

```csharp theme={null}
await connection.ExecuteAsync(
    "INSERT INTO users (id, name, balance) VALUES (@Id, @Name, @Balance)",
    new { Id = 1, Name = "alice", Balance = 3.14 });
```

**POCO classes:**

```csharp theme={null}
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);
```

**Dictionary:**

```csharp theme={null}
var parameters = new Dictionary<string, object> { { "Id", 2 } };
var rows = await connection.QueryAsync<User>(
    "SELECT id, name FROM users WHERE id = @Id", parameters);
```

**`DynamicParameters` (from dictionary or anonymous object):**

```csharp theme={null}
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);
```

<h4 id="dapper-pocos">
  Querying into POCOs
</h4>

Dapper maps columns to properties by name (case-insensitive):

```csharp theme={null}
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();
```

<h4 id="dapper-clickhouse-param-syntax">
  ClickHouse-native parameter syntax
</h4>

When you need explicit type control, use ClickHouse's `{param:Type}` syntax directly in the SQL with a `Dictionary<string, object>` for the parameter values. Don't combine `@param` syntax and `{param:Type}` syntax for the same parameter.

```csharp theme={null}
var parameters = new Dictionary<string, object> { { "value", 42 } };
var result = await connection.QueryAsync<int>("SELECT {value:Int32}", parameters);
```

<h4 id="dapper-where-in">
  WHERE IN
</h4>

**Dapper's native IN expansion works:**

```csharp theme={null}
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 rewrites this to `WHERE id IN (@Ids1, @Ids2, @Ids3)`, and the driver converts each expanded parameter.

**ClickHouse's `has()` with Array parameter also works:**

```csharp theme={null}
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);
```

<h4 id="dapper-type-handlers">
  Custom type handlers
</h4>

Some ClickHouse types, eg `ITuple`, `BigInteger`, and `ClickHouseDecimal` need handlers registered at startup:

```csharp theme={null}
// 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());
```

See the [Dapper example](https://github.com/ClickHouse/clickhouse-cs/blob/main/examples/ORM/ORM_001_Dapper.cs) for an example type handler implementation.

<h4 id="dapper-contrib">
  Dapper.Contrib
</h4>

`GetAll<T>()` and `Get<T>(id)` work. `Insert<T>()` does not — it generates SQL Server syntax (`SCOPE_IDENTITY`, `[]`). It is recommended to use the `ClickHouseClient` native `InsertBinaryAsync` method instead.

```csharp theme={null}
[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);
```

Property names must match ClickHouse column names exactly (case-sensitive).

<h4 id="dapper-limitations">
  Limitations
</h4>

| What                         | Status        | Details                                                             |
| ---------------------------- | ------------- | ------------------------------------------------------------------- |
| Tuple as **result**          | Works         | Requires `SqlMapper.TypeHandler<ITuple>` registration               |
| Tuple as **parameter**       | Not supported | Dapper cannot serialize `ITuple`/`Tuple<>` as a `DbParameter` value |
| Nested types as parameter    | Not supported | Same reason — Dapper rejects complex types as parameter values      |
| Geo types as parameter       | Not supported | Point, Ring, Polygon, LineString, MultiLineString, MultiPolygon     |
| `Dapper.Contrib.Insert<T>()` | Not supported | Generates SQL Server-specific syntax                                |
| `Nothing` type               | Not supported | No meaningful .NET representation                                   |

<h3 id="orm-support-linq2db">
  Linq2db
</h3>

This driver is compatible with [linq2db](https://github.com/linq2db/linq2db), a lightweight ORM and LINQ provider for .NET. See the project website for detailed documentation.

**Example usage:**

Create a `DataConnection` using the ClickHouse provider:

```csharp theme={null}
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);
```

Table mappings can be defined using attributes or fluent configuration. If your class and property names match the table and column names exactly, no configuration is needed:

```csharp theme={null}
public class Product
{
    public int Id { get; set; }
    public string Name { get; set; }
    public decimal Price { get; set; }
}
```

**Querying:**

```csharp theme={null}
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:**

Use `BulkCopyAsync` for efficient bulk inserts.

```csharp theme={null}
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);
```

<h3 id="orm-support-ef-core">
  Entity Framework Core
</h3>

The official Entity Framework Core provider for ClickHouse. Map C# classes to ClickHouse tables, query with LINQ, and insert data via `SaveChanges` — all using familiar EF Core patterns.

* **NuGet**: [`ClickHouse.EntityFrameworkCore`](https://www.nuget.org/packages/ClickHouse.EntityFrameworkCore)
* **Source**: [GitHub](https://github.com/ClickHouse/ClickHouse.EntityFrameworkCore)

<Note>
  This provider is in active development. Current release supports LINQ queries (including JOINs, subqueries, and set operations), `INSERT` via `SaveChanges` / `BulkInsertAsync`, migrations with full DDL (CREATE / ALTER / DROP), and ClickHouse-specific table engine configuration. `UPDATE` / `DELETE` are not supported.
</Note>

<h4 id="ef-core-installation">
  Installation
</h4>

```bash theme={null}
dotnet add package ClickHouse.EntityFrameworkCore
```

Requires .NET 10.0 and EF Core 10.

<h4 id="ef-core-quick-start">
  Quick start
</h4>

Define your entity and `DbContext`, then query with LINQ:

```csharp theme={null}
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();
```

<h4 id="ef-core-types">
  Supported types
</h4>

| Category         | ClickHouse Types                                                                        | CLR Types                                                                                                      |
| ---------------- | --------------------------------------------------------------------------------------- | -------------------------------------------------------------------------------------------------------------- |
| **Integers**     | `Int8`–`Int64`, `UInt8`–`UInt64`                                                        | `sbyte`, `short`, `int`, `long`, `byte`, `ushort`, `uint`, `ulong`                                             |
| **Big integers** | `Int128`, `Int256`, `UInt128`, `UInt256`                                                | `BigInteger`                                                                                                   |
| **Floats**       | `Float32`, `Float64`, `BFloat16`                                                        | `float`, `double`                                                                                              |
| **Decimals**     | `Decimal(P,S)`, `Decimal32(S)`, `Decimal64(S)`, `Decimal128(S)`                         | `decimal` or `ClickHouseDecimal`                                                                               |
| **Bool**         | `Bool`                                                                                  | `bool`                                                                                                         |
| **Strings**      | `String`, `FixedString(N)`                                                              | `string`                                                                                                       |
| **Enums**        | `Enum8(...)`, `Enum16(...)`                                                             | `string` or C# `enum`                                                                                          |
| **Date/time**    | `Date`, `Date32`, `DateTime`, `DateTime64(P, 'TZ')`                                     | `DateOnly`, `DateTime`                                                                                         |
| **Time**         | `Time`, `Time64(N)`                                                                     | `TimeSpan`                                                                                                     |
| **UUID**         | `UUID`                                                                                  | `Guid`                                                                                                         |
| **Network**      | `IPv4`, `IPv6`                                                                          | `IPAddress`                                                                                                    |
| **Arrays**       | `Array(T)`                                                                              | `T[]`, `List<T>`, `IList<T>`, `ICollection<T>`, `IReadOnlyList<T>`, `IReadOnlyCollection<T>`, `IEnumerable<T>` |
| **Maps**         | `Map(K, V)`                                                                             | `Dictionary<K,V>`                                                                                              |
| **Tuples**       | `Tuple(T1, ...)`                                                                        | `Tuple<...>` or `ValueTuple<...>`                                                                              |
| **Variant**      | `Variant(T1, T2, ...)`                                                                  | `object`                                                                                                       |
| **Dynamic**      | `Dynamic`                                                                               | `object`                                                                                                       |
| **JSON**         | `Json`                                                                                  | `JsonNode` or `string`                                                                                         |
| **Geographic**   | `Point`, `Ring`, `LineString`, `Polygon`, `MultiLineString`, `MultiPolygon`, `Geometry` | `Tuple<double,double>` and arrays thereof; `object` for Geometry                                               |
| **Wrappers**     | `Nullable(T)`, `LowCardinality(T)`                                                      | Unwrapped automatically                                                                                        |

Use `ClickHouseDecimal` (from `ClickHouse.Driver.Numerics`) instead of `decimal` when you need the full precision of `Decimal128`/`Decimal256` columns — .NET `decimal` is limited to 28–29 significant digits.

<h4 id="ef-core-linq">
  Supported LINQ operations
</h4>

**Queries:** `Where`, `OrderBy`, `Take`, `Skip`, `Select`, `First`, `Single`, `Any`, `All`, `Count`, `Distinct`, `AsNoTracking`

**GROUP BY & aggregates:** `GroupBy` with `Count`, `LongCount`, `Sum`, `Average`, `Min`, `Max` — including `HAVING` (`.Where()` after `.GroupBy()`), multiple aggregates in a single projection, and `OrderBy` on aggregate results.

**JOINs:** `Join` (INNER), `GroupJoin`/`SelectMany` patterns (LEFT and CROSS). LEFT JOIN returns real `null` for non-matching rows (see [LEFT JOIN null semantics](#ef-core-join-nulls) below).

**Subqueries:** correlated `Contains` / `IN`, `Any` / `EXISTS`, `All`, and scalar subqueries in projections.

**Set operations:** `Concat` (→ `UNION ALL`), `Union` (→ `UNION DISTINCT`), `Intersect`, `Except`.

**Inline local collections:** joins and `Contains` against in-memory collections (`int[]`, `List<T>`, etc.) translate into a series of UNIONs.

**String methods:** `Contains`, `StartsWith`, `EndsWith`, `IndexOf`, `Replace`, `Substring`, `Trim`/`TrimStart`/`TrimEnd`, `ToLower`, `ToUpper`, `Length`, `IsNullOrEmpty`, `Concat` (and `+` operator).

**Math functions:** standard `Math` and `MathF` methods translated to their ClickHouse equivalents — arithmetic, logarithmic, trigonometric, and utility functions.

<h5 id="ef-core-join-nulls">
  LEFT JOIN null semantics
</h5>

The provider injects `set_join_use_nulls=1` into every connection path automatically to match Entity Framework expectations on JOIN behavior.

If your ClickHouse server or profile forbids changing this setting (e.g. a `readonly=1` profile), opt out with:

```csharp theme={null}
optionsBuilder.UseClickHouse(connectionString, o => o.DisableJoinNullSemantics());
```

With the opt-out enabled, LEFT JOIN returns ClickHouse column defaults and EF's null-based navigation detection no longer works as expected. Use explicit comparisons against `0` / `""` instead of `== null`.

<h4 id="ef-core-insert">
  Inserting data
</h4>

`SaveChanges` uses the driver's native `InsertBinaryAsync` API — RowBinary encoding with GZip compression, far more efficient than parameterized SQL:

```csharp theme={null}
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();
```

Entities transition from `Added` to `Unchanged` after save, just like any other EF Core provider.

**Batch size** is configurable (default 1000):

```csharp theme={null}
optionsBuilder.UseClickHouse("Host=localhost", o => o.MaxBatchSize(5000));
```

<h4 id="ef-core-bulk-insert">
  Bulk insert
</h4>

For high-throughput loads, use `BulkInsertAsync` instead of `SaveChanges`. This is an extension method on `DbContext` that bypasses EF Core's change tracker, identity resolution, and state management entirely — it calls the driver's `InsertBinaryAsync` directly with RowBinary encoding and GZip compression.

This makes it suitable for loading large datasets where you don't need entity tracking after insert:

```csharp theme={null}
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);
```

The input can be any `IEnumerable<T>` — it streams through the entities without loading them all into memory. The return value is the number of rows inserted. Entities are **not** attached to the `DbContext` after insert, so there is no `Added` → `Unchanged` state transition.

<h4 id="ef-core-enums">
  Enums
</h4>

ClickHouse `Enum8`/`Enum16` columns can be mapped as `string` properties or as C# `enum` types. When using C# enums, the provider automatically converts between the enum and its string representation:

```csharp theme={null}
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();
```

<h4 id="ef-core-value-converters">
  Custom type conversions
</h4>

EF Core's `ValueConverter` system lets you map custom types to types the provider already supports. The provider never sees your custom type — EF Core converts at the boundary.

**Per-property conversion:**

```csharp theme={null}
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");
```

**Reusable converter class:**

```csharp theme={null}
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>();
}
```

<h4 id="ef-core-column-types">
  Column type annotations
</h4>

For scalar types like `string`, `int`, `DateTime`, etc., the provider infers the ClickHouse type automatically. For parameterized types and wrappers, you need to specify the ClickHouse type explicitly.

**Using data annotations (attributes):**

```csharp theme={null}
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; }
}
```

**Using the fluent API in `OnModelCreating`:**

```csharp theme={null}
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)");
});
```

Nested wrappers like `Array(Nullable(Int32))` and `LowCardinality(Nullable(String))` are supported — the provider unwraps `Nullable` and `LowCardinality` automatically at every nesting level.

<h4 id="ef-core-variant-dynamic">
  Variant and Dynamic columns
</h4>

ClickHouse `Variant(T1, T2, ...)` and `Dynamic` columns map to `object` in .NET. Since `object` is too generic for automatic type inference, you must declare the store type explicitly via `.HasColumnType()`:

```csharp theme={null}
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");
```

When reading, the value is automatically deserialized to the corresponding .NET type for the stored discriminator (e.g. `string`, `ulong`, `ulong[]`).

<h4 id="ef-core-json">
  JSON columns
</h4>

The provider supports ClickHouse's `Json` column type, mapping to `System.Text.Json.Nodes.JsonNode` (primary) or `string` (via automatic `ValueConverter`):

```csharp theme={null}
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");
```

Reading and writing JSON works through both `SaveChanges` and `BulkInsertAsync`:

```csharp theme={null}
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"
```

If you prefer raw JSON strings, map the property as `string` with a `Json` column type — the provider applies a `ValueConverter` automatically:

```csharp theme={null}
public class Event
{
    public long Id { get; set; }
    public string? Data { get; set; }  // raw JSON string
}

entity.Property(e => e.Data).HasColumnType("Json");
```

<Note>
  * **No JSON path translation** — `entity.Data["name"]` in LINQ does not translate to ClickHouse's `data.name` SQL syntax. Filter on non-JSON columns and inspect JSON in memory.
  * **NULL semantics** — ClickHouse's JSON type returns `{}` (empty object) for NULL values rather than SQL NULL.
  * **Integer precision** — ClickHouse JSON stores all integers as `Int64`. When reading via `JsonNode`, use `GetValue<long>()` rather than `GetValue<int>()`.
</Note>

<h4 id="ef-core-engines">
  Table engines
</h4>

Configure ClickHouse table engines and engine-specific clauses via the `ToTable(name, t => ...)` fluent API. When no engine is configured, the provider defaults to `MergeTree` with `ORDER BY` derived from the entity's primary key.

```csharp theme={null}
modelBuilder.Entity<Event>(e =>
{
    e.ToTable("events", t => t
        .HasMergeTreeEngine()
        .WithOrderBy("UserId", "Timestamp")
        .WithPartitionBy("toYYYYMM(Timestamp)")
        .WithPrimaryKey("UserId")
        .WithSettings("index_granularity = 8192"));
});
```

Supported engine families:

| Engine                                  | Fluent method                                                                                             | Notes                                |
| --------------------------------------- | --------------------------------------------------------------------------------------------------------- | ------------------------------------ |
| `MergeTree`                             | `HasMergeTreeEngine()`                                                                                    | Default when none configured         |
| `ReplacingMergeTree`                    | `HasReplacingMergeTreeEngine("Version", "IsDeleted")` or `HasReplacingMergeTreeEngine<T>(e => e.Version)` | Version / IsDeleted columns optional |
| `SummingMergeTree`                      | `HasSummingMergeTreeEngine(…)` or `HasSummingMergeTreeEngine<T>(e => new { … })`                          | Optional columns-to-sum              |
| `AggregatingMergeTree`                  | `HasAggregatingMergeTreeEngine()`                                                                         | —                                    |
| `CollapsingMergeTree`                   | `HasCollapsingMergeTreeEngine("Sign")` or `HasCollapsingMergeTreeEngine<T>(e => e.Sign)`                  | `Sign` column must be `Int8`         |
| `VersionedCollapsingMergeTree`          | `HasVersionedCollapsingMergeTreeEngine("Sign", "Version")` or `<T>(e => e.Sign, e => e.Version)`          | —                                    |
| `GraphiteMergeTree`                     | `HasGraphiteMergeTreeEngine("config_section")`                                                            | —                                    |
| `Log`, `TinyLog`, `StripeLog`, `Memory` | `HasLogEngine()`, `HasTinyLogEngine()`, `HasStripeLogEngine()`, `HasMemoryEngine()`                       | No ORDER BY / PARTITION BY           |

**Engine clauses:** `WithOrderBy`, `WithPartitionBy`, `WithPrimaryKey`, `WithSampleBy`, `WithTtl`, `WithSettings`. All attach to the engine builder returned from `HasXxxEngine()`.

**Column-level features:** `HasCodec`, `HasTtl`, `HasComment`, `HasDefault` — all participate in migrations.

**Data-skipping indexes** — via `HasIndex(...).HasSkippingIndexType(...)`:

```csharp theme={null}
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);
```

Standard (non-skipping) indexes are silently ignored since ClickHouse has no equivalent. Unique indexes throw, as ClickHouse does not enforce uniqueness.

<h4 id="ef-core-migrations">
  Migrations
</h4>

Standard EF Core migrations workflow:

```bash theme={null}
dotnet ef migrations add InitialCreate
dotnet ef database update
```

Supported operations:

| Operation                              | Emits                                                                                         |
| -------------------------------------- | --------------------------------------------------------------------------------------------- |
| `CREATE TABLE`                         | Includes engine clause, ORDER BY, PARTITION BY, SETTINGS, column codecs/TTL/comments/defaults |
| `ALTER TABLE ADD COLUMN`               | —                                                                                             |
| `ALTER TABLE DROP COLUMN`              | —                                                                                             |
| `ALTER TABLE MODIFY COLUMN`            | Handles type change plus annotation add/remove (CODEC, TTL, COMMENT, DEFAULT)                 |
| `ALTER TABLE RENAME COLUMN`            | —                                                                                             |
| `RENAME TABLE`                         | —                                                                                             |
| `ALTER TABLE ADD INDEX` / `DROP INDEX` | Data-skipping indexes only                                                                    |
| `CREATE DATABASE` / `DROP DATABASE`    | Via `EnsureCreated` / `EnsureDeleted` and migrations                                          |

<h4 id="ef-core-limitations">
  Migration limitations
</h4>

| Feature                                               | Reason                                                                                                                                                       |
| ----------------------------------------------------- | ------------------------------------------------------------------------------------------------------------------------------------------------------------ |
| Foreign keys                                          | ClickHouse does not enforce foreign keys. Migrations reject `AddForeignKey`; the model validator emits a warning at model build time.                        |
| Unique constraints / unique indexes                   | ClickHouse does not enforce uniqueness. Unique indexes throw at migration time.                                                                              |
| Server-generated values (auto-increment / `IDENTITY`) | ClickHouse has no equivalent.                                                                                                                                |
| `Nested(…)` columns                                   | Not yet supported as a mapped CLR type.                                                                                                                      |
| Owned entities as JSON (`.ToJson()`)                  | Structural JSON mapping for owned entities is not yet implemented. Use `JsonNode` / `string` on a `Json` column instead (see [JSON columns](#ef-core-json)). |

Beyond migrations, the provider also does not yet support:

* **`UPDATE` / `DELETE`**
* **Transactions**: `BeginTransaction` is a no-op. No support for ACID transactions in ClickHouse.
* **JSON path query translation**: `entity.Data["key"]` in LINQ does not translate to ClickHouse's `data.key` SQL syntax. Filter on non-JSON columns and inspect JSON in memory.

<h2 id="limitations">
  Limitations
</h2>

<h3 id="aggregatefunction-columns">
  AggregateFunction columns
</h3>

Columns of type `AggregateFunction(...)` can't be queried or inserted directly.

To insert:

```sql theme={null}
INSERT INTO t VALUES (uniqState(1));
```

To select:

```sql theme={null}
SELECT uniqMerge(c) FROM t;
```

***
