Logical data replication is only supported in CockroachDB self-hosted clusters.
The CREATE LOGICAL REPLICATION STREAM statement starts logical data replication (LDR) that runs between a source and destination cluster in an active-active setup.
This page is a reference for the CREATE LOGICAL REPLICATION STREAM SQL statement, which includes information on its parameters and possible options. For a step-by-step guide to set up LDR, refer to the Set Up Logical Data Replication page.
If the table you're replicating does not contain user-defined types, we recommend using the CREATE LOGICALLY REPLICATED syntax that provides a fast, offline initial scan and automatic table setup on the destination cluster.
Required privileges
CREATE LOGICAL REPLICATION STREAM creates a one-way LDR stream only. To achieve bidirectional replication, you must manually create two separate streams, one in each direction, with the required privileges set on both clusters.
LDR from cluster A to B represents a one-way stream from a source to a destination cluster. LDR from cluster B to A is the reverse stream for a bidirectional setup.
To run the CREATE LOGICAL REPLICATION STREAM statement to create an LDR stream, the following privileges are required:
| LDR direction | Cluster | User role | Required privilege | 
|---|---|---|---|
| A ➔ B | A | User in the LDR connection string. | REPLICATIONSOURCE | 
| A ➔ B | B | User running the command. | REPLICATIONDEST | 
| B ➔ A | B | User in the LDR connection string. | REPLICATIONSOURCE | 
| B ➔ A | A | User running the command. | REPLICATIONDEST | 
Grant the privilege at the table or system level with the GRANT statement to a user or a role:
GRANT REPLICATIONSOURCE ON TABLE database.public.tablename TO user/role;
As of v25.2, the REPLICATION system privilege is deprecated and will be removed in a future release. Use REPLICATIONSOURCE and REPLICATIONDEST for authorization at the table level.
Synopsis
Parameters
| Parameter | Description | 
|---|---|
| db_object_name | The fully qualified name of the table on the source or destination cluster. Refer to Examples. | 
| logical_replication_resources_list | A list of the fully qualified table names on the source or destination cluster to include in the LDR stream. Refer to the LDR with multiple tables example. | 
| source_connection_string | The connection string to the source cluster. Use an external connection to store the source cluster's connection URI. To start LDR, you run CREATE LOGICAL REPLICATION STREAMfrom the destination cluster. | 
| logical_replication_options | Options to modify the behavior of the LDR stream. | 
Options
| Option | Description | 
|---|---|
| cursor | Emits any changes after the specified timestamp. LDR will not perform an initial backfill with the cursoroption, it will stream any changes after the specified timestamp. The LDR job will encounter an error if you specify acursortimestamp that is before the configured garbage collection window for that table. Warning: Apply thecursoroption carefully to LDR streams. Using a timestamp in error could cause data loss. | 
| discard | (Unidirectional LDR only) Ignore TTL deletes in an LDR stream with discard = ttl-deletes. Note: To ignore row-level TTL deletes in an LDR stream, it is necessary to set thettl_disable_changefeed_replicationstorage parameter on the source table. Refer to the Ignore row-level TTL deletes example. | 
| label | Tracks LDR metrics at the job level. Add a user-specified string with label. Refer to Metrics labels. | 
Bidirectional LDR
Bidirectional LDR consists of two clusters with two LDR jobs running in opposite directions between the clusters. If you're setting up bidirectional LDR, both clusters will act as a source and a destination in the respective LDR jobs.
LDR supports starting with two empty tables, or one non-empty table. LDR does not support starting with two non-empty tables. When you set up bidirectional LDR, if you're starting with one non-empty table, start the first LDR job from empty to non-empty table. Therefore, you would run CREATE LOGICAL REPLICATION STREAM from the destination cluster where the non-empty table exists.
Examples
To start LDR, you must run the CREATE LOGICAL REPLICATION STREAM statement from the destination cluster. Use the fully qualified table name(s). The following examples show statement usage with different options and use cases.
Start an LDR stream
There are some tradeoffs between enabling one table per LDR job versus multiple tables in one LDR job. Multiple tables in one LDR job can be easier to operate. For example, if you pause and resume the single job, LDR will stop and resume for all the tables. However, the most granular level observability will be at the job level. One table in one LDR job will allow for table-level observability.
Single table
CREATE LOGICAL REPLICATION STREAM FROM TABLE {database.public.table_name} ON 'external://{source_external_connection}' INTO TABLE {database.public.table_name};
Multiple tables
CREATE LOGICAL REPLICATION STREAM FROM TABLES ({database.public.table_name},{database.public.table_name},...)  ON 'external://{source_external_connection}' INTO TABLES ({database.public.table_name},{database.public.table_name},...);
Ignore row-level TTL deletes
If you would like to ignore row-level TTL deletes in a unidirectional LDR stream, set the ttl_disable_changefeed_replication storage parameter on the table. On the source cluster, alter the table to set the table storage parameter:
ALTER TABLE {table_name} SET (ttl_disable_changefeed_replication = 'true');
When you start LDR on the destination cluster, include the discard = ttl-deletes option in the statement:
CREATE LOGICAL REPLICATION STREAM FROM TABLE {database.public.table_name} ON 'external://{source_external_connection}' INTO TABLE {database.public.table_name} WITH discard = ttl-deletes;