use serde::{Deserialize, Serialize};
use std::{fmt, str::FromStr};
#[derive(Clone, Debug, Deserialize, Eq, Hash, Ord, PartialEq, PartialOrd, Serialize)]
pub struct StableKey(String);
impl StableKey {
pub fn parse(value: impl AsRef<str>) -> Result<Self, StableKeyError> {
validate(value.as_ref())?;
Ok(Self(value.as_ref().to_string()))
}
#[must_use]
pub fn as_str(&self) -> &str {
&self.0
}
#[must_use]
pub fn into_string(self) -> String {
self.0
}
}
impl AsRef<str> for StableKey {
fn as_ref(&self) -> &str {
self.as_str()
}
}
impl fmt::Display for StableKey {
fn fmt(&self, formatter: &mut fmt::Formatter<'_>) -> fmt::Result {
formatter.write_str(self.as_str())
}
}
impl FromStr for StableKey {
type Err = StableKeyError;
fn from_str(value: &str) -> Result<Self, Self::Err> {
Self::parse(value)
}
}
#[derive(Clone, Debug, Eq, thiserror::Error, PartialEq)]
#[error("stable key '{stable_key}' is invalid: {reason}")]
pub struct StableKeyError {
pub stable_key: String,
pub reason: &'static str,
}
fn validate(stable_key: &str) -> Result<(), StableKeyError> {
if stable_key.is_empty() {
return invalid(stable_key, "must not be empty");
}
if stable_key.len() > 128 {
return invalid(stable_key, "must be at most 128 bytes");
}
if !stable_key.is_ascii() {
return invalid(stable_key, "must be ASCII");
}
if stable_key.bytes().any(|byte| byte.is_ascii_uppercase()) {
return invalid(stable_key, "must be lowercase");
}
if stable_key.contains(char::is_whitespace) {
return invalid(stable_key, "must not contain whitespace");
}
if stable_key.contains('/') || stable_key.contains('-') {
return invalid(stable_key, "must not contain slashes or hyphens");
}
if stable_key.starts_with('.') || stable_key.ends_with('.') {
return invalid(stable_key, "must not start or end with a dot");
}
let Some(version_index) = stable_key.rfind(".v") else {
return invalid(stable_key, "must end with .vN");
};
let version = &stable_key[version_index + 2..];
if version.is_empty()
|| version.starts_with('0')
|| !version.bytes().all(|byte| byte.is_ascii_digit())
{
return invalid(stable_key, "version suffix must be nonzero .vN");
}
let prefix = &stable_key[..version_index];
if prefix.is_empty() {
return invalid(
stable_key,
"must contain at least one segment before version",
);
}
for segment in prefix.split('.') {
validate_segment(stable_key, segment)?;
}
Ok(())
}
fn validate_segment(stable_key: &str, segment: &str) -> Result<(), StableKeyError> {
if segment.is_empty() {
return invalid(stable_key, "must not contain empty segments");
}
let mut bytes = segment.bytes();
let Some(first) = bytes.next() else {
return invalid(stable_key, "must not contain empty segments");
};
if !first.is_ascii_lowercase() {
return invalid(stable_key, "segments must start with a lowercase letter");
}
if !bytes.all(|byte| byte.is_ascii_lowercase() || byte.is_ascii_digit() || byte == b'_') {
return invalid(
stable_key,
"segments may contain only lowercase letters, digits, and underscores",
);
}
Ok(())
}
fn invalid<T>(stable_key: &str, reason: &'static str) -> Result<T, StableKeyError> {
Err(StableKeyError {
stable_key: stable_key.to_string(),
reason,
})
}
#[cfg(test)]
mod tests {
use super::*;
#[test]
fn accepts_canonical_keys() {
assert_eq!(
StableKey::parse("app.users.primary.v1")
.expect("valid key")
.as_str(),
"app.users.primary.v1"
);
assert!(StableKey::parse("canic.core.auth_state.v12").is_ok());
}
#[test]
fn rejects_noncanonical_keys() {
for key in [
"",
"App.users.v1",
"app.users",
"app.users.v0",
"app..users.v1",
".app.users.v1",
"app.users.v1.",
"app-users.v1",
"app/users.v1",
"app.1users.v1",
] {
assert!(StableKey::parse(key).is_err(), "{key} should fail");
}
}
}