1use crate::id::{IDCollection, define_id_type};
6use crate::units::{Dimensionless, Year};
7use anyhow::{Context, Result};
8use indexmap::{IndexMap, IndexSet};
9use itertools::Itertools;
10use serde::de::Error;
11use serde::{Deserialize, Serialize};
12use serde_string_enum::DeserializeLabeledStringEnum;
13use std::fmt::Display;
14use std::iter;
15
16define_id_type! {Season, "season"}
17define_id_type! {TimeOfDay, "time of day"}
18
19#[derive(Hash, Eq, PartialEq, Ord, PartialOrd, Clone, Debug)]
21pub struct TimeSliceID {
22 pub season: Season,
24 pub time_of_day: TimeOfDay,
26}
27
28#[cfg(test)]
30impl From<&str> for TimeSliceID {
31 fn from(value: &str) -> Self {
32 let (season, time_of_day) = value
33 .split('.')
34 .collect_tuple()
35 .expect("Time slice not in form season.time_of_day");
36 TimeSliceID {
37 season: season.into(),
38 time_of_day: time_of_day.into(),
39 }
40 }
41}
42
43impl Display for TimeSliceID {
44 fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result {
45 write!(f, "{}.{}", self.season, self.time_of_day)
46 }
47}
48
49impl<'de> Deserialize<'de> for TimeSliceID {
50 fn deserialize<D>(deserializer: D) -> std::result::Result<Self, D::Error>
51 where
52 D: serde::Deserializer<'de>,
53 {
54 let s: &str = Deserialize::deserialize(deserializer)?;
55 let (season, time_of_day) = s.split('.').collect_tuple().ok_or_else(|| {
56 D::Error::custom(format!(
57 "Invalid input '{s}': Should be in form season.time_of_day"
58 ))
59 })?;
60 Ok(Self {
61 season: season.into(),
62 time_of_day: time_of_day.into(),
63 })
64 }
65}
66
67impl Serialize for TimeSliceID {
68 fn serialize<S>(&self, serializer: S) -> std::result::Result<S::Ok, S::Error>
69 where
70 S: serde::Serializer,
71 {
72 serializer.collect_str(self)
73 }
74}
75
76#[derive(PartialEq, Eq, Hash, Clone, Debug)]
78pub enum TimeSliceSelection {
79 Annual,
81 Season(Season),
83 Single(TimeSliceID),
85}
86
87impl TimeSliceSelection {
88 pub fn level(&self) -> TimeSliceLevel {
90 match self {
91 Self::Annual => TimeSliceLevel::Annual,
92 Self::Season(_) => TimeSliceLevel::Season,
93 Self::Single(_) => TimeSliceLevel::DayNight,
94 }
95 }
96
97 pub fn iter<'a>(
99 &'a self,
100 time_slice_info: &'a TimeSliceInfo,
101 ) -> Box<dyn Iterator<Item = (&'a TimeSliceID, Year)> + 'a> {
102 let ts_info = time_slice_info;
103 match self {
104 Self::Annual => Box::new(ts_info.iter()),
105 Self::Season(season) => {
106 Box::new(ts_info.iter().filter(move |(ts, _)| ts.season == *season))
107 }
108 Self::Single(ts) => Box::new(iter::once((ts, ts_info.time_slices[ts]))),
109 }
110 }
111
112 pub fn iter_at_level<'a>(
123 &'a self,
124 time_slice_info: &'a TimeSliceInfo,
125 level: TimeSliceLevel,
126 ) -> Option<Box<dyn Iterator<Item = (Self, Year)> + 'a>> {
127 if level > self.level() {
128 return None;
129 }
130
131 let ts_info = time_slice_info;
132 let iter: Box<dyn Iterator<Item = _>> = match self {
133 Self::Annual => match level {
134 TimeSliceLevel::Annual => Box::new(iter::once((Self::Annual, Year(1.0)))),
135 TimeSliceLevel::Season => Box::new(
136 ts_info
137 .seasons
138 .iter()
139 .map(|(season, duration)| (season.clone().into(), *duration)),
140 ),
141 TimeSliceLevel::DayNight => Box::new(
142 ts_info
143 .time_slices
144 .iter()
145 .map(|(ts, duration)| (ts.clone().into(), *duration)),
146 ),
147 },
148 Self::Season(season) => match level {
149 TimeSliceLevel::Season => {
150 Box::new(iter::once((self.clone(), ts_info.seasons[season])))
151 }
152 TimeSliceLevel::DayNight => Box::new(
153 ts_info
154 .time_slices
155 .iter()
156 .filter(move |(ts, _)| &ts.season == season)
157 .map(|(ts, duration)| (ts.clone().into(), *duration)),
158 ),
159 TimeSliceLevel::Annual => unreachable!(),
160 },
161 Self::Single(time_slice) => Box::new(iter::once((
162 time_slice.clone().into(),
163 ts_info.time_slices[time_slice],
164 ))),
165 };
166
167 Some(iter)
168 }
169}
170
171impl From<TimeSliceID> for TimeSliceSelection {
172 fn from(value: TimeSliceID) -> Self {
173 Self::Single(value)
174 }
175}
176
177impl From<Season> for TimeSliceSelection {
178 fn from(value: Season) -> Self {
179 Self::Season(value)
180 }
181}
182
183impl Display for TimeSliceSelection {
184 fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result {
185 match self {
186 Self::Annual => write!(f, "annual"),
187 Self::Season(season) => write!(f, "{season}"),
188 Self::Single(ts) => write!(f, "{ts}"),
189 }
190 }
191}
192
193#[derive(
195 PartialEq, PartialOrd, Copy, Clone, Debug, DeserializeLabeledStringEnum, strum::EnumIter,
196)]
197pub enum TimeSliceLevel {
198 #[string = "daynight"]
200 DayNight,
201 #[string = "season"]
203 Season,
204 #[string = "annual"]
206 Annual,
207}
208
209impl TimeSliceLevel {
210 pub fn containing_selection(&self, ts: &TimeSliceID) -> TimeSliceSelection {
212 match self {
213 Self::Annual => TimeSliceSelection::Annual,
214 Self::Season => TimeSliceSelection::Season(ts.season.clone()),
215 Self::DayNight => TimeSliceSelection::Single(ts.clone()),
216 }
217 }
218}
219
220#[derive(PartialEq, Debug)]
222pub struct TimeSliceInfo {
223 pub times_of_day: IndexSet<TimeOfDay>,
225 pub seasons: IndexMap<Season, Year>,
227 pub time_slices: IndexMap<TimeSliceID, Year>,
229}
230
231impl Default for TimeSliceInfo {
232 fn default() -> Self {
234 let id = TimeSliceID {
235 season: "all-year".into(),
236 time_of_day: "all-day".into(),
237 };
238 let time_slices = [(id.clone(), Year(1.0))].into_iter().collect();
239
240 Self {
241 seasons: iter::once((id.season, Year(1.0))).collect(),
242 times_of_day: iter::once(id.time_of_day).collect(),
243 time_slices,
244 }
245 }
246}
247
248impl TimeSliceInfo {
249 #[allow(clippy::doc_markdown)]
250 pub fn get_time_slice_id_from_str(&self, time_slice: &str) -> Result<TimeSliceID> {
254 let (season, time_of_day) = time_slice
255 .split('.')
256 .collect_tuple()
257 .context("Time slice must be in the form season.time_of_day")?;
258 let season = self.seasons.get_id(season)?;
259 let time_of_day = self.times_of_day.get_id(time_of_day)?;
260
261 Ok(TimeSliceID {
262 season: season.clone(),
263 time_of_day: time_of_day.clone(),
264 })
265 }
266
267 pub fn get_selection(&self, time_slice: &str) -> Result<TimeSliceSelection> {
271 if time_slice.eq_ignore_ascii_case("annual") {
272 Ok(TimeSliceSelection::Annual)
273 } else if time_slice.contains('.') {
274 let time_slice = self.get_time_slice_id_from_str(time_slice)?;
275 Ok(TimeSliceSelection::Single(time_slice))
276 } else {
277 let season = self.seasons.get_id(time_slice)?.clone();
278 Ok(TimeSliceSelection::Season(season))
279 }
280 }
281
282 pub fn iter_ids(&self) -> indexmap::map::Keys<'_, TimeSliceID, Year> {
284 self.time_slices.keys()
285 }
286
287 pub fn iter_seasons(&self) -> indexmap::map::Keys<'_, Season, Year> {
289 self.seasons.keys()
290 }
291
292 pub fn iter(&self) -> impl Iterator<Item = (&TimeSliceID, Year)> {
294 self.time_slices
295 .iter()
296 .map(|(ts, duration)| (ts, *duration))
297 }
298
299 pub fn iter_selections_at_level(
304 &self,
305 level: TimeSliceLevel,
306 ) -> Box<dyn Iterator<Item = TimeSliceSelection> + '_> {
307 match level {
308 TimeSliceLevel::Annual => Box::new(iter::once(TimeSliceSelection::Annual)),
309 TimeSliceLevel::Season => {
310 Box::new(self.seasons.keys().cloned().map(TimeSliceSelection::Season))
311 }
312 TimeSliceLevel::DayNight => {
313 Box::new(self.iter_ids().cloned().map(TimeSliceSelection::Single))
314 }
315 }
316 }
317
318 pub fn iter_selection_share<'a>(
330 &'a self,
331 selection: &'a TimeSliceSelection,
332 level: TimeSliceLevel,
333 ) -> Option<impl Iterator<Item = (TimeSliceSelection, Dimensionless)> + use<>> {
334 let selections = selection.iter_at_level(self, level)?.collect_vec();
336
337 let total_duration: Year = selections.iter().map(|(_, duration)| *duration).sum();
339
340 let iter = selections
342 .into_iter()
343 .map(move |(selection, duration)| (selection, duration / total_duration));
344 Some(iter)
345 }
346
347 pub fn length_for_selection(&self, selection: &TimeSliceSelection) -> Result<Year> {
349 let length: Year = selection.iter(self).map(|(_, duration)| duration).sum();
350 Ok(length)
351 }
352
353 pub fn calculate_share<'a>(
367 &'a self,
368 selection: &'a TimeSliceSelection,
369 level: TimeSliceLevel,
370 value: Dimensionless,
371 ) -> Option<impl Iterator<Item = (TimeSliceSelection, Dimensionless)> + use<>> {
372 let iter = self
373 .iter_selection_share(selection, level)?
374 .map(move |(selection, share)| (selection, value * share));
375 Some(iter)
376 }
377}
378
379#[cfg(test)]
380mod tests {
381 use super::*;
382 use crate::units::UnitType;
383 use itertools::assert_equal;
384 use rstest::{fixture, rstest};
385
386 #[fixture]
387 fn time_slices1() -> [TimeSliceID; 2] {
388 [
389 TimeSliceID {
390 season: "winter".into(),
391 time_of_day: "day".into(),
392 },
393 TimeSliceID {
394 season: "summer".into(),
395 time_of_day: "night".into(),
396 },
397 ]
398 }
399
400 #[fixture]
401 fn time_slice_info1(time_slices1: [TimeSliceID; 2]) -> TimeSliceInfo {
402 TimeSliceInfo {
403 seasons: [("winter".into(), Year(0.5)), ("summer".into(), Year(0.5))]
404 .into_iter()
405 .collect(),
406 times_of_day: ["day".into(), "night".into()].into_iter().collect(),
407 time_slices: time_slices1.map(|ts| (ts, Year(0.5))).into_iter().collect(),
408 }
409 }
410
411 #[fixture]
412 fn time_slice_info2() -> TimeSliceInfo {
413 let time_slices = [
414 TimeSliceID {
415 season: "winter".into(),
416 time_of_day: "day".into(),
417 },
418 TimeSliceID {
419 season: "winter".into(),
420 time_of_day: "night".into(),
421 },
422 TimeSliceID {
423 season: "summer".into(),
424 time_of_day: "day".into(),
425 },
426 TimeSliceID {
427 season: "summer".into(),
428 time_of_day: "night".into(),
429 },
430 ];
431 TimeSliceInfo {
432 times_of_day: ["day".into(), "night".into()].into_iter().collect(),
433 seasons: [("winter".into(), Year(0.5)), ("summer".into(), Year(0.5))]
434 .into_iter()
435 .collect(),
436 time_slices: time_slices
437 .iter()
438 .map(|ts| (ts.clone(), Year(0.25)))
439 .collect(),
440 }
441 }
442
443 #[rstest]
444 fn ts_selection_iter_annual(time_slice_info1: TimeSliceInfo, time_slices1: [TimeSliceID; 2]) {
445 assert_equal(
446 TimeSliceSelection::Annual.iter(&time_slice_info1),
447 time_slices1.iter().map(|ts| (ts, Year(0.5))),
448 );
449 }
450
451 #[rstest]
452 fn ts_selection_iter_season(time_slice_info1: TimeSliceInfo, time_slices1: [TimeSliceID; 2]) {
453 assert_equal(
454 TimeSliceSelection::Season("winter".into()).iter(&time_slice_info1),
455 iter::once((&time_slices1[0], Year(0.5))),
456 );
457 }
458
459 #[rstest]
460 fn ts_selection_iter_single(time_slice_info1: TimeSliceInfo, time_slices1: [TimeSliceID; 2]) {
461 let ts = time_slice_info1
462 .get_time_slice_id_from_str("summer.night")
463 .unwrap();
464 assert_equal(
465 TimeSliceSelection::Single(ts).iter(&time_slice_info1),
466 iter::once((&time_slices1[1], Year(0.5))),
467 );
468 }
469
470 fn assert_selection_equal<I, T>(actual: Option<I>, expected: Option<Vec<(&str, T)>>)
471 where
472 T: UnitType,
473 I: Iterator<Item = (TimeSliceSelection, T)>,
474 {
475 let Some(actual) = actual else {
476 assert!(expected.is_none());
477 return;
478 };
479
480 let ts_info = time_slice_info2();
481 let expected = expected
482 .unwrap()
483 .into_iter()
484 .map(move |(sel, frac)| (ts_info.get_selection(sel).unwrap(), frac));
485 assert_equal(actual, expected);
486 }
487
488 #[rstest]
489 #[case(TimeSliceSelection::Annual, TimeSliceLevel::Annual, Some(vec![("annual", Year(1.0))]))]
490 #[case(TimeSliceSelection::Annual, TimeSliceLevel::Season, Some(vec![("winter", Year(0.5)), ("summer", Year(0.5))]))]
491 #[case(TimeSliceSelection::Annual, TimeSliceLevel::DayNight,
492 Some(vec![("winter.day", Year(0.25)), ("winter.night", Year(0.25)), ("summer.day", Year(0.25)), ("summer.night", Year(0.25))]))]
493 #[case(TimeSliceSelection::Season("winter".into()), TimeSliceLevel::Annual, None)]
494 #[case(TimeSliceSelection::Season("winter".into()), TimeSliceLevel::Season, Some(vec![("winter", Year(0.5))]))]
495 #[case(TimeSliceSelection::Season("winter".into()), TimeSliceLevel::DayNight,
496 Some(vec![("winter.day", Year(0.25)), ("winter.night", Year(0.25))]))]
497 #[case(TimeSliceSelection::Single("winter.day".into()), TimeSliceLevel::Annual, None)]
498 #[case(TimeSliceSelection::Single("winter.day".into()), TimeSliceLevel::Season, None)]
499 #[case(TimeSliceSelection::Single("winter.day".into()), TimeSliceLevel::DayNight, Some(vec![("winter.day", Year(0.25))]))]
500 fn ts_selection_iter_at_level(
501 time_slice_info2: TimeSliceInfo,
502 #[case] selection: TimeSliceSelection,
503 #[case] level: TimeSliceLevel,
504 #[case] expected: Option<Vec<(&str, Year)>>,
505 ) {
506 let actual = selection.iter_at_level(&time_slice_info2, level);
507 assert_selection_equal(actual, expected);
508 }
509
510 #[rstest]
511 #[case(TimeSliceSelection::Annual, TimeSliceLevel::Annual, Some(vec![("annual", Dimensionless(8.0))]))]
512 #[case(TimeSliceSelection::Annual, TimeSliceLevel::Season, Some(vec![("winter", Dimensionless(4.0)), ("summer", Dimensionless(4.0))]))]
513 #[case(TimeSliceSelection::Annual, TimeSliceLevel::DayNight,
514 Some(vec![("winter.day", Dimensionless(2.0)), ("winter.night", Dimensionless(2.0)), ("summer.day", Dimensionless(2.0)), ("summer.night", Dimensionless(2.0))]))]
515 #[case(TimeSliceSelection::Season("winter".into()), TimeSliceLevel::Annual, None)]
516 #[case(TimeSliceSelection::Season("winter".into()), TimeSliceLevel::Season, Some(vec![("winter", Dimensionless(8.0))]))]
517 #[case(TimeSliceSelection::Season("winter".into()), TimeSliceLevel::DayNight,
518 Some(vec![("winter.day", Dimensionless(4.0)), ("winter.night", Dimensionless(4.0))]))]
519 #[case(TimeSliceSelection::Single("winter.day".into()), TimeSliceLevel::Annual, None)]
520 #[case(TimeSliceSelection::Single("winter.day".into()), TimeSliceLevel::Season, None)]
521 #[case(TimeSliceSelection::Single("winter.day".into()), TimeSliceLevel::DayNight, Some(vec![("winter.day", Dimensionless(8.0))]))]
522 fn calculate_share(
523 time_slice_info2: TimeSliceInfo,
524 #[case] selection: TimeSliceSelection,
525 #[case] level: TimeSliceLevel,
526 #[case] expected: Option<Vec<(&str, Dimensionless)>>,
527 ) {
528 let actual = time_slice_info2.calculate_share(&selection, level, Dimensionless(8.0));
529 assert_selection_equal(actual, expected);
530 }
531}