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
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 TryAsBlankId: MaybeBlankId {
	/// Returns a reference to the blank node identifier value, if any.
	fn try_as_blank(&self) -> Option<&Self::BlankId>;

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

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

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

/// Types that can be turned into a blank node identifier.
pub trait TryIntoBlankId: MaybeBlankId + Sized {
	fn try_into_blank(self) -> Result<Self::BlankId, Self>;
}

impl<I, B> TryIntoBlankId for Id<I, B> {
	fn try_into_blank(self) -> Result<Self::BlankId, Self> {
		self.try_into_blank().map_err(Self::Iri)
	}
}

impl<I: TryIntoBlankId, L> TryIntoBlankId for Term<I, L> {
	fn try_into_blank(self) -> Result<Self::BlankId, Self> {
		self.try_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))
	}
}