ftml/parsing/element_condition.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
/*
* parsing/element_condition.rs
*
* ftml - Library to parse Wikidot text
* Copyright (C) 2019-2024 Wikijump Team
*
* This program is free software: you can redistribute it and/or modify
* it under the terms of the GNU Affero General Public License as published by
* the Free Software Foundation, either version 3 of the License, or
* (at your option) any later version.
*
* This program is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU Affero General Public License for more details.
*
* You should have received a copy of the GNU Affero General Public License
* along with this program. If not, see <http://www.gnu.org/licenses/>.
*/
use std::borrow::Cow;
use strum_macros::IntoStaticStr;
/// Representation of a single condition to determine element presence.
///
/// A list of these constitutes a full condition specification, and is
/// used in blocks like `[[iftags]]` and `[[ifcategory]]`.
#[derive(Serialize, Deserialize, Debug, Clone, Hash, PartialEq, Eq)]
pub struct ElementCondition<'t> {
#[serde(rename = "condition")]
pub ctype: ElementConditionType,
pub value: Cow<'t, str>,
}
impl<'t> ElementCondition<'t> {
/// Parse out a specification.
///
/// The specification is a space separated list of strings, prefixed with
/// either `+` or `-` or nothing.
pub fn parse(raw_spec: &'t str) -> Vec<ElementCondition<'t>> {
// Helper to get the value and its condition type
fn get_spec(value: &str) -> (ElementConditionType, &str) {
if let Some(value) = value.strip_prefix('+') {
return (ElementConditionType::Required, value);
}
if let Some(value) = value.strip_prefix('-') {
return (ElementConditionType::Prohibited, value);
}
(ElementConditionType::Present, value)
}
raw_spec
.split(' ')
.filter(|s| !s.is_empty())
.map(|s| {
let (ctype, value) = get_spec(s);
ElementCondition {
ctype,
value: cow!(value),
}
})
.collect()
}
/// Determines if this condition is satisfied.
///
/// * `ElementConditionType::Required` -- All values of this kind must be present.
/// * `ElementConditionType::Prohibited` -- All values of this kind must be absent.
/// * `ElementConditionType::Present` -- Some values of this kind must be present.
///
/// The full logic is essentially `all(required) && any(present) && all(prohibited)`.
pub fn check(conditions: &[ElementCondition], values: &[Cow<str>]) -> bool {
let mut required = true;
let mut prohibited = true;
let mut present = false;
let mut had_present = false; // whether there were any present conditions
for condition in conditions {
let has_value = values.contains(&condition.value);
match condition.ctype {
ElementConditionType::Required => required &= has_value,
ElementConditionType::Prohibited => prohibited &= !has_value,
ElementConditionType::Present => {
present |= has_value;
had_present = true;
}
}
}
// Since this is false by default, if there are no present conditions,
// it's effectively true.
//
// Otherwise you have to include a present condition for any iftags to pass!
//
// We could do "required && prohibited && (present || !had_present)" instead,
// but this if block is more readable.
if !had_present {
present = true;
}
required && prohibited && present
}
}
#[derive(
Serialize, Deserialize, IntoStaticStr, Debug, Copy, Clone, Hash, PartialEq, Eq,
)]
#[serde(rename_all = "kebab-case")]
pub enum ElementConditionType {
Required,
Prohibited,
Present,
}
impl ElementConditionType {
#[inline]
pub fn name(self) -> &'static str {
self.into()
}
}