pxar: impl storing/dumping/restoring of quota project ids
Allows to store/dump/restore the quota project id associated with an inode in order to correctly restore project quotas. The project id is obtained/set via ioctl calls getting/setting the fsxattr associated with the given file descriptor. Signed-off-by: Christian Ebner <c.ebner@proxmox.com>
This commit is contained in:
parent
7312ab9eef
commit
e7b60a16c8
@ -9,6 +9,7 @@ use std::collections::HashMap;
|
|||||||
use super::format_definition::*;
|
use super::format_definition::*;
|
||||||
use super::binary_search_tree::*;
|
use super::binary_search_tree::*;
|
||||||
use super::helper::*;
|
use super::helper::*;
|
||||||
|
use crate::tools::fs;
|
||||||
use crate::tools::acl;
|
use crate::tools::acl;
|
||||||
use crate::tools::xattr;
|
use crate::tools::xattr;
|
||||||
|
|
||||||
@ -396,6 +397,34 @@ impl <'a, W: Write> Encoder<'a, W> {
|
|||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/// Read the project quota id for an inode, supported on ext4/XFS/FUSE/(ZFS TODO impl) filesystems
|
||||||
|
fn read_quota_project_id(&self, fd: RawFd, magic: i64, stat: &FileStat) -> Result<Option<CaFormatQuotaProjID>, Error> {
|
||||||
|
if !(is_directory(&stat) || is_reg_file(&stat)) {
|
||||||
|
return Ok(None);
|
||||||
|
}
|
||||||
|
if !self.has_features(CA_FORMAT_WITH_QUOTA_PROJID) {
|
||||||
|
return Ok(None);
|
||||||
|
}
|
||||||
|
let projid = match magic {
|
||||||
|
//TODO ZFS quota
|
||||||
|
EXT4_SUPER_MAGIC | XFS_SUPER_MAGIC | FUSE_SUPER_MAGIC => {
|
||||||
|
let mut fsxattr = fs::FSXAttr::default();
|
||||||
|
unsafe {
|
||||||
|
fs::fs_ioc_fsgetxattr(fd, &mut fsxattr)
|
||||||
|
.map_err(|err| format_err!("error while reading quota project id for {:#?} - {}", self.full_path(), err))?;
|
||||||
|
}
|
||||||
|
fsxattr.fsx_projid as u64
|
||||||
|
},
|
||||||
|
_ => return Ok(None),
|
||||||
|
};
|
||||||
|
println!("Encountered projid: {} for {:#?}", projid, self.full_path());
|
||||||
|
if projid == 0 {
|
||||||
|
return Ok(None);
|
||||||
|
}
|
||||||
|
|
||||||
|
Ok(Some(CaFormatQuotaProjID { projid }))
|
||||||
|
}
|
||||||
|
|
||||||
fn write_entry(&mut self, entry: CaFormatEntry) -> Result<(), Error> {
|
fn write_entry(&mut self, entry: CaFormatEntry) -> Result<(), Error> {
|
||||||
|
|
||||||
self.write_header(CA_FORMAT_ENTRY, std::mem::size_of::<CaFormatEntry>() as u64)?;
|
self.write_header(CA_FORMAT_ENTRY, std::mem::size_of::<CaFormatEntry>() as u64)?;
|
||||||
@ -466,6 +495,13 @@ impl <'a, W: Write> Encoder<'a, W> {
|
|||||||
Ok(())
|
Ok(())
|
||||||
}
|
}
|
||||||
|
|
||||||
|
fn write_quota_project_id(&mut self, projid: CaFormatQuotaProjID) -> Result<(), Error> {
|
||||||
|
self.write_header(CA_FORMAT_QUOTA_PROJID, std::mem::size_of::<CaFormatQuotaProjID>() as u64)?;
|
||||||
|
self.write_item(projid)?;
|
||||||
|
|
||||||
|
Ok(())
|
||||||
|
}
|
||||||
|
|
||||||
fn write_goodbye_table(&mut self, goodbye_offset: usize, goodbye_items: &mut [CaFormatGoodbyeItem]) -> Result<(), Error> {
|
fn write_goodbye_table(&mut self, goodbye_offset: usize, goodbye_items: &mut [CaFormatGoodbyeItem]) -> Result<(), Error> {
|
||||||
|
|
||||||
goodbye_items.sort_unstable_by(|a, b| a.hash.cmp(&b.hash));
|
goodbye_items.sort_unstable_by(|a, b| a.hash.cmp(&b.hash));
|
||||||
@ -522,6 +558,7 @@ impl <'a, W: Write> Encoder<'a, W> {
|
|||||||
let (xattrs, fcaps) = self.read_xattrs(rawfd, &dir_stat)?;
|
let (xattrs, fcaps) = self.read_xattrs(rawfd, &dir_stat)?;
|
||||||
let acl_access = self.read_acl(rawfd, &dir_stat, acl::ACL_TYPE_ACCESS)?;
|
let acl_access = self.read_acl(rawfd, &dir_stat, acl::ACL_TYPE_ACCESS)?;
|
||||||
let acl_default = self.read_acl(rawfd, &dir_stat, acl::ACL_TYPE_DEFAULT)?;
|
let acl_default = self.read_acl(rawfd, &dir_stat, acl::ACL_TYPE_DEFAULT)?;
|
||||||
|
let projid = self.read_quota_project_id(rawfd, magic, &dir_stat)?;
|
||||||
|
|
||||||
self.write_entry(dir_entry)?;
|
self.write_entry(dir_entry)?;
|
||||||
for xattr in xattrs {
|
for xattr in xattrs {
|
||||||
@ -548,6 +585,9 @@ impl <'a, W: Write> Encoder<'a, W> {
|
|||||||
if let Some(default) = acl_default.default {
|
if let Some(default) = acl_default.default {
|
||||||
self.write_acl_default(default)?;
|
self.write_acl_default(default)?;
|
||||||
}
|
}
|
||||||
|
if let Some(projid) = projid {
|
||||||
|
self.write_quota_project_id(projid)?;
|
||||||
|
}
|
||||||
|
|
||||||
let mut dir_count = 0;
|
let mut dir_count = 0;
|
||||||
|
|
||||||
@ -733,6 +773,7 @@ impl <'a, W: Write> Encoder<'a, W> {
|
|||||||
self.read_fat_attr(filefd, magic, &mut entry)?;
|
self.read_fat_attr(filefd, magic, &mut entry)?;
|
||||||
let (xattrs, fcaps) = self.read_xattrs(filefd, &stat)?;
|
let (xattrs, fcaps) = self.read_xattrs(filefd, &stat)?;
|
||||||
let acl_access = self.read_acl(filefd, &stat, acl::ACL_TYPE_ACCESS)?;
|
let acl_access = self.read_acl(filefd, &stat, acl::ACL_TYPE_ACCESS)?;
|
||||||
|
let projid = self.read_quota_project_id(filefd, magic, &stat)?;
|
||||||
|
|
||||||
self.write_entry(entry)?;
|
self.write_entry(entry)?;
|
||||||
for xattr in xattrs {
|
for xattr in xattrs {
|
||||||
@ -748,6 +789,9 @@ impl <'a, W: Write> Encoder<'a, W> {
|
|||||||
if let Some(group_obj) = acl_access.group_obj {
|
if let Some(group_obj) = acl_access.group_obj {
|
||||||
self.write_acl_group_obj(group_obj)?;
|
self.write_acl_group_obj(group_obj)?;
|
||||||
}
|
}
|
||||||
|
if let Some(projid) = projid {
|
||||||
|
self.write_quota_project_id(projid)?;
|
||||||
|
}
|
||||||
|
|
||||||
let include_payload;
|
let include_payload;
|
||||||
if is_virtual_file_system(magic) {
|
if is_virtual_file_system(magic) {
|
||||||
@ -912,6 +956,8 @@ pub const TMPFS_MAGIC: i64 = 0x01021994;
|
|||||||
pub const SYSFS_MAGIC: i64 = 0x62656572;
|
pub const SYSFS_MAGIC: i64 = 0x62656572;
|
||||||
pub const MSDOS_SUPER_MAGIC: i64 = 0x00004d44;
|
pub const MSDOS_SUPER_MAGIC: i64 = 0x00004d44;
|
||||||
pub const FUSE_SUPER_MAGIC: i64 = 0x65735546;
|
pub const FUSE_SUPER_MAGIC: i64 = 0x65735546;
|
||||||
|
pub const EXT4_SUPER_MAGIC: i64 = 0xEF53;
|
||||||
|
pub const XFS_SUPER_MAGIC: i64 = 0x58465342;
|
||||||
|
|
||||||
|
|
||||||
#[inline(always)]
|
#[inline(always)]
|
||||||
|
@ -23,6 +23,7 @@ use nix::NixPath;
|
|||||||
|
|
||||||
use crate::tools::io::ops::*;
|
use crate::tools::io::ops::*;
|
||||||
use crate::tools::vec;
|
use crate::tools::vec;
|
||||||
|
use crate::tools::fs;
|
||||||
use crate::tools::acl;
|
use crate::tools::acl;
|
||||||
use crate::tools::xattr;
|
use crate::tools::xattr;
|
||||||
|
|
||||||
@ -169,6 +170,7 @@ impl <'a, R: Read> SequentialDecoder<'a, R> {
|
|||||||
fn restore_attributes(&mut self, entry: &CaFormatEntry, fd: RawFd) -> Result<CaFormatHeader, Error> {
|
fn restore_attributes(&mut self, entry: &CaFormatEntry, fd: RawFd) -> Result<CaFormatHeader, Error> {
|
||||||
let mut xattrs = Vec::new();
|
let mut xattrs = Vec::new();
|
||||||
let mut fcaps = None;
|
let mut fcaps = None;
|
||||||
|
let mut quota_projid = None;
|
||||||
|
|
||||||
let mut acl_user = Vec::new();
|
let mut acl_user = Vec::new();
|
||||||
let mut acl_group = Vec::new();
|
let mut acl_group = Vec::new();
|
||||||
@ -238,6 +240,13 @@ impl <'a, R: Read> SequentialDecoder<'a, R> {
|
|||||||
self.skip_bytes(size)?;
|
self.skip_bytes(size)?;
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
CA_FORMAT_QUOTA_PROJID => {
|
||||||
|
if self.has_features(CA_FORMAT_WITH_QUOTA_PROJID) {
|
||||||
|
quota_projid = Some(self.read_item::<CaFormatQuotaProjID>()?);
|
||||||
|
} else {
|
||||||
|
self.skip_bytes(size)?;
|
||||||
|
}
|
||||||
|
},
|
||||||
_ => break,
|
_ => break,
|
||||||
}
|
}
|
||||||
head = self.read_item()?;
|
head = self.read_item()?;
|
||||||
@ -288,6 +297,7 @@ impl <'a, R: Read> SequentialDecoder<'a, R> {
|
|||||||
}
|
}
|
||||||
acl.set_file(&proc_path, acl::ACL_TYPE_DEFAULT)?;
|
acl.set_file(&proc_path, acl::ACL_TYPE_DEFAULT)?;
|
||||||
}
|
}
|
||||||
|
self.restore_quota_projid(fd, quota_projid)?;
|
||||||
|
|
||||||
Ok(head)
|
Ok(head)
|
||||||
}
|
}
|
||||||
@ -307,6 +317,23 @@ impl <'a, R: Read> SequentialDecoder<'a, R> {
|
|||||||
Ok(())
|
Ok(())
|
||||||
}
|
}
|
||||||
|
|
||||||
|
fn restore_quota_projid(&mut self, fd: RawFd, projid: Option<CaFormatQuotaProjID>) -> Result<(), Error> {
|
||||||
|
if let Some(projid) = projid {
|
||||||
|
let mut fsxattr = fs::FSXAttr::default();
|
||||||
|
unsafe {
|
||||||
|
fs::fs_ioc_fsgetxattr(fd, &mut fsxattr)
|
||||||
|
.map_err(|err| format_err!("error while getting fsxattr to restore quota project id - {}", err))?;
|
||||||
|
}
|
||||||
|
fsxattr.fsx_projid = projid.projid as u32;
|
||||||
|
unsafe {
|
||||||
|
fs::fs_ioc_fssetxattr(fd, &fsxattr)
|
||||||
|
.map_err(|err| format_err!("error while setting fsxattr to restore quota project id - {}", err))?;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
Ok(())
|
||||||
|
}
|
||||||
|
|
||||||
fn restore_mode(&mut self, entry: &CaFormatEntry, fd: RawFd) -> Result<(), Error> {
|
fn restore_mode(&mut self, entry: &CaFormatEntry, fd: RawFd) -> Result<(), Error> {
|
||||||
|
|
||||||
let mode = Mode::from_bits_truncate((entry.mode as u32) & 0o7777);
|
let mode = Mode::from_bits_truncate((entry.mode as u32) & 0o7777);
|
||||||
@ -827,6 +854,12 @@ impl <'a, R: Read> SequentialDecoder<'a, R> {
|
|||||||
println!("ACLDefaultGroup: {:?}", default_group);
|
println!("ACLDefaultGroup: {:?}", default_group);
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
CA_FORMAT_QUOTA_PROJID => {
|
||||||
|
let quota_projid = self.read_item::<CaFormatQuotaProjID>()?;
|
||||||
|
if verbose && self.has_features(CA_FORMAT_WITH_QUOTA_PROJID) {
|
||||||
|
println!("Quota project id: {:?}", quota_projid);
|
||||||
|
}
|
||||||
|
},
|
||||||
_ => return Ok(false),
|
_ => return Ok(false),
|
||||||
}
|
}
|
||||||
|
|
||||||
|
Loading…
Reference in New Issue
Block a user