mirror of
https://github.com/Steffo99/micronfig.git
synced 2024-11-22 08:04:20 +00:00
Clear the crate and start over
This commit is contained in:
parent
825a20b19a
commit
c6df6cb3af
16 changed files with 0 additions and 1070 deletions
|
@ -1,8 +0,0 @@
|
|||
[package]
|
||||
name = "e01_the_cave"
|
||||
description = "Echoes back the value of ECHO."
|
||||
version = "0.0.0"
|
||||
edition = "2021"
|
||||
|
||||
[dependencies]
|
||||
micronfig = { path = "../.." }
|
|
@ -1,9 +0,0 @@
|
|||
use std::fmt::Display;
|
||||
|
||||
|
||||
micronfig::required!(ECHO, String);
|
||||
|
||||
|
||||
fn main() {
|
||||
println!("{}", *ECHO);
|
||||
}
|
|
@ -1,8 +0,0 @@
|
|||
[package]
|
||||
name = "e02_quick_math"
|
||||
description = "Performs OPERATOR between FIRST and SECOND."
|
||||
version = "0.0.0"
|
||||
edition = "2021"
|
||||
|
||||
[dependencies]
|
||||
micronfig = { path = "../.." }
|
|
@ -1,52 +0,0 @@
|
|||
use std::fmt::{Display, Formatter};
|
||||
use std::str::FromStr;
|
||||
|
||||
|
||||
micronfig::required!(FIRST, u64);
|
||||
micronfig::required!(SECOND, u64);
|
||||
micronfig::required!(OPERATOR, Operator);
|
||||
|
||||
|
||||
fn main() {
|
||||
let result = match *OPERATOR {
|
||||
Operator::Sum => (*FIRST) + (*SECOND),
|
||||
Operator::Subtraction => (*FIRST) - (*SECOND),
|
||||
Operator::Multiplication => (*FIRST) * (*SECOND),
|
||||
Operator::Division => (*FIRST) / (*SECOND),
|
||||
};
|
||||
|
||||
println!("{} {} {} = {}", *FIRST, *OPERATOR, *SECOND, result)
|
||||
}
|
||||
|
||||
|
||||
pub enum Operator {
|
||||
Sum,
|
||||
Subtraction,
|
||||
Multiplication,
|
||||
Division,
|
||||
}
|
||||
|
||||
impl FromStr for Operator {
|
||||
type Err = ();
|
||||
|
||||
fn from_str(s: &str) -> Result<Self, Self::Err> {
|
||||
match s {
|
||||
"+" => Ok(Self::Sum),
|
||||
"-" => Ok(Self::Subtraction),
|
||||
"*" => Ok(Self::Multiplication),
|
||||
"/" => Ok(Self::Division),
|
||||
_ => Err(())
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
impl Display for Operator {
|
||||
fn fmt(&self, f: &mut Formatter<'_>) -> std::fmt::Result {
|
||||
write!(f, "{}", match self {
|
||||
Self::Sum => "+",
|
||||
Self::Subtraction => "-",
|
||||
Self::Multiplication => "*",
|
||||
Self::Division => "/",
|
||||
})
|
||||
}
|
||||
}
|
|
@ -1,8 +0,0 @@
|
|||
[package]
|
||||
name = "e03_order_a_pizza"
|
||||
description = "Order a pizza using micronfig!"
|
||||
version = "0.0.0"
|
||||
edition = "2021"
|
||||
|
||||
[dependencies]
|
||||
micronfig = { path = "../.." }
|
|
@ -1,119 +0,0 @@
|
|||
use std::fmt::Formatter;
|
||||
use std::net::IpAddr;
|
||||
use std::str::FromStr;
|
||||
|
||||
|
||||
// The name of the person who ordered the pizza.
|
||||
micronfig::required!(FULLNAME, String);
|
||||
|
||||
// The (IP) address the pizza should be delivered to.
|
||||
micronfig::required!(DESTINATION, IpAddr);
|
||||
|
||||
// The base of the pizza to add toppings on.
|
||||
micronfig::required!(PIZZABASE, PizzaBase);
|
||||
|
||||
// The toppings to add to the pizza.
|
||||
micronfig::optional!(PIZZATOPPINGS, PizzaToppingsList);
|
||||
// A pizza with no toppings, to use as fallback.
|
||||
const PIZZATOPPINGS_NONE: PizzaToppingsList = PizzaToppingsList{ list: vec![] };
|
||||
|
||||
|
||||
fn main() {
|
||||
// Let's print the order!
|
||||
println!("Pizza Order");
|
||||
println!("===========");
|
||||
println!();
|
||||
println!("Base:");
|
||||
println!("- {}", *PIZZABASE);
|
||||
println!();
|
||||
println!("Toppings:");
|
||||
for topping in &(*PIZZATOPPINGS).as_ref().unwrap_or(&PIZZATOPPINGS_NONE).list {
|
||||
println!("- {}", &topping);
|
||||
};
|
||||
println!();
|
||||
println!("Deliver to:");
|
||||
println!("{} @ {}", *FULLNAME, *DESTINATION)
|
||||
}
|
||||
|
||||
|
||||
/// A possible base of pizza.
|
||||
#[derive(Clone, Copy, Debug)]
|
||||
enum PizzaBase {
|
||||
/// Just the pizza dough, with nothing else on top f it.
|
||||
Blank,
|
||||
/// Pizza dough with tomato on top.
|
||||
Red,
|
||||
/// Pizza dough with mozzarella on top.
|
||||
White,
|
||||
/// Pizza dough with both tomato and mozzarella on top.
|
||||
Margherita,
|
||||
}
|
||||
|
||||
impl FromStr for PizzaBase {
|
||||
type Err = &'static str;
|
||||
|
||||
fn from_str(s: &str) -> Result<Self, Self::Err> {
|
||||
match s.to_lowercase().as_str() {
|
||||
// Italian
|
||||
"vuota" => Ok(Self::Blank),
|
||||
"stria" => Ok(Self::Blank),
|
||||
"rossa" => Ok(Self::Red),
|
||||
"marinara" => Ok(Self::Red),
|
||||
"pomodoro" => Ok(Self::Red),
|
||||
"bianca" => Ok(Self::White),
|
||||
"mozzarella" => Ok(Self::White),
|
||||
"regina" => Ok(Self::Margherita),
|
||||
"margherita" => Ok(Self::Margherita),
|
||||
"normale" => Ok(Self::Margherita),
|
||||
"entrambi" => Ok(Self::Margherita),
|
||||
// English
|
||||
"blank" => Ok(Self::Blank),
|
||||
"red" => Ok(Self::Red),
|
||||
"tomato" => Ok(Self::Red),
|
||||
"white" => Ok(Self::White),
|
||||
"cheese" => Ok(Self::White),
|
||||
"both" => Ok(Self::Margherita),
|
||||
"normal" => Ok(Self::Margherita),
|
||||
// Unknown
|
||||
_ => Err("Unknown pizza base; ensure you have written the name in either English or Italian!"),
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
impl std::fmt::Display for PizzaBase {
|
||||
fn fmt(&self, f: &mut Formatter<'_>) -> std::fmt::Result {
|
||||
write!(f, "{}", match self {
|
||||
PizzaBase::Blank => "Blank (Empty)",
|
||||
PizzaBase::Red => "Red (Tomato)",
|
||||
PizzaBase::White => "White (Mozzarella)",
|
||||
PizzaBase::Margherita => "Margherita (Tomato + Mozzarella)"
|
||||
})
|
||||
}
|
||||
}
|
||||
|
||||
/// The toppings
|
||||
#[derive(Clone, Debug)]
|
||||
struct PizzaToppingsList {
|
||||
pub list: Vec<String>
|
||||
}
|
||||
|
||||
impl FromStr for PizzaToppingsList {
|
||||
type Err = &'static str;
|
||||
|
||||
fn from_str(s: &str) -> Result<Self, Self::Err> {
|
||||
let list: Vec<String> = s.split(",").map(|s| s.to_string()).collect();
|
||||
|
||||
for topping in list.iter() {
|
||||
// Ensure compatibility with https://github.com/rust-lang/rust/pull/70645
|
||||
if ["pineapple", "ananas"].contains(&topping.as_str()) {
|
||||
return Err("Ruining pizzas is not allowed by the Rust compiler.")
|
||||
}
|
||||
}
|
||||
|
||||
Ok(
|
||||
PizzaToppingsList {
|
||||
list
|
||||
}
|
||||
)
|
||||
}
|
||||
}
|
|
@ -1,153 +0,0 @@
|
|||
//! High-level API — Handle errors automatically.
|
||||
//!
|
||||
//! It can be useful if you want to specify when configuration values are loaded in the lifecycle of your binary.
|
||||
|
||||
|
||||
/// Get a value from the first available source, panicking with a human-readable message in case the value is missing or cannot be processed.
|
||||
///
|
||||
/// # Process
|
||||
///
|
||||
/// This function:
|
||||
///
|
||||
/// 1. calls [`crate::multi::get`] with the given key and a file suffix of `_FILE`
|
||||
/// 2. pattern matches errors and [`panic`]s if an error is caught.
|
||||
///
|
||||
/// # Panics
|
||||
///
|
||||
/// Any error encountered by this function causes a panic with a message describing what went wrong.
|
||||
///
|
||||
/// The same thing happens if the configuration value could not be retrieved by any source.
|
||||
///
|
||||
/// # Examples
|
||||
///
|
||||
/// Retrieve a configuration value from either the `USER` environment variable or the `USER_FILE` file, maintaining it as a [`String`]:
|
||||
/// ```
|
||||
/// use micronfig::handle::get_required;
|
||||
/// #
|
||||
/// # std::env::set_var("USER", "steffo");
|
||||
/// # std::env::remove_var("USER_FILE");
|
||||
///
|
||||
/// let user: String = get_required("USER");
|
||||
/// ```
|
||||
///
|
||||
/// Retrieve a configuration value from the `IP_ADDRESS` environment variable or the `IP_ADDRESS_FILE` file, then try to convert it to a [`std::net::IpAddr`]:
|
||||
/// ```
|
||||
/// use std::net::IpAddr;
|
||||
/// use micronfig::handle::get_required;
|
||||
/// #
|
||||
/// # std::env::set_var("IP_ADDRESS", "192.168.1.1");
|
||||
/// # std::env::remove_var("IP_ADDRESS_FILE");
|
||||
///
|
||||
/// let ip_addr: IpAddr = get_required("IP_ADDRESS");
|
||||
/// ```
|
||||
///
|
||||
/// # See also
|
||||
///
|
||||
/// [`get_optional`], which has the same behaviour but does not panic if the value is not found, instead returning [`None`].
|
||||
///
|
||||
/// # Possible future improvements
|
||||
///
|
||||
/// Possibly refactor this to a method of [`crate::multi::Source`].
|
||||
///
|
||||
pub fn get_required<Type>(key: &str) -> Type
|
||||
where Type: std::str::FromStr,
|
||||
<Type as std::str::FromStr>::Err: std::fmt::Debug,
|
||||
{
|
||||
use crate::multi::{get, Source};
|
||||
use crate::single::{envvars, envfiles};
|
||||
|
||||
match get(key, "_FILE") {
|
||||
Source::EnvVar(Ok(v)) => v,
|
||||
Source::EnvVar(Err(envvars::Error::CannotConvertValue(err))) =>
|
||||
panic!("The contents of the {} environment variable could not be converted to a {}: {:?}", &key, &std::any::type_name::<Type>(), &err),
|
||||
Source::EnvVar(Err(envvars::Error::CannotReadEnvVar(_))) =>
|
||||
panic!("Something unexpected happened in micronfig. Please report this as a bug!"),
|
||||
|
||||
Source::EnvFile(Ok(v)) => v,
|
||||
Source::EnvFile(Err(envfiles::Error::CannotConvertValue(err))) =>
|
||||
panic!("The contents of the file at {} could not be converted to a {}: {:?}", &key, &std::any::type_name::<Type>(), &err),
|
||||
Source::EnvFile(Err(envfiles::Error::CannotOpenFile(err))) =>
|
||||
panic!("The file at {} could not be opened: {}", &key, &err),
|
||||
Source::EnvFile(Err(envfiles::Error::CannotReadFile(err))) =>
|
||||
panic!("The contents of the file at {} could not be read: {}", &key, &err),
|
||||
Source::EnvFile(Err(envfiles::Error::CannotReadEnvVar(_))) =>
|
||||
panic!("Something unexpected happened in micronfig. Please report this as a bug!"),
|
||||
|
||||
Source::NotFound =>
|
||||
panic!("The configuration value {} is not defined.", &key),
|
||||
}
|
||||
}
|
||||
|
||||
|
||||
/// Try to get a value from the first available source, panicking with a human-readable message in case it cannot be processed.
|
||||
///
|
||||
/// # Process
|
||||
///
|
||||
/// This function:
|
||||
///
|
||||
/// 1. calls [`crate::multi::get`] with the given key and a file suffix of `_FILE`
|
||||
/// 2. pattern matches errors and [`panic`]s if an error is caught.
|
||||
///
|
||||
/// # Panics
|
||||
///
|
||||
/// Any error encountered by this function causes a panic with a message describing what went wrong.
|
||||
///
|
||||
/// # Examples
|
||||
///
|
||||
/// Retrieve a configuration value from either the `USER` environment variable or the `USER_FILE` file, maintaining it as a [`String`]:
|
||||
/// ```
|
||||
/// use micronfig::handle::get_optional;
|
||||
/// #
|
||||
/// # std::env::set_var("USER", "steffo");
|
||||
/// # std::env::remove_var("USER_FILE");
|
||||
///
|
||||
/// let user: Option<String> = get_optional("USER");
|
||||
/// ```
|
||||
///
|
||||
/// Retrieve a configuration value from the `IP_ADDRESS` environment variable or the `IP_ADDRESS_FILE` file, then try to convert it to a [`std::net::IpAddr`]:
|
||||
/// ```
|
||||
/// use std::net::IpAddr;
|
||||
/// use micronfig::handle::get_optional;
|
||||
/// #
|
||||
/// # std::env::set_var("IP_ADDRESS", "192.168.1.1");
|
||||
/// # std::env::remove_var("IP_ADDRESS_FILE");
|
||||
///
|
||||
/// let ip_addr: Option<IpAddr> = get_optional("IP_ADDRESS");
|
||||
/// ```
|
||||
///
|
||||
/// # See also
|
||||
///
|
||||
/// [`get_required`], which has the same behaviour but does panics if the value is not found.
|
||||
///
|
||||
/// # Possible future improvements
|
||||
///
|
||||
/// Possibly refactor this to a method of [`crate::multi::Source`].
|
||||
///
|
||||
pub fn get_optional<Type>(name: &str) -> Option<Type>
|
||||
where Type: std::str::FromStr,
|
||||
<Type as std::str::FromStr>::Err: std::fmt::Debug,
|
||||
{
|
||||
use crate::multi::{get, Source};
|
||||
use crate::single::{envvars, envfiles};
|
||||
|
||||
match get(name, "_FILE") {
|
||||
Source::EnvVar(Ok(v)) => Some(v),
|
||||
Source::EnvVar(Err(envvars::Error::CannotConvertValue(err))) =>
|
||||
panic!("The contents of the {} environment variable could not be converted to a {}: {:?}", &name, &std::any::type_name::<Type>(), &err),
|
||||
Source::EnvVar(Err(envvars::Error::CannotReadEnvVar(_))) =>
|
||||
panic!("Something unexpected happened in micronfig. Please report this as a bug!"),
|
||||
|
||||
Source::EnvFile(Ok(v)) => Some(v),
|
||||
Source::EnvFile(Err(envfiles::Error::CannotConvertValue(err))) =>
|
||||
panic!("The contents of the file at {} could not be converted to a {}: {:?}", &name, &std::any::type_name::<Type>(), &err),
|
||||
Source::EnvFile(Err(envfiles::Error::CannotOpenFile(err))) =>
|
||||
panic!("The file at {} could not be opened: {}", &name, &err),
|
||||
Source::EnvFile(Err(envfiles::Error::CannotReadFile(err))) =>
|
||||
panic!("The contents of the file at {} could not be read: {}", &name, &err),
|
||||
Source::EnvFile(Err(envfiles::Error::CannotReadEnvVar(_))) =>
|
||||
panic!("Something unexpected happened in micronfig. Please report this as a bug!"),
|
||||
|
||||
Source::NotFound => None,
|
||||
|
||||
}
|
||||
}
|
52
src/lib.rs
52
src/lib.rs
|
@ -1,52 +0,0 @@
|
|||
//! Tiny crate for simple configuration management.
|
||||
//!
|
||||
//! > **Unstable**; I haven't fully committed to the API yet, so it might change wildly in the following minor versions (`0.x.0`).
|
||||
//!
|
||||
//! # Features
|
||||
//!
|
||||
//! This crate handles:
|
||||
//!
|
||||
//! - Retrieval of values of configuration properties from multiple sources, such as environment variables or files
|
||||
//! - Parsing of retrieved data
|
||||
//! - Displaying human-readable errors if case a step does not succeed
|
||||
//!
|
||||
//! # Usage
|
||||
//!
|
||||
//! This crate has four levels of abstraction, each one with a different usage method.
|
||||
//!
|
||||
//! In order from the highest to the lowest, they are:
|
||||
//!
|
||||
//! 1. **Recommended**: [`required`] and [`optional`], macros which allow you to define global, lazily-evaluated, configuration values;
|
||||
//! 2. [`handle::get_required`] and [`handle::get_optional`], functions which allow you to get a configuration value in a specific moment, without having to consider handling errors;
|
||||
//! 3. [`multi::get`], function which behaves in the same way as the previous two, but returns [`multi::Source`] instead, allowing you to handle errors how you prefer;
|
||||
//! 4. [`single`], module containing submodules allowing the retrieval of configuration values from a single source, returning a source-specific [`Result`].
|
||||
//!
|
||||
//! ## Examples
|
||||
//!
|
||||
//! Some examples are provided in the crate source, [inside the `examples/` directory](https://github.com/Steffo99/micronfig/tree/main/examples).
|
||||
|
||||
#![warn(missing_docs)]
|
||||
#![doc(html_logo_url = "https://raw.githubusercontent.com/Steffo99/micronfig/main/icon.png")]
|
||||
#![cfg_attr(docsrs, feature(doc_cfg))]
|
||||
|
||||
|
||||
pub mod single;
|
||||
|
||||
#[cfg(feature = "multi")]
|
||||
#[cfg_attr(doc_cfg, doc(cfg(feature = "multi")))]
|
||||
pub mod multi;
|
||||
|
||||
#[cfg(feature = "handle")]
|
||||
#[cfg_attr(doc_cfg, doc(cfg(feature = "handle")))]
|
||||
pub mod handle;
|
||||
|
||||
#[cfg(feature = "macros")]
|
||||
#[cfg_attr(doc_cfg, doc(cfg(feature = "macros")))]
|
||||
pub use lazy_static;
|
||||
#[cfg(feature = "macros")]
|
||||
#[cfg_attr(doc_cfg, doc(cfg(feature = "macros")))]
|
||||
pub mod macros;
|
||||
|
||||
#[cfg(feature = "testing")]
|
||||
#[cfg_attr(doc_cfg, doc(cfg(feature = "testing")))]
|
||||
pub mod testing;
|
|
@ -1,82 +0,0 @@
|
|||
//! Highest-level API — Define lazy statics.
|
||||
//!
|
||||
//! The recommended way to use the library.
|
||||
|
||||
|
||||
/// Define a required configuration value with a certain type.
|
||||
///
|
||||
/// # Process
|
||||
///
|
||||
/// This macro:
|
||||
///
|
||||
/// 1. uses [`lazy_static::lazy_static`] to define a new static variable (and associated struct)
|
||||
/// 2. uses [`crate::handle::get_required`] to get the configuration value and handle eventual errors
|
||||
///
|
||||
/// # Examples
|
||||
///
|
||||
/// Define a configuration value with the `USER` key, a [`String`]:
|
||||
/// ```
|
||||
/// # std::env::set_var("USER", "steffo");
|
||||
/// # std::env::remove_var("USER_FILE");
|
||||
/// #
|
||||
/// micronfig::required!(USER, String);
|
||||
/// println!("{:?}", *USER);
|
||||
/// ```
|
||||
///
|
||||
/// Retrieve a configuration value from the `IP_ADDRESS` environment variable or the `IP_ADDRESS_FILE` file, then try to convert it to a [`std::net::IpAddr`]:
|
||||
/// ```
|
||||
/// use std::net::IpAddr;
|
||||
/// #
|
||||
/// # std::env::set_var("IP_ADDRESS", "192.168.1.1");
|
||||
/// # std::env::remove_var("IP_ADDRESS_FILE");
|
||||
///
|
||||
/// micronfig::required!(IP_ADDRESS, IpAddr);
|
||||
/// println!("{:?}", *IP_ADDRESS);
|
||||
/// ```
|
||||
#[macro_export]
|
||||
macro_rules! required {
|
||||
($identifier:ident, $kind:ty) => {
|
||||
$crate::lazy_static::lazy_static! {
|
||||
pub(crate) static ref $identifier: $kind = $crate::handle::get_required::<$kind>(stringify!($identifier));
|
||||
}
|
||||
};
|
||||
}
|
||||
|
||||
/// Define a optional configuration value with a certain type.
|
||||
///
|
||||
/// # Process
|
||||
///
|
||||
/// This macro:
|
||||
///
|
||||
/// 1. uses [`lazy_static::lazy_static`] to define a new static variable (and associated struct)
|
||||
/// 2. uses [`crate::handle::get_optional`] to get the configuration value and handle eventual errors
|
||||
///
|
||||
/// # Examples
|
||||
///
|
||||
/// Define a configuration value with the `USER` key, a [`String`]:
|
||||
/// ```
|
||||
/// # std::env::set_var("USER", "steffo");
|
||||
/// # std::env::remove_var("USER_FILE");
|
||||
/// #
|
||||
/// micronfig::optional!(USER, String);
|
||||
/// println!("{:?}", *USER);
|
||||
/// ```
|
||||
///
|
||||
/// Retrieve a configuration value from the `IP_ADDRESS` environment variable or the `IP_ADDRESS_FILE` file, then try to convert it to a [`std::net::IpAddr`]:
|
||||
/// ```
|
||||
/// use std::net::IpAddr;
|
||||
/// #
|
||||
/// # std::env::set_var("IP_ADDRESS", "192.168.1.1");
|
||||
/// # std::env::remove_var("IP_ADDRESS_FILE");
|
||||
///
|
||||
/// micronfig::optional!(IP_ADDRESS, IpAddr);
|
||||
/// println!("{:?}", *IP_ADDRESS);
|
||||
/// ```
|
||||
#[macro_export]
|
||||
macro_rules! optional {
|
||||
($identifier:ident, $kind:ty) => {
|
||||
$crate::lazy_static::lazy_static! {
|
||||
pub(crate) static ref $identifier: Option<$kind> = $crate::handle::get_optional::<$kind>(stringify!($identifier));
|
||||
}
|
||||
}
|
||||
}
|
293
src/multi/mod.rs
293
src/multi/mod.rs
|
@ -1,293 +0,0 @@
|
|||
//! Middle-level API — Use all available configuration sources.
|
||||
//!
|
||||
//! It can be useful if you want more control on how errors are handled or on how the key is passed to the [`crate::single`] sources.
|
||||
|
||||
|
||||
use std::ffi::OsString;
|
||||
#[cfg(feature = "single_envvars")] use crate::single::envvars;
|
||||
#[cfg(feature = "single_envfiles")] use crate::single::envfiles;
|
||||
|
||||
|
||||
/// Get a value from the first available source, additionally returning information about how the value was retrieved.
|
||||
///
|
||||
/// # Process
|
||||
///
|
||||
/// This function tries to `get` a configuration value:
|
||||
///
|
||||
/// 1. with [`envvars::get`], using `key`, returning a [`Source::EnvVar`]
|
||||
/// 2. with [`envfiles::get`], using `key + key_suffix_file`, returning a [`Source::EnvFile`]
|
||||
///
|
||||
/// If none of these options successfully resulted in the successful retrieval of the configuration value, [`Source::NotFound`] is returned instead.
|
||||
///
|
||||
/// # Errors
|
||||
///
|
||||
/// All errors encountered are bubbled up, except the ones surfacing because of the total absence of a configuration value, which make the function immediately try the next available source.
|
||||
///
|
||||
/// Currently, those errors are:
|
||||
/// - [`envvars::Error::CannotReadEnvVar`]
|
||||
/// - [`envfiles::Error::CannotReadEnvVar`]
|
||||
///
|
||||
/// # Examples
|
||||
///
|
||||
/// Retrieve a configuration value from either the `USER` environment variable or the `USER_FILE` file, maintaining it as a [`String`]:
|
||||
/// ```
|
||||
/// use micronfig::multi::get;
|
||||
/// use micronfig::multi::Source;
|
||||
/// #
|
||||
/// # std::env::set_var("USER", "steffo");
|
||||
/// # std::env::remove_var("USER_FILE");
|
||||
///
|
||||
/// let user: Source<u32> = get("USER", "_FILE");
|
||||
/// ```
|
||||
///
|
||||
/// Retrieve a configuration value from the `IP_ADDRESS` environment variable or the `IP_ADDRESS_FILE` file, then try to convert it to a [`std::net::IpAddr`]:
|
||||
/// ```
|
||||
/// use std::net::IpAddr;
|
||||
/// use micronfig::multi::get;
|
||||
/// use micronfig::multi::Source;
|
||||
/// #
|
||||
/// # std::env::set_var("IP_ADDRESS", "192.168.1.1");
|
||||
/// # std::env::remove_var("IP_ADDRESS_FILE");
|
||||
///
|
||||
/// let ip_addr: Source<IpAddr> = get("IP_ADDRESS", "_FILE");
|
||||
/// ```
|
||||
///
|
||||
pub fn get<Key, KeySuffixFile, Type>(key: Key, key_suffix_file: KeySuffixFile) -> Source<Type>
|
||||
where Key: AsRef<std::ffi::OsStr>,
|
||||
KeySuffixFile: AsRef<std::ffi::OsStr>,
|
||||
Type: std::str::FromStr,
|
||||
<Type as std::str::FromStr>::Err: std::fmt::Debug,
|
||||
{
|
||||
|
||||
if cfg!(feature = "single_envvars") {
|
||||
let v = envvars::get(&key);
|
||||
|
||||
match v {
|
||||
Err(envvars::Error::CannotReadEnvVar(_)) => {},
|
||||
_ => return Source::EnvVar(v),
|
||||
}
|
||||
}
|
||||
|
||||
if cfg!(feature = "single_envfiles") {
|
||||
let mut key_file = OsString::new();
|
||||
key_file.push(key);
|
||||
key_file.push(key_suffix_file);
|
||||
let v = envfiles::get(key_file);
|
||||
|
||||
match v {
|
||||
Err(envfiles::Error::CannotReadEnvVar(_)) => {},
|
||||
_ => return Source::EnvFile(v),
|
||||
}
|
||||
}
|
||||
|
||||
Source::NotFound
|
||||
}
|
||||
|
||||
|
||||
/// The way the result returned by [`get`] was obtained.
|
||||
///
|
||||
/// Since more sources might be added in the future, this function is `non_exaustive`.
|
||||
#[non_exhaustive]
|
||||
#[derive(Debug)]
|
||||
pub enum Source<Type>
|
||||
where Type: std::str::FromStr,
|
||||
<Type as std::str::FromStr>::Err: std::fmt::Debug,
|
||||
{
|
||||
/// The result was not obtained, since the configuration value was not defined anywhere.
|
||||
NotFound,
|
||||
|
||||
/// The result was obtained by [`envvars::get`].
|
||||
#[cfg(feature = "single_envvars")]
|
||||
#[cfg_attr(doc_cfg, doc(cfg(feature = "single_envvars")))]
|
||||
EnvVar(envvars::Result<Type>),
|
||||
|
||||
/// The result was obtained by [`envfiles::get`].
|
||||
#[cfg(feature = "single_envfiles")]
|
||||
#[cfg_attr(doc_cfg, doc(cfg(feature = "single_envfiles")))]
|
||||
EnvFile(envfiles::Result<Type>),
|
||||
}
|
||||
|
||||
impl<Type> Source<Type>
|
||||
where Type: std::str::FromStr,
|
||||
<Type as std::str::FromStr>::Err: std::fmt::Debug,
|
||||
{
|
||||
/// Returns any contained [`Ok`] value, consuming both `self` and the [`Source`] inside.
|
||||
///
|
||||
/// # Panics
|
||||
///
|
||||
/// This function panics if `self` is a [`Source::NotFound`], or if the contained value is a [`Err`].
|
||||
///
|
||||
/// The panic message is the `msg` given.
|
||||
///
|
||||
/// # See also
|
||||
///
|
||||
/// Similar to [`Result::expect`].
|
||||
///
|
||||
/// Used by [`Self::unwrap`].
|
||||
///
|
||||
/// # Examples
|
||||
///
|
||||
/// ```
|
||||
/// use micronfig::multi::Source;
|
||||
///
|
||||
/// let value = Source::<u8>::EnvFile(Ok(1)).expect("value to be present");
|
||||
/// assert_eq!(value, 1)
|
||||
/// ```
|
||||
///
|
||||
/// ```should_panic
|
||||
/// use micronfig::multi::Source;
|
||||
/// use micronfig::single::envfiles::Error as FileError;
|
||||
///
|
||||
/// let value = Source::<u8>::EnvFile(Err(FileError::CannotReadEnvVar(std::env::VarError::NotPresent))).expect("value to be present");
|
||||
/// // Panic!
|
||||
/// ```
|
||||
pub fn expect(self, msg: &str) -> Type {
|
||||
match self {
|
||||
#[cfg(feature = "single_envvars")]
|
||||
Self::EnvVar(Ok(v)) => v,
|
||||
|
||||
#[cfg(feature = "single_envfiles")]
|
||||
Self::EnvFile(Ok(v)) => v,
|
||||
|
||||
_ => panic!("{}", msg),
|
||||
}
|
||||
}
|
||||
|
||||
/// Returns any contained [`Ok`] value, consuming both `self` and the [`Source`] inside.
|
||||
///
|
||||
/// # Panics
|
||||
///
|
||||
/// This function panics if `self` is a [`Source::NotFound`], or if the contained value is a [`Err`].
|
||||
///
|
||||
/// # See also
|
||||
///
|
||||
/// Similar to [`Result::unwrap`].
|
||||
///
|
||||
/// Internally, it uses [`Self::expect`].
|
||||
///
|
||||
/// # Examples
|
||||
///
|
||||
/// ```
|
||||
/// use micronfig::multi::Source;
|
||||
///
|
||||
/// let value = Source::<u8>::EnvFile(Ok(1)).unwrap();
|
||||
/// assert_eq!(value, 1)
|
||||
/// ```
|
||||
///
|
||||
/// ```should_panic
|
||||
/// use micronfig::multi::Source;
|
||||
/// use micronfig::single::envfiles::Error as FileError;
|
||||
///
|
||||
/// let value = Source::<u8>::EnvFile(Err(FileError::CannotReadEnvVar(std::env::VarError::NotPresent))).unwrap();
|
||||
/// // Panic!
|
||||
/// ```
|
||||
pub fn unwrap(self) -> Type
|
||||
{
|
||||
self.expect("called `Source::unwrap()` on an invalid variant, such as `NotFound` or `_(Err(_))`")
|
||||
}
|
||||
}
|
||||
|
||||
|
||||
#[cfg(test)]
|
||||
pub(crate) mod tests {
|
||||
use super::*;
|
||||
use crate::testing::tempfile_fixture;
|
||||
|
||||
#[test]
|
||||
#[cfg(feature = "single_envvars")]
|
||||
fn it_works_var() {
|
||||
std::env::set_var("NUMBER", "1");
|
||||
std::env::remove_var("NUMBER_FILE");
|
||||
|
||||
match get::<&str, &str, u32>("NUMBER", "_FILE") {
|
||||
Source::EnvVar(Ok(1u32)) => {},
|
||||
_ => panic!("expected Source::EnvVar(Ok(1u32))")
|
||||
}
|
||||
}
|
||||
|
||||
#[test]
|
||||
#[cfg(feature = "single_envfiles")]
|
||||
fn it_works_file() {
|
||||
let file = tempfile_fixture("1");
|
||||
std::env::remove_var("NUMBER");
|
||||
std::env::set_var("NUMBER_FILE", file.as_os_str());
|
||||
|
||||
let n = get::<&str, &str, u32>("NUMBER", "_FILE");
|
||||
match n {
|
||||
Source::EnvFile(Ok(1u32)) => {},
|
||||
_ => panic!("expected Source::EnvFile(Ok(1u32))")
|
||||
}
|
||||
}
|
||||
|
||||
#[test]
|
||||
#[cfg(feature = "single_envvars")]
|
||||
fn missing_envvar() {
|
||||
match get::<&str, &str, String>("MISSING_ENVVAR", "_FILE") {
|
||||
Source::NotFound => {},
|
||||
_ => panic!("expected Source::NotFound"),
|
||||
}
|
||||
}
|
||||
|
||||
#[test]
|
||||
#[cfg(feature = "single_envfiles")]
|
||||
fn missing_file() {
|
||||
std::env::remove_var("NUMBER");
|
||||
std::env::set_var("NUMBER_FILE", "/this/file/does/not/exist");
|
||||
|
||||
match get::<&str, &str, u32>("NUMBER", "_FILE") {
|
||||
Source::EnvFile(Err(envfiles::Error::CannotOpenFile(_))) => {},
|
||||
_ => panic!("expected Source::EnvFile(Err(envfiles::Error::CannotOpenFile(_)))"),
|
||||
}
|
||||
}
|
||||
|
||||
#[test]
|
||||
#[cfg(feature = "single_envvars")]
|
||||
fn not_a_number_var() {
|
||||
std::env::set_var("NUMBER", "XYZ");
|
||||
std::env::remove_var("NUMBER_FILE");
|
||||
|
||||
match get::<&str, &str, u32>("NUMBER", "_FILE") {
|
||||
Source::EnvVar(Err(envvars::Error::CannotConvertValue(_))) => {},
|
||||
_ => panic!("expected Source::EnvVar(Err(envvars::Error::CannotConvertValue(_)))"),
|
||||
}
|
||||
}
|
||||
|
||||
#[test]
|
||||
#[cfg(feature = "single_envfiles")]
|
||||
fn not_a_number_file() {
|
||||
let file = tempfile_fixture("XYZ");
|
||||
std::env::set_var("NUMBER_FILE", file.as_os_str());
|
||||
std::env::remove_var("NUMBER");
|
||||
|
||||
match get::<&str, &str, u32>("NUMBER", "_FILE") {
|
||||
Source::EnvFile(Err(envfiles::Error::CannotConvertValue(_))) => {},
|
||||
_ => panic!("expected Source::EnvFile(Err(envfiles::Error::CannotConvertValue(_)))"),
|
||||
}
|
||||
}
|
||||
|
||||
#[test]
|
||||
#[cfg(feature = "single_envvars")]
|
||||
fn unwrap_var_ok() {
|
||||
Source::EnvVar(Ok("ok".to_string())).unwrap();
|
||||
}
|
||||
|
||||
#[test]
|
||||
#[cfg(feature = "single_envfiles")]
|
||||
fn unwrap_file_ok() {
|
||||
Source::EnvFile(Ok("ok".to_string())).unwrap();
|
||||
}
|
||||
|
||||
#[test]
|
||||
#[should_panic]
|
||||
#[cfg(feature = "single_envvars")]
|
||||
fn unwrap_var_err() {
|
||||
Source::<String>::EnvVar(Err(envvars::Error::CannotReadEnvVar(std::env::VarError::NotPresent))).unwrap();
|
||||
}
|
||||
|
||||
#[test]
|
||||
#[should_panic]
|
||||
#[cfg(feature = "single_envfiles")]
|
||||
fn unwrap_file_err() {
|
||||
Source::<String>::EnvFile(Err(envfiles::Error::CannotReadEnvVar(std::env::VarError::NotPresent))).unwrap();
|
||||
}
|
||||
}
|
|
@ -1,137 +0,0 @@
|
|||
//! Contents of files at paths defined by environment variables.
|
||||
|
||||
|
||||
/// Get a configuration value from the source.
|
||||
///
|
||||
/// # Process
|
||||
///
|
||||
/// This function:
|
||||
///
|
||||
/// 1. tries to access the environment variable with the given name using [`std::env::var`]
|
||||
/// 2. tries to interpret the contents of the environment variable as a [`std::path::PathBuf`]
|
||||
/// 3. tries to [`std::fs::File::open`] the file at that path
|
||||
/// 4. tries to [`std::io::Read::read_to_string`] the contents of the opened file
|
||||
/// 5. tries to convert the obtained value to another of the given type using [`std::str::FromStr::from_str`]
|
||||
///
|
||||
/// # Examples
|
||||
///
|
||||
/// Retrieve a configuration value from the `USER_FILE` file, maintaining it as a [`String`]:
|
||||
/// ```
|
||||
/// use micronfig::single::envfiles::get;
|
||||
///
|
||||
/// # let filename = micronfig::testing::tempfile_fixture("steffo");
|
||||
/// # std::env::set_var("USER_FILE", filename.as_os_str());
|
||||
/// let user: String = get("USER_FILE").expect("USER_FILE envvar to be defined");
|
||||
/// ```
|
||||
///
|
||||
/// Retrieve a configuration value from the `IP_ADDRESS_FILE` file, then try to convert it to a [`std::net::IpAddr`]:
|
||||
/// ```
|
||||
/// use std::net::IpAddr;
|
||||
/// use micronfig::single::envfiles::get;
|
||||
///
|
||||
/// # let filename = micronfig::testing::tempfile_fixture("192.168.1.1");
|
||||
/// # std::env::set_var("IP_ADDRESS_FILE", filename.as_os_str());
|
||||
/// let ip_addr: IpAddr = get("IP_ADDRESS_FILE").expect("IP_ADDRESS_FILE envvar to be defined");
|
||||
/// ```
|
||||
///
|
||||
pub fn get<Key, Type>(key: Key) -> Result<Type>
|
||||
where Key: AsRef<std::ffi::OsStr>,
|
||||
Type: std::str::FromStr,
|
||||
<Type as std::str::FromStr>::Err: std::fmt::Debug,
|
||||
{
|
||||
let path = std::env::var(key)
|
||||
.map_err(Error::CannotReadEnvVar)?;
|
||||
let path = std::ffi::OsString::from(path);
|
||||
let path = std::path::PathBuf::from(path);
|
||||
|
||||
let mut file = std::fs::File::open(path)
|
||||
.map_err(Error::CannotOpenFile)?;
|
||||
|
||||
use std::io::Read;
|
||||
let mut data = String::new();
|
||||
file.read_to_string(&mut data)
|
||||
.map_err(Error::CannotReadFile)?;
|
||||
|
||||
let value = Type::from_str(&data)
|
||||
.map_err(Error::CannotConvertValue)?;
|
||||
|
||||
Ok(value)
|
||||
}
|
||||
|
||||
|
||||
/// A possible error encountered by [`get`].
|
||||
#[derive(std::fmt::Debug)]
|
||||
pub enum Error<ConversionError>
|
||||
where ConversionError: std::fmt::Debug,
|
||||
{
|
||||
/// The environment variable could not be read.
|
||||
///
|
||||
/// Encountered when the call to [`std::env::var`] fails.
|
||||
CannotReadEnvVar(std::env::VarError),
|
||||
|
||||
/// The specified file could not be opened. (Probably it doesn't exist.)
|
||||
///
|
||||
/// Encountered when the call to [`std::fs::File::open`] fails.
|
||||
CannotOpenFile(std::io::Error),
|
||||
|
||||
/// The specified file could not be read.
|
||||
///
|
||||
/// Encountered when the call to [`std::io::Read::read_to_string`] fails.
|
||||
CannotReadFile(std::io::Error),
|
||||
|
||||
/// The value could not be converted to the desired type.
|
||||
///
|
||||
/// Encountered when the call to [`std::str::FromStr::from_str`] fails.
|
||||
CannotConvertValue(ConversionError),
|
||||
}
|
||||
|
||||
|
||||
/// A possible error encountered by [`get`].
|
||||
pub type Result<Type> = std::result::Result<Type, Error<<Type as std::str::FromStr>::Err>>;
|
||||
|
||||
|
||||
#[cfg(test)]
|
||||
pub(crate) mod tests {
|
||||
use super::*;
|
||||
use crate::testing::tempfile_fixture;
|
||||
|
||||
#[test]
|
||||
fn it_works() {
|
||||
let file = tempfile_fixture("1");
|
||||
std::env::set_var("NUMBER_FILE", file.as_os_str());
|
||||
|
||||
let number = get::<&str, u32>("NUMBER_FILE").unwrap();
|
||||
assert_eq!(number, 1u32);
|
||||
}
|
||||
|
||||
#[test]
|
||||
fn missing_envvar() {
|
||||
std::env::remove_var("THIS_ENVVAR_DOES_NOT_EXIST_FILE");
|
||||
|
||||
match get::<&str, String>("THIS_ENVVAR_DOES_NOT_EXIST_FILE") {
|
||||
Err(Error::CannotReadEnvVar(std::env::VarError::NotPresent)) => {},
|
||||
_ => panic!("expected Err(Error::CannotReadEnvVar(std::env::VarError::NotPresent))"),
|
||||
}
|
||||
}
|
||||
|
||||
#[test]
|
||||
fn missing_file() {
|
||||
std::env::set_var("NUMBER_FILE", "/this/file/does/not/exist");
|
||||
|
||||
match get::<&str, u32>("NUMBER_FILE") {
|
||||
Err(Error::CannotOpenFile(_)) => {},
|
||||
_ => panic!("expected Err(Error::CannotOpenFile(_))"),
|
||||
}
|
||||
}
|
||||
|
||||
#[test]
|
||||
fn not_a_number() {
|
||||
let file = tempfile_fixture("XYZ");
|
||||
std::env::set_var("NUMBER_FILE", file.as_os_str());
|
||||
|
||||
match get::<&str, u32>("NUMBER_FILE") {
|
||||
Err(Error::CannotConvertValue(_)) => {},
|
||||
_ => panic!("expected Err(Error::CannotConvertValue(_))"),
|
||||
}
|
||||
}
|
||||
}
|
|
@ -1,99 +0,0 @@
|
|||
//! Environment variables.
|
||||
|
||||
|
||||
/// Get a configuration value from the source.
|
||||
///
|
||||
/// # Process
|
||||
///
|
||||
/// This function:
|
||||
///
|
||||
/// 1. tries to access the environment variable with the given name using [`std::env::var`]
|
||||
/// 2. tries to convert the obtained value to another of the given type using [`std::str::FromStr::from_str`]
|
||||
///
|
||||
/// # Examples
|
||||
///
|
||||
/// Retrieve a configuration value from the `USER` environment variable, maintaining it as a [`String`]:
|
||||
/// ```
|
||||
/// use micronfig::single::envvars::get;
|
||||
///
|
||||
/// # std::env::set_var("USER", "steffo");
|
||||
/// let user: String = get("USER").expect("USER envvar to be defined");
|
||||
/// ```
|
||||
///
|
||||
/// Retrieve a configuration value from the `IP_ADDRESS` environment variable, then try to convert it to a [`std::net::IpAddr`]:
|
||||
/// ```
|
||||
/// use std::net::IpAddr;
|
||||
/// use micronfig::single::envvars::get;
|
||||
///
|
||||
/// # std::env::set_var("IP_ADDRESS", "192.168.1.1");
|
||||
/// let ip_addr: IpAddr = get("IP_ADDRESS").expect("IP_ADDRESS envvar to be defined");
|
||||
/// ```
|
||||
///
|
||||
pub fn get<Key, Type>(key: Key) -> Result<Type>
|
||||
where Key: AsRef<std::ffi::OsStr>,
|
||||
Type: std::str::FromStr,
|
||||
<Type as std::str::FromStr>::Err: std::fmt::Debug,
|
||||
{
|
||||
let data = std::env::var(key)
|
||||
.map_err(Error::CannotReadEnvVar)?;
|
||||
|
||||
let value = Type::from_str(&data)
|
||||
.map_err(Error::CannotConvertValue)?;
|
||||
|
||||
Ok(value)
|
||||
}
|
||||
|
||||
|
||||
/// A possible error encountered by [`get`].
|
||||
#[derive(std::fmt::Debug)]
|
||||
pub enum Error<ConversionError>
|
||||
where ConversionError: std::fmt::Debug,
|
||||
{
|
||||
/// The environment variable could not be read.
|
||||
///
|
||||
/// Encountered when the call to [`std::env::var`] fails.
|
||||
CannotReadEnvVar(std::env::VarError),
|
||||
|
||||
/// The value could not be converted to the desired type.
|
||||
///
|
||||
/// Encountered when the call to [`std::str::FromStr::from_str`] fails.
|
||||
CannotConvertValue(ConversionError),
|
||||
}
|
||||
|
||||
|
||||
/// The result of [`get`].
|
||||
pub type Result<Type> = std::result::Result<Type, Error<<Type as std::str::FromStr>::Err>>;
|
||||
|
||||
|
||||
#[cfg(test)]
|
||||
pub(crate) mod tests {
|
||||
use super::*;
|
||||
|
||||
#[test]
|
||||
fn it_works() {
|
||||
std::env::set_var("NUMBER", "1");
|
||||
|
||||
let number = get::<&str, u32>("NUMBER").unwrap();
|
||||
assert_eq!(number, 1u32);
|
||||
}
|
||||
|
||||
#[test]
|
||||
fn missing_envvar() {
|
||||
std::env::remove_var("THIS_ENVVAR_DOES_NOT_EXIST");
|
||||
|
||||
match get::<&str, String>("THIS_ENVVAR_DOES_NOT_EXIST") {
|
||||
Err(Error::CannotReadEnvVar(std::env::VarError::NotPresent)) => {},
|
||||
_ => panic!("expected Err(Error::CannotReadEnvVar(std::env::VarError::NotPresent))"),
|
||||
}
|
||||
}
|
||||
|
||||
#[test]
|
||||
fn not_a_number() {
|
||||
std::env::set_var("NUMBER", "XYZ");
|
||||
|
||||
match get::<&str, u32>("NUMBER") {
|
||||
Err(Error::CannotConvertValue(_)) => {},
|
||||
_ => panic!("expected Error::CannotConvertValue(_)"),
|
||||
}
|
||||
}
|
||||
}
|
|
@ -1,15 +0,0 @@
|
|||
//! Lowest-level API — Manually select configuration sources.
|
||||
//!
|
||||
//! It can be useful if you want to specify manually the sources to access when retrieving configuration values.
|
||||
//!
|
||||
//! Each possible source has an associated module, and a feature named `single_{MODULENAME}` enabling it; see the list of modules below to see what sources are available!
|
||||
|
||||
|
||||
#[cfg(feature = "single_envvars")]
|
||||
#[cfg_attr(doc_cfg, doc(cfg(feature = "single_envvars")))]
|
||||
pub mod envvars;
|
||||
|
||||
#[cfg(feature = "single_envfiles")]
|
||||
#[cfg_attr(doc_cfg, doc(cfg(feature = "single_envfiles")))]
|
||||
pub mod envfiles;
|
||||
|
|
@ -1,17 +0,0 @@
|
|||
//! Fixtures for testing.
|
||||
//!
|
||||
//! **Unstable**; not supposed to be used outside this crate; do not add `pub(crate)` or doctests will stop working.
|
||||
|
||||
/// Create a temporary file and write `content` inside it.
|
||||
///
|
||||
/// The file will be deleted as soon as the [`tempfile::TempPath`] is dropped.
|
||||
pub fn tempfile_fixture(content: &str) -> tempfile::TempPath {
|
||||
use std::io::Write;
|
||||
|
||||
let mut file = tempfile::NamedTempFile::new()
|
||||
.expect("the tempfile fixture to be created successfully");
|
||||
write!(file, "{}", content)
|
||||
.expect("to be able to write into the tempfile fixture");
|
||||
|
||||
file.into_temp_path()
|
||||
}
|
|
@ -1,17 +0,0 @@
|
|||
use std::env;
|
||||
|
||||
micronfig::required!(PLAYER_NAME, String);
|
||||
micronfig::required!(PLAYER_ID, u64);
|
||||
micronfig::optional!(IS_SUS, bool);
|
||||
|
||||
|
||||
#[test]
|
||||
fn test_macros() {
|
||||
env::set_var("PLAYER_NAME", "Steffo");
|
||||
env::set_var("PLAYER_ID", "1234");
|
||||
env::remove_var("IS_SUS");
|
||||
|
||||
assert_eq!(*PLAYER_NAME, "Steffo");
|
||||
assert_eq!(*PLAYER_ID, 1234u64);
|
||||
assert_eq!(*IS_SUS, None);
|
||||
}
|
|
@ -1 +0,0 @@
|
|||
pub mod macros;
|
Loading…
Reference in a new issue