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
// Map

use std::collections::HashMap;

use crate::{Edn, EdnTag};

/// Map interface for Edn::Map
#[derive(Debug, Clone, Default, PartialEq, Eq)]
pub struct EdnMapView(pub HashMap<Edn, Edn>);

impl TryFrom<Edn> for EdnMapView {
  type Error = String;

  fn try_from(data: Edn) -> Result<Self, Self::Error> {
    match data {
      Edn::Map(xs) => Ok(xs),
      Edn::Nil => Ok(EdnMapView(HashMap::new())),
      a => Err(format!("data is not map: {}", a)),
    }
  }
}

impl From<HashMap<Edn, Edn>> for EdnMapView {
  fn from(xs: HashMap<Edn, Edn>) -> EdnMapView {
    EdnMapView(xs)
  }
}

impl From<EdnMapView> for HashMap<Edn, Edn> {
  fn from(x: EdnMapView) -> HashMap<Edn, Edn> {
    x.0
  }
}

impl From<EdnMapView> for Edn {
  fn from(x: EdnMapView) -> Edn {
    Edn::Map(EdnMapView(x.0))
  }
}

impl EdnMapView {
  /// get by tag
  pub fn tag_get(&self, key: &str) -> Option<&Edn> {
    self.0.get(&Edn::Tag(EdnTag::from(key)))
  }
  /// get by str
  pub fn str_get(&self, key: &str) -> Option<&Edn> {
    self.0.get(&Edn::str(key))
  }

  /// get reference of element
  pub fn get(&self, key: &Edn) -> Option<&Edn> {
    self.0.get(key)
  }

  /// regardless of key in string or tag
  pub fn get_or_nil(&self, key: &str) -> Edn {
    self
      .0
      .get(&Edn::str(key))
      .cloned()
      .or_else(|| self.0.get(&Edn::tag(key)).cloned())
      .unwrap_or(Edn::Nil)
  }

  pub fn contains_key(&self, key: &str) -> bool {
    self.0.contains_key(&Edn::str(key)) || self.0.contains_key(&Edn::tag(key))
  }

  pub fn insert(&mut self, k: Edn, v: Edn) {
    self.0.insert(k, v);
  }

  /// takes k that impl Into<EdnTag>
  pub fn insert_key(&mut self, k: impl Into<EdnTag>, v: Edn) {
    self.0.insert(k.into().into(), v);
  }

  pub fn len(&self) -> usize {
    self.0.len()
  }

  pub fn is_empty(&self) -> bool {
    self.0.is_empty()
  }
}