aws_types/os_shim_internal.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 210 211 212 213 214 215 216 217 218 219 220 221 222 223 224 225 226 227 228 229 230 231 232 233 234 235 236 237 238 239 240 241 242 243 244 245 246 247 248 249 250 251 252 253 254 255 256 257 258 259 260 261 262 263 264 265 266 267 268 269 270 271 272 273 274 275 276 277 278 279 280 281 282 283 284 285 286 287 288 289 290 291 292 293 294 295 296 297 298 299 300 301 302 303 304 305 306 307 308 309 310 311 312 313 314 315 316 317 318 319 320 321 322 323 324 325 326 327 328
/*
* Copyright Amazon.com, Inc. or its affiliates. All Rights Reserved.
* SPDX-License-Identifier: Apache-2.0
*/
//! Abstractions for testing code that interacts with the operating system:
//! - Reading environment variables
//! - Reading from the file system
use std::collections::HashMap;
use std::env::VarError;
use std::ffi::OsString;
use std::fmt::Debug;
use std::path::{Path, PathBuf};
use std::sync::{Arc, Mutex};
use crate::os_shim_internal::fs::Fake;
/// File system abstraction
///
/// Simple abstraction enabling in-memory mocking of the file system
///
/// # Examples
/// Construct a file system which delegates to `std::fs`:
/// ```rust
/// let fs = aws_types::os_shim_internal::Fs::real();
/// ```
///
/// Construct an in-memory file system for testing:
/// ```rust
/// use std::collections::HashMap;
/// let fs = aws_types::os_shim_internal::Fs::from_map({
/// let mut map = HashMap::new();
/// map.insert("/home/.aws/config".to_string(), "[default]\nregion = us-east-1");
/// map
/// });
/// ```
#[derive(Clone, Debug)]
pub struct Fs(fs::Inner);
impl Default for Fs {
fn default() -> Self {
Fs::real()
}
}
impl Fs {
/// Create `Fs` representing a real file system.
pub fn real() -> Self {
Fs(fs::Inner::Real)
}
/// Create `Fs` from a map of `OsString` to `Vec<u8>`.
pub fn from_raw_map(fs: HashMap<OsString, Vec<u8>>) -> Self {
Fs(fs::Inner::Fake(Arc::new(Fake::MapFs(Mutex::new(fs)))))
}
/// Create `Fs` from a map of `String` to `Vec<u8>`.
pub fn from_map(data: HashMap<String, impl Into<Vec<u8>>>) -> Self {
let fs = data
.into_iter()
.map(|(k, v)| (k.into(), v.into()))
.collect();
Self::from_raw_map(fs)
}
/// Create a test filesystem rooted in real files
///
/// Creates a test filesystem from the contents of `test_directory` rooted into `namespaced_to`.
///
/// Example:
/// Given:
/// ```bash
/// $ ls
/// ./my-test-dir/aws-config
/// ./my-test-dir/aws-config/config
/// $ cat ./my-test-dir/aws-config/config
/// test-config
/// ```
/// ```rust,no_run
/// # async fn docs() {
/// use aws_types::os_shim_internal::{Env, Fs};
/// let env = Env::from_slice(&[("HOME", "/Users/me")]);
/// let fs = Fs::from_test_dir("my-test-dir/aws-config", "/Users/me/.aws/config");
/// assert_eq!(fs.read_to_end("/Users/me/.aws/config").await.unwrap(), b"test-config");
/// # }
pub fn from_test_dir(
test_directory: impl Into<PathBuf>,
namespaced_to: impl Into<PathBuf>,
) -> Self {
Self(fs::Inner::Fake(Arc::new(Fake::NamespacedFs {
real_path: test_directory.into(),
namespaced_to: namespaced_to.into(),
})))
}
/// Create a fake process environment from a slice of tuples.
///
/// # Examples
/// ```rust
/// # async fn example() {
/// use aws_types::os_shim_internal::Fs;
/// let mock_fs = Fs::from_slice(&[
/// ("config", "[default]\nretry_mode = \"standard\""),
/// ]);
/// assert_eq!(mock_fs.read_to_end("config").await.unwrap(), b"[default]\nretry_mode = \"standard\"");
/// # }
/// ```
pub fn from_slice<'a>(files: &[(&'a str, &'a str)]) -> Self {
let fs: HashMap<String, Vec<u8>> = files
.iter()
.map(|(k, v)| {
let k = (*k).to_owned();
let v = v.as_bytes().to_vec();
(k, v)
})
.collect();
Self::from_map(fs)
}
/// Read the entire contents of a file
///
/// _Note: This function is currently `async` primarily for forward compatibility. Currently,
/// this function does not use Tokio (or any other runtime) to perform IO, the IO is performed
/// directly within the function._
pub async fn read_to_end(&self, path: impl AsRef<Path>) -> std::io::Result<Vec<u8>> {
use fs::Inner;
let path = path.as_ref();
match &self.0 {
// TODO(https://github.com/awslabs/aws-sdk-rust/issues/867): Use async IO below
Inner::Real => std::fs::read(path),
Inner::Fake(fake) => match fake.as_ref() {
Fake::MapFs(fs) => fs
.lock()
.unwrap()
.get(path.as_os_str())
.cloned()
.ok_or_else(|| std::io::ErrorKind::NotFound.into()),
Fake::NamespacedFs {
real_path,
namespaced_to,
} => {
let actual_path = path
.strip_prefix(namespaced_to)
.map_err(|_| std::io::Error::from(std::io::ErrorKind::NotFound))?;
std::fs::read(real_path.join(actual_path))
}
},
}
}
/// Write a slice as the entire contents of a file.
///
/// This is equivalent to `std::fs::write`.
pub async fn write(
&self,
path: impl AsRef<Path>,
contents: impl AsRef<[u8]>,
) -> std::io::Result<()> {
use fs::Inner;
match &self.0 {
// TODO(https://github.com/awslabs/aws-sdk-rust/issues/867): Use async IO below
Inner::Real => {
std::fs::write(path, contents)?;
}
Inner::Fake(fake) => match fake.as_ref() {
Fake::MapFs(fs) => {
fs.lock()
.unwrap()
.insert(path.as_ref().as_os_str().into(), contents.as_ref().to_vec());
}
Fake::NamespacedFs {
real_path,
namespaced_to,
} => {
let actual_path = path
.as_ref()
.strip_prefix(namespaced_to)
.map_err(|_| std::io::Error::from(std::io::ErrorKind::NotFound))?;
std::fs::write(real_path.join(actual_path), contents)?;
}
},
}
Ok(())
}
}
mod fs {
use std::collections::HashMap;
use std::ffi::OsString;
use std::path::PathBuf;
use std::sync::{Arc, Mutex};
#[derive(Clone, Debug)]
pub(super) enum Inner {
Real,
Fake(Arc<Fake>),
}
#[derive(Debug)]
pub(super) enum Fake {
MapFs(Mutex<HashMap<OsString, Vec<u8>>>),
NamespacedFs {
real_path: PathBuf,
namespaced_to: PathBuf,
},
}
}
/// Environment variable abstraction
///
/// Environment variables are global to a process, and, as such, are difficult to test with a multi-
/// threaded test runner like Rust's. This enables loading environment variables either from the
/// actual process environment ([`std::env::var`]) or from a hash map.
///
/// Process environments are cheap to clone:
/// - Faked process environments are wrapped in an internal Arc
/// - Real process environments are pointer-sized
#[derive(Clone, Debug)]
pub struct Env(env::Inner);
impl Default for Env {
fn default() -> Self {
Self::real()
}
}
impl Env {
/// Retrieve a value for the given `k` and return `VarError` is that key is not present.
pub fn get(&self, k: &str) -> Result<String, VarError> {
use env::Inner;
match &self.0 {
Inner::Real => std::env::var(k),
Inner::Fake(map) => map.get(k).cloned().ok_or(VarError::NotPresent),
}
}
/// Create a fake process environment from a slice of tuples.
///
/// # Examples
/// ```rust
/// use aws_types::os_shim_internal::Env;
/// let mock_env = Env::from_slice(&[
/// ("HOME", "/home/myname"),
/// ("AWS_REGION", "us-west-2")
/// ]);
/// assert_eq!(mock_env.get("HOME").unwrap(), "/home/myname");
/// ```
pub fn from_slice<'a>(vars: &[(&'a str, &'a str)]) -> Self {
let map: HashMap<_, _> = vars
.iter()
.map(|(k, v)| (k.to_string(), v.to_string()))
.collect();
Self::from(map)
}
/// Create a process environment that uses the real process environment
///
/// Calls will be delegated to [`std::env::var`].
pub fn real() -> Self {
Self(env::Inner::Real)
}
}
impl From<HashMap<String, String>> for Env {
fn from(hash_map: HashMap<String, String>) -> Self {
Self(env::Inner::Fake(Arc::new(hash_map)))
}
}
mod env {
use std::collections::HashMap;
use std::sync::Arc;
#[derive(Clone, Debug)]
pub(super) enum Inner {
Real,
Fake(Arc<HashMap<String, String>>),
}
}
#[cfg(test)]
mod test {
use std::env::VarError;
use crate::os_shim_internal::{Env, Fs};
#[test]
fn env_works() {
let env = Env::from_slice(&[("FOO", "BAR")]);
assert_eq!(env.get("FOO").unwrap(), "BAR");
assert_eq!(
env.get("OTHER").expect_err("no present"),
VarError::NotPresent
)
}
#[tokio::test]
async fn fs_from_test_dir_works() {
let fs = Fs::from_test_dir(".", "/users/test-data");
let _ = fs
.read_to_end("/users/test-data/Cargo.toml")
.await
.expect("file exists");
let _ = fs
.read_to_end("doesntexist")
.await
.expect_err("file doesnt exists");
}
#[tokio::test]
async fn fs_round_trip_file_with_real() {
let temp = tempfile::tempdir().unwrap();
let path = temp.path().join("test-file");
let fs = Fs::real();
fs.read_to_end(&path)
.await
.expect_err("file doesn't exist yet");
fs.write(&path, b"test").await.expect("success");
let result = fs.read_to_end(&path).await.expect("success");
assert_eq!(b"test", &result[..]);
}
}