1#![deny(rust_2018_idioms)]
4
5pub const SCHEMA_VERSION: &str = "1.0";
9
10pub fn new_task_id() -> String {
16 format!(
17 "tj-{}",
18 &ulid::Ulid::new().to_string()[10..20].to_lowercase()
19 )
20}
21
22#[cfg(test)]
23mod task_id_tests {
24 use super::new_task_id;
25 use std::collections::HashSet;
26
27 #[test]
28 fn new_task_id_has_expected_shape() {
29 let id = new_task_id();
30 assert!(id.starts_with("tj-"), "{id}");
31 assert_eq!(id.len(), 13, "{id}");
32 assert!(
33 id[3..]
34 .chars()
35 .all(|c| c.is_ascii_lowercase() || c.is_ascii_digit()),
36 "{id}"
37 );
38 }
39
40 #[test]
41 fn new_task_id_unique_over_ten_thousand() {
42 let mut seen = HashSet::with_capacity(10_000);
43 for _ in 0..10_000 {
44 let id = new_task_id();
45 assert!(seen.insert(id.clone()), "collision: {id}");
46 }
47 }
48}
49
50pub mod artifacts;
51pub mod classifier;
52pub mod db;
53pub mod event;
54pub mod fts;
55pub mod pack;
56pub mod paths;
57pub mod project_hash;
58pub mod session;
59pub mod storage;
60
61#[cfg(test)]
62mod schema_version_tests {
63 #[test]
67 fn pack_assembler_does_not_inline_schema_version_literal() {
68 let pack_src = include_str!("pack.rs");
69 assert!(
70 !pack_src.contains("schema_version: \""),
71 "pack.rs has an inline schema_version string literal — use crate::SCHEMA_VERSION"
72 );
73 }
74
75 #[test]
76 fn schema_version_matches_event_default() {
77 let evt = crate::event::Event::new(
78 "tj-x",
79 crate::event::EventType::Open,
80 crate::event::Author::User,
81 crate::event::Source::Cli,
82 "x".into(),
83 );
84 assert_eq!(evt.schema_version, super::SCHEMA_VERSION);
85 }
86}