use fidl_fuchsia_diagnostics::StringSelector;
use moniker::{ExtendedMoniker, Moniker, EXTENDED_MONIKER_COMPONENT_MANAGER_STR};
use selectors::FastError;
use serde::de::Unexpected;
use serde::{Deserialize, Deserializer};
use std::sync::atomic::{AtomicU64, Ordering};
use std::sync::Arc;
#[derive(Clone, Debug, PartialEq)]
pub struct SelectorList(pub Vec<Option<ParsedSelector>>);
impl std::ops::Deref for SelectorList {
type Target = Vec<Option<ParsedSelector>>;
fn deref(&self) -> &Self::Target {
&self.0
}
}
impl std::ops::DerefMut for SelectorList {
fn deref_mut(&mut self) -> &mut Self::Target {
&mut self.0
}
}
impl IntoIterator for SelectorList {
type Item = Option<ParsedSelector>;
type IntoIter = std::vec::IntoIter<Self::Item>;
fn into_iter(self) -> Self::IntoIter {
self.0.into_iter()
}
}
#[derive(Clone, Debug)]
pub struct ParsedSelector {
pub selector_string: String,
pub selector: fidl_fuchsia_diagnostics::Selector,
pub moniker: ExtendedMoniker,
upload_count: Arc<AtomicU64>,
}
impl ParsedSelector {
pub fn increment_upload_count(&self) {
self.upload_count.fetch_add(1, Ordering::Relaxed);
}
pub fn get_upload_count(&self) -> u64 {
self.upload_count.load(Ordering::Relaxed)
}
}
impl PartialEq for ParsedSelector {
fn eq(&self, other: &Self) -> bool {
self.selector_string == other.selector_string
&& self.selector == other.selector
&& self.moniker == other.moniker
&& self.upload_count.load(Ordering::Relaxed)
== other.upload_count.load(Ordering::Relaxed)
}
}
pub(crate) fn parse_selector<E>(selector_str: &str) -> Result<ParsedSelector, E>
where
E: serde::de::Error,
{
let selector = selectors::parse_selector::<FastError>(selector_str)
.or(Err(E::invalid_value(Unexpected::Str(selector_str), &"need a valid selector")))?;
let component_selector = selector.component_selector.as_ref().ok_or(E::invalid_value(
Unexpected::Str(selector_str),
&"selector must specify component",
))?;
let moniker_segments = component_selector.moniker_segments.as_ref().ok_or(E::invalid_value(
Unexpected::Str(selector_str),
&"selector must specify component",
))?;
let moniker_strings = moniker_segments
.iter()
.map(|segment| match segment {
StringSelector::StringPattern(_) => Err(E::invalid_value(
Unexpected::Str(selector_str),
&"component monikers cannot contain wildcards",
)),
StringSelector::ExactMatch(text) => Ok(text.as_ref()),
_ => Err(E::invalid_value(Unexpected::Str(selector_str), &"Unexpected moniker type")),
})
.collect::<Result<Vec<_>, _>>()?;
let moniker = if moniker_strings.len() == 1
&& moniker_strings[0] == EXTENDED_MONIKER_COMPONENT_MANAGER_STR
{
ExtendedMoniker::ComponentManager
} else {
ExtendedMoniker::ComponentInstance(
Moniker::try_from(moniker_strings)
.or(Err(E::invalid_value(Unexpected::Str(selector_str), &"invalid moniker")))?,
)
};
Ok(ParsedSelector {
selector,
selector_string: selector_str.to_string(),
moniker,
upload_count: Arc::new(AtomicU64::new(0)),
})
}
impl<'de> Deserialize<'de> for SelectorList {
fn deserialize<D>(d: D) -> Result<Self, D::Error>
where
D: Deserializer<'de>,
{
struct SelectorVec(std::marker::PhantomData<Vec<Option<ParsedSelector>>>);
impl<'de> serde::de::Visitor<'de> for SelectorVec {
type Value = Vec<Option<ParsedSelector>>;
fn expecting(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result {
f.write_str("either a single selector or an array of selectors")
}
fn visit_str<E>(self, value: &str) -> Result<Self::Value, E>
where
E: serde::de::Error,
{
Ok(vec![Some(parse_selector::<E>(value)?)])
}
fn visit_seq<A>(self, mut value: A) -> Result<Self::Value, A::Error>
where
A: serde::de::SeqAccess<'de>,
{
let mut out = vec![];
while let Some(s) = value.next_element::<String>()? {
out.push(Some(parse_selector::<A::Error>(&s)?));
}
if out.is_empty() {
use serde::de::Error;
Err(A::Error::invalid_length(0, &"expected at least one selector"))
} else {
Ok(out)
}
}
}
Ok(SelectorList(d.deserialize_any(SelectorVec(std::marker::PhantomData))?))
}
}
#[cfg(test)]
mod tests {
use super::*;
use anyhow::Error;
use fidl_fuchsia_diagnostics::TreeSelector;
fn require_string(data: &StringSelector, required: &str) {
match data {
StringSelector::ExactMatch(string) => assert_eq!(string, required),
_ => assert!(false, "Expected an exact match"),
}
}
fn require_strings(data: &Vec<StringSelector>, required: Vec<&str>) {
assert_eq!(data.len(), required.len());
for (data, required) in data.iter().zip(required.iter()) {
require_string(data, required);
}
}
#[fuchsia::test]
fn parse_valid_single_selector() -> Result<(), Error> {
let json = "\"core/foo:root/branch:leaf\"";
let selectors: SelectorList = serde_json5::from_str(json)?;
assert_eq!(selectors.len(), 1);
let ParsedSelector { selector_string, selector, moniker, .. } =
selectors[0].as_ref().unwrap();
assert_eq!(selector_string, "core/foo:root/branch:leaf");
assert_eq!(moniker.to_string(), "core/foo");
let moniker =
selector.component_selector.as_ref().unwrap().moniker_segments.as_ref().unwrap();
require_strings(moniker, vec!["core", "foo"]);
match &selector.tree_selector {
Some(TreeSelector::PropertySelector(selector)) => {
require_strings(&selector.node_path, vec!["root", "branch"]);
require_string(&selector.target_properties, "leaf");
}
_ => assert!(false, "Expected a property selector"),
}
Ok(())
}
#[fuchsia::test]
fn parse_valid_multiple_selectors() -> Result<(), Error> {
let json = "[ \"core/foo:root/branch:leaf\", \"core/bar:root/twig:leaf\"]";
let selectors: SelectorList = serde_json5::from_str(json)?;
assert_eq!(selectors.len(), 2);
let ParsedSelector { selector_string, selector, moniker, .. } =
selectors[0].as_ref().unwrap();
assert_eq!(selector_string, "core/foo:root/branch:leaf");
assert_eq!(moniker.to_string(), "core/foo");
let moniker =
selector.component_selector.as_ref().unwrap().moniker_segments.as_ref().unwrap();
require_strings(moniker, vec!["core", "foo"]);
match &selector.tree_selector {
Some(TreeSelector::PropertySelector(selector)) => {
require_strings(&selector.node_path, vec!["root", "branch"]);
require_string(&selector.target_properties, "leaf");
}
_ => assert!(false, "Expected a property selector"),
}
let ParsedSelector { selector_string, selector, moniker, .. } =
selectors[1].as_ref().unwrap();
assert_eq!(selector_string, "core/bar:root/twig:leaf");
assert_eq!(moniker.to_string(), "core/bar");
let moniker =
selector.component_selector.as_ref().unwrap().moniker_segments.as_ref().unwrap();
require_strings(moniker, vec!["core", "bar"]);
match &selector.tree_selector {
Some(TreeSelector::PropertySelector(selector)) => {
require_strings(&selector.node_path, vec!["root", "twig"]);
require_string(&selector.target_properties, "leaf");
}
_ => assert!(false, "Expected a property selector"),
}
Ok(())
}
#[fuchsia::test]
fn refuse_invalid_selectors() {
let bad_selector = "\"core/foo:wrong:root/branch:leaf\"";
let not_string = "42";
let bad_list = "[ \"core/foo:root/branch:leaf\", \"core/bar:wrong:root/twig:leaf\"]";
serde_json5::from_str::<SelectorList>(bad_selector).expect_err("this should fail");
serde_json5::from_str::<SelectorList>(not_string).expect_err("this should fail");
serde_json5::from_str::<SelectorList>(bad_list).expect_err("this should fail");
}
}