The Perl Toolchain Summit needs more sponsors. If your company depends on Perl, please support this very important event.
h1. Redis RDB File Format

Redis *.rdb file is a binary representation of the in-memory store. This binary file is sufficient to completely restore Redis' state.

The rdb file format is optimized for fast read and writes. Where possible LZF compression is used to reduce the file size. In general, objects are prefixed with their lengths, so before reading the object you know exactly how much memory to allocate.

Optimizing for fast read/writes means the on-disk format should be as close as possible to the in-memory representation. This is the approach taken by the rdb file. As a consequence, you cannot parse the rdb file without some understanding of Redis' in-memory representation of data structures

h2. High Level Algorithm to parse RDB

At a high level, the RDB file has the following structure
<pre><code>
----------------------------# RDB is a binary format. There are no new lines or spaces in the file.
52 45 44 49 53              # Magic String "REDIS"
30 30 30 33                 # RDB Version Number in big endian. In this case, version = 0003 = 3
----------------------------
FE 00                       # FE = code that indicates database selector. db number = 00
----------------------------# Key-Value pair starts
FD $unsigned int            # FD indicates "expiry time in seconds". After that, expiry time is read as a 4 byte unsigned int
$value-type                 # 1 byte flag indicating the type of value - set, map, sorted set etc.
$string-encoded-key         # The key, encoded as a redis string
$encoded-value              # The value. Encoding depends on $value-type
----------------------------
FC $unsigned long           # FC indicates "expiry time in ms". After that, expiry time is read as a 8 byte unsigned long
$value-type                 # 1 byte flag indicating the type of value - set, map, sorted set etc.
$string-encoded-key         # The key, encoded as a redis string
$encoded-value              # The value. Encoding depends on $value-type
----------------------------
$value-type                 # This key value pair doesn't have an expiry. $value_type guaranteed != to FD, FC, FE and FF
$string-encoded-key
$encoded-value
----------------------------
FE $length-encoding         # Previos db ends, next db starts. Database number read using length encoding.
----------------------------
...                         # Key value pairs for this database, additonal database
                            
FF                          ## End of RDB file indicator
8 byte checksum             ## CRC 32 checksum of the entire file.
</code></pre>

h3. Magic Number

The file starts off with the magic string "REDIS". This is a quick sanity check to know we are dealing with a redis rdb file.

@52 45 44 49 53  # "REDIS"@

h3. RDB Version Number

The next 4 bytes store the version number of the rdb format. The 4 bytes are interpreted as ascii characters and then converted to an integer using string to integer conversion.

@00 00 00 03 # Version = 3@

h3. Database Selector

A Redis instance can have multiple databases.

A single byte @0xFE@ flags the start of the database selector. After this byte, a variable length field indicates the database number. See the section "Length Encoding" to understand how to read this database number.

h3. Key Value Pairs

After the database selector, the file contains a sequence of key value pairs.

Each key value pair has 4 parts -
# Key Expiry Timestamp. This is optional
# One byte flag indicating the value type
# The key, encoded as a Redis String. See "Redis String Encoding"
# The value, encoded according to the value type. See "Redis Value Encoding"

h4. Key Expiry Timestamp

This section starts with a one byte flag. A value of @FD@ indicates expiry is specified in seconds. A value of @FC@ indicates expiry in specified in millseconds.

If time is specified in ms, the next 8 bytes represent the unix time  This number is a unix time stamp in either seconds or milliseconds precision, and represents the expiry of this key.

See the section "Redis Length Encoding" on how this number is encoded.

During the import process, keys that have expired must be discarded.

h4. Value Type

A one byte flag indicates encoding used to save the Value.
# 0 =  "String Encoding"
# 1 =  "List Encoding"
# 2 =  "Set Encoding"
# 3 =  "Sorted Set Encoding"
# 4 =  "Hash Encoding"
# 9 =  "Zipmap Encoding"
# 10 = "Ziplist Encoding"
# 11 = "Intset Encoding"
# 12 = "Sorted Set in Ziplist Encoding"
# 13 = "Hashmap in Ziplist Encoding" (Introduced in rdb version 4)

h4. Key 

The key is simply encoded as a Redis string. See the section "String Encoding" to learn how the key is encoded.

h4. Value

The encoding of the value depends on the value type flag.

* When value type = 0, the value is a simple string.
* When value type is one of 9, 10, 11 or 12, the value is wrapped in a string. After reading the string, it must be parsed further.
* When value type is one of 1, 2, 3 or 4, the value is a sequence of strings. This sequence of strings is used to construct a list, set, sorted set or hashmap.

h2. Length Encoding

Length encoding is used to store the length of the next object in the stream. Length encoding is a variable byte encoding designed to use as few bytes as possible.

This is how length encoding works :
# One byte is read from the stream, and the two most significant bits are read.
# If starting bits are @00@, then the next 6 bits represent the length
# If starting bits are @01@, then an additional byte is read from the stream. The combined 14 bits represent the length
# If starting bits are @10@, then the remaining 6 bits are discared. Additional 4 bytes are read from the stream, and those 4 bytes represent the length
# If starting bits are @11@, then the next object is encoded in a special format. The remaining 6 bits indicate the format. This encoding is generally used to store numbers as strings, or to store encoded strings. See String Encoding

As a result of this encoding - 
# Numbers upto and including 63 can be stored in 1 byte
# Numbers upto and including 16383 can be stored in 2 bytes
# Numbers upto 2^32 -1 can be stored in 4 bytes

h2. String Encoding

Redis Strings are binary safe - which means you can store anything in them. They do not have any special end-of-string token. It is best to think of Redis Strings as a byte array.

There are three types of Strings in Redis - 
# Length prefixed strings
# An 8, 16 or 32 bit integer
# A LZF compressed string

h4. Length Prefixed String

Length prefixed strings are quite simple. The length of the string in bytes is first encoded using "Length Encoding". After this, the raw bytes of the string are stored.

h4. Integers as String

First read the section "Length Encoding", specifically the part when the first two bits are @11@. In this case, the remaining 6 bits are read.
If the value of those 6 bits is - 
# 0 indicates that an 8 bit integer follows
# 1 indicates that a 16 bit integer follows
# 2 indicates that a 32 bit integer follows

h4. Compressed Strings

First read the section "Length Encoding", specifically the part when the first two bits are @11@. In this case, the remaining 6 bits are read.
If the value of those 6 bits is 4, it indicates that a compressed string follows.

The compressed string is read as follows - 
# The compressed length @clen@ is read from the stream using "Length Encoding"
# The uncompressed length is read from the stream using "Length Encoding"
# The next @clen@ bytes are read from the stream
# Finally, these bytes are decompressed using LZF algorithm

h2. List Encoding

A redis list is represented as a sequence of strings.

# First, the size of the list @size@ is read from the stream using "Length Encoding"
# Next, @size@ strings are read from the stream using "String Encoding"
# The list is then re-constructed using these Strings

h2. Set Encoding

Sets are encoded exactly like lists. 

h2. Sorted Set Encoding

# First, the size of the sorted set @size@ is read from the stream using "Length Encoding"
# TODO

h2. Hash Encoding

# First, the size of the hash @size@ is read from the stream using "Length Encoding"
# Next, @ 2 * size @ strings are read from the stream using "String Encoding"
# Alternate strings are key and values
# For example, @ 2 us washington india delhi @ represents the map @{"us" => "washington", "india" => "delhi"}@

h2. Zipmap Encoding

A Zipmap is a hashmap that has been serialized to a string. In essence, the key value pairs are stored sequentially. Looking up a key in this structure is O(N). This structure is used instead of a dictionary when the number of key value pairs are small.

To parse a zipmap, first a string is read from the stream using "String Encoding". This string is the envelope of the zipmap. The contents of this string represent the zipmap.

The structure of a zipmap within this string is as follows - 
@<zmlen><len>"foo"<len><free>"bar"<len>"hello"<len><free>"world"<zmend>@

# *zmlen* : Is a 1 byte length that holds the size of the zip map. If it is greater than or equal to 254, value is not used. You will have to iterate the entire zip map to find the length.
# *len* : Is the length of the following string, which can be either a key or a value. This length is stored in either 1 byte or 5 bytes (yes, it differs from "Length Encoding" described above). If the first byte is between 0 and 252, that is the length of the zipmap. If the first byte is 253, then the next 4 bytes read as an unsigned integer represent the length of the zipmap. 254 and 255 are invalid values for this field.
# *free* : This is always 1 byte, and indicates the number of free bytes _after_ the value. For example, if the value of a key is "America" and its get updated to "USA", 4 free bytes will be available.
# *zmend* : Always 255. Indicates the end of the zipmap.

*Worked Example*
@18 02 06 4d 4b 44 31 47 36 01 00 32 05 59 4e 4e 58 4b 04 00 46 37 54 49 ff ..@

# Start by decoding this using "String Encoding". You will notice that @18@ is the length of the string. Accordingly, we will read the next 24 bytes i.e. upto @ff@
# Now, we are parsing the string starting at @02 06... @ using the "Zipmap Encoding"
# @02@ is the number of entries in the hashmap.
# @06@ is the length of the next string. Since this is less than 254, we don't have to read any additional bytes
# We read the next 6 bytes i.e. @4d 4b 44 31 47 36@ to get the key "MKD1G6"
# @01@ is the length of the next string, which would be the value
# @00@ is the number of free bytes
# We read the next 1 byte(s), which is @0x32@. Thus, we get our value "2"
# In this case, the free bytes is 0, so we don't skip anything
# @05@ is the length of the next string, in this case a key. 
# We read the next 5 bytes @59 4e 4e 58 4b@, to get the key "YNNXK"
# @04@ is the length of the next string, which is a value
# @00@ is the number of free bytes after the value
# We read the next 4 bytes i.e. @46 37 54 49@ to get the value "F7TI"
# Finally, we encounter @FF@, which indicates the end of this zip map
# Thus, this zip map represents the hash @{"MKD1G6" => "2", "YNNXK" => "F7TI"}@

h2. Ziplist Encoding

A Ziplist is a list that has been serialized to a string. In essence, the elements of the list are stored sequentially along with flags and offsets to allow efficient traversal of the list in both directions.

To parse a ziplist, first a string is read from thee stream using "String Encoding". This string is the envelope of the ziplist. The contents of this string represent the ziplist.

The structure of a ziplist within this string is as follows -
@<zlbytes><zltail><zllen><entry><entry><zlend>@

# *zlbytes* : This is a 4 byte unsigned integer representing the total size in bytes of the zip list. The 4 bytes are in little endian format - the least signinficant bit comes first.
# *zltail* : This is a 4 byte unsigned integer in little endian format. It represents the offset to the tail (i.e. last) entry in the zip list
# *zllen* : This is a 2 byte unsigned integer in little endian format. It represents the number of entries in this zip list
# *entry* : An entry represents an element in the zip list. Details below
# *zlend* : Is always equal to @255@. It represents the end of the zip list.

Each entry in the zip list has the following format :
@<length-prev-entry><special-flag><raw-bytes-of-entry>@

*length-prev-entry* : This field stores the length of the previous entry, or 0 if this is the first entry. This allows easy traversal of the list in the reverse direction. This length is stored in either 1 byte or in 5 bytes. If the first byte is less than or equal to 253, it is considered as the length. If the first byte is 254, then the next 4 bytes are used to store the length. The 4 bytes are read as an unsigned integer.

*Special flag* : This flag indicates whether the entry is a string or an integer. It also indicates the length of the string, or the size of the integer. 
The various encodings of this flag are shown below :
# |00pppppp| - 1 byte : String value with length less than or equal to 63 bytes (6 bits).
# |01pppppp|qqqqqqqq| - 2 bytes : String value with length less than or equal to 16383 bytes (14 bits).
# |10______|qqqqqqqq|rrrrrrrr|ssssssss|tttttttt| - 5 bytes : String value with length greater than or equal to 16384 bytes.
# |1100____| - 1 byte : Integer encoded as int16_t (2 bytes).
# |1101____| - 1 byte : Integer encoded as int32_t (4 bytes).
# |1110____| - 1 byte : Integer encoded as int64_t (8 bytes).

*Raw Bytes* : After the special flag, the raw bytes of entry follow. The number of bytes was previously determined as part of the special flag.

*Worked Example 1*
@23 23 00 00 00 1e 00 00 00 04 00 00 e0 ff ff ff ff ff ff ff 7f 0a d0 ff ff 00 00 06 c0 fc 3f 04 c0 3f 00 ff ... @
@  |           |           |     |                             |                 |           |           |       @

# Start by decoding this using "String Encoding". @23@ is the length of the string, therefore we will read the next 35 bytes till @ff@
# Now, we are parsing the string starting at @23 00 00 ...@ using "Ziplist encoding"
# The first 4 bytes @23 00 00 00@ represent the total length in bytes of this ziplist. Notice that this is in little endian format
# The next 4 bytes @1e 00 00 00@ represent the offset to the tail entry. @1e@ = 30, and this is a 0 based offset. 0th position = @23@, 1st position = @00@ and so on. It follows that the last entry starts at @04 c0 3f 00 ..@
# The next 2 bytes @04 00@ represent the number of entries in this list.
# From now on, we start reading the entries
# @00@ represents the lenght of previous entry. 0 indicates this is the first entry.
# @e0@ is the special flag. Since it starts with the bit pattern @1110____@, we read the next 8 bytes as an integer. This is the first entry of the list.
# We now start the second entry
# @0a@ is the length of the previous entry. 10 bytes = 1 byte for prev. length + 1 byte for special flag + 8 bytes for integer.
# @d0@ is the special flag. Since it starts with the bit pattern @1101____@, we read the next 4 bytes as an integer. This is the second entry of the list
# We now start the third entry
# @06@ is the length of previous entry. 6 bytes = 1 byte for prev. length + 1 byte for special flag + 4 bytes for integer
# @c0@ is the special flag. Since it starts with the bit pattern @1100____@, we read the next 2 bytes as an integer. This is the third entry of the list
# We now start the last entry
# @04@ is length of previous entry
# @c0@ indicates a 2 byte number
# We read the next 2 bytes, which gives us our fourth entry
# Finally, we encounter @ff@, which tells us we have consumed all elements in this ziplist.
# Thus, this ziplist stores the values @[0x7fffffffffffffff, 65535, 16380, 63]

h2. Intset Encoding

An Intset is a binary search tree of integers. The binary tree is implemented in an array of integers. An intset is used when all the elements of the set are integers. An Intset has support for upto 64 bit integers. As an optimization, if the integers can be represented in fewer bytes, the array of integers will be constructed from 16 bit or 32 bit integers. When a new element is inserted, the implementation takes care to upgrade if necessary.

Since an Intset is a binary search tree, the numbers in this set will always be sorted.

An Intset has an external interface of a Set. 

To parse an Intset, first a string is read from thee stream using "String Encoding". This string is the envelope of the Intset. The contents of this string represent the Intset.

Within this string, the Intset has a very simple layout :
@<encoding><length-of-contents><contents>@

# *encoding* : is a 32 bit unsigned integer. It has 3 possible values - 2, 4 or 8. It indicates the size in bytes of each integer stored in contents. And yes, this is wasteful - we could have stored the same information in 2 bits.
# *length-of-contents* : is a 32 bit unsigned integer, and indicates the length of the contents array
# *contents* : is an array of $length-of-contents bytes. It contains the binary tree of integers

*Example*
@14 04 00 00 00 03 00 00 00 fc ff 00 00 fd ff 00 00 fe ff 00 00 ...@

# Start by decoding this using "String Encoding". @14@ is the length of the string, therefore we will read the next 20 bytes till @00@
# Now, we start interpreting the string starting at @04 00 00 ... @
# The first 4 bytes @04 00 00 00@ is the encoding. Since this evaluates to 4, we know we are dealing with 32 bit integers
# The next 4 bytes @03 00 00 00@ is the length of contents. So, we know we are dealing with 3 integers, each 4 byte long
# From now on, we read in groups of 4 bytes, and convert it into a unsigned integer
# Thus, our intset looks like - @0x0000FFFC, 0x0000FFFD, 0x0000FFFE@. Notice that the integers are in little endian format i.e. least signinficant bit came first.

h2. Sorted Set as Ziplist Encoding

A sorted list in ziplist encoding is stored just like the Ziplist described above. Each element in the sorted set is followed by its score in the ziplist.

*Example*
['Manchester City', 1, 'Manchester United', 2, 'Totenham', 3]

As you see, the scores follow each element.

h2. Hashmap in Ziplist Encoding
In this, key=value pairs of a hashmap are stored as successive entries in a ziplist. 

Note : This was introduced in rdb version 4. This deprecates zipmap encoding that was used in earlier versions.

*Example*
   {"us" => "washington", "india" => "delhi"} 
   
   is stored in a ziplist as :
   
   ["us", "washington", "india", "delhi"]
   

h3. CRC32 Check Sum

Starting with RDB Version 5, an 8 byte CRC 32 checksum is added to the end of the file. It is possible to disable this checksum via a parameter in redis.conf.
When checksum is disabled, this field will have zeroes.