Import / ETL

GenomicsDB supports importing genomics data in several common formats, and with a variety of methods.

GenomicsDB can ingest data in VCF, BCF, gVCF and CSV formats. When importing VCF/BCFs or gVCFs, you may need to run a few preprocessing steps before importing.

The primary and suggested method of importing is to use the native vcf2genomicsdb importer tool (see CLI Tools). There is also a Java API for importing (see here).

VCF/gVCF

The import program can handle block compressed and indexed VCFs, BCFs and gVCFs. For brevity, we will only use the term VCF.

Organizing your data

  • All your VCFs must be block compressed and indexed. Bcftools is one good option for compressing and indexing.

  • The VCF format allows you to have multiple lines with the same position (identical chromosome+pos). The import program can NOT handle such VCFs. Make sure you use tools like bcftools to collapse multiple lines with the same position into a single line. Example command:

    bcftools norm -m +any [-O <output_format> -o <output>] <input_file>
    
  • In a multi-node environment, you must decide:
    • How to partition your data in GenomicsDB <#Multi-node-setup>

    • How your VCF files are accessed by the import program:
      • On a shared filesystem (NFS, Lustre etc) accessible from all nodes.

      • If you are partitioning data by rows, your files can be scattered across local filesystems on multiple machines; each filesystem accessible only by the node on which it is mounted. Currently, such a setup isn’t supported for column partitioning.

CSV

Importing CSVs into GenomicsDB

Preliminaries

  • You need libcsv while compiling.

  • License terms: We use libcsv to parse CSV files. libcsv is licensed under the GNU Library or Lesser General Public License version 2.0 (LGPLv2). So, if you are re-distributing binaries or object files, they may be subject to LGPLv2 terms. Please ensure that any binaries/object files you distribute are compliant with LGPLv2. You can disable libcsv usage by not setting the USE_LIBCSV and LIBCSV_DIR flags during compilation. However, your binaries/executables will not be able to import CSV files into GenomicsDB.

CSV format

Given a variant call at a specific position in the genome for a particular CallSet/sample, the CSV file format supported by GenomicsDB contains one line describing the call. Essentially, each line in the CSV describes one cell in the GenomicsDB array.

The exact format of the CSV is shown below:

<row>,<begin_column>,<end_column>,<REF>,<concatenated_ALT>,<QUAL>,<FILTER_field_specification>[,<other_fields_specification>]

Fixed fields:

  • row (mandatory, type: int64): Row number in GenomicsDB for this sample/CallSet

  • begin_column (mandatory, type: int64): Column in GenomicsDB at which the variant call begins

  • end_column (mandatory, type: int64): Column in GenomicsDB at which the variant call ends (inclusive).

  • REF (mandatory, type:string): Reference allele at the given position.

  • concatenated_ALT (mandatory, type: string): Concatenated alternate alleles separated by the character ‘|’. For example, if a given call has two alternate alleles TAG and TG, then the concatenated string would be TAG|TG.

  • QUAL (optional, type: float): Represents the QUAL field in a VCF. It can be left empty implying that the value was missing for this call.

  • FILTER_field_specification (mandatory, type: variable length list of integers): Each element of the FILTER field list is an integer representing the FILTERs that this call failed (similar to how a BCF represents FILTERs). The first element of this field is an integer displaying the number of elements in the list. A value of 0 indicates that the list is empty.

Additional fields can be optionally specified * <other_fields_specification>: The format depends on the type of field:

  • String type fields (fixed or variable length strings): The field should contain the string - an empty token indicates that the field is missing.

  • Fixed length field (int or float): The field should contain exactly N elements where N is the length of the field (fixed constant). One or more tokens may be left empty to indicate that those elements are missing.

  • Variable length field (int or float): The first element of this field should be an integer denoting the number of elements in the field for this call. It should then be followed by the elements of this field. An empty or missing field can be specified by setting the first element (field length) to 0 - no other elements should follow an empty field.

Example

The following line contains 2 fields in addition to the fixed fields:

  • SB: Fixed length field of 4 integers

  • PL: Variable length field of integers

    2,1857210,1857210,G,A|T,894.77,0,,,,,6,923,0,599,996,701,1697
    

The line specifies the variant call for row id 2, beginning at column 1857210 and ending at 1857210. The REF allele is ‘G’ and the call has 2 alternate alleles ‘A’ and ‘T’ (SNVs). The QUAL value is 894.77 and there are no FILTERs specified (hence FILTER field length = 0). The SB field is missing - denoted by the 4 empty tokens. The PL field consists of 6 integers - the length appears first (since PL is a variable length field) followed by the elements [923,0,599,996,701,1697].

Special fields

  • GT is represented in the CSV as a variable length list of integers - each element of the list refers to the allele index (0 for reference allele, 1 for the first alternate allele and so on). The length of the list represents the ploidy of the sample/CallSet and must be specified in the CSV line (since GT is treated as a variable length list).

Organizing your data

  • All CSV files imported into a GenomicsDB array must respect the number and order of fields as defined in the vid_mapping_file.

  • The import program cannot handle CSV files where multiple lines have the same value of row and begin_column - this restriction is similar to that imposed on loading VCFs. Consolidate these multiple lines into a single line to continue.

Multi-node setup

GenomicsDB can be setup to store variant data across multiple partitions of an array. All the data belonging to one partition of an array lives on a single filesystem. Thus, by creating multiple partitions, users can store data possibly across multiple hosts/nodes in a cluster. Array partitioning is useful when the data to be stored and queried is very large and cannot fit within a single machine/node. Or the user might wish to store array partitions in different nodes so that downstream queries and analysis can be run in a distributed manner for scalability and/or performance.

The user must decide how to partition data across multiple nodes in a cluster:

  • How many nodes should be used to store the data?

  • How many partitions should reside on each node? A single node can hold multiple partitions (assuming the node has enough disk space).

  • What mode should be used for partitioning the data? Two modes of partitioning are supported by various import/query tools.

    • Row partitioning: In this mode, for a given sample/CallSet (row), all the variant data resides in a single partition. Data belonging to different samples/CallSets may be scattered across different partitions.

    • Column partitioning: In this mode, for a given genomic position (column), all the variant data across all samples/CallSets resides in a single partition. Data is partitioned by genomic positions.

Which partitioning scheme is better to use is dependent on the queries/analysis performed by downstream tools. Here are some example queries for which the ‘best’ partitioning schemes are suggested.

  • Query: fetch attribute X from all samples/CallSets for position Y (or small interval [Y1-Y2])

    • Row-based partitioning
      • For single position queries (or small intervals), partitioning the data by rows would likely provide higher performance. By accessing data across multiple partitions that may be located in multiple nodes in parallel, the system will be able to utilize higher aggregate disk and memory bandwidth. In a column based partitioning, only a single partition would service the request.

      • Simple data import step if the original data is organized as a file per sample/CallSet (for example VCFs). Just import data from the required subset of files to the correct partition.

      • Con(s). A final aggregator may be needed since the data for a given position is scattered across machines. Some of the query tools we provide use MPI to collect the final output into a single node.

  • Query: run analysis tool T on all variants (grouped by column position) found in a large column interval [Z1-Z2] (or scan across the whole array)

    • Column-based partitioning
      • The user is running a query/analysis for every position in the queried interval. Hence, for each position, the system must fetch data from all samples/CallSets and run T. Partitioning by column reduces/eliminates any communication between partitions. For a sufficiently large query interval, the aggregate disk and memory bandwidth across multiple nodes can still be utilized.

      • No/minimal data aggregation step as all the data for a given column is located within a single partition.

      • Con(s). Importing data into GenomicsDB may become complex, especially if the initial data is organized as a file per sample/CallSet.