Skip to content

Latest commit

 

History

History
249 lines (191 loc) · 10.6 KB

20170314_sql_split_syntax.md

File metadata and controls

249 lines (191 loc) · 10.6 KB
  • Feature Name: SQL Syntax for splitting tables
  • Status: completed
  • Start Date: 2017-03-14
  • Authors: Radu Berinde
  • RFC PR: #14146
  • Cockroach Issue: #13665

Summary

This RFC proposes new syntax for introducing table splits and relocating ranges via SQL.

Specifically, we want to be able to:

  • introduce range splits at specific points in a table or index;
  • trigger a "scattering" of ranges in a table or index (or in an area of a table/index);
  • relocate specific ranges to a specified set of replicas.

Motivation

The main motivation is to allow setting up tests, benchmarks, and reproducible testbeds, especially for DistSQL. One set of some sample tests that we want to be able to set up directly from logic tests are in TestDistSQLPlanner. These tests need to split tables in a specific way, and they need to reprogram the replicas and range leaders of each split.

The secondary motivation is for restore (as in backup/restore), which needs to introduce a specific set of key splits. Currently this is done by custom code which works as follows: the keys are sorted; a split is introduced on the middle key, then the splits to the left and respectively to the right are processed recursively (in parallel). We want the new syntax to implement this algorithm so that backup/restore can use it (assuming they can switch to using split point specified via column values as opposed to keys).

We currently have the ALTER TABLE/INDEX SPLIT AT statement which allows introducing a table or index split at a specific tuple of column values. The main drawback is that it cannot be used programmatically: each split must be in its own statement; we cannot generate (e.g. via a SELECT) a set of values where to split. We want to write tests like the one linked above without hardcoding each split in the test file; and we want to be able to easily change the test table sizes. In addition, the SPLIT AT statements don't support control of replication.

Detailed design

1. ALTER TABLE/INDEX SPLIT AT enhancement

The existing SPLIT AT syntax is changed: instead of taking a tuple of column values for a single split, SPLIT AT takes an arbitrary select clause where each row contains the primary key values; a split is issued for each row.

ALTER TABLE <table> SPLIT AT <select_statement>
ALTER INDEX <table>@index SPLIT AT <select_statement>

Existing uses of ALTER TABLE SPLIT AT (pk1,pk2,...) are changed to ALTER TABLE SPLIT AT VALUES (pk1,pk2,...) (and similarly for ALTER INDEX); examples:

Old New
ALTER TABLE d.t SPLIT AT ('c', 3) ALTER TABLE d.t SPLIT AT VALUES ('c', 3)
ALTER INDEX d.t@s_idx SPLIT AT ('f') ALTER INDEX d.t@s_idx SPLIT AT VALUES ('f')

The new syntax allows introducing multiple split points, as well as calculated or table-driven split points; for example:

ALTER TABLE t1 SPLIT AT VALUES ('c', 3), ('d', 4), ('e', 5)
ALTER INDEX t2@idx SPLIT AT SELECT i*10 FROM GENERATE_SERIES(1, 5) AS g(i)
ALTER TABLE t3 SPLIT AT SELECT pk1, pk2 FROM d.t WHERE split_here

The SPLIT AT statement is also extended to support providing only a prefix of the primary key or index values. Example:

CREATE TABLE t (k1, k2, k3, v INT, PRIMARY KEY (k1, k2, k3))

-- Introduce a split at at /t/primary/1
ALTER TABLE t SPLIT AT VALUES (1)

-- Introduce a split at at /t/primary/1/2
ALTER TABLE t SPLIT AT VALUES (1,2)

-- Introduce a split at at /t/primary/1/2/3
ALTER TABLE t SPLIT AT VALUES (1,2,3)

Note that the statement returns only after the splits have been preformed.

Implementation note: the SPLIT AT implementation can issue the splits in any order; for example, if there are many splits, it is advantageous to sort the split points, split at the middle point, then recursively process the left and right sides (in parallel).

Interleaved tables: the command works as expected; the split will inherently cause a corresponding split in the parent or child tables/indexes.

Return values

ALTER TABLE/INDEX SPLIT AT currently returns a row with two columns: the key in raw and pretty-printed form. The same schema is kept; a row for each split is output.

If a split already exists, SPLIT AT currently returns an error.

TBD: what is the correct behavior with multiple split points? Ignore this error?

2. ALTER TABLE/INDEX SCATTER

A new pair of statements similar to SPLIT AT are introduced. Each has two forms. The first form causes all the ranges for that table or index to be "scattered": for each range, a new random set of stores are chosen for replicas (in accordance with the zone config).

ALTER TABLE <table> SCATTER
ALTER INDEX <table>@index SCATTER

The second form allows only a specific area of a table or index to be scattered. The area is specified using a pair of tuples (primary key or index column values) for the start and end of the area. The values are interpreted similar to SPLIT AT, and similarly a prefix of the primary key or index columns can be specified:

ALTER TABLE <table> SCATTER
ALTER TABLE <table> SCATTER FROM (startPK1, startPK2, ...) TO (endPK1, endPK2, ...)
ALTER INDEX <table>@index SCATTER
ALTER INDEX <table>@index SCATTER FROM (startCol1, startCol2, ...) TO (endCol1, endCol2, ...)

Note that no new split points are introduced by SCATTER.

Examples:

ALTER TABLE t SCATTER
ALTER TABLE t SCATTER FROM (1,1) TO (1,2)
ALTER INDEX t@idx SCATTER FROM (1) TO (2)

The statement returns only after the relocations are complete.

Interleaved tables: the command works as expected (the ranges may contain rows for parent or child tables/indexes).

3. ALTER TABLE/INDEX EXPERIMENTAL_RELOCATE

The EXPERIMENTAL_RELOCATE statements can be used to relocate specific ranges to specific stores. This is very low-level functionality and is intended to be used sparingly, mainly for setting up tests which benefit from a predetermined data distribution. The rebalancing queues should be stopped in order to make the relocations "stick".

ALTER TABLE <table> EXPERIMENTAL_RELOCATE <select_statement>
ALTER INDEX <table>@<index> EXPERIMENTAL_RELOCATE <select_statement>

EXPERIMENTAL_RELOCATE takes a select statement with the following result schema:

  • the first column is the relocation information: an array of integers, where each integer is a store ID. This indicates the set of replicas for the range; the first replica in the array will be the new lease owner.

  • the rest of the columns indicate a point in the table as a list of primary key or index columns (similar to SPLIT AT). The range that contains this point is relocated according to the first column. Note that, just like SPLIT AT, a row with this particular set of values doesn't have to exist in the table.

Examples:

CREATE TABLE t (k1, k2, k3, v INT, PRIMARY KEY (k1, k2, k3))

-- Move the range containing /t/primary/1/2/3 to store 1:
ALTER TABLE t EXPERIMENTAL_RELOCATE VALUES (ARRAY[1], 1, 2, 3)

-- Move the range containing /t/primary/1/2 to stores 5,6,7 (with 5 as lease owner):
ALTER TABLE t EXPERIMENTAL_RELOCATE VALUES (ARRAY[5,6,7], 1, 2)

-- Move even k1s to store 1, odd k1s to store 2:
ALTER TABLE t EXPERIMENTAL_RELOCATE SELECT ARRAY[1+i%2], i FROM GENERATE_SERIES(1, 10) AS g(i)

The statement returns only after the relocations are complete.

Interleaved tables: the command works as expected (the ranges may contain rows for parent or child tables/indexes).

4. crdb_internal.ranges and ranges_cached system table

To facilitate testing the implementation of the new commands (as well as allow a user to verify what the commands did), we introduce a crdb_internal.ranges system table that can be used to look at all the ranges on the system, or the ranges from a table.

The schema of the table is as follows:

Column Type Description
start_key BYTES Range start key (raw)
start_pretty STRING Range start key (pretty-printed)
end_key BYTES Range end key (raw)
end_pretty STRING Range end key (pretty-printed)
database STRING Database name (if range is part of a table)
table STRING Table name (if range is part of a table); for interleaved tables this is always the root table.
index STRING Index name (if range is part of a non-primary index);
replicas ARRAY(INT) Replica store IDs
lease_holder INT Lease holder store ID

The last two columns could be hidden (so they are only available if SELECTed for specifically).

Implementation notes:

  • the system table infrastructure will be improved so the row producing function has access to filters; specifying a table or index filter that should be optimized to only look at the ranges for that table.
  • the row producing function should also have access to needed columns; that way the more expensive lease holder determination can be omitted if the column is not needed.

A second table, crdb_internal.ranges_cached has the same schema, but it returns data from the range and lease holder caches. Specifically: the ranges along with replicas information are populated from the range cache; for each range, if that range ID has an entry in the lease holder cache, lease_holder is set according to that entry; otherwise it is NULL.

Alternatives

An alternative considered was to create statements that operate on keys (key prefixes) and provide functions that generate key prefixes from column values. This was deemed as too low-level for the SQL interface.

Another alternative considered was to also use a function for the splitting functionality, e.g. SELECT split_at(..). The problem is that it forces the splits to happen sequentially; we cannot implement the algorithm mentioned above that parallelizes the splits. One way around this would be to introduce split_at as an aggregation function (akin to sum).

Alternatives considered for crdb_internal.ranges:

  • a SHOW RANGES FOR TABLE/INDEX statement; the system table was deemed more useful.
  • having multiple system tables (e.g. a separate one for lease holders) and using joins as necessary; this requires too many changes to make sure we only generate the parts of the table that are needed.

Unresolved questions