summaryrefslogtreecommitdiffstats
path: root/vendor/gix-config-value/src/boolean.rs
diff options
context:
space:
mode:
authorDaniel Baumann <daniel.baumann@progress-linux.org>2024-05-04 12:41:41 +0000
committerDaniel Baumann <daniel.baumann@progress-linux.org>2024-05-04 12:41:41 +0000
commit10ee2acdd26a7f1298c6f6d6b7af9b469fe29b87 (patch)
treebdffd5d80c26cf4a7a518281a204be1ace85b4c1 /vendor/gix-config-value/src/boolean.rs
parentReleasing progress-linux version 1.70.0+dfsg1-9~progress7.99u1. (diff)
downloadrustc-10ee2acdd26a7f1298c6f6d6b7af9b469fe29b87.tar.xz
rustc-10ee2acdd26a7f1298c6f6d6b7af9b469fe29b87.zip
Merging upstream version 1.70.0+dfsg2.
Signed-off-by: Daniel Baumann <daniel.baumann@progress-linux.org>
Diffstat (limited to 'vendor/gix-config-value/src/boolean.rs')
-rw-r--r--vendor/gix-config-value/src/boolean.rs99
1 files changed, 99 insertions, 0 deletions
diff --git a/vendor/gix-config-value/src/boolean.rs b/vendor/gix-config-value/src/boolean.rs
new file mode 100644
index 000000000..908e11a30
--- /dev/null
+++ b/vendor/gix-config-value/src/boolean.rs
@@ -0,0 +1,99 @@
+use std::{borrow::Cow, convert::TryFrom, ffi::OsString, fmt::Display};
+
+use bstr::{BStr, BString, ByteSlice};
+
+use crate::{Boolean, Error};
+
+fn bool_err(input: impl Into<BString>) -> Error {
+ Error::new(
+ "Booleans need to be 'no', 'off', 'false', '' or 'yes', 'on', 'true' or any number",
+ input,
+ )
+}
+
+impl TryFrom<OsString> for Boolean {
+ type Error = Error;
+
+ fn try_from(value: OsString) -> Result<Self, Self::Error> {
+ let value = gix_path::os_str_into_bstr(&value)
+ .map_err(|_| Error::new("Illformed UTF-8", std::path::Path::new(&value).display().to_string()))?;
+ Self::try_from(value)
+ }
+}
+
+/// # Warning
+///
+/// The direct usage of `try_from("string")` is discouraged as it will produce the wrong result for values
+/// obtained from `core.bool-implicit-true`, which have no separator and are implicitly true.
+/// This method chooses to work correctly for `core.bool-empty=`, which is an empty string and resolves
+/// to being `false`.
+///
+/// Instead of this, obtain booleans with `config.boolean(…)`, which handles the case were no separator is
+/// present correctly.
+impl TryFrom<&BStr> for Boolean {
+ type Error = Error;
+
+ fn try_from(value: &BStr) -> Result<Self, Self::Error> {
+ if parse_true(value) {
+ Ok(Boolean(true))
+ } else if parse_false(value) {
+ Ok(Boolean(false))
+ } else {
+ use std::str::FromStr;
+ if let Some(integer) = value.to_str().ok().and_then(|s| i64::from_str(s).ok()) {
+ Ok(Boolean(integer != 0))
+ } else {
+ Err(bool_err(value))
+ }
+ }
+ }
+}
+
+impl Boolean {
+ /// Return true if the boolean is a true value.
+ ///
+ /// Note that the inner value is accessible directly as well.
+ pub fn is_true(self) -> bool {
+ self.0
+ }
+}
+
+impl TryFrom<Cow<'_, BStr>> for Boolean {
+ type Error = Error;
+ fn try_from(c: Cow<'_, BStr>) -> Result<Self, Self::Error> {
+ Self::try_from(c.as_ref())
+ }
+}
+
+impl Display for Boolean {
+ fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result {
+ self.0.fmt(f)
+ }
+}
+
+impl From<Boolean> for bool {
+ fn from(b: Boolean) -> Self {
+ b.0
+ }
+}
+
+#[cfg(feature = "serde")]
+impl serde::Serialize for Boolean {
+ fn serialize<S>(&self, serializer: S) -> Result<S::Ok, S::Error>
+ where
+ S: serde::Serializer,
+ {
+ serializer.serialize_bool(self.0)
+ }
+}
+
+fn parse_true(value: &BStr) -> bool {
+ value.eq_ignore_ascii_case(b"yes") || value.eq_ignore_ascii_case(b"on") || value.eq_ignore_ascii_case(b"true")
+}
+
+fn parse_false(value: &BStr) -> bool {
+ value.eq_ignore_ascii_case(b"no")
+ || value.eq_ignore_ascii_case(b"off")
+ || value.eq_ignore_ascii_case(b"false")
+ || value.is_empty()
+}