Skip to content

Latest commit

 

History

History
83 lines (65 loc) · 5.6 KB

DESIGN.md

File metadata and controls

83 lines (65 loc) · 5.6 KB

Design

Replication

This replication system is built such that there is a primary table and a replica table. All writes are performed against the primary table, and changes made to the primary table are pushed to the replica table. Reads can be performed against either the primary or replica table.

Dyno, the client that we use for interactions with DynamoDB can be configured to read from one table and write to another. The primary table has a DynaoDB stream associated with it, while the replica table does not need to.

Replication is performed via an AWS Lambda function which reads from the primary table's DynamoDB stream and duplicates changes onto the replica table.

 us-east-1                         eu-west-1
-----------                       -----------
|   api   |                       |   api   |
-----------                       -----------
  ↑     |                           |     ↑
  |     |                           |     |
  |     ↓                           |     |
  |   writes ←----------------------+     |
  |   |                                   |
reads |                                 reads
  |   |  +~~~~ DynamoDB stream ~~~~+      |
  |   |  |                         |      |
  |   |  |                       writes   |
  |   |  |                         |      |
  |   ↓  ↑                         ↓      |
-----------                       -----------
|         |                       |         |
| primary |                       | replica |
|  table  |                       |  table  |
|         |                       |         |
-----------                       -----------

Consistency and repair

The diff-tables script provided by dynamodb-replicator scans the primary table, and performs record-by-record checks that the replica table's data is up-to-date. If it encounters any discrepancies, the data in the replica table is updated to match the primary table. This script can also be used to backfill a brand new replica table.

        primary                                     replica
------------------------                    ------------------------
| hash | range | value |                    | hash | range | value |
------------------------ --+                ------------------------
|  10  |   1   |   1   |   |  get set   --→ |  10  |   1   |   1   | ✔
------------------------   | of primary     ------------------------
|  10  |   2   |   7   |   |  records   --→ |  10  |   2   |   7   | ✔
------------------------   |                ------------------------
|  11  |   1   |   3   |   | check each --→ |  11  |   1   |   3   | ✔
------------------------   | individual     ------------------------
|  12  |   1   |   4   |   | in replica --→ |  12  |   1   |   8   | ✘ Repair this!
------------------------ --+                ------------------------

                            ...repeat...

------------------------ --+                ------------------------
|  13  |   1   |   3   |   |  get set   --→ |  13  |   1   |   3   | ✔
------------------------   | of primary     ------------------------
|  13  |   2   |   5   |   |  records   --→ |  13  |   2   |   7   | ✘ Repair this!
------------------------   |                ------------------------
|  14  |   1   |   4   |   | check each --→ |  14  |   1   |   4   | ✔
------------------------   | individual     ------------------------
|  15  |   1   |   6   |   | in replica --→ |  15  |   1   |   6   | ✔
------------------------ --+                ------------------------

This gives us a system where changes to the primary table are rapidly implemented in the replica tables via DynamoDB stream + Lambda replication. The consistency check system gives us additional certainty that replication is doing its job, and provides a system by which we can recover a database in one region by reading the data out of a table in another region.

Backup and Restore

The library provides scripts for two approaches to backup:

  1. A backup-table script can scan the primary table and write a dump file to S3. Running this script on a regular basis helps build resiliency against data corruption: if corrupt data were located, a trail of past states exists on S3 that can be recovered from.

  2. An incremental backup system involves a Lambda function that reads changes from the primary table's DynamoDB stream and writes each change to a file on S3. The result is an S3 "clone" of the current state of the DynamoDB table where one table record = one S3 object. In a versioned S3 bucket, this can proved a complete history of changes made to the table.

An incremental-snapshot script is capable of reading each item in the S3 folder and writing the aggregate to a file, resulting a snapshot of your table's state similar to the direct scan approach.

Dyno includes a CLI tool with an import command that can be used to read a backup file back into a table. This can be used to recover an entire database from an dump. Running a restore looks something like:

$ npm install -g s3print dyno
$ s3print s3://my-bucket/my-database-backup | dyno put us-east-1/my-new-database