#[cfg(not(all(target_arch = "wasm32", target_os = "unknown")))]
use std::{convert::TryInto, time::SystemTime};
use std::{
fmt,
str::FromStr,
sync::atomic::{AtomicUsize, Ordering},
};
use once_cell::sync::Lazy;
use rand::{random, rng, Rng};
use crate::error::{Error, Result};
const TIMESTAMP_SIZE: usize = 4;
const PROCESS_ID_SIZE: usize = 5;
const COUNTER_SIZE: usize = 3;
const TIMESTAMP_OFFSET: usize = 0;
const PROCESS_ID_OFFSET: usize = TIMESTAMP_OFFSET + TIMESTAMP_SIZE;
const COUNTER_OFFSET: usize = PROCESS_ID_OFFSET + PROCESS_ID_SIZE;
const MAX_U24: usize = 0xFF_FFFF;
static OID_COUNTER: Lazy<AtomicUsize> =
Lazy::new(|| AtomicUsize::new(rng().random_range(0..=MAX_U24)));
#[derive(Clone, Copy, PartialEq, PartialOrd, Eq, Ord, Hash)]
pub struct ObjectId {
id: [u8; 12],
}
impl Default for ObjectId {
fn default() -> Self {
Self::new()
}
}
impl FromStr for ObjectId {
type Err = Error;
fn from_str(s: &str) -> std::result::Result<Self, Self::Err> {
Self::parse_str(s)
}
}
impl From<[u8; 12]> for ObjectId {
fn from(bytes: [u8; 12]) -> Self {
Self { id: bytes }
}
}
impl ObjectId {
pub fn new() -> Self {
let timestamp = Self::gen_timestamp();
let process_id = Self::gen_process_id();
let counter = Self::gen_count();
Self::from_parts(timestamp, process_id, counter)
}
pub const fn from_bytes(bytes: [u8; 12]) -> ObjectId {
ObjectId { id: bytes }
}
pub fn from_parts(seconds_since_epoch: u32, process_id: [u8; 5], counter: [u8; 3]) -> Self {
let mut bytes = [0; 12];
bytes[TIMESTAMP_OFFSET..(TIMESTAMP_OFFSET + TIMESTAMP_SIZE)]
.clone_from_slice(&u32::to_be_bytes(seconds_since_epoch));
bytes[PROCESS_ID_OFFSET..(PROCESS_ID_OFFSET + PROCESS_ID_SIZE)]
.clone_from_slice(&process_id);
bytes[COUNTER_OFFSET..(COUNTER_OFFSET + COUNTER_SIZE)].clone_from_slice(&counter);
Self::from_bytes(bytes)
}
pub fn parse_str(s: impl AsRef<str>) -> Result<ObjectId> {
let s = s.as_ref();
let bytes: Vec<u8> =
hex::decode(s.as_bytes()).map_err(|e| Error::from_hex_error(e, s.len()))?;
if bytes.len() != 12 {
Err(Error::oid_invalid_length(bytes.len()))
} else {
let mut byte_array: [u8; 12] = [0; 12];
byte_array[..].copy_from_slice(&bytes[..]);
Ok(ObjectId::from_bytes(byte_array))
}
}
pub fn timestamp(&self) -> crate::DateTime {
let mut buf = [0; 4];
buf.copy_from_slice(&self.id[0..4]);
let seconds_since_epoch = u32::from_be_bytes(buf);
crate::DateTime::from_millis(seconds_since_epoch as i64 * 1000)
}
pub const fn bytes(&self) -> [u8; 12] {
self.id
}
pub fn to_hex(self) -> String {
hex::encode(self.id)
}
fn gen_timestamp() -> u32 {
#[cfg(all(target_arch = "wasm32", target_os = "unknown"))]
let timestamp: u32 = (js_sys::Date::now() / 1000.0) as u32;
#[cfg(not(all(target_arch = "wasm32", target_os = "unknown")))]
let timestamp: u32 = SystemTime::now()
.duration_since(SystemTime::UNIX_EPOCH)
.expect("system clock is before 1970")
.as_secs()
.try_into()
.unwrap();
timestamp
}
fn gen_process_id() -> [u8; 5] {
static BUF: Lazy<[u8; 5]> = Lazy::new(random);
*BUF
}
fn gen_count() -> [u8; 3] {
let u_counter = OID_COUNTER.fetch_add(1, Ordering::SeqCst);
let u = u_counter % (MAX_U24 + 1);
let u_int = u as u64;
let buf = u_int.to_be_bytes();
let buf_u24: [u8; 3] = [buf[5], buf[6], buf[7]];
buf_u24
}
}
impl fmt::Display for ObjectId {
fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result {
f.write_str(&self.to_hex())
}
}
impl fmt::Debug for ObjectId {
fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result {
f.debug_tuple("ObjectId").field(&self.to_hex()).finish()
}
}
#[cfg(test)]
use crate::tests::LOCK;
#[test]
fn count_generated_is_big_endian() {
let _guard = LOCK.run_exclusively();
let start = 1_122_866;
OID_COUNTER.store(start, Ordering::SeqCst);
let count_bytes = ObjectId::gen_count();
let mut buf: [u8; 4] = [0; 4];
buf[1..=COUNTER_SIZE].clone_from_slice(&count_bytes[..COUNTER_SIZE]);
let count = u32::from_be_bytes(buf);
assert_eq!(start as u32, count);
let oid = ObjectId::new();
assert_eq!(0x11u8, oid.bytes()[COUNTER_OFFSET]);
assert_eq!(0x22u8, oid.bytes()[COUNTER_OFFSET + 1]);
assert_eq!(0x33u8, oid.bytes()[COUNTER_OFFSET + 2]);
}
#[test]
fn test_counter_overflow_u24_max() {
let _guard = LOCK.run_exclusively();
let start = MAX_U24;
OID_COUNTER.store(start, Ordering::SeqCst);
let oid = ObjectId::new();
assert_eq!(0xFFu8, oid.bytes()[COUNTER_OFFSET]);
assert_eq!(0xFFu8, oid.bytes()[COUNTER_OFFSET + 1]);
assert_eq!(0xFFu8, oid.bytes()[COUNTER_OFFSET + 2]);
let oid_new = ObjectId::new();
assert_eq!(0x00u8, oid_new.bytes()[COUNTER_OFFSET]);
assert_eq!(0x00u8, oid_new.bytes()[COUNTER_OFFSET + 1]);
assert_eq!(0x00u8, oid_new.bytes()[COUNTER_OFFSET + 2]);
}
#[test]
fn test_counter_overflow_usize_max() {
let _guard = LOCK.run_exclusively();
let start = usize::MAX;
OID_COUNTER.store(start, Ordering::SeqCst);
let oid = ObjectId::new();
assert_eq!(0xFFu8, oid.bytes()[COUNTER_OFFSET]);
assert_eq!(0xFFu8, oid.bytes()[COUNTER_OFFSET + 1]);
assert_eq!(0xFFu8, oid.bytes()[COUNTER_OFFSET + 2]);
let oid_new = ObjectId::new();
assert_eq!(0x00u8, oid_new.bytes()[COUNTER_OFFSET]);
assert_eq!(0x00u8, oid_new.bytes()[COUNTER_OFFSET + 1]);
assert_eq!(0x00u8, oid_new.bytes()[COUNTER_OFFSET + 2]);
}
#[cfg(test)]
mod test {
use time::macros::datetime;
#[test]
fn test_display() {
let id = super::ObjectId::parse_str("53e37d08776f724e42000000").unwrap();
assert_eq!(format!("{}", id), "53e37d08776f724e42000000")
}
#[test]
fn test_debug() {
let id = super::ObjectId::parse_str("53e37d08776f724e42000000").unwrap();
assert_eq!(
format!("{:?}", id),
"ObjectId(\"53e37d08776f724e42000000\")"
);
assert_eq!(
format!("{:#?}", id),
"ObjectId(\n \"53e37d08776f724e42000000\",\n)"
);
}
#[test]
fn test_timestamp() {
let id = super::ObjectId::parse_str("000000000000000000000000").unwrap();
assert_eq!(datetime!(1970-01-01 0:00 UTC), id.timestamp().to_time_0_3());
let id = super::ObjectId::parse_str("7FFFFFFF0000000000000000").unwrap();
assert_eq!(
datetime!(2038-01-19 3:14:07 UTC),
id.timestamp().to_time_0_3()
);
let id = super::ObjectId::parse_str("800000000000000000000000").unwrap();
assert_eq!(
datetime!(2038-01-19 3:14:08 UTC),
id.timestamp().to_time_0_3()
);
let id = super::ObjectId::parse_str("FFFFFFFF0000000000000000").unwrap();
assert_eq!(
datetime!(2106-02-07 6:28:15 UTC),
id.timestamp().to_time_0_3()
);
}
}