A lightweight, extensible system for creating type-safe IDs, email addresses, and domain-specific values using Rust's type system.
rust-tagged
provides a simple way to define strongly typed wrappers over primitive types like String
, i32
, Uuid
, chrono::DateTime
, etc. It helps eliminate bugs caused by misusing raw primitives for conceptually distinct fields such as UserId
, Email
, ProductId
, and more.
- Eliminate accidental mixups between similar types (e.g.
OrgId
vsUserId
) - Enforce domain modeling in code via the type system
- Ergonomic
.into()
support for primitive conversions - Optional serde and macro support for clean
#[derive(Tagged)]
- Phantom types (Rust Nomicon)
- Tagged unions (Wikipedia)
- Phantom types in Haskell
- Newtype pattern in Rust
- Algebraic data types
- Lightweight
Tagged<T, Tag>
abstraction From<T>
andInto<T>
implementations for easy use- Optional
Deref
,Display
,Serialize
, andDeserialize
support
[dependencies]
rust-tagged = "0.4.0"
To enable serde support:
[dependencies.rust-tagged]
version = "0.4.0"
features = ["serde"]
use tagged_core::Tagged;
#[derive(Debug)]
struct UserIdTag {
a: Tagged<u32, Self>,
b: Tagged<u32, Self>,
}
fn main() {
let instance = UserIdTag { a: 1.into(), b: 2.into() };
println!("{}", instance.a);
println!("{:?}", instance.b);
}
use tagged_core::Tagged;
use std::collections::HashSet;
#[derive(Clone, Hash, Debug, PartialEq, Eq)]
struct User {
id: Tagged<String, Self>,
}
fn main() {
let mut s: HashSet<User> = HashSet::new();
let user = User { id: "[email protected]".into() };
s.insert(user.clone());
assert!(s.contains(&user));
}
use tagged_core::Tagged;
#[derive(Debug)]
struct Org;
type EmployeeNames = Tagged<Vec<String>, Org>;
fn main() {
let names: EmployeeNames = Tagged::new(vec!["Alice".into(), "Bob".into()]);
for name in &names {
println!("Name: {name}");
}
for name in names {
println!("Owned: {name}");
}
}
use tagged_core::Tagged;
#[derive(Debug)]
struct Org;
type OrgName = Tagged<String, Org>;
fn main() {
let mut name = OrgName::new("Codefonsi".into());
name.set("New Org Name".into());
println!("Updated Org Name: {}", name.value());
}
use tagged_core::Tagged;
#[derive(Debug)]
struct EmailTag;
type Email = Tagged<String, EmailTag>;
fn main() {
let email: Email = "[email protected]".into();
println!("Email inner value: {}", email.value());
// Convert back to String
let raw: String = email.into();
println!("Raw String: {raw}");
}
use tagged_core::Tagged;
struct Employee {
id: Tagged<i32, Self>,
employee_email_id: Tagged<String, Self>,
name: String,
org: Org,
}
struct Org {
org_email_id: Tagged<String, Self>,
name: String,
}
fn send_mail_employee(mail_id: &Tagged<String, Employee>, message: &str) {
send_mail(mail_id, message);
}
fn send_mail_org(mail_id: &Tagged<String, Org>, message: &str) {
send_mail(mail_id, message);
}
fn send_mail(mail_id: &str, message: &str) {
println!("Mail Sent.{}", message);
}
fn main() {
let emp = Employee {
id: 12.into(),
employee_email_id: "[email protected]".into(),
name: "Akash".into(),
org: Org {
org_email_id: "[email protected]".into(),
name: "Codefonsi".into(),
},
};
send_mail_org(&emp.org.org_email_id, "This is ok");
send_mail_employee(&emp.employee_email_id, "This is ok");
}
Mail Sent.This is ok
Mail Sent.This is ok
use tagged_core::*;
use uuid::Uuid;
struct Org;
type OrgId = Tagged<Uuid, Org>;
type OrgEmail = Tagged<String, Org>;
#[derive(Debug)]
struct Organization {
id: OrgId,
email: OrgEmail,
}
fn main() {
let org = Organization {
id: Uuid::new_v4().into(),
email: "[email protected]".into(),
};
println!("Org ID: {}", org.id);
println!("Org Email: {}", org.email);
}
use tagged_core::*;
use chrono::{DateTime, Utc};
use serde::{Serialize, Deserialize};
struct Audit;
type CreatedAt = Tagged<DateTime<Utc>, Audit>;
type UpdatedAt = Tagged<DateTime<Utc>, Audit>;
#[derive(Serialize, Deserialize, Debug)]
struct BlogPost {
title: String,
created: CreatedAt,
updated: UpdatedAt,
}
fn main() {
let post = BlogPost {
title: "Type-Safe Rust APIs".into(),
created: Utc::now().into(),
updated: Utc::now().into(),
};
let json = serde_json::to_string_pretty(&post).unwrap();
println!("Serialized: \n{json}");
}
Licensed under either of:
- Mozilla Public License 2.0