1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
|
use std::path::{Path, PathBuf};
/// A file system entry.
///
/// Represent all backup-relevant the metadata about a file system
/// object: fully qualified pathname, type, length (if applicable),
/// etc. Everything except the content of a regular file or the
/// contents of a directory.
///
/// This is everything Obnam cares about each file system object, when
/// making a backup.
#[derive(Debug)]
pub struct FilesystemEntry {
kind: FilesystemKind,
path: PathBuf,
len: u64,
}
impl FilesystemEntry {
fn new(kind: FilesystemKind, path: &Path, len: u64) -> Self {
Self {
path: path.to_path_buf(),
kind,
len,
}
}
pub fn regular<P>(path: P, len: u64) -> Self
where
P: AsRef<Path>,
{
Self::new(FilesystemKind::Regular, path.as_ref(), len)
}
pub fn directory<P>(path: P) -> Self
where
P: AsRef<Path>,
{
Self::new(FilesystemKind::Directory, path.as_ref(), 0)
}
pub fn kind(&self) -> FilesystemKind {
self.kind
}
pub fn path(&self) -> &Path {
&self.path
}
pub fn len(&self) -> u64 {
self.len
}
}
/// Different types of file system entries.
#[derive(Debug, Copy, Clone, PartialEq)]
pub enum FilesystemKind {
Regular,
Directory,
}
impl FilesystemKind {
pub fn as_code(&self) -> u8 {
match self {
FilesystemKind::Regular => 0,
FilesystemKind::Directory => 1,
}
}
pub fn from_code(code: u8) -> anyhow::Result<Self> {
match code {
0 => Ok(FilesystemKind::Regular),
1 => Ok(FilesystemKind::Directory),
_ => Err(Error::UnknownFileKindCode(code).into()),
}
}
}
#[derive(Debug, thiserror::Error)]
pub enum Error {
#[error("unknown file kind code {0}")]
UnknownFileKindCode(u8),
}
#[cfg(test)]
mod test {
use super::{FilesystemEntry, FilesystemKind};
use std::path::Path;
#[test]
fn regular_file() {
let filename = Path::new("foo.dat");
let len = 123;
let e = FilesystemEntry::regular(filename, len);
assert_eq!(e.kind(), FilesystemKind::Regular);
assert_eq!(e.path(), filename);
assert_eq!(e.len(), len);
}
#[test]
fn directory() {
let filename = Path::new("foo.dat");
let e = FilesystemEntry::directory(filename);
assert_eq!(e.kind(), FilesystemKind::Directory);
assert_eq!(e.path(), filename);
assert_eq!(e.len(), 0);
}
#[test]
fn file_kind_regular_round_trips() {
one_file_kind_round_trip(FilesystemKind::Regular);
one_file_kind_round_trip(FilesystemKind::Directory);
}
fn one_file_kind_round_trip(kind: FilesystemKind) {
assert_eq!(kind, FilesystemKind::from_code(kind.as_code()).unwrap());
}
}
|