Overview
If you are already using multi-region SQL statements to control your multi-region cluster, you can ignore this page.
CockroachDB v21.1 added support for improved multi-region capabilities that make it easier to run global applications. Using high-level SQL statements, you can control where your data is stored and how it is accessed to provide good performance and tuneable latency for your application's users.
Prior to v21.1, the only way to accomplish these goals in a multi-region cluster involved using lower-level mechanisms called replication zones in specific patterns called Duplicate Indexes, Geo-partitioned Replicas, and Geo-partitioned leaseholders.
These patterns and the use of replication zones are still fully supported. However, for most users, they are harder to use and in some cases can result in worse performance than the multi-region SQL abstractions.
This page describes how to migrate a multi-region cluster from using replication zones to using multi-region SQL abstractions. It contains:
- Mappings from each of the legacy replication-zone-based patterns to the multi-region SQL abstractions that are designed to replace them. 
- Instructions for migrating from replication-zone-based workflows to multi-region SQL abstractions. 
- A review of the underlying zone configuration changes that result from running the multi-region SQL statements. 
Replication zone patterns and multi-region SQL abstractions
| Replication Zone Pattern | Multi-Region SQL | 
|---|---|
| Duplicate indexes | GLOBALtables | 
| Geo-partitioned replicas | REGIONAL BY ROWtables withZONEsurvival goals | 
| Geo-partitioned leaseholders | REGIONAL BY ROWtables withREGIONsurvival goals | 
CockroachDB will no longer provide the Follow-the-Workload pattern's behavior for a database if you use the multi-region SQL statements with that database. In other words, the multi-region SQL statements do not provide a behavior that is analogous to Follow-the-Workload.
For more information about how to use ZONE vs. REGION survival goals, see When to use ZONE vs REGION survival goals.
For more information about when to use GLOBAL vs. REGIONAL tables, see When to use REGIONAL vs GLOBAL tables.
How to migrate a database to the multi-region SQL abstractions
The following instructions assume that you will re-use your existing cluster regions (that is, regions added during cluster setup using cockroach start --locality).
Performance considerations
Expect performance to be temporarily affected by this process. When you run the commands described below, the cluster may need to do a significant amount of work to meet the new replica placement constraints it has just been given. Therefore, we recommend running this procedure at a time when you would normally perform scheduled maintenance.
Depending on how long you are willing to wait between steps for the replica rebalancing process to settle down, data movement may still be occurring when the next set of constraints is added using the multi-region SQL abstractions; this may lead to further data movement.
In other words, the process described here may result in a significant increase in CPU usage, IOPS, and network traffic while the cluster rebalances replicas to meet the final set of constraints you have provided it with. Until this process completes, the cluster may not be able to handle its normal workload.
Note that the process of dropping the old zone configs that occurs when the old configurations are removed must be complete before you can add regions to the database and set table localities. You can ensure this process is complete by waiting for the ALTER DATABASE ... CONFIGURE ZONE DISCARD statement shown below to finish successfully.
For a tutorial that shows how to transition a database to using multi-region SQL statements, see Low Latency Reads and Writes in a Multi-Region Cluster.
As part of this migration, data may move temporarily out of the geography where you have constrained it to be placed. For more information about CockroachDB's support for data domiciling, see Data Domiciling with CockroachDB.
Step 1. Remove the old replication zone configurations
Depending on which legacy multi-region topology pattern you are migrating from, the procedure will vary. For instructions showing how to remove the existing zone configuration for each pattern, see below.
You can check the state of any schema object's replication zone configuration at any time using SHOW ZONE CONFIGURATIONS.
Duplicate indexes
If you used the [duplicate indexes pattern][dupe_index], the steps for backing out the old configuration are:
- Remove the replication zone configurations you added using the - ALTER DATABASE ... CONFIGURE ZONE DISCARDstatement. Note that this will remove all zone configurations from the table. If you had any additional customizations beyond what are required for the [duplicate indexes][dupe_index] pattern, you will have to reapply them.- ALTER TABLE postal_codes CONFIGURE ZONE DISCARD;
- Drop the extra indexes you added. This will have the side effect of also deleting the zone configurations you added to those indexes. - DROP INDEX postal_codes@idx_central; DROP INDEX postal_codes@idx_east;
The latency and resiliency benefits of the duplicate indexes pattern can be replaced by making postal_codes a GLOBAL table with a survival goal that meets your needs.
Geo-partitioned replicas
If you applied the [geo-partitioned replicas][geo_replicas] pattern, the steps for backing out the old configuration are:
- Remove the manually created table partition. This will also automatically remove the replication zone configurations that were applied to the partition as part of the instructions. - ALTER TABLE users PARTITION BY NOTHING;
- Remove the manually created partition on the secondary indexes. This will also automatically remove the replication zone configurations that were applied to the partition as part of the instructions. - ALTER INDEX users_last_name_index PARTITION BY NOTHING;
The latency and resiliency benefits of the geo-partitioned replicas pattern can be replaced by making users a REGIONAL BY ROW table with a ZONE survival goal.
The multi-region SQL abstractions use a hidden crdb_region column to represent the row's home region. You may need to modify your existing schema to take this into account. For example, if you already have a column you are using to denote each row's home region, you can use that name instead of crdb_region by following the instructions on the ALTER TABLE ... SET LOCALITY page.
Geo-partitioned leaseholders
If you applied the [geo-partitioned leaseholders][geo_leaseholders] pattern, the steps for backing out the old configuration are:
- Remove the manually created table partition. This will also automatically remove the replication zone configurations that were applied to the partition as part of the instructions. - ALTER TABLE users PARTITION BY NOTHING;
- Remove the manually created partition on the secondary indexes. This will also automatically remove the replication zone configurations that were applied to the partition as part of the instructions. - ALTER INDEX users_last_name_index PARTITION BY NOTHING;
The latency and resiliency benefits of the geo-partitioned leaseholders pattern can be replaced by making users a REGIONAL BY ROW table with a ZONE survival goal.
Step 2. Add a primary region to your database
The steps from this point forward assume that you have cleared your prior replication zone configurations and will be using the multi-region SQL abstractions to work with a cluster that has existing cluster regions.
Every multi-region database needs to have a primary region. To set the primary region, issue the SET PRIMARY REGION statement:
ALTER DATABASE foo SET PRIMARY REGION = "us-west1"
Step 3. Add more regions to the database
To add another region to the database, issue the ADD REGION statement:
ALTER database foo ADD REGION "us-central1";
Step 4. (Optional) Configure your database survival goal
Depending on your desired database survival goal, you can choose from the following settings:
- ZONEsurvival (Default): the database will remain fully available for reads and writes if one zone in a region goes down. More than one zone going down concurrently may affect availability.
- REGIONsurvival: the database will remain fully available for reads and writes if one region goes down. More than one region going down concurrently may affect availability.
For example, to set a region survival goal, issue the following SQL statement:
ALTER DATABASE foo SURVIVE REGION FAILURE;
For more information about when to use ZONE vs. REGION survival goals, see When to use ZONE vs REGION survival goals.
Step 5. Configure table localities
For each table in your database, apply the table locality that provides the latency and resiliency requirements you need for that table.
As described above, the mapping from legacy replication zone patterns to multi-region SQL abstractions is:
| Replication Zone Pattern | Multi-Region SQL | 
|---|---|
| Duplicate indexes | GLOBALtables | 
| Geo-partitioned replicas | REGIONAL BY ROWtables withZONEsurvival goals | 
| Geo-partitioned leaseholders | REGIONAL BY ROWtables withREGIONsurvival goals | 
For example, to configure the postal_codes table from the duplicate indexes example above to use multi-region SQL, you would enter the following statements to make the postal_codes table a GLOBAL table:
ALTER TABLE postal_codes SET LOCALITY GLOBAL;
For more information about when to use GLOBAL vs. REGIONAL tables, see When to use REGIONAL vs GLOBAL tables.
Step 6. (Optional) View the updated zone configurations
The multi-region SQL statements operate on the same replication zone configurations that you have access to via the ALTER TABLE ... CONFIGURE ZONE statement. If you are interested in seeing how they work with the lower-level zone config mechanisms, you can use the SHOW ZONE CONFIGURATIONS statement to view the zone configurations.
For example, given a multi-region demo cluster set up using the instructions in Low latency Reads and Writes in a Multi-Region Cluster, here is what the zone configs for several tables in the MovR schema look like.
Regional by row tables
A REGIONAL BY ROW table differs from the default by setting the following zone configuration settings:
SHOW ZONE CONFIGURATION FROM TABLE users;
     target     |                                      raw_config_sql
----------------+-------------------------------------------------------------------------------------------
  DATABASE movr | ALTER DATABASE movr CONFIGURE ZONE USING
                |     range_min_bytes = 134217728,
                |     range_max_bytes = 536870912,
                |     gc.ttlseconds = 90000,
                |     num_replicas = 5,
                |     num_voters = 3,
                |     constraints = '{+region=europe-west1: 1, +region=us-east1: 1, +region=us-west1: 1}',
                |     voter_constraints = '[+region=us-east1]',
                |     lease_preferences = '[[+region=us-east1]]'
(1 row)
The same table, loaded in a demo cluster with no zone configuration changes, looks like this:
SHOW ZONE CONFIGURATION FROM TABLE users;
     target     |              raw_config_sql
----------------+-------------------------------------------
  RANGE default | ALTER RANGE default CONFIGURE ZONE USING
                |     range_min_bytes = 134217728,
                |     range_max_bytes = 536870912,
                |     gc.ttlseconds = 90000,
                |     num_replicas = 3,
                |     constraints = '[]',
                |     lease_preferences = '[]'
(1 row)
Global tables
A GLOBAL table differs from the default by setting the following zone configuration settings:
SHOW ZONE CONFIGURATION FROM TABLE promo_codes;
       target       |                                      raw_config_sql
--------------------+-------------------------------------------------------------------------------------------
  TABLE promo_codes | ALTER TABLE promo_codes CONFIGURE ZONE USING
                    |     range_min_bytes = 134217728,
                    |     range_max_bytes = 536870912,
                    |     gc.ttlseconds = 90000,
                    |     global_reads = true,
                    |     num_replicas = 5,
                    |     num_voters = 3,
                    |     constraints = '{+region=europe-west1: 1, +region=us-east1: 1, +region=us-west1: 1}',
                    |     voter_constraints = '[+region=us-east1]',
                    |     lease_preferences = '[[+region=us-east1]]'
(1 row)
The same table, loaded in a demo cluster with no zone configuration changes, looks like this:
SHOW ZONE CONFIGURATION FROM TABLE promo_codes;
     target     |              raw_config_sql
----------------+-------------------------------------------
  RANGE default | ALTER RANGE default CONFIGURE ZONE USING
                |     range_min_bytes = 134217728,
                |     range_max_bytes = 536870912,
                |     gc.ttlseconds = 90000,
                |     num_replicas = 3,
                |     constraints = '[]',
                |     lease_preferences = '[]'
(1 row)