use super::validate_identity;
use crate::error::ModelError;
pub const TASK_ID_MAX_LEN: usize = 256;
arc_str_newtype! {
pub struct TaskId;
}
impl TaskId {
pub fn validate_format(&self) -> Result<(), ModelError> {
validate_identity("task_id", self.as_str(), TASK_ID_MAX_LEN)
}
}
#[cfg(test)]
mod tests {
use super::*;
use std::sync::Arc;
#[test]
fn task_id_from_string() {
let id = TaskId::from("subprocess-slot-2a");
assert_eq!(id.as_str(), "subprocess-slot-2a");
}
#[test]
fn task_id_display() {
let id = TaskId::new("test-id");
assert_eq!(format!("{}", id), "test-id");
}
#[test]
fn task_id_serde_transparent() {
let id = TaskId::from("runner-slot-ff");
let json = serde_json::to_string(&id).unwrap();
assert_eq!(json, r#""runner-slot-ff""#);
let back: TaskId = serde_json::from_str(&json).unwrap();
assert_eq!(back, id);
}
#[test]
fn task_id_hash_equality() {
use std::collections::HashSet;
let mut set = HashSet::new();
set.insert(TaskId::from("id-1"));
set.insert(TaskId::from("id-2"));
set.insert(TaskId::from("id-1"));
assert_eq!(set.len(), 2);
assert!(set.contains(&TaskId::from("id-1")));
}
#[test]
fn clone_is_cheap() {
let id = TaskId::new("shared-task");
let cloned = id.clone();
let a: Arc<str> = id.into_inner();
let b: Arc<str> = cloned.into_inner();
assert!(Arc::ptr_eq(&a, &b));
}
#[test]
fn validate_format_accepts_runner_generated() {
TaskId::new("subprocess-build-1").validate_format().unwrap();
TaskId::new("subprocess-build.frontend-ff")
.validate_format()
.unwrap();
}
#[test]
fn validate_format_rejects_invalid() {
assert!(TaskId::new("").validate_format().is_err());
assert!(TaskId::new("with/slash").validate_format().is_err());
assert!(TaskId::new("with space").validate_format().is_err());
assert!(TaskId::new(&"x".repeat(257)).validate_format().is_err());
}
}