-
-
Notifications
You must be signed in to change notification settings - Fork 7
/
Copy pathexample.rs
209 lines (176 loc) · 6.37 KB
/
example.rs
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
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
// Copyright (c) The datatest-stable Contributors
// SPDX-License-Identifier: MIT OR Apache-2.0
use camino::Utf8Path;
use datatest_stable::Result;
use std::{fs::File, io::Read, path::Path};
fn test_artifact(path: &Path) -> Result<()> {
let mut file = File::open(path)?;
let mut contents = String::new();
file.read_to_string(&mut contents)?;
Ok(())
}
fn test_artifact_utf8(path: &Utf8Path) -> Result<()> {
test_artifact(path.as_ref())
}
fn test_artifact_utf8_abs(path: &Utf8Path) -> Result<()> {
// path must be absolute
assert!(path.is_absolute(), "path must be absolute: {:?}", path);
// On Windows, the path must be normalized to use backslashes.
#[cfg(windows)]
{
assert!(
!path.as_str().contains('/'),
"path must not contain forward slashes: {:?}",
path
);
}
test_artifact(path.as_ref())
}
#[cfg(feature = "include-dir")]
#[macro_use]
mod with_contents {
use super::*;
/// Returns an `include_dir::Dir` instance.
macro_rules! maybe_include_dir {
() => {
include_dir::include_dir!("$CARGO_MANIFEST_DIR/tests/files")
};
}
/// A `&'static include_dir::Dir` instance.
pub(crate) static MAYBE_INCLUDE_STATIC: include_dir::Dir =
include_dir::include_dir!("$CARGO_MANIFEST_DIR/tests/files");
pub(crate) fn test_artifact_string(path: &Path, contents: String) -> Result<()> {
compare_contents(path, contents.as_bytes())
}
pub(crate) fn test_artifact_utf8_string(path: &Utf8Path, contents: String) -> Result<()> {
compare_contents(path.as_std_path(), contents.as_bytes())
}
pub(crate) fn test_artifact_bytes(path: &Path, contents: Vec<u8>) -> Result<()> {
compare_contents(path, &contents)
}
pub(crate) fn test_artifact_utf8_bytes(path: &Utf8Path, contents: Vec<u8>) -> Result<()> {
compare_contents(path.as_std_path(), &contents)
}
fn compare_contents(path: &Path, expected: &[u8]) -> Result<()> {
// The path must not begin with "tests/files".
assert!(
!path.to_string_lossy().starts_with("tests/files"),
"path must not start with 'tests/files': {:?}",
path
);
// Prepend tests/files to the path to get the expected contents. In
// general we can't verify the contents, but in this case we can do so
// because the paths are also available on disk.
let path = format!("tests/files/{}", path.to_str().unwrap());
compare(path.as_ref(), expected)
}
}
#[cfg(not(feature = "include-dir"))]
#[macro_use]
mod with_contents {
use super::*;
/// Returns an `include_dir::Dir` instance.
macro_rules! maybe_include_dir {
() => {
"tests/files"
};
}
/// A `&'static include_dir::Dir` instance.
pub(crate) static MAYBE_INCLUDE_STATIC: &str = "tests/files";
pub(crate) fn test_artifact_string(path: &Path, contents: String) -> Result<()> {
compare_contents(path, contents.as_bytes())
}
pub(crate) fn test_artifact_utf8_string(path: &Utf8Path, contents: String) -> Result<()> {
compare_contents(path.as_std_path(), contents.as_bytes())
}
pub(crate) fn test_artifact_bytes(path: &Path, contents: Vec<u8>) -> Result<()> {
compare_contents(path, &contents)
}
pub(crate) fn test_artifact_utf8_bytes(path: &Utf8Path, contents: Vec<u8>) -> Result<()> {
compare_contents(path.as_std_path(), &contents)
}
fn compare_contents(path: &Path, expected: &[u8]) -> Result<()> {
// The path must begin with "tests/files".
assert!(
path.to_string_lossy().starts_with("tests/files"),
"path must start with 'tests/files': {:?}",
path
);
compare(&path, expected)
}
}
fn compare(path: &Path, expected: &[u8]) -> Result<()> {
// The path must be relative.
assert!(path.is_relative(), "path must be relative: {:?}", path);
// The path must not have any backslashes on Windows.
assert!(
!path.to_string_lossy().contains('\\'),
"path must not contain backslashes: {:?}",
path
);
let actual =
std::fs::read(path).map_err(|error| format!("failed to read file: {:?}: {error}", path))?;
assert_eq!(expected, &actual, "file contents match for {:?}", path);
Ok(())
}
#[cfg(windows)]
static TESTS_FILES_MAIN_SEP: &str = "tests\\files";
#[cfg(not(windows))]
static TESTS_FILES_MAIN_SEP: &str = "tests/files";
fn tests_files_abs() -> String {
std::env::current_dir()
.expect("current dir obtained")
.join("tests/files")
.to_string_lossy()
.into_owned()
}
datatest_stable::harness! {
{
test = test_artifact,
root = "tests/files",
// This regex pattern skips .skip.txt files.
pattern = r"^.*(?<!\.skip)\.txt$",
},
{
test = test_artifact_utf8,
// Ensure that tests\files is normalized to tests/files on Windows.
root = TESTS_FILES_MAIN_SEP,
// This regex pattern matches all .txt files.
pattern = r"^.*\.txt$",
},
{
test = test_artifact_utf8_abs,
// Ensure that tests/files in an absolute path is normalized to
// tests\files on Windows.
root = tests_files_abs(),
// This regex matches exactly dir/a.txt, b.txt, and c.skip.txt -- this
// ensures that patterns are relative to the include dir and not the
// crate root.
pattern = r"^(dir/a|b|c\.skip)\.txt$",
},
{
test = with_contents::test_artifact_string,
root = maybe_include_dir!(),
// This regex matches exactly dir/a.txt, b.txt, and c.skip.txt -- this
// ensures that patterns are relative to the include dir and not the
// crate root.
pattern = r"^(dir/a|b|c\.skip)\.txt$",
},
{
test = with_contents::test_artifact_utf8_string,
// Test out some combinations with &'static include_dir::Dir.
root = &with_contents::MAYBE_INCLUDE_STATIC,
pattern = r"^.*\.txt$",
},
{
test = with_contents::test_artifact_bytes,
root = &with_contents::MAYBE_INCLUDE_STATIC,
// No pattern specified, so all files are included, including non-.txt
// files.
},
{
test = with_contents::test_artifact_utf8_bytes,
root = maybe_include_dir!(),
pattern = r"^.*\.txt$",
},
}