2023-05-26 10:00:09 +02:00
|
|
|
//! ID generation utility based on [cuid2]
|
|
|
|
|
2023-08-06 08:34:44 +02:00
|
|
|
use basen::BASE36;
|
2023-06-02 17:55:14 +02:00
|
|
|
use cfg_if::cfg_if;
|
2024-04-02 12:08:32 +02:00
|
|
|
use chrono::NaiveDateTime;
|
2023-05-26 10:00:09 +02:00
|
|
|
use once_cell::sync::OnceCell;
|
2023-07-10 07:39:33 +02:00
|
|
|
use std::cmp;
|
2023-05-26 10:00:09 +02:00
|
|
|
|
2023-06-02 17:55:14 +02:00
|
|
|
use crate::impl_into_napi_error;
|
|
|
|
|
2023-05-26 10:00:09 +02:00
|
|
|
#[derive(thiserror::Error, Debug, PartialEq, Eq)]
|
|
|
|
#[error("ID generator has not been initialized yet")]
|
|
|
|
pub struct ErrorUninitialized;
|
|
|
|
|
2023-06-02 17:55:14 +02:00
|
|
|
impl_into_napi_error!(ErrorUninitialized);
|
|
|
|
|
|
|
|
static FINGERPRINT: OnceCell<String> = OnceCell::new();
|
|
|
|
static GENERATOR: OnceCell<cuid2::CuidConstructor> = OnceCell::new();
|
2023-05-26 10:00:09 +02:00
|
|
|
|
2023-07-10 07:39:33 +02:00
|
|
|
const TIME_2000: i64 = 946_684_800_000;
|
|
|
|
const TIMESTAMP_LENGTH: u16 = 8;
|
|
|
|
|
2023-06-02 17:55:14 +02:00
|
|
|
/// Initializes Cuid2 generator. Must be called before any [create_id].
|
2024-03-02 06:24:05 +01:00
|
|
|
pub fn init_id(length: u16, fingerprint: &str) {
|
2023-07-10 07:39:33 +02:00
|
|
|
FINGERPRINT.get_or_init(move || format!("{}{}", fingerprint, cuid2::create_id()));
|
2023-06-02 17:55:14 +02:00
|
|
|
GENERATOR.get_or_init(move || {
|
|
|
|
cuid2::CuidConstructor::new()
|
2023-07-10 07:39:33 +02:00
|
|
|
// length to pass shoule be greater than or equal to 8.
|
|
|
|
.with_length(cmp::max(length - TIMESTAMP_LENGTH, 8))
|
2023-06-02 17:55:14 +02:00
|
|
|
.with_fingerprinter(|| FINGERPRINT.get().unwrap().clone())
|
|
|
|
});
|
2023-05-26 10:00:09 +02:00
|
|
|
}
|
|
|
|
|
2023-06-02 17:55:14 +02:00
|
|
|
/// Returns Cuid2 with the length specified by [init_id]. Must be called after
|
|
|
|
/// [init_id], otherwise returns [ErrorUninitialized].
|
2024-04-02 12:08:32 +02:00
|
|
|
pub fn create_id(datetime: &NaiveDateTime) -> Result<String, ErrorUninitialized> {
|
2023-05-26 10:00:09 +02:00
|
|
|
match GENERATOR.get() {
|
|
|
|
None => Err(ErrorUninitialized),
|
2023-07-10 07:39:33 +02:00
|
|
|
Some(gen) => {
|
2024-04-02 12:08:32 +02:00
|
|
|
let date_num = cmp::max(0, datetime.and_utc().timestamp_millis() - TIME_2000) as u64;
|
2023-07-10 07:39:33 +02:00
|
|
|
Ok(format!(
|
|
|
|
"{:0>8}{}",
|
2024-04-02 12:08:32 +02:00
|
|
|
BASE36.encode_var_len(&date_num),
|
2023-07-10 07:39:33 +02:00
|
|
|
gen.create_id()
|
|
|
|
))
|
|
|
|
}
|
2023-05-26 10:00:09 +02:00
|
|
|
}
|
2023-05-25 18:11:59 +02:00
|
|
|
}
|
|
|
|
|
2023-08-06 08:34:44 +02:00
|
|
|
pub fn get_timestamp(id: &str) -> i64 {
|
|
|
|
let n: Option<u64> = BASE36.decode_var_len(&id[0..8]);
|
|
|
|
match n {
|
|
|
|
None => -1,
|
|
|
|
Some(n) => n as i64 + TIME_2000,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-06-02 17:55:14 +02:00
|
|
|
cfg_if! {
|
|
|
|
if #[cfg(feature = "napi")] {
|
|
|
|
use napi_derive::napi;
|
2024-04-02 12:08:32 +02:00
|
|
|
use chrono::{DateTime, Utc};
|
2023-06-02 17:55:14 +02:00
|
|
|
|
|
|
|
/// Calls [init_id] inside. Must be called before [native_create_id].
|
2024-04-02 12:08:32 +02:00
|
|
|
#[napi(js_name = "initIdGenerator")]
|
2023-06-02 17:55:14 +02:00
|
|
|
pub fn native_init_id_generator(length: u16, fingerprint: String) {
|
2023-07-10 07:39:33 +02:00
|
|
|
init_id(length, &fingerprint);
|
2023-06-02 17:55:14 +02:00
|
|
|
}
|
|
|
|
|
2024-04-02 12:08:32 +02:00
|
|
|
/// The generated ID results in the form of `[8 chars timestamp] + [cuid2]`.
|
|
|
|
/// The minimum and maximum lengths are 16 and 24, respectively.
|
|
|
|
/// With the length of 16, namely 8 for cuid2, roughly 1427399 IDs are needed
|
|
|
|
/// in the same millisecond to reach 50% chance of collision.
|
|
|
|
///
|
|
|
|
/// Ref: https://github.com/paralleldrive/cuid2#parameterized-length
|
2023-06-02 17:55:14 +02:00
|
|
|
#[napi]
|
2024-04-02 12:08:32 +02:00
|
|
|
pub fn gen_id(date: Option<DateTime<Utc>>) -> String {
|
|
|
|
create_id(&date.unwrap_or_else(|| Utc::now()).naive_utc()).unwrap()
|
2023-06-02 17:55:14 +02:00
|
|
|
}
|
2023-08-06 08:34:44 +02:00
|
|
|
|
2024-04-02 12:08:32 +02:00
|
|
|
#[napi(js_name = "getTimestamp")]
|
2023-08-06 08:34:44 +02:00
|
|
|
pub fn native_get_timestamp(id: String) -> i64 {
|
|
|
|
get_timestamp(&id)
|
|
|
|
}
|
2023-06-02 17:55:14 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-05-25 18:11:59 +02:00
|
|
|
#[cfg(test)]
|
2023-05-27 12:52:15 +02:00
|
|
|
mod unit_test {
|
2023-06-02 22:10:01 +02:00
|
|
|
use crate::util::id;
|
2023-08-06 08:34:44 +02:00
|
|
|
use chrono::Utc;
|
2023-06-02 13:08:58 +02:00
|
|
|
use pretty_assertions::{assert_eq, assert_ne};
|
2023-05-26 10:00:09 +02:00
|
|
|
use std::thread;
|
2023-05-25 18:11:59 +02:00
|
|
|
|
2023-07-10 07:39:33 +02:00
|
|
|
#[test]
|
2024-04-02 12:08:32 +02:00
|
|
|
fn can_create_and_decode_id() {
|
|
|
|
let now = Utc::now().naive_utc();
|
|
|
|
assert_eq!(id::create_id(&now), Err(id::ErrorUninitialized));
|
2023-07-10 07:39:33 +02:00
|
|
|
id::init_id(16, "");
|
2024-04-02 12:08:32 +02:00
|
|
|
assert_eq!(id::create_id(&now).unwrap().len(), 16);
|
|
|
|
assert_ne!(id::create_id(&now).unwrap(), id::create_id(&now).unwrap());
|
|
|
|
let id1 = thread::spawn(move || id::create_id(&now).unwrap());
|
|
|
|
let id2 = thread::spawn(move || id::create_id(&now).unwrap());
|
2023-07-10 07:39:33 +02:00
|
|
|
assert_ne!(id1.join().unwrap(), id2.join().unwrap());
|
2023-08-06 08:34:44 +02:00
|
|
|
|
2024-04-02 12:08:32 +02:00
|
|
|
let test_id = id::create_id(&now).unwrap();
|
2023-08-06 08:34:44 +02:00
|
|
|
let timestamp = id::get_timestamp(&test_id);
|
2024-04-02 12:08:32 +02:00
|
|
|
assert_eq!(now.and_utc().timestamp_millis(), timestamp);
|
2023-05-25 18:11:59 +02:00
|
|
|
}
|
|
|
|
}
|