[ ˈrɛplɪkət ]
Configurable and lightweight backup utility with deduplication and encryption.
Python 3.8 (or newer) running on Linux, MacOS, or Windows.
- local path
- Backblaze B2
- Amazon S3
- any S3-compatible service
You can implement and use your own adapter for pretty much any backup destination without changing the source code of Replicat (more on that later).
pip install replicat
For various reasons, I wasn't entirely happy with any of the similar projects that I've tried.
Highlights/goals of Replicat:
- efficient, concise, easily auditable implementation
- high customisability
- few external dependencies
- well-documented behaviour
- unified repository layout
- API that exists
This project borrows a few ideas from those other projects, but not enough to be considered a copycat.
You can use Replicat to backup files from your machine to a repository, located on a supported backend such as a local directory or cloud storage (like Backblaze B2). Files are transferred and stored in an optionally encrypted and chunked form, and references to chunks are stored in snapshots, along with file name and metadata.
To restore files from a snapshot, Replicat will download referenced chunks from the backend and use them to assemble the original files locally.
Replicat supports two types of repositories: encrypted (the default) and unencrypted. You may want to disable encryption if you trust your backend provider and network, for example. Duplicate chunks are reused between snapshots to save on bandwidth and storage costs.
See Encryption for a more in-depth look into this, or Functional flow overview for the extremely cool and colorful diagrams that I worked really hard on.
The installer will create the replicat
command (same as python -m replicat
).
There are several available subcommands:
init
- initialises the repository using the provided settingssnapshot
- creates a new snapshot in the repositorylist-snapshots
/ls
- lists snapshotslist-files
/lf
- lists files across snapshotsrestore
- restores files from snapshotsadd-key
- creates a new key for the encrypted repositorydelete
- deletes snapshots by their namesclean
- performs garbage collectionupload-objects
- uploads objects to the backend (a low-level command)download-objects
- downloads objects from the backend (a low-level command)list-objects
- lists objects at the backend (a low-level command)delete-objects
- deletes objects from the backend (a low-level command)
⚠️ WARNING: commands that read from or upload to the repository can safely be run concurrently; however, there are presently no guards in place that would make it safe for you to run destructive actions (delete
,clean
) concurrently with those actions unless you use independent keys (see the explanation above). I do plan to implement them eventually, but in the meantime DO NOT use shared keys (or, naturally, the same key) tosnapshot
andclean
at the same time, for example.As far as the upcoming implementation of such guards, it'll be based on locks. I'm familiar with the lock-free deduplication strategy (like in Duplicacy), but I don't like it much.
There are several command line arguments that are common to all subcommands:
-
-r
/--repository
- used to specify the type and location of the repository backend (backup destination). The format is<backend>:<connection string>
, where<backend>
is the short name of a Replicat-compatible backend and<connection string>
is open to interpretation by the adapter for the selected backend. Examples:b2:bucket-name
for the B2 backend orlocal:some/local/path
for the local backend (or justsome/local/path
, since the<backend>:
part can be omitted for local destinations). If the backend requires additional arguments, they will appear in the--help
output. Refer to the section on backends for more detailed information. -
-q
/--hide-progress
- suppresses progress indication for commands that support it -
-c
/--concurrent
- the number of concurrent connections to the backend -
--cache-directory
- specifies the directory to use for cache.--no-cache
disables cache completely. -
-v
/--verbose
- increases the logging verbosity. The default verbosity iswarning
,-v
meansinfo
,-vv
meansdebug
.
Encrypted repositories require a key and a matching password for every operation:
-K
/--key-file
- the path to the key file-p
/--password
- the password in plaintext. However, it's more secure to provide the password in a file via the-P
/--password-file
argument, or as an environment variableREPLICAT_PASSWORD
.
If the backend requires additional parameters (account name, client secret, some boolean flag, or literally anything else), you'll also be able to set them via command line arguments or in the configuration file. Refer to Backends section to learn more.
If you often use many of these arguments, and their values mostly stay the same between invocations, you may find it easier to put them in a configuration file instead. There are three arguments related to that:
--profile
- load settings from this profile in the configuration file--config
- path to the configuration file (check--help
for the default config location)--ignore-config
- ignore the configuration file
Note that values from CLI always take precedence over options from the configuration file.
Specifically, to build the final configuration, Replicat considers command line arguments, environment
variables, the configuration file (either the default one or the one supplied via --config
),
and global defaults, in that order.
Names of configuration file options mostly match the long names of command line arguments
(e.g., hide-progress = true
matches --hide-progress
, repository = "s3:bucket"
matches
-r s3:bucket
), but you can always refer to the
Configuration file section for full reference.
# Unencrypted repository 'some:repository'. The --encryption none flag disables encryption
$ replicat init -r some:repository --encryption none
# Encrypted repository with initial password taken from string.
# The new key will be printed to stdout
$ replicat init -r some:repository -p 'password string'
# Encrypted repository with initial password taken from a file.
# The new key will be written to 'path/to/key/file'
$ replicat init -r some:repository -P path/to/password/file -o path/to/key/file
# Specifies the cipher
$ replicat init -r some:repository -p '...' --encryption.cipher.name chacha20_poly1305
# Specifies settings for the cipher and the hash algorithm
$ replicat init -r some:repository \
-p '...' \
--encryption.cipher.name aes_gcm \
--encryption.cipher.key_bits 128 \
--hashing.name sha2 \
--hashing.bits 256
# Specifies the KDF name and parameters (for the key)
$ replicat init -r some:repository \
-p '...' \
--encryption.kdf.name scrypt \
--encryption.kdf.n 2097152
# Specifies the chunking parameters
$ replicat init -r some:repository \
-p '...' \
--chunking.min-length 128_000 \
--chunking.max-length 2_048_000
# Equivalent (dashes in argument names are converted to underscores)
$ replicat init -r some:repository \
-p '...' \
--chunking.min_length 128_000 \
--chunking.max_length 2_048_000
The main takeaway here is that you can disable encryption simply by setting --encryption none
.
Check the Custom settings section for more.
# Unlocks the repository, uploads provided files in encrypted chunks,
# using no more than 10 concurrent connections, creating a snapshot
$ replicat snapshot -r some:repository \
-P path/to/password/file \
-K path/to/key/file \
-c 10 \
-n 'A note (optional)'
image.jpg some-directory another-directory and/more.text
# Uploads one file in chunks using 5 concurrent connections, while restricting
# the bandswith to 512 KB per second. Creates a new snapshot without any note
$ replicat snapshot -r some:repository --limit-rate 512K file.log
# Unlocks the repository and lists all of the snapshots
$ replicat list-snapshots -r some:repository -P path/to/password/file -K path/to/key/file
# Same, but without the table header
$ replicat ls -r some:repository -P path/to/password/file -K path/to/key/file --no-header
# Lists snapshots with names that match any of the regexes passed via -S/--snapshot-regex
# In this example, Replicat will only list snapshots whose names start with '123456'
# OR include substring 'abcdef'
$ replicat ls -r some:repository \
-P path/to/password/file \
-K path/to/key/file \
-S '^123456' \
-S abcdef
# Lists the snapshots, but instead of the default set of columns, displays just the
# snapshot name, the number of files in the snapshot, and the total size of the snapshot
# (in that order). Oh, there's also --no-header
$ replicat ls -r some:repository \
-P path/to/password/file \
-K path/to/key/file \
--no-header \
--columns name,file_count,size
# Unlocks the repository and lists all versions of all the files
$ replicat list-files -r some:repository -P path/to/password/file -K path/to/key/file
# Same, but without the table header
$ replicat lf -r some:repository -P path/to/password/file -K path/to/key/file --no-header
# Only lists files with paths that match any of the regexes passed via -F/--file-regex
# (in this example, PNGs and text files) IF they are included in snapshots that match
# the -S regex(es) (i.e., snapshot names that start with '1234beef')
$ replicat lf -r some:repository \
-P path/to/password/file \
-K path/to/key/file \
-F '\.t[e]?xt$' \
-F '\.png$' \
-S '^1234beef'
# Lists all versions of all the files, but instead of the default set of columns,
# displays the original path of the file first, then the snapshot name, the hash digest
# of the file, and the file access time (recorded at the time of the snapshot creation)
$ replicat lf -r some:repository \
-P path/to/password/file \
-K path/to/key/file \
--columns snapshot_name,path,digest,atime
# Unlocks the repository and restores the latest versions of all files to target directory,
# while limiting the download speed to at most 10 MB per second
$ replicat restore -r some:repository \
-P path/to/password/file \
-K path/to/key/file \
--limit-rate 10M \
target-directory
# Unlocks the repository and restores the latest versions of files that match any of the
# -F regex(es) from snapshots that match any of the -S regex(es)
$ replicat restore -r some:repository \
-P path/to/password/file \
-K path/to/key/file \
-F '\.jpg$' \
-F '^/root' \
-S 'abcdef' \
-S '12345' \
target-directory
# Creates an independent key, which will be printed to stdout
$ replicat add-key -N path/to/file/with/new/password
# Unlocks the repository and creates a shared key (i.e. with shared secrets)
$ replicat add-key -r some:repository \
-p 'your password' \
-K your/key/file \
-n 'new password as a string' \
---shared
# Creates an independent key, which will be written to path/to/new/key/file
$ replicat add-key -r some:repository -n 'new password as a string' -o path/to/new/key/file
# Creates an independent key with some custom encryption key derivation settings
$ replicat add-key -r some:repository \
--encryption.kdf.name scrypt \
--encryption.kdf.n 4194304
# Uses your password to create a new key with the copy of data in your key, but with
# different encryption key derivation settings (same as in the previous example).
# The new key will be printed to stdout
$ replicat add-key -r some:repository \
-P your/password/file \
-K your/key/file \
--encryption.kdf.n 4_194_304 \
--clone
This shows a way to customize the KDF parameters for the key, arguably making the data stored in your key better protected (to the point of probable overkill with values like this). Use the Custom settings section as reference.
# Unlocks the repository and deletes snapshots by name (as returned by ls/list-snapshots).
# Chunks that aren't referenced by any other snapshot will be deleted automatically
$ replicat delete -r some:repository \
-P path/to/password/file \
-K path/to/key/file \
NAME1 NAME2 NAME3 ...
# Same, but doesn't ask for confirmation
$ replicat delete -r some:repository \
-P path/to/password/file \
-K path/to/key/file \
NAME1 NAME2 NAME3 ... \
--yes
# Unlocks the repository and deletes all chunks that are not referenced by any snapshot
$ replicat clean -r some:repository -P path/to/password/file -K path/to/key/file
# Uploads local files directly to the repository without any additional processing.
# File path -> resulting name:
# '/working/directory/some/file' -> 'some/file'
# '/working/directory/another/file' -> 'another/file'
# '/working/directory/another/directory/another-file' -> 'another/directory/another-file'
# '/absolute/directory/path/with-file' -> 'absolute/directory/path/with-file'
# '/absolute/file' -> 'absolute/file'
/working/directory$ replicat upload-objects -r some:repository \
some/file \
/working/directory/another/directory \
/absolute/directory/path \
/absolute/file
# Uploads local files that do not yet exist in the repository (only checks the file names)
$ replicat upload-objects -r some:repository --skip-existing some/file some/directory
# Same, but additionally limits the upload speed to 2 MiB per second
$ replicat upload-objects \
-r some:repository \
--skip-existing \
-L 2MiB \
some/file some/directory
# Downloads all objects from the repository directly to the current working directory
# without any additional processing
$ replicat download-objects -r some:repository
# Same, but it downloads objects to 'different/directory' instead
$ replicat download-objects -r some:repository different/directory
# Same, but it skips objects that already exist locally (only checks the file names)
$ replicat download-objects -r some:repository --skip-existing different/directory
# Downloads objects whose paths match any the -O regex(es) (i.e., all objects inside of
# 'data' OR 'snapshots' top-level directories in the repository) to the current working
# directory, skipping existing objects and limiting the download speed to mere
# 1 gigabit per second
$ replicat download-objects -r some:repository -O '^data/' -O '^snapshots/' -S -L 1Gb
# Lists all objects currently in the repository
$ replicat list-objects -r some:repository
# Only lists objects whose paths match any of the regexes passed via -O/--object-regex
# (i.e., all objects within 'data' AND 'snapshots' top-level directories in the repository)
$ replicat list-objects -r some:repository -O '^data/' -O '^snapshots/'
# Deletes objects by their full paths as returned by list-objects
$ replicat delete-objects -r some:repository object/path/1 object/path/2 ...
# Same, but doesn't ask for confirmation
$ replicat delete-objects -r some:repository object/path/1 object/path/2 ... -y
replicat --version
As mentioned in the Command line interface section, options that you can put in the configuration file mostly match CLI arguments, with few exceptions.
Option name | Type | Supported values | Notes |
---|---|---|---|
repository |
string | <backend>:<connection string> |
|
concurrent |
integer | Integers greater than 0 | |
hide‑progress |
boolean | true , false |
|
cache‑directory |
path | Relative or absolute path | |
no‑cache |
boolean | true , false |
|
password |
string | Password as a string | Cannot be used together with password-file |
password‑file |
path | Relative or absolute path | Cannot be used together with password |
key |
string | JSON as a string | Cannot be used together with key-file |
key‑file |
path | Relative or absolute path | Cannot be used together with key |
log‑level |
string | debug , info , warning , error , critical , fatal |
CLI option -v increases logging verbosity starting from warning , while this option lets you set lower logging verbosity, such as error |
If the backend requires additional parameters (account id, access key, numeric port, or literally
anything else), Replicat lets you provide them via the configuration file. For example, if you see
a backend-specific argument --some-backend-option
in the --help
output, the equivalent
configuration file option will be called some-backend-option
.
Here's an example configuration file (it uses TOML syntax)
concurrent = 10
# Relative paths work
cache-directory = "~/.cache/directory/for/replicat"
[debugging]
log-level = "info"
hide-progress = true
[my-local-repo]
repository = "some/local/path"
password = "<secret>"
key = """
{
"kdf": { ... },
"kdf_params": { "!b": "..." },
"private": { "!b": "..." }
}
"""
concurrent = 15
no-cache = true
[some-s3-repo]
repository = "s3:bucket-name"
key-id = "..."
access-key = "..."
region = "..."
Options that you specify at the top of the configuration file are defaults and they will be
inherited by all of the profiles. In the example above there are three profiles
(not including the default one): debugging
, my-local-repo
, some-s3-repo
. You can tell
Replicat which profile to use via the --profile
CLI argument.
Notice that some-s3-repo
includes options that were not listed in the table. key-id
,
access-key
, region
are the aforementioned backend-specific options for the S3 backend.
See Backends.
Run replicat
commands with -r <backend>:<connection string>
and additional arguments
that are specific to the selected backend. Those arguments may have defaults and may also
be provided via environment variables or profiles. Use
replicat <command> -r <backend>:<connection string> --help
to see them.
The format is -r local:some/local/path
or simply -r some/local/path
.
The format is -r b2:bucket-id
or -r b2:bucket-name
. This backend uses B2 native API and
requires
- key ID (
--key-id
argument, orB2_KEY_ID
environment variable, orkey-id
option in a profile) - application key (
--application-key
argument, orB2_APPLICATION_KEY
environment variable, orapplication-key
option in a profile)
Sign into your Backblaze B2 account to generate them. Note that you can use the master application key or a normal (non-master) application key (which can also be restricted to a single bucket). Refer to official B2 docs for more information.
The format is -r s3:bucket-name
. Requires
- AWS key ID (
--key-id
argument, orS3_KEY_ID
environment variable, or thekey-id
option in a profile) - AWS access key (
--access-key
argument, orS3_ACCESS_KEY
environment variable, oraccess-key
option in a profile) - region (
--region
argument, orS3_REGION
environment variable, orregion
option in a profile)
The format is -r s3c:bucket-name
. Requires
- key ID (
--key-id
argument, orS3C_KEY_ID
environment variable, or thekey-id
option in a profile) - access key (
--access-key
argument, orS3C_ACCESS_KEY
environment variable, oraccess-key
option in a profile) - host (
--host
argument, orS3C_HOST
environment variable, orhost
option in a profile) - region (
--region
argument, orS3C_REGION
environment variable, orregion
option in a profile)
Host must not include the scheme. The default scheme is https
, but can be changed via the
--scheme
argument (or, equivalently, the S3C_SCHEME
environment variable or scheme
option
in a profile).
You can use S3-compatible backend to connect to B2, S3, and many other cloud storage providers that offer S3-compatible API.
replicat.backends
is a namespace package, making it possible to add custom backends
without changing replicat
source code.
Suppose your backend of choice is a hypothetical low low cost cloud storage
Proud Cloud (pc
for short). The most barebones implementation of the
Replicat-compatible adapter for the pc
backend will require a directory with
the following structure:
$ tree proud-cloud/
proud-cloud/
└── replicat
└── backends
└── pc.py
The -r
argument of replicat
commands will take the form of -r pc:<connection string>
.
Replicat will use it to locate the pc
module inside the replicat.backends
package,
load the replicat.backends.pc.Client
class, and pass the <connection string>
to its constructor to create the backend instance. In case there are some additional
parameters that are required to connect to Proud Cloud (account id, secret token, etc.),
you should add them to the replicat.backends.pc.Client
constructor as keyword-only arguments.
If present, Replicat will generate the corresponding command line arguments with defaults and
you'll even be able to use environment variables or profiles to provide them.
replicat.backends.pc.Client
must subclass replicat.backends.base.Backend
and implement all
of the methods marked as abstract. You could use implementations of existing
replicat.backends.base.Backend
subclasses as examples. To make your implementation visible
to Replicat, you'll need to add it to the module search path before invoking replicat
(you could use the
PYTHONPATH
environment
variable for that).
Here's an example:
# ./proud-cloud/replicat/backends/pc.py
from .base import Backend
class ProudCloud(Backend):
def __init__(self, connection_string, *, account_id, secret, port=9_876, legacy=False):
print(f'PC args: {connection_string=}, {account_id=}, {secret=}, {port=}, {legacy=}')
...
...
Client = ProudCloud
$ PYTHONPATH=proud-cloud replicat init -r pc:... --help
usage: replicat init [-h] [-r REPO] [-q] [-c CONCURRENT] [-v] [-K KEYFILE]
[-p PASSWORD | -P PASSWORD_FILE_PATH] [--account-id ACCOUNT_ID]
[--secret SECRET] [--port PORT] [--legacy LEGACY] [-o KEY_OUTPUT_FILE]
optional arguments:
...
arguments specific to the ProudCloud backend:
--account-id ACCOUNT_ID
or the PROUDCLOUD_ACCOUNT_ID environment variable
--secret SECRET or the PROUDCLOUD_SECRET environment variable
--port PORT or the PROUDCLOUD_PORT environment variable, or the constructor default 9876
--legacy LEGACY or the PROUDCLOUD_LEGACY environment variable, or the constructor default False
$ PYTHONPATH=proud-cloud PROUDCLOUD_LEGACY=true PROUDCLOUD_SECRET='pr0ud' \
replicat init -r pc:... \
--account-id 12345 \
--port 9877
PC args: connection_string='...', account_id=12345, secret='pr0ud', port=9877, legacy=True
...
If you've created a Replicat-compatible adapter for a backend that Replicat doesn't already support and your implementation doesn't depend on additional third-party libraries (or at least they are not too heavy and can be moved to extras), consider submitting a PR to include it in this repository.
Replicat's default parameters and selection of cryptographic primitives should work well for most users but they do allow for some customisation if you know what you are doing.
Supported primitives and their parameters:
Name | Use | Parameters (with defaults) | Description | Notes |
---|---|---|---|---|
aes_gcm |
AEAD | key_bits (256 [bits]), nonce_bits (96 [bits]) |
AES in GCM mode | Supports 128, 192, and 256 bit keys |
chacha20_poly1305 |
AEAD | ChaCha20‑Poly1305 | ||
scrypt |
Key derivation | n (2 ^ 20), r (8), p (1) |
Scrypt KDF | Preferred for low-entropy inputs |
blake2b |
Hash, HMAC, key derivation | length (64 [bytes]) |
BLAKE2b | BLAKE2b-based "KDF" should only be used for high-entropy inputs |
sha2 |
Hash | bits (512 [bits]) |
SHA2 | Supported digest sizes are 224, 256, 384, and 512 bits |
sha3 |
Hash | bits (512 [bits]) |
SHA3 | Supported digest sizes are 224, 256, 384, and 512 bits |
gclmulchunker |
CDC | min_length (128K [bytes]), max_length (5.12M [bytes]) |
Content-defined chunking algorithm based on carry-less multiplication (CLMUL) |
Encryption algorithm, hash algorithm, chunker, as well as their parameters are set for the entire
repository and so apply to all users of the repository. By default, Replicat uses AES256-GCM
for encryption, Scrypt (n
is 2 ^ 20, r
is 8, p
is 1) as the KDF for the user's encryption
key, BLAKE2b (as hash, "KDF", and HMAC), and GCLMULChunker as the chunker, which produces chunks
between 128K and 5.12M in size with default parameters. KDF for the user's encryption key and
KDF params can vary from key to key, though.
Repository-wide settings are stored in the file called config
that gets uploaded to the root
of the repository when you initialise it. Here's the default config
:
{
"hashing": {
"name": "blake2b",
"length": 64
},
"chunking": {
"name": "gclmulchunker",
"min_length": 128000,
"max_length": 5120000
},
"encryption": {
"cipher": {
"name": "aes_gcm",
"key_bits": 256,
"nonce_bits": 96
}
}
}
Coincidentally, this hierarchy is how you're expected to provide your custom repository settings via CLI -- only in a flat representation.
For example, to disable encryption for the repository, you'd pass --encryption none
to
the init
command. If instead you wish to reduce AES-GCM key size from the default
256 bits to 128 bits, all you need to do is pass --encryption.cipher.key_bits 128
(or, equivalently, --encryption.cipher.key-bits 128
) during initialisation.
To change the cipher from the default AES256-GCM to ChaCha20‑Poly1305, you'd use
--encryption.cipher.name chacha20_poly1305
. Note that when you set the name
attribute
like that, Replicat will load the default parameters for the new algorithm and also check
if your settings are valid. It would be an error to provide any parameters
for chacha20_poly1305
, say.
Changing config
after the repository has already been initialised may render
existing data inaccessible.
You may specify the KDF and its params whenever you create a new key, so either via init
or via add-key
. Similar to repository-wide settings, you'd use a flat hierarchical format
for that. For example, in order to increase the work factor for Scrypt (the default KDF),
you could pass --encryption.kdf.n 2097152
(next power of two after 1048576) to the command,
or you could tweak Scrypt's r
parameter the same way.
It's also possible to change the parameters for the existing key by cloning it via
the add-key
command. Simply set the --clone
flag and provide the new KDF settings
as shown in the previous paragraph.
If you believe you've found a security issue with Replicat, please report it to [email protected] (or DM me on Twitter or Telegram).