diff --git a/src/dbus_api/pool/mod.rs b/src/dbus_api/pool/mod.rs index 3fb20d179d..66d66a9469 100644 --- a/src/dbus_api/pool/mod.rs +++ b/src/dbus_api/pool/mod.rs @@ -260,6 +260,7 @@ pub fn create_dbus_pool<'a>( .add_m(pool_3_0::rename_method(&f)) .add_m(pool_3_3::grow_physical_device_method(&f)) .add_m(pool_3_7::get_metadata_method(&f)) + .add_m(pool_3_7::get_fs_metadata_method(&f)) .add_p(pool_3_0::name_property(&f)) .add_p(pool_3_0::uuid_property(&f)) .add_p(pool_3_0::encrypted_property(&f)) diff --git a/src/dbus_api/pool/pool_3_7/api.rs b/src/dbus_api/pool/pool_3_7/api.rs index 910c0c40e5..0271b02a9a 100644 --- a/src/dbus_api/pool/pool_3_7/api.rs +++ b/src/dbus_api/pool/pool_3_7/api.rs @@ -5,7 +5,7 @@ use dbus_tree::{Factory, MTSync, Method}; use crate::dbus_api::{ - pool::pool_3_7::methods::{destroy_filesystems, metadata}, + pool::pool_3_7::methods::{destroy_filesystems, fs_metadata, metadata}, types::TData, }; @@ -34,3 +34,16 @@ pub fn get_metadata_method(f: &Factory, TData>) -> Method, TData>) -> Method, TData> { + f.method("FilesystemMetadata", (), fs_metadata) + .in_arg(("fs_name", "(bs)")) + .in_arg(("current", "b")) + // A string representing the pool's filesystem metadata in serialized + // JSON format. + // + // Rust representation: String + .out_arg(("results", "s")) + .out_arg(("return_code", "q")) + .out_arg(("return_string", "s")) +} diff --git a/src/dbus_api/pool/pool_3_7/methods.rs b/src/dbus_api/pool/pool_3_7/methods.rs index 587fb8789c..68d8b61b21 100644 --- a/src/dbus_api/pool/pool_3_7/methods.rs +++ b/src/dbus_api/pool/pool_3_7/methods.rs @@ -11,7 +11,7 @@ use crate::{ dbus_api::{ consts::filesystem_interface_list, types::{DbusErrorEnum, TData, OK_STRING}, - util::{engine_to_dbus_err_tuple, get_next_arg}, + util::{engine_to_dbus_err_tuple, get_next_arg, tuple_to_option}, }, engine::{EngineAction, FilesystemUuid}, }; @@ -134,3 +134,47 @@ pub fn metadata(m: &MethodInfo<'_, MTSync, TData>) -> MethodResult { }; Ok(vec![msg]) } + +pub fn fs_metadata(m: &MethodInfo<'_, MTSync, TData>) -> MethodResult { + let default_return = String::new(); + + let message: &Message = m.msg; + + let return_message = message.method_return(); + + let dbus_context = m.tree.get_data(); + let object_path = m.path.get_name(); + + let pool_path = m + .tree + .get(object_path) + .expect("implicit argument must be in tree"); + let pool_uuid = typed_uuid!( + get_data!(pool_path; default_return; return_message).uuid; + Pool; + default_return; + return_message + ); + + let mut iter = message.iter_init(); + let filesystem_name: Option<&str> = tuple_to_option(get_next_arg(&mut iter, 0)?); + let current: bool = get_next_arg(&mut iter, 1)?; + + let guard = get_pool!(dbus_context.engine; pool_uuid; default_return; return_message); + let (_, _, pool) = guard.as_tuple(); + + let result = if current { + pool.current_fs_metadata(filesystem_name) + } else { + pool.last_fs_metadata(filesystem_name) + }; + + let msg = match result { + Ok(v) => return_message.append3(v, DbusErrorEnum::OK as u16, OK_STRING.to_string()), + Err(err) => { + let (rc, rs) = engine_to_dbus_err_tuple(&err); + return_message.append3(default_return, rc, rs) + } + }; + Ok(vec![msg]) +} diff --git a/src/dbus_api/pool/pool_3_7/mod.rs b/src/dbus_api/pool/pool_3_7/mod.rs index 063e004350..49572d6114 100644 --- a/src/dbus_api/pool/pool_3_7/mod.rs +++ b/src/dbus_api/pool/pool_3_7/mod.rs @@ -5,4 +5,4 @@ mod api; mod methods; -pub use api::{destroy_filesystems_method, get_metadata_method}; +pub use api::{destroy_filesystems_method, get_fs_metadata_method, get_metadata_method}; diff --git a/src/engine/engine.rs b/src/engine/engine.rs index 917ea94b82..794488c417 100644 --- a/src/engine/engine.rs +++ b/src/engine/engine.rs @@ -347,6 +347,12 @@ pub trait Pool: Debug + Send + Sync { /// Return the metadata that was last written to pool devices. fn last_metadata(&self) -> StratisResult; + + /// Get the filesystem metadata that would be written if written now. + fn current_fs_metadata(&self, fs_name: Option<&str>) -> StratisResult; + + /// Get the last written filesystem metadata. + fn last_fs_metadata(&self, fs_name: Option<&str>) -> StratisResult; } pub type HandleEvents

= ( diff --git a/src/engine/sim_engine/filesystem.rs b/src/engine/sim_engine/filesystem.rs index 5198f234bf..b6661bab3a 100644 --- a/src/engine/sim_engine/filesystem.rs +++ b/src/engine/sim_engine/filesystem.rs @@ -10,10 +10,25 @@ use serde_json::{Map, Value}; use devicemapper::{Bytes, Sectors}; use crate::{ - engine::{types::FilesystemUuid, Filesystem}, + engine::{ + types::{FilesystemUuid, Name}, + Filesystem, + }, stratis::{StratisError, StratisResult}, }; +#[derive(Debug, Eq, PartialEq, Serialize)] +pub struct FilesystemSave { + name: String, + uuid: FilesystemUuid, + size: Sectors, + created: u64, + #[serde(skip_serializing_if = "Option::is_none")] + fs_size_limit: Option, + #[serde(skip_serializing_if = "Option::is_none")] + origin: Option, +} + #[derive(Debug)] pub struct SimFilesystem { rand: u32, @@ -73,6 +88,17 @@ impl SimFilesystem { self.origin = None; changed } + + pub fn record(&self, name: &Name, uuid: FilesystemUuid) -> FilesystemSave { + FilesystemSave { + name: name.to_owned(), + uuid, + size: self.size, + created: self.created.timestamp() as u64, + fs_size_limit: self.size_limit, + origin: self.origin, + } + } } impl Filesystem for SimFilesystem { diff --git a/src/engine/sim_engine/pool.rs b/src/engine/sim_engine/pool.rs index 6b8441ffec..f6b951a59a 100644 --- a/src/engine/sim_engine/pool.rs +++ b/src/engine/sim_engine/pool.rs @@ -764,6 +764,29 @@ impl Pool for SimPool { // Just invent a name for the pool; a sim pool has no real metadata serde_json::to_string(&self.record("")).map_err(|e| e.into()) } + + fn current_fs_metadata(&self, fs_name: Option<&str>) -> StratisResult { + serde_json::to_string( + &self + .filesystems + .iter() + .filter_map(|(name, uuid, fs)| { + if fs_name.map(|n| *n == **name).unwrap_or(true) { + Some((uuid, fs.record(name, *uuid))) + } else { + None + } + }) + .collect::>(), + ) + .map_err(|e| e.into()) + } + + fn last_fs_metadata(&self, fs_name: Option<&str>) -> StratisResult { + // The sim pool doesn't write data, so the last fs metadata and the + // current fs metadata are, by definition, the same. + self.current_fs_metadata(fs_name) + } } #[cfg(test)] diff --git a/src/engine/strat_engine/pool.rs b/src/engine/strat_engine/pool.rs index 8840d0ea5b..469a881895 100644 --- a/src/engine/strat_engine/pool.rs +++ b/src/engine/strat_engine/pool.rs @@ -1273,6 +1273,14 @@ impl Pool for StratPool { .map_err(|_| StratisError::Msg("metadata byte array is not utf-8".into())) }) } + + fn current_fs_metadata(&self, fs_name: Option<&str>) -> StratisResult { + self.thin_pool.current_fs_metadata(fs_name) + } + + fn last_fs_metadata(&self, fs_name: Option<&str>) -> StratisResult { + self.thin_pool.last_fs_metadata(fs_name) + } } pub struct StratPoolState { diff --git a/src/engine/strat_engine/thinpool/thinpool.rs b/src/engine/strat_engine/thinpool/thinpool.rs index bf2d45df61..5383e049ce 100644 --- a/src/engine/strat_engine/thinpool/thinpool.rs +++ b/src/engine/strat_engine/thinpool/thinpool.rs @@ -1588,6 +1588,43 @@ impl ThinPool { } Ok(changed) } + + /// Calculate filesystem metadata from current state + pub fn current_fs_metadata(&self, fs_name: Option<&str>) -> StratisResult { + serde_json::to_string( + &self + .filesystems + .iter() + .filter_map(|(name, uuid, fs)| { + if fs_name.map(|n| *n == **name).unwrap_or(true) { + Some((*uuid, fs.record(name, *uuid))) + } else { + None + } + }) + .collect::>(), + ) + .map_err(|e| e.into()) + } + + /// Read filesystem metadata from mdv + pub fn last_fs_metadata(&self, fs_name: Option<&str>) -> StratisResult { + serde_json::to_string( + &self + .mdv + .filesystems()? + .iter() + .filter_map(|fssave| { + if fs_name.map(|n| *n == fssave.name).unwrap_or(true) { + Some((fssave.uuid, fssave)) + } else { + None + } + }) + .collect::>(), + ) + .map_err(|e| e.into()) + } } impl<'a> Into for &'a ThinPool { diff --git a/tests/client-dbus/src/stratisd_client_dbus/_introspect.py b/tests/client-dbus/src/stratisd_client_dbus/_introspect.py index 7e899437fe..2c4be82fcd 100644 --- a/tests/client-dbus/src/stratisd_client_dbus/_introspect.py +++ b/tests/client-dbus/src/stratisd_client_dbus/_introspect.py @@ -175,6 +175,13 @@ + + + + + + +