Skip to content

Commit

Permalink
Add D-Bus method to get filesystem metadata
Browse files Browse the repository at this point in the history
Signed-off-by: mulhern <[email protected]>
  • Loading branch information
mulkieran committed Sep 24, 2024
1 parent 9f927a3 commit a82ca07
Show file tree
Hide file tree
Showing 10 changed files with 169 additions and 4 deletions.
1 change: 1 addition & 0 deletions src/dbus_api/pool/mod.rs
Original file line number Diff line number Diff line change
Expand Up @@ -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))
Expand Down
15 changes: 14 additions & 1 deletion src/dbus_api/pool/pool_3_7/api.rs
Original file line number Diff line number Diff line change
Expand Up @@ -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,
};

Expand Down Expand Up @@ -34,3 +34,16 @@ pub fn get_metadata_method(f: &Factory<MTSync<TData>, TData>) -> Method<MTSync<T
.out_arg(("return_code", "q"))
.out_arg(("return_string", "s"))
}

pub fn get_fs_metadata_method(f: &Factory<MTSync<TData>, TData>) -> Method<MTSync<TData>, 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"))
}
46 changes: 45 additions & 1 deletion src/dbus_api/pool/pool_3_7/methods.rs
Original file line number Diff line number Diff line change
Expand Up @@ -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},
};
Expand Down Expand Up @@ -134,3 +134,47 @@ pub fn metadata(m: &MethodInfo<'_, MTSync<TData>, TData>) -> MethodResult {
};
Ok(vec![msg])
}

pub fn fs_metadata(m: &MethodInfo<'_, MTSync<TData>, 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])
}
2 changes: 1 addition & 1 deletion src/dbus_api/pool/pool_3_7/mod.rs
Original file line number Diff line number Diff line change
Expand Up @@ -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};
6 changes: 6 additions & 0 deletions src/engine/engine.rs
Original file line number Diff line number Diff line change
Expand Up @@ -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<String>;

/// Get the filesystem metadata that would be written if written now.
fn current_fs_metadata(&self, fs_name: Option<&str>) -> StratisResult<String>;

/// Get the last written filesystem metadata.
fn last_fs_metadata(&self, fs_name: Option<&str>) -> StratisResult<String>;
}

pub type HandleEvents<P> = (
Expand Down
28 changes: 27 additions & 1 deletion src/engine/sim_engine/filesystem.rs
Original file line number Diff line number Diff line change
Expand Up @@ -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<Sectors>,
#[serde(skip_serializing_if = "Option::is_none")]
origin: Option<FilesystemUuid>,
}

#[derive(Debug)]
pub struct SimFilesystem {
rand: u32,
Expand Down Expand Up @@ -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 {
Expand Down
23 changes: 23 additions & 0 deletions src/engine/sim_engine/pool.rs
Original file line number Diff line number Diff line change
Expand Up @@ -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("<name>")).map_err(|e| e.into())
}

fn current_fs_metadata(&self, fs_name: Option<&str>) -> StratisResult<String> {
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::<HashMap<_, _>>(),
)
.map_err(|e| e.into())
}

fn last_fs_metadata(&self, fs_name: Option<&str>) -> StratisResult<String> {
// 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)]
Expand Down
8 changes: 8 additions & 0 deletions src/engine/strat_engine/pool.rs
Original file line number Diff line number Diff line change
Expand Up @@ -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<String> {
self.thin_pool.current_fs_metadata(fs_name)
}

fn last_fs_metadata(&self, fs_name: Option<&str>) -> StratisResult<String> {
self.thin_pool.last_fs_metadata(fs_name)
}
}

pub struct StratPoolState {
Expand Down
37 changes: 37 additions & 0 deletions src/engine/strat_engine/thinpool/thinpool.rs
Original file line number Diff line number Diff line change
Expand Up @@ -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<String> {
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::<HashMap<_, _>>(),
)
.map_err(|e| e.into())
}

/// Read filesystem metadata from mdv
pub fn last_fs_metadata(&self, fs_name: Option<&str>) -> StratisResult<String> {
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::<HashMap<_, _>>(),
)
.map_err(|e| e.into())
}
}

impl<'a> Into<Value> for &'a ThinPool {
Expand Down
7 changes: 7 additions & 0 deletions tests/client-dbus/src/stratisd_client_dbus/_introspect.py
Original file line number Diff line number Diff line change
Expand Up @@ -175,6 +175,13 @@
<arg name="return_code" type="q" direction="out" />
<arg name="return_string" type="s" direction="out" />
</method>
<method name="FilesystemMetadata">
<arg name="fs_name" type="(bs)" direction="in" />
<arg name="current" type="b" direction="in" />
<arg name="results" type="s" direction="out" />
<arg name="return_code" type="q" direction="out" />
<arg name="return_string" type="s" direction="out" />
</method>
<method name="GrowPhysicalDevice">
<arg name="dev" type="s" direction="in" />
<arg name="results" type="b" direction="out" />
Expand Down

0 comments on commit a82ca07

Please sign in to comment.