2020-10-15 10:49:16 +00:00
//! This module implements the data storage and access layer.
2019-02-12 12:27:11 +00:00
//!
2020-10-15 10:49:16 +00:00
//! # Data formats
2019-02-12 12:27:11 +00:00
//!
2020-10-15 10:49:16 +00:00
//! PBS splits large files into chunks, and stores them deduplicated using
//! a content addressable storage format.
2019-08-14 12:08:27 +00:00
//!
2020-10-15 10:49:16 +00:00
//! Backup snapshots are stored as folders containing a manifest file and
//! potentially one or more index or blob files.
2019-03-22 09:14:50 +00:00
//!
2020-10-15 10:49:16 +00:00
//! The manifest contains hashes of all other files and can be signed by
//! the client.
2019-03-22 09:14:50 +00:00
//!
2020-10-15 10:49:16 +00:00
//! Blob files contain data directly. They are used for config files and
//! the like.
2019-03-22 09:14:50 +00:00
//!
2020-10-15 10:49:16 +00:00
//! Index files are used to reconstruct an original file. They contain a
//! list of SHA256 checksums. The `DynamicIndex*` format is able to deal
//! with dynamic chunk sizes (CT and host backups), whereas the
//! `FixedIndex*` format is an optimization to store a list of equal sized
//! chunks (VMs, whole block devices).
2019-03-22 09:14:50 +00:00
//!
2020-10-15 10:49:16 +00:00
//! A chunk is defined as a binary blob, which is stored inside a
//! [ChunkStore](struct.ChunkStore.html) instead of the backup directory
//! directly, and can be addressed by its SHA256 digest.
2019-03-22 09:14:50 +00:00
//!
//!
2020-10-15 10:49:16 +00:00
//! # Garbage Collection (GC)
2019-03-22 09:14:50 +00:00
//!
2020-10-15 10:49:16 +00:00
//! Deleting backups is as easy as deleting the corresponding .idx files.
//! However, this does not free up any storage, because those files just
//! contain references to chunks.
2019-03-22 09:14:50 +00:00
//!
2020-10-15 10:49:16 +00:00
//! To free up some storage, we run a garbage collection process at
//! regular intervals. The collector uses a mark and sweep approach. In
//! the first phase, it scans all .idx files to mark used chunks. The
//! second phase then removes all unmarked chunks from the store.
2019-03-22 09:14:50 +00:00
//!
2020-10-15 10:49:16 +00:00
//! The locking mechanisms mentioned below make sure that we are the only
//! process running GC. We still want to be able to create backups during
//! GC, so there may be multiple backup threads/tasks running, either
//! started before GC, or while GC is running.
2019-03-22 09:14:50 +00:00
//!
2020-10-15 10:49:16 +00:00
//! ## `atime` based GC
2019-03-22 09:14:50 +00:00
//!
2020-10-15 10:49:16 +00:00
//! The idea here is to mark chunks by updating the `atime` (access
//! timestamp) on the chunk file. This is quite simple and does not need
//! additional RAM.
2019-03-22 09:14:50 +00:00
//!
2020-10-15 10:49:16 +00:00
//! One minor problem is that recent Linux versions use the `relatime`
//! mount flag by default for performance reasons (and we want that). When
//! enabled, `atime` data is written to the disk only if the file has been
//! modified since the `atime` data was last updated (`mtime`), or if the
//! file was last accessed more than a certain amount of time ago (by
//! default 24h). So we may only delete chunks with `atime` older than 24
//! hours.
//!
//! Another problem arises from running backups. The mark phase does not
//! find any chunks from those backups, because there is no .idx file for
//! them (created after the backup). Chunks created or touched by those
//! backups may have an `atime` as old as the start time of those backups.
//! Please note that the backup start time may predate the GC start time.
//! So we may only delete chunks older than the start time of those
//! running backup jobs, which might be more than 24h back (this is the
//! reason why ProcessLocker exclusive locks only have to be exclusive
//! between processes, since within one we can determine the age of the
//! oldest shared lock).
2019-03-22 09:14:50 +00:00
//!
2020-10-15 10:49:16 +00:00
//! ## Store `marks` in RAM using a HASH
2019-03-22 09:14:50 +00:00
//!
2020-10-15 10:49:16 +00:00
//! Might be better. Under investigation.
2019-03-22 09:14:50 +00:00
//!
//!
2020-10-15 10:49:16 +00:00
//! # Locking
2019-03-22 09:14:50 +00:00
//!
2020-10-15 10:49:16 +00:00
//! Since PBS allows multiple potentially interfering operations at the
//! same time (e.g. garbage collect, prune, multiple backup creations
2021-03-10 15:37:09 +00:00
//! (only in separate groups), forget, ...), these need to lock against
2020-10-15 10:49:16 +00:00
//! each other in certain scenarios. There is no overarching global lock
//! though, instead always the finest grained lock possible is used,
//! because running these operations concurrently is treated as a feature
//! on its own.
2019-03-22 09:14:50 +00:00
//!
2020-10-15 10:49:16 +00:00
//! ## Inter-process Locking
2019-03-30 15:26:52 +00:00
//!
2020-10-15 10:49:16 +00:00
//! We need to be able to restart the proxmox-backup service daemons, so
//! that we can update the software without rebooting the host. But such
//! restarts must not abort running backup jobs, so we need to keep the
//! old service running until those jobs are finished. This implies that
//! we need some kind of locking for modifying chunks and indices in the
//! ChunkStore.
2019-03-30 15:26:52 +00:00
//!
2020-10-15 10:49:16 +00:00
//! Please note that it is perfectly valid to have multiple
//! parallel ChunkStore writers, even when they write the same chunk
//! (because the chunk would have the same name and the same data, and
//! writes are completed atomically via a rename). The only problem is
//! garbage collection, because we need to avoid deleting chunks which are
//! still referenced.
2019-03-30 15:26:52 +00:00
//!
2020-10-15 10:49:16 +00:00
//! To do this we use the
//! [ProcessLocker](../tools/struct.ProcessLocker.html).
2019-03-30 15:26:52 +00:00
//!
2020-10-15 10:49:16 +00:00
//! ### ChunkStore-wide
2019-03-30 15:26:52 +00:00
//!
2020-10-15 10:49:16 +00:00
//! * Create Index Files:
2019-03-30 15:26:52 +00:00
//!
2020-10-15 10:49:16 +00:00
//! Acquire shared lock for ChunkStore.
2019-03-30 16:21:40 +00:00
//!
2020-10-15 10:49:16 +00:00
//! Note: When creating .idx files, we create a temporary .tmp file,
//! then do an atomic rename.
2019-03-30 16:21:40 +00:00
//!
2020-10-15 10:49:16 +00:00
//! * Garbage Collect:
2019-03-30 16:21:40 +00:00
//!
2020-10-15 10:49:16 +00:00
//! Acquire exclusive lock for ChunkStore. If we have
//! already a shared lock for the ChunkStore, try to upgrade that
//! lock.
2019-03-30 16:21:40 +00:00
//!
2020-10-15 10:49:16 +00:00
//! Exclusive locks only work _between processes_. It is valid to have an
//! exclusive and one or more shared locks held within one process. Writing
//! chunks within one process is synchronized using the gc_mutex.
//!
//! On server restart, we stop any running GC in the old process to avoid
//! having the exclusive lock held for too long.
//!
//! ## Locking table
//!
//! Below table shows all operations that play a role in locking, and which
//! mechanisms are used to make their concurrent usage safe.
//!
//! | starting ><br>v during | read index file | create index file | GC mark | GC sweep | update manifest | forget | prune | create backup | verify | reader api |
//! |-|-|-|-|-|-|-|-|-|-|-|
//! | **read index file** | / | / | / | / | / | mmap stays valid, oldest_shared_lock prevents GC | see forget column | / | / | / |
//! | **create index file** | / | / | / | / | / | / | / | /, happens at the end, after all chunks are touched | /, only happens without a manifest | / |
//! | **GC mark** | / | Datastore process-lock shared | gc_mutex, exclusive ProcessLocker | gc_mutex | /, GC only cares about index files, not manifests | tells GC about removed chunks | see forget column | /, index files don’ t exist yet | / | / |
//! | **GC sweep** | / | Datastore process-lock shared | gc_mutex, exclusive ProcessLocker | gc_mutex | / | /, chunks already marked | see forget column | chunks get touched; chunk_store.mutex; oldest PL lock | / | / |
//! | **update manifest** | / | / | / | / | update_manifest lock | update_manifest lock, remove dir under lock | see forget column | /, “write manifest” happens at the end | /, can call “write manifest”, see that column | / |
//! | **forget** | / | / | removed_during_gc mutex is held during unlink | marking done, doesn’ t matter if forgotten now | update_manifest lock, forget waits for lock | /, unlink is atomic | causes forget to fail, but that’ s OK | running backup has snapshot flock | /, potentially detects missing folder | shared snap flock |
//! | **prune** | / | / | see forget row | see forget row | see forget row | causes warn in prune, but no error | see forget column | running and last non-running can’ t be pruned | see forget row | shared snap flock |
//! | **create backup** | / | only time this happens, thus has snapshot flock | / | chunks get touched; chunk_store.mutex; oldest PL lock | no lock, but cannot exist beforehand | snapshot flock, can’ t be forgotten | running and last non-running can’ t be pruned | snapshot group flock, only one running per group | /, won’ t be verified since manifest missing | / |
//! | **verify** | / | / | / | / | see “update manifest” row | /, potentially detects missing folder | see forget column | / | /, but useless (“update manifest” protects itself) | / |
//! | **reader api** | / | / | / | /, open snap can’ t be forgotten, so ref must exist | / | prevented by shared snap flock | prevented by shared snap flock | / | / | /, lock is shared |!
//! * / = no interaction
//! * shared/exclusive from POV of 'starting' process
2018-12-31 15:08:04 +00:00
2020-04-17 12:11:25 +00:00
use anyhow ::{ bail , Error } ;
2019-12-19 09:20:13 +00:00
2019-11-08 09:35:48 +00:00
// Note: .pcat1 => Proxmox Catalog Format version 1
pub const CATALOG_NAME : & str = " catalog.pcat1.didx " ;
2019-09-03 11:15:44 +00:00
2019-06-05 06:41:20 +00:00
#[ macro_export ]
macro_rules ! PROXMOX_BACKUP_PROTOCOL_ID_V1 {
( ) = > { " proxmox-backup-protocol-v1 " }
}
2019-06-05 06:12:13 +00:00
2019-06-27 07:01:41 +00:00
#[ macro_export ]
macro_rules ! PROXMOX_BACKUP_READER_PROTOCOL_ID_V1 {
( ) = > { " proxmox-backup-reader-protocol-v1 " }
}
2019-12-19 09:20:13 +00:00
/// Unix system user used by proxmox-backup-proxy
pub const BACKUP_USER_NAME : & str = " backup " ;
2020-08-26 08:02:18 +00:00
/// Unix system group used by proxmox-backup-proxy
pub const BACKUP_GROUP_NAME : & str = " backup " ;
2019-12-19 09:20:13 +00:00
/// Return User info for the 'backup' user (``getpwnam_r(3)``)
pub fn backup_user ( ) -> Result < nix ::unistd ::User , Error > {
match nix ::unistd ::User ::from_name ( BACKUP_USER_NAME ) ? {
Some ( user ) = > Ok ( user ) ,
None = > bail! ( " Unable to lookup backup user. " ) ,
}
}
2020-08-26 08:02:18 +00:00
/// Return Group info for the 'backup' group (``getgrnam(3)``)
pub fn backup_group ( ) -> Result < nix ::unistd ::Group , Error > {
match nix ::unistd ::Group ::from_name ( BACKUP_GROUP_NAME ) ? {
Some ( group ) = > Ok ( group ) ,
None = > bail! ( " Unable to lookup backup user. " ) ,
}
}
2021-07-06 10:49:10 +00:00
pub use pbs_datastore ::catalog ;
2021-07-07 07:04:09 +00:00
pub use pbs_datastore ::catalog ::* ;
pub use pbs_datastore ::checksum_reader ;
pub use pbs_datastore ::checksum_reader ::* ;
pub use pbs_datastore ::checksum_writer ;
pub use pbs_datastore ::checksum_writer ::* ;
pub use pbs_datastore ::chunker ;
pub use pbs_datastore ::chunker ::* ;
2021-07-06 10:49:10 +00:00
pub use pbs_datastore ::crypt_config ;
pub use pbs_datastore ::crypt_config ::* ;
pub use pbs_datastore ::crypt_reader ;
pub use pbs_datastore ::crypt_reader ::* ;
pub use pbs_datastore ::crypt_writer ;
pub use pbs_datastore ::crypt_writer ::* ;
pub use pbs_datastore ::data_blob ;
pub use pbs_datastore ::data_blob ::* ;
pub use pbs_datastore ::data_blob_reader ;
pub use pbs_datastore ::data_blob_reader ::* ;
pub use pbs_datastore ::data_blob_writer ;
pub use pbs_datastore ::data_blob_writer ::* ;
2021-07-07 07:04:09 +00:00
pub use pbs_datastore ::file_formats ;
pub use pbs_datastore ::file_formats ::* ;
pub use pbs_datastore ::index ;
pub use pbs_datastore ::index ::* ;
pub use pbs_datastore ::key_derivation ;
pub use pbs_datastore ::key_derivation ::* ;
2019-06-18 09:17:22 +00:00
2021-07-06 10:49:10 +00:00
mod manifest ;
pub use manifest ::* ;
2019-08-16 10:27:17 +00:00
2019-05-14 08:05:29 +00:00
mod chunk_stream ;
pub use chunk_stream ::* ;
2019-02-25 11:52:10 +00:00
mod chunk_stat ;
pub use chunk_stat ::* ;
2019-07-02 06:22:29 +00:00
mod read_chunk ;
pub use read_chunk ::* ;
2019-02-12 13:13:31 +00:00
mod chunk_store ;
pub use chunk_store ::* ;
mod fixed_index ;
pub use fixed_index ::* ;
mod dynamic_index ;
pub use dynamic_index ::* ;
2021-05-06 12:20:03 +00:00
#[ macro_use ]
2019-03-05 06:18:12 +00:00
mod backup_info ;
pub use backup_info ::* ;
2019-12-06 07:05:40 +00:00
mod prune ;
pub use prune ::* ;
2019-02-12 13:13:31 +00:00
mod datastore ;
pub use datastore ::* ;
2019-11-21 11:47:51 +00:00
2020-12-01 05:34:33 +00:00
mod store_progress ;
pub use store_progress ::* ;
2020-06-24 11:11:45 +00:00
mod verify ;
pub use verify ::* ;
2019-11-21 11:47:51 +00:00
mod catalog_shell ;
pub use catalog_shell ::* ;
2020-06-18 11:55:27 +00:00
2021-06-07 15:35:26 +00:00
mod cached_chunk_reader ;
pub use cached_chunk_reader ::* ;