//! ID generation utility based on [cuid2] use cfg_if::cfg_if; use once_cell::sync::OnceCell; use crate::impl_into_napi_error; #[derive(thiserror::Error, Debug, PartialEq, Eq)] #[error("ID generator has not been initialized yet")] pub struct ErrorUninitialized; impl_into_napi_error!(ErrorUninitialized); static FINGERPRINT: OnceCell = OnceCell::new(); static GENERATOR: OnceCell = OnceCell::new(); /// Initializes Cuid2 generator. Must be called before any [create_id]. pub fn init_id(length: u16, fingerprint: impl Into) { FINGERPRINT.get_or_init(move || format!("{}{}", fingerprint.into(), cuid2::create_id())); GENERATOR.get_or_init(move || { cuid2::CuidConstructor::new() .with_length(length) .with_fingerprinter(|| FINGERPRINT.get().unwrap().clone()) }); } /// Returns Cuid2 with the length specified by [init_id]. Must be called after /// [init_id], otherwise returns [ErrorUninitialized]. pub fn create_id() -> Result { match GENERATOR.get() { None => Err(ErrorUninitialized), Some(gen) => Ok(gen.create_id()), } } cfg_if! { if #[cfg(feature = "napi")] { use radix_fmt::radix_36; use std::cmp; use napi::bindgen_prelude::BigInt; use napi_derive::napi; const TIME_2000: u64 = 946_684_800_000; /// Calls [init_id] inside. Must be called before [native_create_id]. #[napi] pub fn native_init_id_generator(length: u16, fingerprint: String) { init_id(length, fingerprint); } /// Generates #[napi] pub fn native_create_id(date_num: BigInt) -> String { let time = cmp::max(date_num.get_u64().1 - TIME_2000, 0); format!("{:0>8}{}", radix_36(time).to_string(), create_id().unwrap()) } } } #[cfg(test)] mod unit_test { use pretty_assertions::{assert_eq, assert_ne}; use std::thread; use crate::util::id; #[test] fn can_generate_unique_ids() { assert_eq!(id::create_id(), Err(id::ErrorUninitialized)); id::init_id(12, ""); assert_eq!(id::create_id().unwrap().len(), 12); assert_ne!(id::create_id().unwrap(), id::create_id().unwrap()); let id1 = thread::spawn(|| id::create_id().unwrap()); let id2 = thread::spawn(|| id::create_id().unwrap()); assert_ne!(id1.join().unwrap(), id2.join().unwrap()) } }