865 lines
27 KiB
Rust
865 lines
27 KiB
Rust
// Copyright 2013-2014 The Rust Project Developers. See the COPYRIGHT
|
|
// file at the top-level directory of this distribution and at
|
|
// http://rust-lang.org/COPYRIGHT.
|
|
//
|
|
// Licensed under the Apache License, Version 2.0 <LICENSE-APACHE or
|
|
// http://www.apache.org/licenses/LICENSE-2.0> or the MIT license
|
|
// <LICENSE-MIT or http://opensource.org/licenses/MIT>, at your
|
|
// option. This file may not be copied, modified, or distributed
|
|
// except according to those terms.
|
|
|
|
/*!
|
|
Generate and parse UUIDs
|
|
|
|
Provides support for Universally Unique Identifiers (UUIDs). A UUID is a
|
|
unique 128-bit number, stored as 16 octets. UUIDs are used to assign unique
|
|
identifiers to entities without requiring a central allocating authority.
|
|
|
|
They are particularly useful in distributed systems, though can be used in
|
|
disparate areas, such as databases and network protocols. Typically a UUID is
|
|
displayed in a readable string form as a sequence of hexadecimal digits,
|
|
separated into groups by hyphens.
|
|
|
|
The uniqueness property is not strictly guaranteed, however for all practical
|
|
purposes, it can be assumed that an unintentional collision would be extremely
|
|
unlikely.
|
|
|
|
# Examples
|
|
|
|
To create a new random (V4) UUID and print it out in hexadecimal form:
|
|
|
|
```rust
|
|
# #![allow(deprecated)]
|
|
# extern crate uuid;
|
|
use uuid::Uuid;
|
|
|
|
fn main() {
|
|
let uuid1 = Uuid::new_v4();
|
|
println!("{}", uuid1.to_string());
|
|
}
|
|
```
|
|
|
|
# Strings
|
|
|
|
Examples of string representations:
|
|
|
|
* simple: `936DA01F9ABD4d9d80C702AF85C822A8`
|
|
* hyphenated: `550e8400-e29b-41d4-a716-446655440000`
|
|
* urn: `urn:uuid:F9168C5E-CEB2-4faa-B6BF-329BF39FA1E4`
|
|
|
|
# References
|
|
|
|
* [Wikipedia: Universally Unique Identifier](
|
|
http://en.wikipedia.org/wiki/Universally_unique_identifier)
|
|
* [RFC4122: A Universally Unique IDentifier (UUID) URN Namespace](
|
|
http://tools.ietf.org/html/rfc4122)
|
|
|
|
*/
|
|
|
|
#![crate_name = "uuid"]
|
|
#![deprecated = "This is now a cargo package located at: \
|
|
https://github.com/rust-lang/uuid"]
|
|
#![allow(deprecated)]
|
|
#![crate_type = "rlib"]
|
|
#![crate_type = "dylib"]
|
|
#![license = "MIT/ASL2"]
|
|
#![doc(html_logo_url = "http://www.rust-lang.org/logos/rust-logo-128x128-blk-v2.png",
|
|
html_favicon_url = "http://www.rust-lang.org/favicon.ico",
|
|
html_root_url = "http://doc.rust-lang.org/master/",
|
|
html_playground_url = "http://play.rust-lang.org/")]
|
|
|
|
#![feature(default_type_params)]
|
|
|
|
// test harness access
|
|
#[cfg(test)]
|
|
extern crate test;
|
|
extern crate serialize;
|
|
|
|
use std::char::Char;
|
|
use std::default::Default;
|
|
use std::fmt;
|
|
use std::from_str::FromStr;
|
|
use std::hash;
|
|
use std::mem::{transmute,transmute_copy};
|
|
use std::num::FromStrRadix;
|
|
use std::rand;
|
|
use std::rand::Rng;
|
|
use std::slice;
|
|
|
|
use serialize::{Encoder, Encodable, Decoder, Decodable};
|
|
|
|
/// A 128-bit (16 byte) buffer containing the ID
|
|
pub type UuidBytes = [u8, ..16];
|
|
|
|
/// The version of the UUID, denoting the generating algorithm
|
|
#[deriving(PartialEq)]
|
|
pub enum UuidVersion {
|
|
/// Version 1: MAC address
|
|
Version1Mac = 1,
|
|
/// Version 2: DCE Security
|
|
Version2Dce = 2,
|
|
/// Version 3: MD5 hash
|
|
Version3Md5 = 3,
|
|
/// Version 4: Random
|
|
Version4Random = 4,
|
|
/// Version 5: SHA-1 hash
|
|
Version5Sha1 = 5,
|
|
}
|
|
|
|
/// The reserved variants of UUIDs
|
|
#[deriving(PartialEq)]
|
|
pub enum UuidVariant {
|
|
/// Reserved by the NCS for backward compatibility
|
|
VariantNCS,
|
|
/// As described in the RFC4122 Specification (default)
|
|
VariantRFC4122,
|
|
/// Reserved by Microsoft for backward compatibility
|
|
VariantMicrosoft,
|
|
/// Reserved for future expansion
|
|
VariantFuture,
|
|
}
|
|
|
|
/// A Universally Unique Identifier (UUID)
|
|
pub struct Uuid {
|
|
/// The 128-bit number stored in 16 bytes
|
|
bytes: UuidBytes
|
|
}
|
|
|
|
impl<S: hash::Writer> hash::Hash<S> for Uuid {
|
|
fn hash(&self, state: &mut S) {
|
|
self.bytes.hash(state)
|
|
}
|
|
}
|
|
|
|
/// A UUID stored as fields (identical to UUID, used only for conversions)
|
|
struct UuidFields {
|
|
/// First field, 32-bit word
|
|
data1: u32,
|
|
/// Second field, 16-bit short
|
|
data2: u16,
|
|
/// Third field, 16-bit short
|
|
data3: u16,
|
|
/// Fourth field, 8 bytes
|
|
data4: [u8, ..8]
|
|
}
|
|
|
|
/// Error details for string parsing failures
|
|
#[allow(missing_doc)]
|
|
pub enum ParseError {
|
|
ErrorInvalidLength(uint),
|
|
ErrorInvalidCharacter(char, uint),
|
|
ErrorInvalidGroups(uint),
|
|
ErrorInvalidGroupLength(uint, uint, uint),
|
|
}
|
|
|
|
/// Converts a ParseError to a string
|
|
impl fmt::Show for ParseError {
|
|
fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result {
|
|
match *self {
|
|
ErrorInvalidLength(found) =>
|
|
write!(f, "Invalid length; expecting 32, 36 or 45 chars, \
|
|
found {}", found),
|
|
ErrorInvalidCharacter(found, pos) =>
|
|
write!(f, "Invalid character; found `{}` (0x{:02x}) at \
|
|
offset {}", found, found as uint, pos),
|
|
ErrorInvalidGroups(found) =>
|
|
write!(f, "Malformed; wrong number of groups: expected 1 \
|
|
or 5, found {}", found),
|
|
ErrorInvalidGroupLength(group, found, expecting) =>
|
|
write!(f, "Malformed; length of group {} was {}, \
|
|
expecting {}", group, found, expecting),
|
|
}
|
|
}
|
|
}
|
|
|
|
// Length of each hyphenated group in hex digits
|
|
static UuidGroupLens: [uint, ..5] = [8u, 4u, 4u, 4u, 12u];
|
|
|
|
/// UUID support
|
|
impl Uuid {
|
|
/// Returns a nil or empty UUID (containing all zeroes)
|
|
pub fn nil() -> Uuid {
|
|
let uuid = Uuid{ bytes: [0, .. 16] };
|
|
uuid
|
|
}
|
|
|
|
/// Create a new UUID of the specified version
|
|
pub fn new(v: UuidVersion) -> Option<Uuid> {
|
|
match v {
|
|
Version4Random => Some(Uuid::new_v4()),
|
|
_ => None
|
|
}
|
|
}
|
|
|
|
/// Creates a new random UUID
|
|
///
|
|
/// Uses the `rand` module's default RNG task as the source
|
|
/// of random numbers. Use the rand::Rand trait to supply
|
|
/// a custom generator if required.
|
|
pub fn new_v4() -> Uuid {
|
|
let ub = rand::task_rng().gen_iter::<u8>().take(16).collect::<Vec<_>>();
|
|
let mut uuid = Uuid{ bytes: [0, .. 16] };
|
|
slice::bytes::copy_memory(uuid.bytes, ub.as_slice());
|
|
uuid.set_variant(VariantRFC4122);
|
|
uuid.set_version(Version4Random);
|
|
uuid
|
|
}
|
|
|
|
/// Creates a UUID using the supplied field values
|
|
///
|
|
/// # Arguments
|
|
/// * `d1` A 32-bit word
|
|
/// * `d2` A 16-bit word
|
|
/// * `d3` A 16-bit word
|
|
/// * `d4` Array of 8 octets
|
|
pub fn from_fields(d1: u32, d2: u16, d3: u16, d4: &[u8]) -> Uuid {
|
|
// First construct a temporary field-based struct
|
|
let mut fields = UuidFields {
|
|
data1: 0,
|
|
data2: 0,
|
|
data3: 0,
|
|
data4: [0, ..8]
|
|
};
|
|
|
|
fields.data1 = d1.to_be();
|
|
fields.data2 = d2.to_be();
|
|
fields.data3 = d3.to_be();
|
|
slice::bytes::copy_memory(fields.data4, d4);
|
|
|
|
unsafe {
|
|
transmute(fields)
|
|
}
|
|
}
|
|
|
|
/// Creates a UUID using the supplied bytes
|
|
///
|
|
/// # Arguments
|
|
/// * `b` An array or slice of 16 bytes
|
|
pub fn from_bytes(b: &[u8]) -> Option<Uuid> {
|
|
if b.len() != 16 {
|
|
return None
|
|
}
|
|
|
|
let mut uuid = Uuid{ bytes: [0, .. 16] };
|
|
slice::bytes::copy_memory(uuid.bytes, b);
|
|
Some(uuid)
|
|
}
|
|
|
|
/// Specifies the variant of the UUID structure
|
|
fn set_variant(&mut self, v: UuidVariant) {
|
|
// Octet 8 contains the variant in the most significant 3 bits
|
|
match v {
|
|
VariantNCS => // b0xx...
|
|
self.bytes[8] = self.bytes[8] & 0x7f,
|
|
VariantRFC4122 => // b10x...
|
|
self.bytes[8] = (self.bytes[8] & 0x3f) | 0x80,
|
|
VariantMicrosoft => // b110...
|
|
self.bytes[8] = (self.bytes[8] & 0x1f) | 0xc0,
|
|
VariantFuture => // b111...
|
|
self.bytes[8] = (self.bytes[8] & 0x1f) | 0xe0,
|
|
}
|
|
}
|
|
|
|
/// Returns the variant of the UUID structure
|
|
///
|
|
/// This determines the interpretation of the structure of the UUID.
|
|
/// Currently only the RFC4122 variant is generated by this module.
|
|
///
|
|
/// * [Variant Reference](http://tools.ietf.org/html/rfc4122#section-4.1.1)
|
|
pub fn get_variant(&self) -> Option<UuidVariant> {
|
|
if self.bytes[8] & 0x80 == 0x00 {
|
|
Some(VariantNCS)
|
|
} else if self.bytes[8] & 0xc0 == 0x80 {
|
|
Some(VariantRFC4122)
|
|
} else if self.bytes[8] & 0xe0 == 0xc0 {
|
|
Some(VariantMicrosoft)
|
|
} else if self.bytes[8] & 0xe0 == 0xe0 {
|
|
Some(VariantFuture)
|
|
} else {
|
|
None
|
|
}
|
|
}
|
|
|
|
/// Specifies the version number of the UUID
|
|
fn set_version(&mut self, v: UuidVersion) {
|
|
self.bytes[6] = (self.bytes[6] & 0xF) | ((v as u8) << 4);
|
|
}
|
|
|
|
/// Returns the version number of the UUID
|
|
///
|
|
/// This represents the algorithm used to generate the contents.
|
|
///
|
|
/// Currently only the Random (V4) algorithm is supported by this
|
|
/// module. There are security and privacy implications for using
|
|
/// older versions - see [Wikipedia: Universally Unique Identifier](
|
|
/// http://en.wikipedia.org/wiki/Universally_unique_identifier) for
|
|
/// details.
|
|
///
|
|
/// * [Version Reference](http://tools.ietf.org/html/rfc4122#section-4.1.3)
|
|
pub fn get_version_num(&self) -> uint {
|
|
(self.bytes[6] >> 4) as uint
|
|
}
|
|
|
|
/// Returns the version of the UUID
|
|
///
|
|
/// This represents the algorithm used to generate the contents
|
|
pub fn get_version(&self) -> Option<UuidVersion> {
|
|
let v = self.bytes[6] >> 4;
|
|
match v {
|
|
1 => Some(Version1Mac),
|
|
2 => Some(Version2Dce),
|
|
3 => Some(Version3Md5),
|
|
4 => Some(Version4Random),
|
|
5 => Some(Version5Sha1),
|
|
_ => None
|
|
}
|
|
}
|
|
|
|
/// Return an array of 16 octets containing the UUID data
|
|
pub fn as_bytes(&self) -> &[u8] {
|
|
self.bytes.as_slice()
|
|
}
|
|
|
|
/// Returns the UUID as a string of 16 hexadecimal digits
|
|
///
|
|
/// Example: `936DA01F9ABD4d9d80C702AF85C822A8`
|
|
pub fn to_simple_str(&self) -> String {
|
|
let mut s: Vec<u8> = Vec::from_elem(32, 0u8);
|
|
for i in range(0u, 16u) {
|
|
let digit = format!("{:02x}", self.bytes[i] as uint);
|
|
*s.get_mut(i*2+0) = digit.as_bytes()[0];
|
|
*s.get_mut(i*2+1) = digit.as_bytes()[1];
|
|
}
|
|
String::from_utf8(s).unwrap()
|
|
}
|
|
|
|
/// Returns a string of hexadecimal digits, separated into groups with a hyphen.
|
|
///
|
|
/// Example: `550e8400-e29b-41d4-a716-446655440000`
|
|
pub fn to_hyphenated_str(&self) -> String {
|
|
// Convert to field-based struct as it matches groups in output.
|
|
// Ensure fields are in network byte order, as per RFC.
|
|
let mut uf: UuidFields;
|
|
unsafe {
|
|
uf = transmute_copy(&self.bytes);
|
|
}
|
|
uf.data1 = uf.data1.to_be();
|
|
uf.data2 = uf.data2.to_be();
|
|
uf.data3 = uf.data3.to_be();
|
|
let s = format!("{:08x}-{:04x}-{:04x}-{:02x}{:02x}-\
|
|
{:02x}{:02x}{:02x}{:02x}{:02x}{:02x}",
|
|
uf.data1,
|
|
uf.data2, uf.data3,
|
|
uf.data4[0], uf.data4[1],
|
|
uf.data4[2], uf.data4[3], uf.data4[4],
|
|
uf.data4[5], uf.data4[6], uf.data4[7]);
|
|
s
|
|
}
|
|
|
|
/// Returns the UUID formatted as a full URN string
|
|
///
|
|
/// This is the same as the hyphenated format, but with the "urn:uuid:" prefix.
|
|
///
|
|
/// Example: `urn:uuid:F9168C5E-CEB2-4faa-B6BF-329BF39FA1E4`
|
|
pub fn to_urn_str(&self) -> String {
|
|
format!("urn:uuid:{}", self.to_hyphenated_str())
|
|
}
|
|
|
|
/// Parses a UUID from a string of hexadecimal digits with optional hyphens
|
|
///
|
|
/// Any of the formats generated by this module (simple, hyphenated, urn) are
|
|
/// supported by this parsing function.
|
|
pub fn parse_string(us: &str) -> Result<Uuid, ParseError> {
|
|
|
|
let mut us = us.clone();
|
|
let orig_len = us.len();
|
|
|
|
// Ensure length is valid for any of the supported formats
|
|
if orig_len != 32 && orig_len != 36 && orig_len != 45 {
|
|
return Err(ErrorInvalidLength(orig_len));
|
|
}
|
|
|
|
// Strip off URN prefix if present
|
|
if us.starts_with("urn:uuid:") {
|
|
us = us.slice(9, orig_len);
|
|
}
|
|
|
|
// Make sure all chars are either hex digits or hyphen
|
|
for (i, c) in us.chars().enumerate() {
|
|
match c {
|
|
'0'..'9' | 'A'..'F' | 'a'..'f' | '-' => {},
|
|
_ => return Err(ErrorInvalidCharacter(c, i)),
|
|
}
|
|
}
|
|
|
|
// Split string up by hyphens into groups
|
|
let hex_groups: Vec<&str> = us.split_str("-").collect();
|
|
|
|
// Get the length of each group
|
|
let group_lens: Vec<uint> = hex_groups.iter().map(|&v| v.len()).collect();
|
|
|
|
// Ensure the group lengths are valid
|
|
match group_lens.len() {
|
|
// Single group, no hyphens
|
|
1 => {
|
|
if group_lens[0] != 32 {
|
|
return Err(ErrorInvalidLength(group_lens[0]));
|
|
}
|
|
},
|
|
// Five groups, hyphens in between each
|
|
5 => {
|
|
// Ensure each group length matches the expected
|
|
for (i, (&gl, &expected)) in
|
|
group_lens.iter().zip(UuidGroupLens.iter()).enumerate() {
|
|
if gl != expected {
|
|
return Err(ErrorInvalidGroupLength(i, gl, expected))
|
|
}
|
|
}
|
|
},
|
|
_ => {
|
|
return Err(ErrorInvalidGroups(group_lens.len()));
|
|
}
|
|
}
|
|
|
|
// Normalise into one long hex string
|
|
let vs = hex_groups.concat();
|
|
|
|
// At this point, we know we have a valid hex string, without hyphens
|
|
assert!(vs.len() == 32);
|
|
assert!(vs.as_slice().chars().all(|c| c.is_digit_radix(16)));
|
|
|
|
// Allocate output UUID buffer
|
|
let mut ub = [0u8, ..16];
|
|
|
|
// Extract each hex digit from the string
|
|
for i in range(0u, 16u) {
|
|
ub[i] = FromStrRadix::from_str_radix(vs.as_slice()
|
|
.slice(i*2, (i+1)*2),
|
|
16).unwrap();
|
|
}
|
|
|
|
Ok(Uuid::from_bytes(ub).unwrap())
|
|
}
|
|
|
|
/// Tests if the UUID is nil
|
|
pub fn is_nil(&self) -> bool {
|
|
return self.bytes.iter().all(|&b| b == 0);
|
|
}
|
|
}
|
|
|
|
impl Default for Uuid {
|
|
/// Returns the nil UUID, which is all zeroes
|
|
fn default() -> Uuid {
|
|
Uuid::nil()
|
|
}
|
|
}
|
|
|
|
impl Clone for Uuid {
|
|
/// Returns a copy of the UUID
|
|
fn clone(&self) -> Uuid { *self }
|
|
}
|
|
|
|
impl FromStr for Uuid {
|
|
/// Parse a hex string and interpret as a UUID
|
|
///
|
|
/// Accepted formats are a sequence of 32 hexadecimal characters,
|
|
/// with or without hyphens (grouped as 8, 4, 4, 4, 12).
|
|
fn from_str(us: &str) -> Option<Uuid> {
|
|
let result = Uuid::parse_string(us);
|
|
match result {
|
|
Ok(u) => Some(u),
|
|
Err(_) => None
|
|
}
|
|
}
|
|
}
|
|
|
|
/// Convert the UUID to a hexadecimal-based string representation
|
|
impl fmt::Show for Uuid {
|
|
fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result {
|
|
write!(f, "{}", self.to_simple_str())
|
|
}
|
|
}
|
|
|
|
/// Test two UUIDs for equality
|
|
///
|
|
/// UUIDs are equal only when they are byte-for-byte identical
|
|
impl PartialEq for Uuid {
|
|
fn eq(&self, other: &Uuid) -> bool {
|
|
self.bytes == other.bytes
|
|
}
|
|
}
|
|
|
|
impl Eq for Uuid {}
|
|
|
|
// FIXME #9845: Test these more thoroughly
|
|
impl<T: Encoder<E>, E> Encodable<T, E> for Uuid {
|
|
/// Encode a UUID as a hyphenated string
|
|
fn encode(&self, e: &mut T) -> Result<(), E> {
|
|
e.emit_str(self.to_hyphenated_str().as_slice())
|
|
}
|
|
}
|
|
|
|
impl<T: Decoder<E>, E> Decodable<T, E> for Uuid {
|
|
/// Decode a UUID from a string
|
|
fn decode(d: &mut T) -> Result<Uuid, E> {
|
|
match from_str(try!(d.read_str()).as_slice()) {
|
|
Some(decode) => Ok(decode),
|
|
None => Err(d.error("Unable to decode UUID"))
|
|
}
|
|
}
|
|
}
|
|
|
|
/// Generates a random instance of UUID (V4 conformant)
|
|
impl rand::Rand for Uuid {
|
|
#[inline]
|
|
fn rand<R: rand::Rng>(rng: &mut R) -> Uuid {
|
|
let ub = rng.gen_iter::<u8>().take(16).collect::<Vec<_>>();
|
|
let mut uuid = Uuid{ bytes: [0, .. 16] };
|
|
slice::bytes::copy_memory(uuid.bytes, ub.as_slice());
|
|
uuid.set_variant(VariantRFC4122);
|
|
uuid.set_version(Version4Random);
|
|
uuid
|
|
}
|
|
}
|
|
|
|
#[cfg(test)]
|
|
mod uuidtest {
|
|
use super::{Uuid, VariantMicrosoft, VariantNCS, VariantRFC4122,
|
|
Version1Mac, Version2Dce, Version3Md5, Version4Random,
|
|
Version5Sha1};
|
|
use std::rand;
|
|
|
|
#[test]
|
|
fn test_nil() {
|
|
let nil = Uuid::nil();
|
|
let not_nil = Uuid::new_v4();
|
|
|
|
assert!(nil.is_nil());
|
|
assert!(!not_nil.is_nil());
|
|
}
|
|
|
|
#[test]
|
|
fn test_new() {
|
|
// Supported
|
|
let uuid1 = Uuid::new(Version4Random).unwrap();
|
|
let s = uuid1.to_simple_str();
|
|
|
|
assert!(s.len() == 32);
|
|
assert!(uuid1.get_version().unwrap() == Version4Random);
|
|
|
|
// Test unsupported versions
|
|
assert!(Uuid::new(Version1Mac) == None);
|
|
assert!(Uuid::new(Version2Dce) == None);
|
|
assert!(Uuid::new(Version3Md5) == None);
|
|
assert!(Uuid::new(Version5Sha1) == None);
|
|
}
|
|
|
|
#[test]
|
|
fn test_new_v4() {
|
|
let uuid1 = Uuid::new_v4();
|
|
|
|
assert!(uuid1.get_version().unwrap() == Version4Random);
|
|
assert!(uuid1.get_variant().unwrap() == VariantRFC4122);
|
|
}
|
|
|
|
#[test]
|
|
fn test_get_version() {
|
|
let uuid1 = Uuid::new_v4();
|
|
|
|
assert!(uuid1.get_version().unwrap() == Version4Random);
|
|
assert!(uuid1.get_version_num() == 4);
|
|
}
|
|
|
|
#[test]
|
|
fn test_get_variant() {
|
|
let uuid1 = Uuid::new_v4();
|
|
let uuid2 = Uuid::parse_string("550e8400-e29b-41d4-a716-446655440000").unwrap();
|
|
let uuid3 = Uuid::parse_string("67e55044-10b1-426f-9247-bb680e5fe0c8").unwrap();
|
|
let uuid4 = Uuid::parse_string("936DA01F9ABD4d9dC0C702AF85C822A8").unwrap();
|
|
let uuid5 = Uuid::parse_string("F9168C5E-CEB2-4faa-D6BF-329BF39FA1E4").unwrap();
|
|
let uuid6 = Uuid::parse_string("f81d4fae-7dec-11d0-7765-00a0c91e6bf6").unwrap();
|
|
|
|
assert!(uuid1.get_variant().unwrap() == VariantRFC4122);
|
|
assert!(uuid2.get_variant().unwrap() == VariantRFC4122);
|
|
assert!(uuid3.get_variant().unwrap() == VariantRFC4122);
|
|
assert!(uuid4.get_variant().unwrap() == VariantMicrosoft);
|
|
assert!(uuid5.get_variant().unwrap() == VariantMicrosoft);
|
|
assert!(uuid6.get_variant().unwrap() == VariantNCS);
|
|
}
|
|
|
|
#[test]
|
|
fn test_parse_uuid_v4() {
|
|
use super::{ErrorInvalidCharacter, ErrorInvalidGroups,
|
|
ErrorInvalidGroupLength, ErrorInvalidLength};
|
|
|
|
// Invalid
|
|
assert!(Uuid::parse_string("").is_err());
|
|
assert!(Uuid::parse_string("!").is_err());
|
|
assert!(Uuid::parse_string("F9168C5E-CEB2-4faa-B6BF-329BF39FA1E45").is_err());
|
|
assert!(Uuid::parse_string("F9168C5E-CEB2-4faa-BBF-329BF39FA1E4").is_err());
|
|
assert!(Uuid::parse_string("F9168C5E-CEB2-4faa-BGBF-329BF39FA1E4").is_err());
|
|
assert!(Uuid::parse_string("F9168C5E-CEB2-4faa-B6BFF329BF39FA1E4").is_err());
|
|
assert!(Uuid::parse_string("F9168C5E-CEB2-4faa").is_err());
|
|
assert!(Uuid::parse_string("F9168C5E-CEB2-4faaXB6BFF329BF39FA1E4").is_err());
|
|
assert!(Uuid::parse_string("F9168C5E-CEB-24fa-eB6BFF32-BF39FA1E4").is_err());
|
|
assert!(Uuid::parse_string("01020304-1112-2122-3132-41424344").is_err());
|
|
assert!(Uuid::parse_string("67e5504410b1426f9247bb680e5fe0c").is_err());
|
|
assert!(Uuid::parse_string("67e5504410b1426f9247bb680e5fe0c88").is_err());
|
|
assert!(Uuid::parse_string("67e5504410b1426f9247bb680e5fe0cg8").is_err());
|
|
assert!(Uuid::parse_string("67e5504410b1426%9247bb680e5fe0c8").is_err());
|
|
|
|
// Valid
|
|
assert!(Uuid::parse_string("00000000000000000000000000000000").is_ok());
|
|
assert!(Uuid::parse_string("67e55044-10b1-426f-9247-bb680e5fe0c8").is_ok());
|
|
assert!(Uuid::parse_string("67e55044-10b1-426f-9247-bb680e5fe0c8").is_ok());
|
|
assert!(Uuid::parse_string("F9168C5E-CEB2-4faa-B6BF-329BF39FA1E4").is_ok());
|
|
assert!(Uuid::parse_string("67e5504410b1426f9247bb680e5fe0c8").is_ok());
|
|
assert!(Uuid::parse_string("01020304-1112-2122-3132-414243444546").is_ok());
|
|
assert!(Uuid::parse_string("urn:uuid:67e55044-10b1-426f-9247-bb680e5fe0c8").is_ok());
|
|
|
|
// Nil
|
|
let nil = Uuid::nil();
|
|
assert!(Uuid::parse_string("00000000000000000000000000000000").unwrap() == nil);
|
|
assert!(Uuid::parse_string("00000000-0000-0000-0000-000000000000").unwrap() == nil);
|
|
|
|
// Round-trip
|
|
let uuid_orig = Uuid::new_v4();
|
|
let orig_str = uuid_orig.to_string();
|
|
let uuid_out = Uuid::parse_string(orig_str.as_slice()).unwrap();
|
|
assert!(uuid_orig == uuid_out);
|
|
|
|
// Test error reporting
|
|
let e = Uuid::parse_string("67e5504410b1426f9247bb680e5fe0c").unwrap_err();
|
|
assert!(match e { ErrorInvalidLength(n) => n==31, _ => false });
|
|
|
|
let e = Uuid::parse_string("67e550X410b1426f9247bb680e5fe0cd").unwrap_err();
|
|
assert!(match e { ErrorInvalidCharacter(c, n) => c=='X' && n==6, _ => false });
|
|
|
|
let e = Uuid::parse_string("67e550-4105b1426f9247bb680e5fe0c").unwrap_err();
|
|
assert!(match e { ErrorInvalidGroups(n) => n==2, _ => false });
|
|
|
|
let e = Uuid::parse_string("F9168C5E-CEB2-4faa-B6BF1-02BF39FA1E4").unwrap_err();
|
|
assert!(match e { ErrorInvalidGroupLength(g, n, e) => g==3 && n==5 && e==4, _ => false });
|
|
}
|
|
|
|
#[test]
|
|
fn test_to_simple_str() {
|
|
let uuid1 = Uuid::new_v4();
|
|
let s = uuid1.to_simple_str();
|
|
|
|
assert!(s.len() == 32);
|
|
assert!(s.as_slice().chars().all(|c| c.is_digit_radix(16)));
|
|
}
|
|
|
|
#[test]
|
|
fn test_to_string() {
|
|
let uuid1 = Uuid::new_v4();
|
|
let s = uuid1.to_string();
|
|
|
|
assert!(s.len() == 32);
|
|
assert!(s.as_slice().chars().all(|c| c.is_digit_radix(16)));
|
|
}
|
|
|
|
#[test]
|
|
fn test_to_hyphenated_str() {
|
|
let uuid1 = Uuid::new_v4();
|
|
let s = uuid1.to_hyphenated_str();
|
|
|
|
assert!(s.len() == 36);
|
|
assert!(s.as_slice().chars().all(|c| c.is_digit_radix(16) || c == '-'));
|
|
}
|
|
|
|
#[test]
|
|
fn test_to_urn_str() {
|
|
let uuid1 = Uuid::new_v4();
|
|
let ss = uuid1.to_urn_str();
|
|
let s = ss.as_slice().slice(9, ss.len());
|
|
|
|
assert!(ss.as_slice().starts_with("urn:uuid:"));
|
|
assert!(s.len() == 36);
|
|
assert!(s.as_slice()
|
|
.chars()
|
|
.all(|c| c.is_digit_radix(16) || c == '-'));
|
|
}
|
|
|
|
#[test]
|
|
fn test_to_str_matching() {
|
|
let uuid1 = Uuid::new_v4();
|
|
|
|
let hs = uuid1.to_hyphenated_str();
|
|
let ss = uuid1.to_string();
|
|
|
|
let hsn = String::from_chars(hs.as_slice()
|
|
.chars()
|
|
.filter(|&c| c != '-')
|
|
.collect::<Vec<char>>()
|
|
.as_slice());
|
|
|
|
assert!(hsn == ss);
|
|
}
|
|
|
|
#[test]
|
|
fn test_string_roundtrip() {
|
|
let uuid = Uuid::new_v4();
|
|
|
|
let hs = uuid.to_hyphenated_str();
|
|
let uuid_hs = Uuid::parse_string(hs.as_slice()).unwrap();
|
|
assert!(uuid_hs == uuid);
|
|
|
|
let ss = uuid.to_string();
|
|
let uuid_ss = Uuid::parse_string(ss.as_slice()).unwrap();
|
|
assert!(uuid_ss == uuid);
|
|
}
|
|
|
|
#[test]
|
|
fn test_compare() {
|
|
let uuid1 = Uuid::new_v4();
|
|
let uuid2 = Uuid::new_v4();
|
|
|
|
assert!(uuid1 == uuid1);
|
|
assert!(uuid2 == uuid2);
|
|
assert!(uuid1 != uuid2);
|
|
assert!(uuid2 != uuid1);
|
|
}
|
|
|
|
#[test]
|
|
fn test_from_fields() {
|
|
let d1: u32 = 0xa1a2a3a4;
|
|
let d2: u16 = 0xb1b2;
|
|
let d3: u16 = 0xc1c2;
|
|
let d4: Vec<u8> = vec!(0xd1, 0xd2, 0xd3, 0xd4, 0xd5, 0xd6, 0xd7, 0xd8);
|
|
|
|
let u = Uuid::from_fields(d1, d2, d3, d4.as_slice());
|
|
|
|
let expected = "a1a2a3a4b1b2c1c2d1d2d3d4d5d6d7d8".to_string();
|
|
let result = u.to_simple_str();
|
|
assert!(result == expected);
|
|
}
|
|
|
|
#[test]
|
|
fn test_from_bytes() {
|
|
let b = vec!( 0xa1, 0xa2, 0xa3, 0xa4, 0xb1, 0xb2, 0xc1, 0xc2,
|
|
0xd1, 0xd2, 0xd3, 0xd4, 0xd5, 0xd6, 0xd7, 0xd8 );
|
|
|
|
let u = Uuid::from_bytes(b.as_slice()).unwrap();
|
|
let expected = "a1a2a3a4b1b2c1c2d1d2d3d4d5d6d7d8".to_string();
|
|
|
|
assert!(u.to_simple_str() == expected);
|
|
}
|
|
|
|
#[test]
|
|
fn test_as_bytes() {
|
|
let u = Uuid::new_v4();
|
|
let ub = u.as_bytes();
|
|
|
|
assert!(ub.len() == 16);
|
|
assert!(! ub.iter().all(|&b| b == 0));
|
|
}
|
|
|
|
#[test]
|
|
fn test_bytes_roundtrip() {
|
|
let b_in: [u8, ..16] = [ 0xa1, 0xa2, 0xa3, 0xa4, 0xb1, 0xb2, 0xc1, 0xc2,
|
|
0xd1, 0xd2, 0xd3, 0xd4, 0xd5, 0xd6, 0xd7, 0xd8 ];
|
|
|
|
let u = Uuid::from_bytes(b_in.clone()).unwrap();
|
|
|
|
let b_out = u.as_bytes();
|
|
|
|
assert!(b_in == b_out);
|
|
}
|
|
|
|
#[test]
|
|
fn test_operator_eq() {
|
|
let u1 = Uuid::new_v4();
|
|
let u2 = u1.clone();
|
|
let u3 = Uuid::new_v4();
|
|
|
|
assert!(u1 == u1);
|
|
assert!(u1 == u2);
|
|
assert!(u2 == u1);
|
|
|
|
assert!(u1 != u3);
|
|
assert!(u3 != u1);
|
|
assert!(u2 != u3);
|
|
assert!(u3 != u2);
|
|
}
|
|
|
|
#[test]
|
|
fn test_rand_rand() {
|
|
let mut rng = rand::task_rng();
|
|
let u: Uuid = rand::Rand::rand(&mut rng);
|
|
let ub = u.as_bytes();
|
|
|
|
assert!(ub.len() == 16);
|
|
assert!(! ub.iter().all(|&b| b == 0));
|
|
}
|
|
|
|
#[test]
|
|
fn test_serialize_round_trip() {
|
|
use serialize::json;
|
|
|
|
let u = Uuid::new_v4();
|
|
let s = json::encode(&u);
|
|
let u2 = json::decode(s.as_slice()).unwrap();
|
|
assert_eq!(u, u2);
|
|
}
|
|
|
|
#[test]
|
|
fn test_bad_decode() {
|
|
use serialize::json;
|
|
use serialize::{Decodable};
|
|
|
|
let js_good = json::String("a1a2a3a4a5a6a7a8a1a2a3a4a5a6a7a8".to_string());
|
|
let js_bad1 = json::String("a1a2a3a4a5a6a7a8a1a2a3a4a5a6a7ah".to_string());
|
|
let js_bad2 = json::String("a1a2a3a4a5a6a7a8a1a2a3a4a5a6a7a".to_string());
|
|
|
|
let u_good: Result<Uuid, _> = Decodable::decode(&mut json::Decoder::new(js_good));
|
|
let u_bad1: Result<Uuid, _> = Decodable::decode(&mut json::Decoder::new(js_bad1));
|
|
let u_bad2: Result<Uuid, _> = Decodable::decode(&mut json::Decoder::new(js_bad2));
|
|
assert!(u_good.is_ok());
|
|
assert!(u_bad1.is_err());
|
|
assert!(u_bad2.is_err());
|
|
}
|
|
|
|
#[test]
|
|
fn test_iterbytes_impl_for_uuid() {
|
|
use std::collections::HashSet;
|
|
let mut set = HashSet::new();
|
|
let id1 = Uuid::new_v4();
|
|
let id2 = Uuid::new_v4();
|
|
set.insert(id1);
|
|
assert!(set.contains(&id1));
|
|
assert!(!set.contains(&id2));
|
|
}
|
|
}
|
|
|
|
#[cfg(test)]
|
|
mod bench {
|
|
extern crate test;
|
|
use self::test::Bencher;
|
|
use super::Uuid;
|
|
|
|
#[bench]
|
|
pub fn create_uuids(b: &mut Bencher) {
|
|
b.iter(|| {
|
|
Uuid::new_v4();
|
|
})
|
|
}
|
|
|
|
#[bench]
|
|
pub fn uuid_to_string(b: &mut Bencher) {
|
|
let u = Uuid::new_v4();
|
|
b.iter(|| {
|
|
u.to_string();
|
|
})
|
|
}
|
|
|
|
#[bench]
|
|
pub fn parse_str(b: &mut Bencher) {
|
|
let s = "urn:uuid:F9168C5E-CEB2-4faa-B6BF-329BF39FA1E4";
|
|
b.iter(|| {
|
|
Uuid::parse_string(s).unwrap();
|
|
})
|
|
}
|
|
}
|