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
use crate::{Id, Term};

/// Types that may represent a blank node identifier.
pub trait MaybeBlankId {
	/// Inner blank node identifier type.
	type BlankId;
}

impl<I, B> MaybeBlankId for Id<I, B> {
	type BlankId = B;
}

impl<I: MaybeBlankId, L> MaybeBlankId for Term<I, L> {
	type BlankId = I::BlankId;
}

/// Types that may have a blank node identifier representation that can be
/// borrowed.
pub trait AsBlankId: MaybeBlankId {
	/// Returns a reference to the blank node identifier value, if any.
	fn as_blank(&self) -> Option<&Self::BlankId>;

	fn is_blank(&self) -> bool {
		self.as_blank().is_some()
	}
}

impl<I, B> AsBlankId for Id<I, B> {
	fn as_blank(&self) -> Option<&Self::BlankId> {
		self.as_blank()
	}
}

impl<I: AsBlankId, L> AsBlankId for Term<I, L> {
	fn as_blank(&self) -> Option<&Self::BlankId> {
		self.as_blank()
	}
}

/// Types that can be turned into a blank node identifier.
pub trait IntoBlankId: MaybeBlankId {
	/// Converts the value into a blank node identifier, if any.
	fn into_blank(self) -> Option<Self::BlankId>;
}

impl<I, B> IntoBlankId for Id<I, B> {
	fn into_blank(self) -> Option<Self::BlankId> {
		self.into_blank()
	}
}

impl<I: IntoBlankId, L> IntoBlankId for Term<I, L> {
	fn into_blank(self) -> Option<Self::BlankId> {
		self.into_blank()
	}
}

/// Types that can be constructed from a blank node identifier.
pub trait FromBlankId: MaybeBlankId {
	/// Builds a value from a blank node identifier.
	fn from_blank(b: Self::BlankId) -> Self;
}

impl<I, B> FromBlankId for Id<I, B> {
	fn from_blank(b: Self::BlankId) -> Self {
		Self::Blank(b)
	}
}

impl<I: FromBlankId, L> FromBlankId for Term<I, L> {
	fn from_blank(b: Self::BlankId) -> Self {
		Self::Id(I::from_blank(b))
	}
}