chunk readers: ensure chunk/index CryptMode matches
an encrypted Index should never reference a plain-text chunk, and an unencrypted Index should never reference an encrypted chunk. Signed-off-by: Fabian Grünbichler <f.gruenbichler@proxmox.com>
This commit is contained in:
		
				
					committed by
					
						
						Dietmar Maurer
					
				
			
			
				
	
			
			
			
						parent
						
							2d55beeca0
						
					
				
				
					commit
					14f6c9cb8b
				
			@ -973,7 +973,7 @@ fn download_file_decoded(
 | 
				
			|||||||
                let (csum, size) = index.compute_csum();
 | 
					                let (csum, size) = index.compute_csum();
 | 
				
			||||||
                manifest.verify_file(&file_name, &csum, size)?;
 | 
					                manifest.verify_file(&file_name, &csum, size)?;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
                let chunk_reader = LocalChunkReader::new(datastore, None);
 | 
					                let chunk_reader = LocalChunkReader::new(datastore, None, CryptMode::None);
 | 
				
			||||||
                let reader = AsyncIndexReader::new(index, chunk_reader);
 | 
					                let reader = AsyncIndexReader::new(index, chunk_reader);
 | 
				
			||||||
                Body::wrap_stream(AsyncReaderStream::new(reader)
 | 
					                Body::wrap_stream(AsyncReaderStream::new(reader)
 | 
				
			||||||
                    .map_err(move |err| {
 | 
					                    .map_err(move |err| {
 | 
				
			||||||
@ -988,7 +988,7 @@ fn download_file_decoded(
 | 
				
			|||||||
                let (csum, size) = index.compute_csum();
 | 
					                let (csum, size) = index.compute_csum();
 | 
				
			||||||
                manifest.verify_file(&file_name, &csum, size)?;
 | 
					                manifest.verify_file(&file_name, &csum, size)?;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
                let chunk_reader = LocalChunkReader::new(datastore, None);
 | 
					                let chunk_reader = LocalChunkReader::new(datastore, None, CryptMode::None);
 | 
				
			||||||
                let reader = AsyncIndexReader::new(index, chunk_reader);
 | 
					                let reader = AsyncIndexReader::new(index, chunk_reader);
 | 
				
			||||||
                Body::wrap_stream(AsyncReaderStream::with_buffer_size(reader, 4*1024*1024)
 | 
					                Body::wrap_stream(AsyncReaderStream::with_buffer_size(reader, 4*1024*1024)
 | 
				
			||||||
                    .map_err(move |err| {
 | 
					                    .map_err(move |err| {
 | 
				
			||||||
@ -1159,7 +1159,7 @@ fn catalog(
 | 
				
			|||||||
    let (csum, size) = index.compute_csum();
 | 
					    let (csum, size) = index.compute_csum();
 | 
				
			||||||
    manifest.verify_file(&file_name, &csum, size)?;
 | 
					    manifest.verify_file(&file_name, &csum, size)?;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    let chunk_reader = LocalChunkReader::new(datastore, None);
 | 
					    let chunk_reader = LocalChunkReader::new(datastore, None, CryptMode::None);
 | 
				
			||||||
    let reader = BufferedDynamicReader::new(index, chunk_reader);
 | 
					    let reader = BufferedDynamicReader::new(index, chunk_reader);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    let mut catalog_reader = CatalogReader::new(reader);
 | 
					    let mut catalog_reader = CatalogReader::new(reader);
 | 
				
			||||||
@ -1282,7 +1282,7 @@ fn pxar_file_download(
 | 
				
			|||||||
        let (csum, size) = index.compute_csum();
 | 
					        let (csum, size) = index.compute_csum();
 | 
				
			||||||
        manifest.verify_file(&pxar_name, &csum, size)?;
 | 
					        manifest.verify_file(&pxar_name, &csum, size)?;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
        let chunk_reader = LocalChunkReader::new(datastore, None);
 | 
					        let chunk_reader = LocalChunkReader::new(datastore, None, CryptMode::None);
 | 
				
			||||||
        let reader = BufferedDynamicReader::new(index, chunk_reader);
 | 
					        let reader = BufferedDynamicReader::new(index, chunk_reader);
 | 
				
			||||||
        let archive_size = reader.archive_size();
 | 
					        let archive_size = reader.archive_size();
 | 
				
			||||||
        let reader = LocalDynamicReadAt::new(reader);
 | 
					        let reader = LocalDynamicReadAt::new(reader);
 | 
				
			||||||
 | 
				
			|||||||
@ -49,6 +49,20 @@ pub struct FileInfo {
 | 
				
			|||||||
    pub csum: [u8; 32],
 | 
					    pub csum: [u8; 32],
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					impl FileInfo {
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    /// Return expected CryptMode of referenced chunks
 | 
				
			||||||
 | 
					    ///
 | 
				
			||||||
 | 
					    /// Encrypted Indices should only reference encrypted chunks, while signed or plain indices
 | 
				
			||||||
 | 
					    /// should only reference plain chunks.
 | 
				
			||||||
 | 
					    pub fn chunk_crypt_mode (&self) -> CryptMode {
 | 
				
			||||||
 | 
					        match self.crypt_mode {
 | 
				
			||||||
 | 
					            CryptMode::Encrypt => CryptMode::Encrypt,
 | 
				
			||||||
 | 
					            CryptMode::SignOnly | CryptMode::None => CryptMode::None,
 | 
				
			||||||
 | 
					        }
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
#[derive(Serialize, Deserialize)]
 | 
					#[derive(Serialize, Deserialize)]
 | 
				
			||||||
#[serde(rename_all="kebab-case")]
 | 
					#[serde(rename_all="kebab-case")]
 | 
				
			||||||
pub struct BackupManifest {
 | 
					pub struct BackupManifest {
 | 
				
			||||||
 | 
				
			|||||||
@ -2,9 +2,9 @@ use std::future::Future;
 | 
				
			|||||||
use std::pin::Pin;
 | 
					use std::pin::Pin;
 | 
				
			||||||
use std::sync::Arc;
 | 
					use std::sync::Arc;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
use anyhow::Error;
 | 
					use anyhow::{bail, Error};
 | 
				
			||||||
 | 
					
 | 
				
			||||||
use super::crypt_config::CryptConfig;
 | 
					use super::crypt_config::{CryptConfig, CryptMode};
 | 
				
			||||||
use super::data_blob::DataBlob;
 | 
					use super::data_blob::DataBlob;
 | 
				
			||||||
use super::datastore::DataStore;
 | 
					use super::datastore::DataStore;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
@ -21,20 +21,41 @@ pub trait ReadChunk {
 | 
				
			|||||||
pub struct LocalChunkReader {
 | 
					pub struct LocalChunkReader {
 | 
				
			||||||
    store: Arc<DataStore>,
 | 
					    store: Arc<DataStore>,
 | 
				
			||||||
    crypt_config: Option<Arc<CryptConfig>>,
 | 
					    crypt_config: Option<Arc<CryptConfig>>,
 | 
				
			||||||
 | 
					    crypt_mode: CryptMode,
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
impl LocalChunkReader {
 | 
					impl LocalChunkReader {
 | 
				
			||||||
    pub fn new(store: Arc<DataStore>, crypt_config: Option<Arc<CryptConfig>>) -> Self {
 | 
					    pub fn new(store: Arc<DataStore>, crypt_config: Option<Arc<CryptConfig>>, crypt_mode: CryptMode) -> Self {
 | 
				
			||||||
        Self {
 | 
					        Self {
 | 
				
			||||||
            store,
 | 
					            store,
 | 
				
			||||||
            crypt_config,
 | 
					            crypt_config,
 | 
				
			||||||
 | 
					            crypt_mode,
 | 
				
			||||||
 | 
					        }
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    fn ensure_crypt_mode(&self, chunk_mode: CryptMode) -> Result<(), Error> {
 | 
				
			||||||
 | 
					        match self.crypt_mode {
 | 
				
			||||||
 | 
					            CryptMode::Encrypt => {
 | 
				
			||||||
 | 
					                match chunk_mode {
 | 
				
			||||||
 | 
					                    CryptMode::Encrypt => Ok(()),
 | 
				
			||||||
 | 
					                    CryptMode::SignOnly | CryptMode::None => bail!("Index and chunk CryptMode don't match."),
 | 
				
			||||||
 | 
					                }
 | 
				
			||||||
 | 
					            },
 | 
				
			||||||
 | 
					            CryptMode::SignOnly | CryptMode::None => {
 | 
				
			||||||
 | 
					                match chunk_mode {
 | 
				
			||||||
 | 
					                    CryptMode::Encrypt => bail!("Index and chunk CryptMode don't match."),
 | 
				
			||||||
 | 
					                    CryptMode::SignOnly | CryptMode::None => Ok(()),
 | 
				
			||||||
 | 
					                }
 | 
				
			||||||
 | 
					            },
 | 
				
			||||||
        }
 | 
					        }
 | 
				
			||||||
    }
 | 
					    }
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
impl ReadChunk for LocalChunkReader {
 | 
					impl ReadChunk for LocalChunkReader {
 | 
				
			||||||
    fn read_raw_chunk(&self, digest: &[u8; 32]) -> Result<DataBlob, Error> {
 | 
					    fn read_raw_chunk(&self, digest: &[u8; 32]) -> Result<DataBlob, Error> {
 | 
				
			||||||
        self.store.load_chunk(digest)
 | 
					        let chunk = self.store.load_chunk(digest)?;
 | 
				
			||||||
 | 
					        self.ensure_crypt_mode(chunk.crypt_mode()?)?;
 | 
				
			||||||
 | 
					        Ok(chunk)
 | 
				
			||||||
    }
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    fn read_chunk(&self, digest: &[u8; 32]) -> Result<Vec<u8>, Error> {
 | 
					    fn read_chunk(&self, digest: &[u8; 32]) -> Result<Vec<u8>, Error> {
 | 
				
			||||||
@ -71,6 +92,7 @@ impl AsyncReadChunk for LocalChunkReader {
 | 
				
			|||||||
            let raw_data = tokio::fs::read(&path).await?;
 | 
					            let raw_data = tokio::fs::read(&path).await?;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
            let chunk = DataBlob::load_from_reader(&mut &raw_data[..])?;
 | 
					            let chunk = DataBlob::load_from_reader(&mut &raw_data[..])?;
 | 
				
			||||||
 | 
					            self.ensure_crypt_mode(chunk.crypt_mode()?)?;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
            Ok(chunk)
 | 
					            Ok(chunk)
 | 
				
			||||||
        })
 | 
					        })
 | 
				
			||||||
 | 
				
			|||||||
@ -1182,6 +1182,7 @@ fn complete_backup_source(arg: &str, param: &HashMap<String, String>) -> Vec<Str
 | 
				
			|||||||
async fn dump_image<W: Write>(
 | 
					async fn dump_image<W: Write>(
 | 
				
			||||||
    client: Arc<BackupReader>,
 | 
					    client: Arc<BackupReader>,
 | 
				
			||||||
    crypt_config: Option<Arc<CryptConfig>>,
 | 
					    crypt_config: Option<Arc<CryptConfig>>,
 | 
				
			||||||
 | 
					    crypt_mode: CryptMode,
 | 
				
			||||||
    index: FixedIndexReader,
 | 
					    index: FixedIndexReader,
 | 
				
			||||||
    mut writer: W,
 | 
					    mut writer: W,
 | 
				
			||||||
    verbose: bool,
 | 
					    verbose: bool,
 | 
				
			||||||
@ -1189,7 +1190,7 @@ async fn dump_image<W: Write>(
 | 
				
			|||||||
 | 
					
 | 
				
			||||||
    let most_used = index.find_most_used_chunks(8);
 | 
					    let most_used = index.find_most_used_chunks(8);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    let chunk_reader = RemoteChunkReader::new(client.clone(), crypt_config, most_used);
 | 
					    let chunk_reader = RemoteChunkReader::new(client.clone(), crypt_config, crypt_mode, most_used);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    // Note: we avoid using BufferedFixedReader, because that add an additional buffer/copy
 | 
					    // Note: we avoid using BufferedFixedReader, because that add an additional buffer/copy
 | 
				
			||||||
    // and thus slows down reading. Instead, directly use RemoteChunkReader
 | 
					    // and thus slows down reading. Instead, directly use RemoteChunkReader
 | 
				
			||||||
@ -1340,7 +1341,12 @@ async fn restore(param: Value) -> Result<Value, Error> {
 | 
				
			|||||||
                .map_err(|err| format_err!("unable to pipe data - {}", err))?;
 | 
					                .map_err(|err| format_err!("unable to pipe data - {}", err))?;
 | 
				
			||||||
        }
 | 
					        }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    } else if archive_type == ArchiveType::Blob {
 | 
					        return Ok(Value::Null);
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    let file_info = manifest.lookup_file_info(&archive_name)?;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    if archive_type == ArchiveType::Blob {
 | 
				
			||||||
 | 
					
 | 
				
			||||||
        let mut reader = client.download_blob(&manifest, &archive_name).await?;
 | 
					        let mut reader = client.download_blob(&manifest, &archive_name).await?;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
@ -1365,7 +1371,7 @@ async fn restore(param: Value) -> Result<Value, Error> {
 | 
				
			|||||||
 | 
					
 | 
				
			||||||
        let most_used = index.find_most_used_chunks(8);
 | 
					        let most_used = index.find_most_used_chunks(8);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
        let chunk_reader = RemoteChunkReader::new(client.clone(), crypt_config, most_used);
 | 
					        let chunk_reader = RemoteChunkReader::new(client.clone(), crypt_config, file_info.chunk_crypt_mode(), most_used);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
        let mut reader = BufferedDynamicReader::new(index, chunk_reader);
 | 
					        let mut reader = BufferedDynamicReader::new(index, chunk_reader);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
@ -1412,7 +1418,7 @@ async fn restore(param: Value) -> Result<Value, Error> {
 | 
				
			|||||||
                .map_err(|err| format_err!("unable to open /dev/stdout - {}", err))?
 | 
					                .map_err(|err| format_err!("unable to open /dev/stdout - {}", err))?
 | 
				
			||||||
        };
 | 
					        };
 | 
				
			||||||
 | 
					
 | 
				
			||||||
        dump_image(client.clone(), crypt_config.clone(), index, &mut writer, verbose).await?;
 | 
					        dump_image(client.clone(), crypt_config.clone(), file_info.chunk_crypt_mode(), index, &mut writer, verbose).await?;
 | 
				
			||||||
    }
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    Ok(Value::Null)
 | 
					    Ok(Value::Null)
 | 
				
			||||||
 | 
				
			|||||||
@ -97,7 +97,9 @@ async fn dump_catalog(param: Value) -> Result<Value, Error> {
 | 
				
			|||||||
 | 
					
 | 
				
			||||||
    let most_used = index.find_most_used_chunks(8);
 | 
					    let most_used = index.find_most_used_chunks(8);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    let chunk_reader = RemoteChunkReader::new(client.clone(), crypt_config, most_used);
 | 
					    let file_info = manifest.lookup_file_info(&CATALOG_NAME)?;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    let chunk_reader = RemoteChunkReader::new(client.clone(), crypt_config, file_info.chunk_crypt_mode(), most_used);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    let mut reader = BufferedDynamicReader::new(index, chunk_reader);
 | 
					    let mut reader = BufferedDynamicReader::new(index, chunk_reader);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
@ -200,7 +202,9 @@ async fn catalog_shell(param: Value) -> Result<(), Error> {
 | 
				
			|||||||
 | 
					
 | 
				
			||||||
    let index = client.download_dynamic_index(&manifest, &server_archive_name).await?;
 | 
					    let index = client.download_dynamic_index(&manifest, &server_archive_name).await?;
 | 
				
			||||||
    let most_used = index.find_most_used_chunks(8);
 | 
					    let most_used = index.find_most_used_chunks(8);
 | 
				
			||||||
    let chunk_reader = RemoteChunkReader::new(client.clone(), crypt_config.clone(), most_used);
 | 
					
 | 
				
			||||||
 | 
					    let file_info = manifest.lookup_file_info(&server_archive_name)?;
 | 
				
			||||||
 | 
					    let chunk_reader = RemoteChunkReader::new(client.clone(), crypt_config.clone(), file_info.chunk_crypt_mode(), most_used);
 | 
				
			||||||
    let reader = BufferedDynamicReader::new(index, chunk_reader);
 | 
					    let reader = BufferedDynamicReader::new(index, chunk_reader);
 | 
				
			||||||
    let archive_size = reader.archive_size();
 | 
					    let archive_size = reader.archive_size();
 | 
				
			||||||
    let reader: proxmox_backup::pxar::fuse::Reader =
 | 
					    let reader: proxmox_backup::pxar::fuse::Reader =
 | 
				
			||||||
@ -216,7 +220,9 @@ async fn catalog_shell(param: Value) -> Result<(), Error> {
 | 
				
			|||||||
    manifest.verify_file(CATALOG_NAME, &csum, size)?;
 | 
					    manifest.verify_file(CATALOG_NAME, &csum, size)?;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    let most_used = index.find_most_used_chunks(8);
 | 
					    let most_used = index.find_most_used_chunks(8);
 | 
				
			||||||
    let chunk_reader = RemoteChunkReader::new(client.clone(), crypt_config, most_used);
 | 
					
 | 
				
			||||||
 | 
					    let file_info = manifest.lookup_file_info(&CATALOG_NAME)?;
 | 
				
			||||||
 | 
					    let chunk_reader = RemoteChunkReader::new(client.clone(), crypt_config, file_info.chunk_crypt_mode(), most_used);
 | 
				
			||||||
    let mut reader = BufferedDynamicReader::new(index, chunk_reader);
 | 
					    let mut reader = BufferedDynamicReader::new(index, chunk_reader);
 | 
				
			||||||
    let mut catalogfile = std::fs::OpenOptions::new()
 | 
					    let mut catalogfile = std::fs::OpenOptions::new()
 | 
				
			||||||
        .write(true)
 | 
					        .write(true)
 | 
				
			||||||
 | 
				
			|||||||
@ -141,10 +141,12 @@ async fn mount_do(param: Value, pipe: Option<RawFd>) -> Result<Value, Error> {
 | 
				
			|||||||
 | 
					
 | 
				
			||||||
    let (manifest, _) = client.download_manifest().await?;
 | 
					    let (manifest, _) = client.download_manifest().await?;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    let file_info = manifest.lookup_file_info(&archive_name)?;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    if server_archive_name.ends_with(".didx") {
 | 
					    if server_archive_name.ends_with(".didx") {
 | 
				
			||||||
        let index = client.download_dynamic_index(&manifest, &server_archive_name).await?;
 | 
					        let index = client.download_dynamic_index(&manifest, &server_archive_name).await?;
 | 
				
			||||||
        let most_used = index.find_most_used_chunks(8);
 | 
					        let most_used = index.find_most_used_chunks(8);
 | 
				
			||||||
        let chunk_reader = RemoteChunkReader::new(client.clone(), crypt_config, most_used);
 | 
					        let chunk_reader = RemoteChunkReader::new(client.clone(), crypt_config, file_info.chunk_crypt_mode(), most_used);
 | 
				
			||||||
        let reader = BufferedDynamicReader::new(index, chunk_reader);
 | 
					        let reader = BufferedDynamicReader::new(index, chunk_reader);
 | 
				
			||||||
        let archive_size = reader.archive_size();
 | 
					        let archive_size = reader.archive_size();
 | 
				
			||||||
        let reader: proxmox_backup::pxar::fuse::Reader =
 | 
					        let reader: proxmox_backup::pxar::fuse::Reader =
 | 
				
			||||||
 | 
				
			|||||||
@ -224,8 +224,6 @@ async fn pull_snapshot(
 | 
				
			|||||||
 | 
					
 | 
				
			||||||
    let manifest = BackupManifest::try_from(tmp_manifest_blob)?;
 | 
					    let manifest = BackupManifest::try_from(tmp_manifest_blob)?;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    let mut chunk_reader = RemoteChunkReader::new(reader.clone(), None, HashMap::new());
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
    for item in manifest.files() {
 | 
					    for item in manifest.files() {
 | 
				
			||||||
        let mut path = tgt_store.base_path();
 | 
					        let mut path = tgt_store.base_path();
 | 
				
			||||||
        path.push(snapshot.relative_path());
 | 
					        path.push(snapshot.relative_path());
 | 
				
			||||||
@ -266,6 +264,8 @@ async fn pull_snapshot(
 | 
				
			|||||||
            }
 | 
					            }
 | 
				
			||||||
        }
 | 
					        }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        let mut chunk_reader = RemoteChunkReader::new(reader.clone(), None, item.chunk_crypt_mode(), HashMap::new());
 | 
				
			||||||
 | 
					
 | 
				
			||||||
        pull_single_archive(
 | 
					        pull_single_archive(
 | 
				
			||||||
            worker,
 | 
					            worker,
 | 
				
			||||||
            &reader,
 | 
					            &reader,
 | 
				
			||||||
 | 
				
			|||||||
@ -3,10 +3,10 @@ use std::collections::HashMap;
 | 
				
			|||||||
use std::pin::Pin;
 | 
					use std::pin::Pin;
 | 
				
			||||||
use std::sync::{Arc, Mutex};
 | 
					use std::sync::{Arc, Mutex};
 | 
				
			||||||
 | 
					
 | 
				
			||||||
use anyhow::Error;
 | 
					use anyhow::{bail, Error};
 | 
				
			||||||
 | 
					
 | 
				
			||||||
use super::BackupReader;
 | 
					use super::BackupReader;
 | 
				
			||||||
use crate::backup::{AsyncReadChunk, CryptConfig, DataBlob, ReadChunk};
 | 
					use crate::backup::{AsyncReadChunk, CryptConfig, CryptMode, DataBlob, ReadChunk};
 | 
				
			||||||
use crate::tools::runtime::block_on;
 | 
					use crate::tools::runtime::block_on;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
/// Read chunks from remote host using ``BackupReader``
 | 
					/// Read chunks from remote host using ``BackupReader``
 | 
				
			||||||
@ -14,6 +14,7 @@ use crate::tools::runtime::block_on;
 | 
				
			|||||||
pub struct RemoteChunkReader {
 | 
					pub struct RemoteChunkReader {
 | 
				
			||||||
    client: Arc<BackupReader>,
 | 
					    client: Arc<BackupReader>,
 | 
				
			||||||
    crypt_config: Option<Arc<CryptConfig>>,
 | 
					    crypt_config: Option<Arc<CryptConfig>>,
 | 
				
			||||||
 | 
					    crypt_mode: CryptMode,
 | 
				
			||||||
    cache_hint: HashMap<[u8; 32], usize>,
 | 
					    cache_hint: HashMap<[u8; 32], usize>,
 | 
				
			||||||
    cache: Arc<Mutex<HashMap<[u8; 32], Vec<u8>>>>,
 | 
					    cache: Arc<Mutex<HashMap<[u8; 32], Vec<u8>>>>,
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
@ -25,11 +26,13 @@ impl RemoteChunkReader {
 | 
				
			|||||||
    pub fn new(
 | 
					    pub fn new(
 | 
				
			||||||
        client: Arc<BackupReader>,
 | 
					        client: Arc<BackupReader>,
 | 
				
			||||||
        crypt_config: Option<Arc<CryptConfig>>,
 | 
					        crypt_config: Option<Arc<CryptConfig>>,
 | 
				
			||||||
 | 
					        crypt_mode: CryptMode,
 | 
				
			||||||
        cache_hint: HashMap<[u8; 32], usize>,
 | 
					        cache_hint: HashMap<[u8; 32], usize>,
 | 
				
			||||||
    ) -> Self {
 | 
					    ) -> Self {
 | 
				
			||||||
        Self {
 | 
					        Self {
 | 
				
			||||||
            client,
 | 
					            client,
 | 
				
			||||||
            crypt_config,
 | 
					            crypt_config,
 | 
				
			||||||
 | 
					            crypt_mode,
 | 
				
			||||||
            cache_hint,
 | 
					            cache_hint,
 | 
				
			||||||
            cache: Arc::new(Mutex::new(HashMap::new())),
 | 
					            cache: Arc::new(Mutex::new(HashMap::new())),
 | 
				
			||||||
        }
 | 
					        }
 | 
				
			||||||
@ -46,7 +49,20 @@ impl RemoteChunkReader {
 | 
				
			|||||||
 | 
					
 | 
				
			||||||
        let chunk = DataBlob::load_from_reader(&mut &chunk_data[..])?;
 | 
					        let chunk = DataBlob::load_from_reader(&mut &chunk_data[..])?;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
        Ok(chunk)
 | 
					        match self.crypt_mode {
 | 
				
			||||||
 | 
					            CryptMode::Encrypt => {
 | 
				
			||||||
 | 
					                match chunk.crypt_mode()? {
 | 
				
			||||||
 | 
					                    CryptMode::Encrypt => Ok(chunk),
 | 
				
			||||||
 | 
					                    CryptMode::SignOnly | CryptMode::None => bail!("Index and chunk CryptMode don't match."),
 | 
				
			||||||
 | 
					                }
 | 
				
			||||||
 | 
					            },
 | 
				
			||||||
 | 
					            CryptMode::SignOnly | CryptMode::None => {
 | 
				
			||||||
 | 
					                match chunk.crypt_mode()? {
 | 
				
			||||||
 | 
					                    CryptMode::Encrypt => bail!("Index and chunk CryptMode don't match."),
 | 
				
			||||||
 | 
					                    CryptMode::SignOnly | CryptMode::None => Ok(chunk),
 | 
				
			||||||
 | 
					                }
 | 
				
			||||||
 | 
					            },
 | 
				
			||||||
 | 
					        }
 | 
				
			||||||
    }
 | 
					    }
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
				
			|||||||
		Reference in New Issue
	
	Block a user