file-restore: add 'extract' command for VM file restore
The data on the restore daemon is either encoded into a pxar archive, to provide the most accurate data for local restore, or encoded directly into a zip file (or written out unprocessed for files), depending on the 'pxar' argument to the 'extract' API call. Signed-off-by: Stefan Reiter <s.reiter@proxmox.com>
This commit is contained in:
		
				
					committed by
					
						 Thomas Lamprecht
						Thomas Lamprecht
					
				
			
			
				
	
			
			
			
						parent
						
							1f03196c0b
						
					
				
				
					commit
					b13089cdf5
				
			| @ -65,7 +65,7 @@ syslog = "4.0" | |||||||
| tokio = { version = "1.0", features = [ "fs", "io-util", "io-std", "macros", "net", "parking_lot", "process", "rt", "rt-multi-thread", "signal", "time" ] } | tokio = { version = "1.0", features = [ "fs", "io-util", "io-std", "macros", "net", "parking_lot", "process", "rt", "rt-multi-thread", "signal", "time" ] } | ||||||
| tokio-openssl = "0.6.1" | tokio-openssl = "0.6.1" | ||||||
| tokio-stream = "0.1.0" | tokio-stream = "0.1.0" | ||||||
| tokio-util = { version = "0.6", features = [ "codec" ] } | tokio-util = { version = "0.6", features = [ "codec", "io" ] } | ||||||
| tower-service = "0.3.0" | tower-service = "0.3.0" | ||||||
| udev = ">= 0.3, <0.5" | udev = ">= 0.3, <0.5" | ||||||
| url = "2.1" | url = "2.1" | ||||||
|  | |||||||
							
								
								
									
										1
									
								
								debian/control
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										1
									
								
								debian/control
									
									
									
									
										vendored
									
									
								
							| @ -68,6 +68,7 @@ Build-Depends: debhelper (>= 11), | |||||||
|  librust-tokio-stream-0.1+default-dev, |  librust-tokio-stream-0.1+default-dev, | ||||||
|  librust-tokio-util-0.6+codec-dev, |  librust-tokio-util-0.6+codec-dev, | ||||||
|  librust-tokio-util-0.6+default-dev, |  librust-tokio-util-0.6+default-dev, | ||||||
|  |  librust-tokio-util-0.6+io-dev, | ||||||
|  librust-tower-service-0.3+default-dev, |  librust-tower-service-0.3+default-dev, | ||||||
|  librust-udev-0.4+default-dev | librust-udev-0.3+default-dev, |  librust-udev-0.4+default-dev | librust-udev-0.3+default-dev, | ||||||
|  librust-url-2+default-dev (>= 2.1-~~), |  librust-url-2+default-dev (>= 2.1-~~), | ||||||
|  | |||||||
| @ -14,6 +14,7 @@ use proxmox::api::{ | |||||||
|     }, |     }, | ||||||
| }; | }; | ||||||
| use pxar::accessor::aio::Accessor; | use pxar::accessor::aio::Accessor; | ||||||
|  | use pxar::decoder::aio::Decoder; | ||||||
|  |  | ||||||
| use proxmox_backup::api2::{helpers, types::ArchiveEntry}; | use proxmox_backup::api2::{helpers, types::ArchiveEntry}; | ||||||
| use proxmox_backup::backup::{ | use proxmox_backup::backup::{ | ||||||
| @ -21,7 +22,7 @@ use proxmox_backup::backup::{ | |||||||
|     DirEntryAttribute, IndexFile, LocalDynamicReadAt, CATALOG_NAME, |     DirEntryAttribute, IndexFile, LocalDynamicReadAt, CATALOG_NAME, | ||||||
| }; | }; | ||||||
| use proxmox_backup::client::{BackupReader, RemoteChunkReader}; | use proxmox_backup::client::{BackupReader, RemoteChunkReader}; | ||||||
| use proxmox_backup::pxar::{create_zip, extract_sub_dir}; | use proxmox_backup::pxar::{create_zip, extract_sub_dir, extract_sub_dir_seq}; | ||||||
| use proxmox_backup::tools; | use proxmox_backup::tools; | ||||||
|  |  | ||||||
| // use "pub" so rust doesn't complain about "unused" functions in the module | // use "pub" so rust doesn't complain about "unused" functions in the module | ||||||
| @ -277,7 +278,11 @@ async fn list( | |||||||
|                description: "Print verbose information", |                description: "Print verbose information", | ||||||
|                optional: true, |                optional: true, | ||||||
|                default: false, |                default: false, | ||||||
|            } |            }, | ||||||
|  |            "driver": { | ||||||
|  |                type: BlockDriverType, | ||||||
|  |                optional: true, | ||||||
|  |            }, | ||||||
|        } |        } | ||||||
|    } |    } | ||||||
| )] | )] | ||||||
| @ -314,20 +319,21 @@ async fn extract( | |||||||
|         } |         } | ||||||
|     }; |     }; | ||||||
|  |  | ||||||
|  |     let client = connect(&repo)?; | ||||||
|  |     let client = BackupReader::start( | ||||||
|  |         client, | ||||||
|  |         crypt_config.clone(), | ||||||
|  |         repo.store(), | ||||||
|  |         &snapshot.group().backup_type(), | ||||||
|  |         &snapshot.group().backup_id(), | ||||||
|  |         snapshot.backup_time(), | ||||||
|  |         true, | ||||||
|  |     ) | ||||||
|  |     .await?; | ||||||
|  |     let (manifest, _) = client.download_manifest().await?; | ||||||
|  |  | ||||||
|     match path { |     match path { | ||||||
|         ExtractPath::Pxar(archive_name, path) => { |         ExtractPath::Pxar(archive_name, path) => { | ||||||
|             let client = connect(&repo)?; |  | ||||||
|             let client = BackupReader::start( |  | ||||||
|                 client, |  | ||||||
|                 crypt_config.clone(), |  | ||||||
|                 repo.store(), |  | ||||||
|                 &snapshot.group().backup_type(), |  | ||||||
|                 &snapshot.group().backup_id(), |  | ||||||
|                 snapshot.backup_time(), |  | ||||||
|                 true, |  | ||||||
|             ) |  | ||||||
|             .await?; |  | ||||||
|             let (manifest, _) = client.download_manifest().await?; |  | ||||||
|             let file_info = manifest.lookup_file_info(&archive_name)?; |             let file_info = manifest.lookup_file_info(&archive_name)?; | ||||||
|             let index = client |             let index = client | ||||||
|                 .download_dynamic_index(&manifest, &archive_name) |                 .download_dynamic_index(&manifest, &archive_name) | ||||||
| @ -344,31 +350,33 @@ async fn extract( | |||||||
|             let archive_size = reader.archive_size(); |             let archive_size = reader.archive_size(); | ||||||
|             let reader = LocalDynamicReadAt::new(reader); |             let reader = LocalDynamicReadAt::new(reader); | ||||||
|             let decoder = Accessor::new(reader, archive_size).await?; |             let decoder = Accessor::new(reader, archive_size).await?; | ||||||
|  |             extract_to_target(decoder, &path, target, verbose).await?; | ||||||
|  |         } | ||||||
|  |         ExtractPath::VM(file, path) => { | ||||||
|  |             let details = SnapRestoreDetails { | ||||||
|  |                 manifest, | ||||||
|  |                 repo, | ||||||
|  |                 snapshot, | ||||||
|  |             }; | ||||||
|  |             let driver: Option<BlockDriverType> = match param.get("driver") { | ||||||
|  |                 Some(drv) => Some(serde_json::from_value(drv.clone())?), | ||||||
|  |                 None => None, | ||||||
|  |             }; | ||||||
|  |  | ||||||
|             let root = decoder.open_root().await?; |             if let Some(mut target) = target { | ||||||
|             let file = root |                 let reader = data_extract(driver, details, file, path.clone(), true).await?; | ||||||
|                 .lookup(OsStr::from_bytes(&path)) |                 let decoder = Decoder::from_tokio(reader).await?; | ||||||
|                 .await? |                 extract_sub_dir_seq(&target, decoder, verbose).await?; | ||||||
|                 .ok_or(format_err!("error opening '{:?}'", path))?; |  | ||||||
|  |  | ||||||
|             if let Some(target) = target { |                 // we extracted a .pxarexclude-cli file auto-generated by the VM when encoding the | ||||||
|                 extract_sub_dir(target, decoder, OsStr::from_bytes(&path), verbose).await?; |                 // archive, this file is of no use for the user, so try to remove it | ||||||
|  |                 target.push(".pxarexclude-cli"); | ||||||
|  |                 std::fs::remove_file(target).map_err(|e| { | ||||||
|  |                     format_err!("unable to remove temporary .pxarexclude-cli file - {}", e) | ||||||
|  |                 })?; | ||||||
|             } else { |             } else { | ||||||
|                 match file.kind() { |                 let mut reader = data_extract(driver, details, file, path.clone(), false).await?; | ||||||
|                     pxar::EntryKind::File { .. } => { |                 tokio::io::copy(&mut reader, &mut tokio::io::stdout()).await?; | ||||||
|                         tokio::io::copy(&mut file.contents().await?, &mut tokio::io::stdout()) |  | ||||||
|                             .await?; |  | ||||||
|                     } |  | ||||||
|                     _ => { |  | ||||||
|                         create_zip( |  | ||||||
|                             tokio::io::stdout(), |  | ||||||
|                             decoder, |  | ||||||
|                             OsStr::from_bytes(&path), |  | ||||||
|                             verbose, |  | ||||||
|                         ) |  | ||||||
|                         .await?; |  | ||||||
|                     } |  | ||||||
|                 } |  | ||||||
|             } |             } | ||||||
|         } |         } | ||||||
|         _ => { |         _ => { | ||||||
| @ -379,6 +387,43 @@ async fn extract( | |||||||
|     Ok(()) |     Ok(()) | ||||||
| } | } | ||||||
|  |  | ||||||
|  | async fn extract_to_target<T>( | ||||||
|  |     decoder: Accessor<T>, | ||||||
|  |     path: &[u8], | ||||||
|  |     target: Option<PathBuf>, | ||||||
|  |     verbose: bool, | ||||||
|  | ) -> Result<(), Error> | ||||||
|  | where | ||||||
|  |     T: pxar::accessor::ReadAt + Clone + Send + Sync + Unpin + 'static, | ||||||
|  | { | ||||||
|  |     let root = decoder.open_root().await?; | ||||||
|  |     let file = root | ||||||
|  |         .lookup(OsStr::from_bytes(&path)) | ||||||
|  |         .await? | ||||||
|  |         .ok_or_else(|| format_err!("error opening '{:?}'", path))?; | ||||||
|  |  | ||||||
|  |     if let Some(target) = target { | ||||||
|  |         extract_sub_dir(target, decoder, OsStr::from_bytes(&path), verbose).await?; | ||||||
|  |     } else { | ||||||
|  |         match file.kind() { | ||||||
|  |             pxar::EntryKind::File { .. } => { | ||||||
|  |                 tokio::io::copy(&mut file.contents().await?, &mut tokio::io::stdout()).await?; | ||||||
|  |             } | ||||||
|  |             _ => { | ||||||
|  |                 create_zip( | ||||||
|  |                     tokio::io::stdout(), | ||||||
|  |                     decoder, | ||||||
|  |                     OsStr::from_bytes(&path), | ||||||
|  |                     verbose, | ||||||
|  |                 ) | ||||||
|  |                 .await?; | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     Ok(()) | ||||||
|  | } | ||||||
|  |  | ||||||
| fn main() { | fn main() { | ||||||
|     let list_cmd_def = CliCommand::new(&API_METHOD_LIST) |     let list_cmd_def = CliCommand::new(&API_METHOD_LIST) | ||||||
|         .arg_param(&["snapshot", "path"]) |         .arg_param(&["snapshot", "path"]) | ||||||
|  | |||||||
| @ -41,6 +41,19 @@ pub trait BlockRestoreDriver { | |||||||
|         path: Vec<u8>, |         path: Vec<u8>, | ||||||
|     ) -> Async<Result<Vec<ArchiveEntry>, Error>>; |     ) -> Async<Result<Vec<ArchiveEntry>, Error>>; | ||||||
|  |  | ||||||
|  |     /// pxar=true: | ||||||
|  |     /// Attempt to create a pxar archive of the given file path and return a reader instance for it | ||||||
|  |     /// pxar=false: | ||||||
|  |     /// Attempt to read the file or folder at the given path and return the file content or a zip | ||||||
|  |     /// file as a stream | ||||||
|  |     fn data_extract( | ||||||
|  |         &self, | ||||||
|  |         details: SnapRestoreDetails, | ||||||
|  |         img_file: String, | ||||||
|  |         path: Vec<u8>, | ||||||
|  |         pxar: bool, | ||||||
|  |     ) -> Async<Result<Box<dyn tokio::io::AsyncRead + Unpin + Send>, Error>>; | ||||||
|  |  | ||||||
|     /// Return status of all running/mapped images, result value is (id, extra data), where id must |     /// Return status of all running/mapped images, result value is (id, extra data), where id must | ||||||
|     /// match with the ones returned from list() |     /// match with the ones returned from list() | ||||||
|     fn status(&self) -> Async<Result<Vec<DriverStatus>, Error>>; |     fn status(&self) -> Async<Result<Vec<DriverStatus>, Error>>; | ||||||
| @ -79,6 +92,17 @@ pub async fn data_list( | |||||||
|     driver.data_list(details, img_file, path).await |     driver.data_list(details, img_file, path).await | ||||||
| } | } | ||||||
|  |  | ||||||
|  | pub async fn data_extract( | ||||||
|  |     driver: Option<BlockDriverType>, | ||||||
|  |     details: SnapRestoreDetails, | ||||||
|  |     img_file: String, | ||||||
|  |     path: Vec<u8>, | ||||||
|  |     pxar: bool, | ||||||
|  | ) -> Result<Box<dyn tokio::io::AsyncRead + Send + Unpin>, Error> { | ||||||
|  |     let driver = driver.unwrap_or(DEFAULT_DRIVER).resolve(); | ||||||
|  |     driver.data_extract(details, img_file, path, pxar).await | ||||||
|  | } | ||||||
|  |  | ||||||
| #[api( | #[api( | ||||||
|    input: { |    input: { | ||||||
|        properties: { |        properties: { | ||||||
|  | |||||||
| @ -204,6 +204,38 @@ impl BlockRestoreDriver for QemuBlockDriver { | |||||||
|         .boxed() |         .boxed() | ||||||
|     } |     } | ||||||
|  |  | ||||||
|  |     fn data_extract( | ||||||
|  |         &self, | ||||||
|  |         details: SnapRestoreDetails, | ||||||
|  |         img_file: String, | ||||||
|  |         mut path: Vec<u8>, | ||||||
|  |         pxar: bool, | ||||||
|  |     ) -> Async<Result<Box<dyn tokio::io::AsyncRead + Unpin + Send>, Error>> { | ||||||
|  |         async move { | ||||||
|  |             let client = ensure_running(&details).await?; | ||||||
|  |             if !path.is_empty() && path[0] != b'/' { | ||||||
|  |                 path.insert(0, b'/'); | ||||||
|  |             } | ||||||
|  |             let path = base64::encode(img_file.bytes().chain(path).collect::<Vec<u8>>()); | ||||||
|  |             let (mut tx, rx) = tokio::io::duplex(1024 * 4096); | ||||||
|  |             tokio::spawn(async move { | ||||||
|  |                 if let Err(err) = client | ||||||
|  |                     .download( | ||||||
|  |                         "api2/json/extract", | ||||||
|  |                         Some(json!({ "path": path, "pxar": pxar })), | ||||||
|  |                         &mut tx, | ||||||
|  |                     ) | ||||||
|  |                     .await | ||||||
|  |                 { | ||||||
|  |                     eprintln!("reading file extraction stream failed - {}", err); | ||||||
|  |                 } | ||||||
|  |             }); | ||||||
|  |  | ||||||
|  |             Ok(Box::new(rx) as Box<dyn tokio::io::AsyncRead + Unpin + Send>) | ||||||
|  |         } | ||||||
|  |         .boxed() | ||||||
|  |     } | ||||||
|  |  | ||||||
|     fn status(&self) -> Async<Result<Vec<DriverStatus>, Error>> { |     fn status(&self) -> Async<Result<Vec<DriverStatus>, Error>> { | ||||||
|         async move { |         async move { | ||||||
|             let mut state_map = VMStateMap::load()?; |             let mut state_map = VMStateMap::load()?; | ||||||
|  | |||||||
| @ -1,16 +1,29 @@ | |||||||
| ///! File-restore API running inside the restore VM | ///! File-restore API running inside the restore VM | ||||||
| use anyhow::{bail, Error}; | use anyhow::{bail, Error}; | ||||||
|  | use futures::FutureExt; | ||||||
|  | use hyper::http::request::Parts; | ||||||
|  | use hyper::{header, Body, Response, StatusCode}; | ||||||
|  | use log::error; | ||||||
|  | use pathpatterns::{MatchEntry, MatchPattern, MatchType, Pattern}; | ||||||
|  | use serde_json::Value; | ||||||
|  |  | ||||||
| use std::ffi::OsStr; | use std::ffi::OsStr; | ||||||
| use std::fs; | use std::fs; | ||||||
| use std::os::unix::ffi::OsStrExt; | use std::os::unix::ffi::OsStrExt; | ||||||
| use std::path::{Path, PathBuf}; | use std::path::{Path, PathBuf}; | ||||||
|  |  | ||||||
| use proxmox::api::{api, ApiMethod, Permission, Router, RpcEnvironment, SubdirMap}; | use proxmox::api::{ | ||||||
| use proxmox::list_subdirs_api_method; |     api, schema::*, ApiHandler, ApiMethod, ApiResponseFuture, Permission, Router, RpcEnvironment, | ||||||
|  |     SubdirMap, | ||||||
|  | }; | ||||||
|  | use proxmox::{identity, list_subdirs_api_method, sortable}; | ||||||
|  |  | ||||||
| use proxmox_backup::api2::types::*; | use proxmox_backup::api2::types::*; | ||||||
| use proxmox_backup::backup::DirEntryAttribute; | use proxmox_backup::backup::DirEntryAttribute; | ||||||
| use proxmox_backup::tools::fs::read_subdir; | use proxmox_backup::pxar::{create_archive, Flags, PxarCreateOptions, ENCODER_MAX_ENTRIES}; | ||||||
|  | use proxmox_backup::tools::{self, fs::read_subdir, zip::zip_directory}; | ||||||
|  |  | ||||||
|  | use pxar::encoder::aio::TokioWriter; | ||||||
|  |  | ||||||
| use super::{disk::ResolveResult, watchdog_remaining, watchdog_ping}; | use super::{disk::ResolveResult, watchdog_remaining, watchdog_ping}; | ||||||
|  |  | ||||||
| @ -18,6 +31,7 @@ use super::{disk::ResolveResult, watchdog_remaining, watchdog_ping}; | |||||||
| // not exist within the restore VM. Safety is guaranteed by checking a ticket via a custom ApiAuth. | // not exist within the restore VM. Safety is guaranteed by checking a ticket via a custom ApiAuth. | ||||||
|  |  | ||||||
| const SUBDIRS: SubdirMap = &[ | const SUBDIRS: SubdirMap = &[ | ||||||
|  |     ("extract", &Router::new().get(&API_METHOD_EXTRACT)), | ||||||
|     ("list", &Router::new().get(&API_METHOD_LIST)), |     ("list", &Router::new().get(&API_METHOD_LIST)), | ||||||
|     ("status", &Router::new().get(&API_METHOD_STATUS)), |     ("status", &Router::new().get(&API_METHOD_STATUS)), | ||||||
|     ("stop", &Router::new().get(&API_METHOD_STOP)), |     ("stop", &Router::new().get(&API_METHOD_STOP)), | ||||||
| @ -197,3 +211,159 @@ fn list( | |||||||
|  |  | ||||||
|     Ok(res) |     Ok(res) | ||||||
| } | } | ||||||
|  |  | ||||||
|  | #[sortable] | ||||||
|  | pub const API_METHOD_EXTRACT: ApiMethod = ApiMethod::new( | ||||||
|  |     &ApiHandler::AsyncHttp(&extract), | ||||||
|  |     &ObjectSchema::new( | ||||||
|  |         "Extract a file or directory from the VM as a pxar archive.", | ||||||
|  |         &sorted!([ | ||||||
|  |             ( | ||||||
|  |                 "path", | ||||||
|  |                 false, | ||||||
|  |                 &StringSchema::new("base64-encoded path to list files and directories under") | ||||||
|  |                     .schema() | ||||||
|  |             ), | ||||||
|  |             ( | ||||||
|  |                 "pxar", | ||||||
|  |                 true, | ||||||
|  |                 &BooleanSchema::new(concat!( | ||||||
|  |                     "if true, return a pxar archive, otherwise either the ", | ||||||
|  |                     "file content or the directory as a zip file" | ||||||
|  |                 )) | ||||||
|  |                 .default(true) | ||||||
|  |                 .schema() | ||||||
|  |             ) | ||||||
|  |         ]), | ||||||
|  |     ), | ||||||
|  | ) | ||||||
|  | .access(None, &Permission::Superuser); | ||||||
|  |  | ||||||
|  | fn extract( | ||||||
|  |     _parts: Parts, | ||||||
|  |     _req_body: Body, | ||||||
|  |     param: Value, | ||||||
|  |     _info: &ApiMethod, | ||||||
|  |     _rpcenv: Box<dyn RpcEnvironment>, | ||||||
|  | ) -> ApiResponseFuture { | ||||||
|  |     watchdog_ping(); | ||||||
|  |     async move { | ||||||
|  |         let path = tools::required_string_param(¶m, "path")?; | ||||||
|  |         let mut path = base64::decode(path)?; | ||||||
|  |         if let Some(b'/') = path.last() { | ||||||
|  |             path.pop(); | ||||||
|  |         } | ||||||
|  |         let path = Path::new(OsStr::from_bytes(&path[..])); | ||||||
|  |  | ||||||
|  |         let pxar = param["pxar"].as_bool().unwrap_or(true); | ||||||
|  |  | ||||||
|  |         let query_result = { | ||||||
|  |             let mut disk_state = crate::DISK_STATE.lock().unwrap(); | ||||||
|  |             disk_state.resolve(&path)? | ||||||
|  |         }; | ||||||
|  |  | ||||||
|  |         let vm_path = match query_result { | ||||||
|  |             ResolveResult::Path(vm_path) => vm_path, | ||||||
|  |             _ => bail!("invalid path, cannot restore meta-directory: {:?}", path), | ||||||
|  |         }; | ||||||
|  |  | ||||||
|  |         // check here so we can return a real error message, failing in the async task will stop | ||||||
|  |         // the transfer, but not return a useful message | ||||||
|  |         if !vm_path.exists() { | ||||||
|  |             bail!("file or directory {:?} does not exist", path); | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         let (mut writer, reader) = tokio::io::duplex(1024 * 64); | ||||||
|  |  | ||||||
|  |         if pxar { | ||||||
|  |             tokio::spawn(async move { | ||||||
|  |                 let result = async move { | ||||||
|  |                     // pxar always expects a directory as it's root, so to accommodate files as | ||||||
|  |                     // well we encode the parent dir with a filter only matching the target instead | ||||||
|  |                     let mut patterns = vec![MatchEntry::new( | ||||||
|  |                         MatchPattern::Pattern(Pattern::path(b"*").unwrap()), | ||||||
|  |                         MatchType::Exclude, | ||||||
|  |                     )]; | ||||||
|  |  | ||||||
|  |                     let name = match vm_path.file_name() { | ||||||
|  |                         Some(name) => name, | ||||||
|  |                         None => bail!("no file name found for path: {:?}", vm_path), | ||||||
|  |                     }; | ||||||
|  |  | ||||||
|  |                     if vm_path.is_dir() { | ||||||
|  |                         let mut pat = name.as_bytes().to_vec(); | ||||||
|  |                         patterns.push(MatchEntry::new( | ||||||
|  |                             MatchPattern::Pattern(Pattern::path(pat.clone())?), | ||||||
|  |                             MatchType::Include, | ||||||
|  |                         )); | ||||||
|  |                         pat.extend(b"/**/*".iter()); | ||||||
|  |                         patterns.push(MatchEntry::new( | ||||||
|  |                             MatchPattern::Pattern(Pattern::path(pat)?), | ||||||
|  |                             MatchType::Include, | ||||||
|  |                         )); | ||||||
|  |                     } else { | ||||||
|  |                         patterns.push(MatchEntry::new( | ||||||
|  |                             MatchPattern::Literal(name.as_bytes().to_vec()), | ||||||
|  |                             MatchType::Include, | ||||||
|  |                         )); | ||||||
|  |                     } | ||||||
|  |  | ||||||
|  |                     let dir_path = vm_path.parent().unwrap_or_else(|| Path::new("/")); | ||||||
|  |                     let dir = nix::dir::Dir::open( | ||||||
|  |                         dir_path, | ||||||
|  |                         nix::fcntl::OFlag::O_NOFOLLOW, | ||||||
|  |                         nix::sys::stat::Mode::empty(), | ||||||
|  |                     )?; | ||||||
|  |  | ||||||
|  |                     let options = PxarCreateOptions { | ||||||
|  |                         entries_max: ENCODER_MAX_ENTRIES, | ||||||
|  |                         device_set: None, | ||||||
|  |                         patterns, | ||||||
|  |                         verbose: false, | ||||||
|  |                         skip_lost_and_found: false, | ||||||
|  |                     }; | ||||||
|  |  | ||||||
|  |                     let pxar_writer = TokioWriter::new(writer); | ||||||
|  |                     create_archive(dir, pxar_writer, Flags::DEFAULT, |_| Ok(()), None, options) | ||||||
|  |                         .await | ||||||
|  |                 } | ||||||
|  |                 .await; | ||||||
|  |                 if let Err(err) = result { | ||||||
|  |                     error!("pxar streaming task failed - {}", err); | ||||||
|  |                 } | ||||||
|  |             }); | ||||||
|  |         } else { | ||||||
|  |             tokio::spawn(async move { | ||||||
|  |                 let result = async move { | ||||||
|  |                     if vm_path.is_dir() { | ||||||
|  |                         zip_directory(&mut writer, &vm_path).await?; | ||||||
|  |                         Ok(()) | ||||||
|  |                     } else if vm_path.is_file() { | ||||||
|  |                         let mut file = tokio::fs::OpenOptions::new() | ||||||
|  |                             .read(true) | ||||||
|  |                             .open(vm_path) | ||||||
|  |                             .await?; | ||||||
|  |                         tokio::io::copy(&mut file, &mut writer).await?; | ||||||
|  |                         Ok(()) | ||||||
|  |                     } else { | ||||||
|  |                         bail!("invalid entry type for path: {:?}", vm_path); | ||||||
|  |                     } | ||||||
|  |                 } | ||||||
|  |                 .await; | ||||||
|  |                 if let Err(err) = result { | ||||||
|  |                     error!("file or dir streaming task failed - {}", err); | ||||||
|  |                 } | ||||||
|  |             }); | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         let stream = tokio_util::io::ReaderStream::new(reader); | ||||||
|  |  | ||||||
|  |         let body = Body::wrap_stream(stream); | ||||||
|  |         Ok(Response::builder() | ||||||
|  |             .status(StatusCode::OK) | ||||||
|  |             .header(header::CONTENT_TYPE, "application/octet-stream") | ||||||
|  |             .body(body) | ||||||
|  |             .unwrap()) | ||||||
|  |     } | ||||||
|  |     .boxed() | ||||||
|  | } | ||||||
|  | |||||||
		Reference in New Issue
	
	Block a user