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 125 126 127 128 129 130 131 132 133 134 135 136 137 138 139 140 141 142 143 144 145 146 147 148 149 150 151 152 153 154 155 156 157 158 159 160 161 162 163 164 165 166 167 168 169 170 171 172 173 174 175 176 177 178 179 180 181 182
/*!
The [`Extent`] type.
An extent is the time for which an event is active. It may be either a point for an event that occurred at a particular time, or a span for an event that was active over a particular period.
Extents can be constructed directly, or generically through the [`ToExtent`] trait.
*/
use crate::{empty::Empty, timestamp::Timestamp};
use core::{fmt, ops::Range, time::Duration};
/**
Either a single [`Timestamp`] for a point in time, or a pair of [`Timestamp`]s for a timespan.
*/
#[derive(Clone)]
pub struct Extent {
range: Range<Timestamp>,
is_span: bool,
}
impl Extent {
/**
Create an extent for a point in time.
*/
pub fn point(ts: Timestamp) -> Self {
Extent {
range: ts..ts,
is_span: false,
}
}
/**
Create an extent for a timespan.
The end of the range should be after the start, but an empty range is still considered a span.
*/
pub fn span(ts: Range<Timestamp>) -> Self {
Extent {
range: ts,
is_span: true,
}
}
/**
Get the extent as a range of timestamps.
For point extents, this will return an empty range with the start and end bounds being equal. For span extents, this will return exactly the range the extent was created from.
*/
pub fn as_range(&self) -> &Range<Timestamp> {
&self.range
}
/**
Get the extent as a point in time.
For point extents, this will return exactly the value the extent was created from. For span extents, this will return the end bound.
*/
pub fn as_point(&self) -> &Timestamp {
&self.range.end
}
/**
Try get the extent as a timespan.
This method will return `Some` if the extent is a span, even if that span is empty. It will return `None` for point extents.
*/
pub fn as_span(&self) -> Option<&Range<Timestamp>> {
if self.is_span() {
Some(&self.range)
} else {
None
}
}
/**
Try get the length of the extent.
This method will return `Some` if the extent is a span, even if that span is empty. It will return `None` for point extents.
*/
pub fn len(&self) -> Option<Duration> {
if self.is_span() {
self.range.end.duration_since(self.range.start)
} else {
None
}
}
/**
Whether the extent is a point in time.
*/
pub fn is_point(&self) -> bool {
!self.is_span()
}
/**
Whether the extent is a timespan.
*/
pub fn is_span(&self) -> bool {
self.is_span
}
}
impl fmt::Debug for Extent {
fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
if self.is_span() {
fmt::Debug::fmt(&self.range.start, f)?;
f.write_str("..")?;
fmt::Debug::fmt(&self.range.end, f)
} else {
fmt::Debug::fmt(&self.range.end, f)
}
}
}
impl fmt::Display for Extent {
fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result {
if self.is_span() {
fmt::Display::fmt(&self.range.start, f)?;
f.write_str("..")?;
fmt::Display::fmt(&self.range.end, f)
} else {
fmt::Display::fmt(&self.range.end, f)
}
}
}
/**
Try convert a value into an [`Extent`].
*/
pub trait ToExtent {
/**
Perform the conversion.
*/
fn to_extent(&self) -> Option<Extent>;
}
impl<'a, T: ToExtent + ?Sized> ToExtent for &'a T {
fn to_extent(&self) -> Option<Extent> {
(**self).to_extent()
}
}
impl ToExtent for Empty {
fn to_extent(&self) -> Option<Extent> {
None
}
}
impl<T: ToExtent> ToExtent for Option<T> {
fn to_extent(&self) -> Option<Extent> {
self.as_ref().and_then(|ts| ts.to_extent())
}
}
impl ToExtent for Extent {
fn to_extent(&self) -> Option<Extent> {
Some(self.clone())
}
}
impl ToExtent for Timestamp {
fn to_extent(&self) -> Option<Extent> {
Some(Extent::point(*self))
}
}
impl ToExtent for Range<Timestamp> {
fn to_extent(&self) -> Option<Extent> {
Some(Extent::span(self.clone()))
}
}
impl ToExtent for Range<Option<Timestamp>> {
fn to_extent(&self) -> Option<Extent> {
match (self.start, self.end) {
(Some(start), Some(end)) => (start..end).to_extent(),
(Some(start), None) => start.to_extent(),
(None, Some(end)) => end.to_extent(),
(None, None) => None::<Timestamp>.to_extent(),
}
}
}