The IMPORT INTO statement imports CSV, Avro, or delimited data into an existing table by appending new rows to the table.
Considerations
IMPORT INTOtakes the table offline before importing the data. The table will be online again once the job has completed successfully.IMPORT INTOworks with existing tables. To import data into a new table, see Import into a new table from a CSV file.IMPORT INTOcannot be used during a rolling upgrade.IMPORT INTOis a blocking statement. To run anIMPORT INTOjob asynchronously, use theDETACHEDoption.IMPORT INTOinvalidates all foreign keys on the target table. To validate the foreign key(s), use theALTER TABLE ... VALIDATE CONSTRAINTstatement.IMPORT INTOis an insert-only statement; it cannot be used to update existing rows. Imported rows cannot conflict with primary keys in the existing table, or any otherUNIQUEconstraint on the table.IMPORT INTOdoes not offerSELECTorWHEREclauses to specify subsets of rows. To add a subset of rows to a table, useINSERT.IMPORT INTOwill cause any changefeeds running on the targeted table to fail.IMPORT INTOsupports importing intoREGIONAL BY ROWtables.- At the beginning of an
IMPORT INTOjob, CockroachDB calculates and stores all user-defined types referenced by the table being imported into, including thecrdb_internal_regiontype descriptor used in multi-region clusters. At the end of the job, CockroachDB checks if any user-defined types have been modified. If so, theIMPORT INTOjob fails. Do not make modifications to your multi-region configuration or create or alter multi-region tables while runningIMPORT INTOjobs.
Optimize import operations in your applications by following our Import Performance Best Practices.
Before you begin
Before using IMPORT INTO, you should have:
An existing CockroachDB table to import into.
IMPORT INTOsupports computed columns and certainDEFAULTexpressions.Sufficient capacity in the CockroachDB store for the imported data.
The CSV or Avro data you want to import, preferably hosted on cloud storage. The import file location must be equally accessible to all nodes using the same import file location. This is necessary because the
IMPORT INTOstatement is issued once by the client, but is executed concurrently across all nodes of the cluster.
Supported DEFAULT expressions
IMPORT INTO supports the following DEFAULT expressions:
DEFAULTexpressions with user-defined types.Constant
DEFAULTexpressions, which are expressions that return the same value in different statements. Examples include:- Literals (booleans, strings, integers, decimals, dates).
- Functions where each argument is a constant expression and the functions themselves depend solely on their arguments (e.g., arithmetic operations, boolean logical operations, string operations).
random()gen_random_uuid()unique_rowid()nextval()Current
TIMESTAMPfunctions that record the transaction timestamp, which include:current_date()current_timestamp()localtimestamp()now()statement_timestamp()timeofday()transaction_timestamp()
Available storage
Each node in the cluster is assigned an equal part of the imported data, and so must have enough temp space to store it. In addition, data is persisted as a normal table, and so there must also be enough space to hold the final, replicated data. The node's first-listed/default store directory must have enough available storage to hold its portion of the data.
On cockroach start, if you set --max-disk-temp-storage, it must also be greater than the portion of the data a node will store in temp space.
Import file location
CockroachDB uses the URL provided to construct a secure API call to the service you specify. The URL structure depends on the type of file storage you are using. For more information, see the following:
You can create an external connection to represent an external storage or sink URI. This allows you to specify the external connection's name in statements rather than the provider-specific URI. For detail on using external connections, see the CREATE EXTERNAL CONNECTION page.
Required privileges
Table privileges
The user must have the INSERT and DROP privileges on the specified table. (DROP is required because the table is taken offline during the IMPORT INTO.)
Source privileges
You can grant a user the EXTERNALIOIMPLICITACCESS system-level privilege to interact with external resources that require implicit access.
Either the EXTERNALIOIMPLICITACCESS system-level privilege or the admin role is required for the following scenarios:
- Interacting with a cloud storage resource using
IMPLICITauthentication. - Using a custom endpoint on S3.
- Using the
cockroach nodelocal uploadcommand. - Using HTTP or HTTPS.
No special privilege is required for:
- Interacting with an Amazon S3 and Google Cloud Storage resource using
SPECIFIEDcredentials. Azure Storage is alwaysSPECIFIEDby default. - Using
userfilestorage.
We recommend using cloud storage. You also need to ensure that the permissions at your storage destination are configured for the operation. See Storage Permissions for a list of the necessary permissions that each bulk operation requires.
While Cockroach Labs actively tests Amazon S3, Google Cloud Storage, and Azure Storage, we do not test S3-compatible services (e.g., MinIO, Red Hat Ceph).
Synopsis
While importing into an existing table, the table is taken offline.
Parameters
| Parameter | Description |
|---|---|
table_name |
The name of the table you want to import into. |
column_name |
The table columns you want to import. Note: Currently, target columns are not enforced. |
file_location |
The URL of a CSV or Avro file containing the table data. This can be a comma-separated list of URLs. For an example, see Import into an existing table from multiple CSV files below. |
<option> [= <value>] |
Control your import's behavior with import options. |
Delimited data files
The DELIMITED DATA format can be used to import delimited data from any text file type, while ignoring characters that need to be escaped, like the following:
- The file's delimiter (
\tby default). - Double quotes (
"). - Newline (
\n). - Carriage return (
\r).
For examples showing how to use the DELIMITED DATA format, see the Examples section below.
Import options
You can control the IMPORT process's behavior using any of the following key-value pairs as a <option> [= <value>].
| Key | Context |
Value |
|---|---|---|
allow_quoted_null |
CSV DATA |
If included with nullif, both quoted and unquoted CSV input values can match the nullif setting.Example: To interpret both empty columns and "" as NULL: IMPORT INTO ... CSV DATA ('file.csv') WITH nullif = '', allow_quoted_null; |
comment |
CSV DATA |
The unicode character that identifies rows to skip. Example: IMPORT INTO ... CSV DATA ('file.csv') WITH comment = '#'; |
data_as_binary_records |
AVRO DATA |
Use when importing a binary file containing Avro records. The schema is not included in the file, so you need to specify the schema with either the schema or schema_uri option. Example: IMPORT INTO ... AVRO DATA ('file.bjson') WITH data_as_binary_records, schema_uri='..'; |
data_as_json_records |
AVRO DATA |
Use when importing a JSON file containing Avro records. The schema is not included in the file, so you need to specify the schema with either the schema or schema_uri option. Example: IMPORT INTO ... AVRO DATA ('file.bjson') WITH data_as_json_records, schema='{ "type": "record",..}'; |
decompress |
General | The decompression codec to be used: gzip, bzip, auto, or none. Default: 'auto', which guesses based on file extension (.gz, .bz, .bz2). none disables decompression. Example: IMPORT INTO ... WITH decompress = 'bzip'; |
delimiter |
CSV DATA |
The unicode character that delimits columns in your rows. Default: ,. Example: To use tab-delimited values: IMPORT INTO ... CSV DATA ('file.csv') WITH delimiter = e'\t'; |
DETACHED |
N/A | When an import runs in DETACHED mode, it will execute asynchronously and the job ID will be returned immediately without waiting for the job to finish. Note that with DETACHED specified, further job information and the job completion status will not be returned. To check on the job status, use the SHOW JOBS statement. To run an import within a transaction, use the DETACHED option. |
fields_enclosed_by |
DELIMITED DATA |
The unicode character that encloses fields. Default: " Example: IMPORT INTO ... WITH fields_enclosed_by='"'; |
fields_escaped_by |
DELIMITED DATA |
The unicode character, when preceding one of the above DELIMITED DATA options, to be interpreted literally. Example: IMPORT INTO ... WITH fields_escaped_by='\'; |
fields_terminated_by |
DELIMITED DATA |
The unicode character used to separate fields in each input line. Default: \t Example: IMPORT INTO ... WITH fields_terminated_by='.'; |
nullif |
CSV DATA, DELIMITED DATA |
Use nullif to specify a column value that should be interpreted as NULL.Example: To interpret an empty column as NULL: IMPORT INTO ... CSV DATA ('file.csv') WITH nullif = '';Example: To interpret value as NULL: IMPORT INTO ... CSV DATA ('file.csv') WITH nullif = 'value';Note: When importing CSV files, nullif matches column values that are unquoted. Any CSV input value that is enclosed in double quotes, including an empty value, is interpreted as a string. To match both quoted and unquoted CSV input values to the nullif setting, include the allow_quoted_null option.When importing from CSV without nullif, empty values are interpreted as NULL by default, unless enclosed in double quotes. When importing DELIMITED DATA files without nullif, empty values are interpreted as empty strings by default. |
records_terminated_by |
AVRO DATA |
The unicode character to indicate new lines in the input binary or JSON file. This is not needed for Avro OCF. Default: \n Example: To use tab-terminated records: IMPORT INTO ... AVRO DATA ('file.csv') WITH records_terminated_by = e'\t'; |
row_limit |
General | The number of rows to import. Useful for doing a test run of an import and finding errors quickly. This option will import the first n rows from each table in the dump file. |
rows_terminated_by |
DELIMITED DATA |
The unicode character to indicate new lines in the input file. Default: \n Example: IMPORT INTO ... WITH rows_terminated_by='\m'; |
schema |
AVRO DATA |
The schema of the Avro records included in the binary or JSON file. This is not needed for Avro OCF. See data_as_json_records example above. |
schema_uri |
AVRO DATA |
The URI of the file containing the schema of the Avro records include in the binary or JSON file. This is not needed for Avro OCF. See data_as_binary_records example above. |
skip |
CSV DATA, DELIMITED DATA |
The number of rows to be skipped while importing a file. Default: '0'. Example: To import CSV files with column headers: IMPORT INTO ... CSV DATA ('file.csv') WITH skip = '1'; |
strict_validation |
AVRO DATA |
Rejects Avro records that do not have a one-to-one mapping between Avro fields to the target CockroachDB schema. By default, CockroachDB ignores unknown Avro fields and sets missing SQL fields to NULL. CockroachDB will also attempt to convert the Avro field to the CockroachDB [data type][datatypes]; otherwise, it will report an error. Example: IMPORT INTO ... AVRO DATA ('file.avro') WITH strict_validation; |
For examples showing how to use these options, see the Examples section.
For instructions and working examples showing how to migrate data from other databases and formats, see the Migration Overview.
View and control import jobs
After CockroachDB successfully initiates an import into an existing table, it registers the import as a job, which you can view with SHOW JOBS.
After the import has been initiated, you can control it with PAUSE JOB, RESUME JOB, and CANCEL JOB.
If initiated correctly, the statement returns when the import is finished or if it encounters an error. In some cases, the import can continue after an error has been returned (the error message will tell you that the import has resumed in background).
Pausing and then resuming an IMPORT INTO job will cause it to restart from the beginning.
Examples
The following examples make use of:
- Amazon S3 connection strings. For guidance on connecting to other storage options or using other authentication parameters instead, read Use Cloud Storage.
- The default
AUTH=specifiedparameter. For guidance on usingAUTH=implicitauthentication with Amazon S3 buckets instead, read Cloud Storage Authentication.
Also, note the following features for connecting and authenticating to cloud storage:
- External connections, which allow you to represent an external storage or sink URI. You can then specify the external connection's name in statements rather than the provider-specific URI. For detail on using external connections, see the
CREATE EXTERNAL CONNECTIONpage. - Assume role authentication, which allows you to limit the control specific users have over your storage buckets. See Assume role authentication for more information.
Import into a new table from a CSV file
To import into a new table, use CREATE TABLE followed by IMPORT INTO.
Create the new table with the necessary columns and data types:
CREATE TABLE users ( id UUID PRIMARY KEY, city STRING, name STRING, address STRING, credit_card STRING );Use
IMPORT INTOto import the data into the new table:IMPORT INTO users (id, city, name, address, credit_card) CSV DATA ( 's3://{BUCKET NAME}/{customers.csv}?AWS_ACCESS_KEY_ID={KEY ID}&AWS_SECRET_ACCESS_KEY={SECRET ACCESS KEY}' );
Import into an existing table from a CSV file
> IMPORT INTO customers (id, name)
CSV DATA (
's3://{BUCKET NAME}/{customers.csv}?AWS_ACCESS_KEY_ID={KEY ID}&AWS_SECRET_ACCESS_KEY={SECRET ACCESS KEY}'
);
The column order in your IMPORT statement must match the column order in the CSV being imported, regardless of the order in the existing table's schema.
Import into an existing table from multiple CSV files
> IMPORT INTO customers (id, name)
CSV DATA (
's3://{BUCKET NAME}/{customers.csv}?AWS_ACCESS_KEY_ID={KEY ID}&AWS_SECRET_ACCESS_KEY={SECRET ACCESS KEY}',
's3://{BUCKET NAME}/{customers2.csv}?AWS_ACCESS_KEY_ID={KEY ID}&AWS_SECRET_ACCESS_KEY={SECRET ACCESS KEY}',
's3://{BUCKET NAME}/{customers3.csv}?AWS_ACCESS_KEY_ID={KEY ID}&AWS_SECRET_ACCESS_KEY={SECRET ACCESS KEY}',
's3://{BUCKET NAME}/{customers4.csv}?AWS_ACCESS_KEY_ID={KEY ID}&AWS_SECRET_ACCESS_KEY={SECRET ACCESS KEY}',
);
Import into an existing table using a wildcard
You can specify file patterns to match instead of explicitly listing every file. Paths are matched using the * wildcard character to include matching files directly under the specified path. Use a wildcard to include:
- All files in a given directory (e.g.,
s3://bucket-name/path/to/data/*). - All files in a given directory that end with a given string (e.g.,
s3://bucket-name/files/*.csv). - All files in a given directory that start with a given string (e.g.,
s3://bucket-name/files/data*). - All files in a given directory that start and end with a given string (e.g.,
s3://bucket-name/files/data*.csv).
These only match files directly under the specified path and do not descend into additional directories recursively.
IMPORT INTO users (id, city, name, address, credit_card)
CSV DATA (
's3://{BUCKET NAME}/*.csv?AWS_ACCESS_KEY_ID={KEY ID}&AWS_SECRET_ACCESS_KEY={SECRET ACCESS KEY}'
);
Import into an existing table from an Avro file
Avro OCF data, JSON records, or binary records can be imported. The following are examples of importing Avro OCF data.
To specify the table schema in-line:
> IMPORT INTO customers
AVRO DATA (
's3://{BUCKET NAME}/{customers.avro}?AWS_ACCESS_KEY_ID={KEY ID}&AWS_SECRET_ACCESS_KEY={SECRET ACCESS KEY}'
);
For more information about importing data from Avro, including examples, see Migrate from Avro.
Import into an existing table from a delimited data file
> IMPORT INTO customers
DELIMITED DATA (
's3://{BUCKET NAME}/{customers.csv}?AWS_ACCESS_KEY_ID={KEY ID}&AWS_SECRET_ACCESS_KEY={SECRET ACCESS KEY}'
)
WITH
fields_terminated_by='|',
fields_enclosed_by='"',
fields_escaped_by='\';
Known limitations
IMPORT INTO has the following limitations:
- While importing into an existing table, the table is taken offline.
- After importing into an existing table, constraints will be un-validated and need to be re-validated.
- Imported rows must not conflict with existing rows in the table or any unique secondary indexes.
IMPORT INTOworks for only a single existing table.IMPORT INTOcan sometimes fail with a "context canceled" error, or can restart itself many times without ever finishing. If this is happening, it is likely due to a high amount of disk contention. This can be mitigated by setting thekv.bulk_io_write.max_ratecluster setting to a value below your max disk write speed. For example, to set it to 10MB/s, execute:SET CLUSTER SETTING kv.bulk_io_write.max_rate = '10MB';