2018-12-13 14:47:20 -05:00
---
layout: doc_page
title: "Segments"
---
2018-11-13 12:38:37 -05:00
<!--
~ Licensed to the Apache Software Foundation (ASF) under one
~ or more contributor license agreements. See the NOTICE file
~ distributed with this work for additional information
~ regarding copyright ownership. The ASF licenses this file
~ to you under the Apache License, Version 2.0 (the
~ "License"); you may not use this file except in compliance
~ with the License. You may obtain a copy of the License at
~
~ http://www.apache.org/licenses/LICENSE-2.0
~
~ Unless required by applicable law or agreed to in writing,
~ software distributed under the License is distributed on an
~ "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY
~ KIND, either express or implied. See the License for the
~ specific language governing permissions and limitations
~ under the License.
-->
2018-12-12 23:42:12 -05:00
# Segments
2015-05-05 17:07:32 -04:00
2015-06-17 19:06:44 -04:00
Druid stores its index in *segment files* , which are partitioned by
time. In a basic setup, one segment file is created for each time
2017-02-03 11:31:14 -05:00
interval, where the time interval is configurable in the
2015-06-17 19:06:44 -04:00
`segmentGranularity` parameter of the `granularitySpec` , which is
2018-09-04 15:54:41 -04:00
documented [here ](../ingestion/ingestion-spec.html#granularityspec ). For druid to
2015-06-17 19:06:44 -04:00
operate well under heavy query load, it is important for the segment
file size to be within the recommended range of 300mb-700mb. If your
segment files are larger than this range, then consider either
2017-02-03 11:31:14 -05:00
changing the granularity of the time interval or partitioning your
2018-02-12 14:03:40 -05:00
data and tweaking the `targetPartitionSize` in your `partitionsSpec`
2015-06-17 19:06:44 -04:00
(a good starting point for this parameter is 5 million rows). See the
sharding section below and the 'Partitioning specification' section of
2018-09-04 15:54:41 -04:00
the [Batch ingestion ](../ingestion/hadoop.html#partitioning-specification ) documentation
2015-06-17 19:06:44 -04:00
for more information.
### A segment file's core data structures
Here we describe the internal structure of segment files, which is
essentially *columnar* : the data for each column is laid out in
separate data structures. By storing each column separately, Druid can
decrease query latency by scanning only those columns actually needed
for a query. There are three basic column types: the timestamp
column, dimension columns, and metric columns, as illustrated in the
image below:
2015-06-29 13:17:42 -04:00
![Druid column types ](../../img/druid-column-types.png "Druid Column Types" )
2015-06-17 19:06:44 -04:00
The timestamp and metric columns are simple: behind the scenes each of
these is an array of integer or floating point values compressed with
LZ4. Once a query knows which rows it needs to select, it simply
decompresses these, pulls out the relevant rows, and applies the
desired aggregation operator. As with all columns, if a query doesn’ t
require a column, then that column’ s data is just skipped over.
Dimensions columns are different because they support filter and
group-by operations, so each dimension requires the following
three data structures:
1. A dictionary that maps values (which are always treated as strings) to integer IDs,
2015-11-04 00:40:28 -05:00
2. A list of the column’ s values, encoded using the dictionary in 1, and
3. For each distinct value in the column, a bitmap that indicates which rows contain that value.
2015-11-03 11:18:19 -05:00
2015-06-17 19:06:44 -04:00
Why these three data structures? The dictionary simply maps string
values to integer ids so that the values in 2 and 3 can be
2015-11-03 11:18:19 -05:00
represented compactly. The bitmaps in 3 -- also known as *inverted
2015-06-17 19:06:44 -04:00
indexes* allow for quick filtering operations (specifically, bitmaps
are convenient for quickly applying AND and OR operators). Finally,
2015-11-04 00:40:28 -05:00
the list of values in 2 is needed for *group by* and *TopN*
2015-06-17 19:06:44 -04:00
queries. In other words, queries that solely aggregate metrics based
on filters do not need to touch the list of dimension values stored in
2015-11-03 11:18:19 -05:00
2.
2015-06-17 19:06:44 -04:00
To get a concrete sense of these data structures, consider the ‘ page’
column from the example data above. The three data structures that
represent this dimension are illustrated in the diagram below.
```
1: Dictionary that encodes column values
{
"Justin Bieber": 0,
"Ke$ha": 1
}
2: Column data
[0,
0,
1,
1]
3: Bitmaps - one for each unique value of the column
value="Justin Bieber": [1,1,0,0]
value="Ke$ha": [0,0,1,1]
```
Note that the bitmap is different from the first two data structures:
whereas the first two grow linearly in the size of the data (in the
worst case), the size of the bitmap section is the product of data
size * column cardinality. Compression will help us here though
2015-09-11 17:32:32 -04:00
because we know that for each row in 'column data', there will only be a
single bitmap that has non-zero entry. This means that high cardinality
columns will have extremely sparse, and therefore highly compressible,
bitmaps. Druid exploits this using compression algorithms that are
specially suited for bitmaps, such as roaring bitmap compression.
2015-06-17 19:06:44 -04:00
### Multi-value columns
If a data source makes use of multi-value columns, then the data
structures within the segment files look a bit different. Let's
imagine that in the example above, the second row were tagged with
both the 'Ke$ha' *and* 'Justin Bieber' topics. In this case, the three
data structures would now look as follows:
```
1: Dictionary that encodes column values
{
"Justin Bieber": 0,
"Ke$ha": 1
}
2: Column data
[0,
[0,1], < --Row value of multi-value column can have array of values
1,
1]
3: Bitmaps - one for each unique value
value="Justin Bieber": [1,1,0,0]
value="Ke$ha": [0,1,1,1]
^
|
|
Multi-value column has multiple non-zero entries
```
Note the changes to the second row in the column data and the Ke$ha
bitmap. If a row has more than one value for a column, its entry in
the 'column data' is an array of values. Additionally, a row with *n*
2015-09-11 17:32:32 -04:00
values in 'column data' will have *n* non-zero valued entries in
bitmaps.
2015-05-05 17:07:32 -04:00
Naming Convention
-----------------
Identifiers for segments are typically constructed using the segment datasource, interval start time (in ISO 8601 format), interval end time (in ISO 8601 format), and a version. If data is additionally sharded beyond a time range, the segment identifier will also contain a partition number.
An example segment identifier may be:
datasource_intervalStart_intervalEnd_version_partitionNum
Segment Components
------------------
2015-06-17 19:06:44 -04:00
Behind the scenes, a segment is comprised of several files, listed below.
2015-05-05 17:07:32 -04:00
* `version.bin`
4 bytes representing the current segment version as an integer. E.g., for v9 segments, the version is 0x0, 0x0, 0x0, 0x9
* `meta.smoosh`
A file with metadata (filenames and offsets) about the contents of the other `smoosh` files
* `XXXXX.smoosh`
There are some number of these files, which are concatenated binary data
The `smoosh` files represent multiple files "smooshed" together in order to minimize the number of file descriptors that must be open to house the data. They are files of up to 2GB in size (to match the limit of a memory mapped ByteBuffer in Java). The `smoosh` files house individual files for each of the columns in the data as well as an `index.drd` file with extra metadata about the segment.
There is also a special column called `__time` that refers to the time column of the segment. This will hopefully become less and less special as the code evolves, but for now it’ s as special as my Mommy always told me I am.
In the codebase, segments have an internal format version. The current segment format version is `v9` .
Format of a column
------------------
Each column is stored as two parts:
1. A Jackson-serialized ColumnDescriptor
2. The rest of the binary for the column
2016-03-22 17:16:34 -04:00
A ColumnDescriptor is essentially an object that allows us to use jackson’ s polymorphic deserialization to add new and interesting methods of serialization with minimal impact to the code. It consists of some metadata about the column (what type is it, is it multi-value, etc.) and then a list of serde logic that can deserialize the rest of the binary.
2015-05-05 17:07:32 -04:00
Sharding Data to Create Segments
--------------------------------
2016-01-06 00:27:52 -05:00
### Sharding
Multiple segments may exist for the same interval of time for the same datasource. These segments form a `block` for an interval.
Depending on the type of `shardSpec` that is used to shard the data, Druid queries may only complete if a `block` is complete. That is to say, if a block consists of 3 segments, such as:
`sampleData_2011-01-01T02:00:00:00Z_2011-01-01T03:00:00:00Z_v1_0`
`sampleData_2011-01-01T02:00:00:00Z_2011-01-01T03:00:00:00Z_v1_1`
`sampleData_2011-01-01T02:00:00:00Z_2011-01-01T03:00:00:00Z_v1_2`
All 3 segments must be loaded before a query for the interval `2011-01-01T02:00:00:00Z_2011-01-01T03:00:00:00Z` completes.
2015-05-05 17:07:32 -04:00
2016-01-06 00:27:52 -05:00
The exception to this rule is with using linear shard specs. Linear shard specs do not force 'completeness' and queries can complete even if shards are not loaded in the system.
For example, if your real-time ingestion creates 3 segments that were sharded with linear shard spec, and only two of the segments were loaded in the system, queries would return results only for those 2 segments.