poco/Foundation/src/UUID.cpp
2022-07-07 04:18:20 -05:00

371 lines
7.3 KiB
C++

//
// UUID.cpp
//
// Library: Foundation
// Package: UUID
// Module: UUID
//
// Copyright (c) 2004-2006, Applied Informatics Software Engineering GmbH.
// and Contributors.
//
// SPDX-License-Identifier: BSL-1.0
//
#include "Poco/UUID.h"
#include "Poco/ByteOrder.h"
#include "Poco/Exception.h"
#include <algorithm>
#include <cstring>
namespace Poco {
UUID::UUID():
_timeLow(0),
_timeMid(0),
_timeHiAndVersion(0),
_clockSeq(0)
{
std::memset(_node, 0, sizeof(_node));
}
UUID::UUID(const UUID& uuid):
_timeLow(uuid._timeLow),
_timeMid(uuid._timeMid),
_timeHiAndVersion(uuid._timeHiAndVersion),
_clockSeq(uuid._clockSeq)
{
std::memcpy(_node, uuid._node, sizeof(_node));
}
UUID::UUID(const std::string& uuid)
{
parse(uuid);
}
UUID::UUID(const char* uuid)
{
poco_check_ptr (uuid);
parse(std::string(uuid));
}
UUID::UUID(UInt32 timeLow, UInt32 timeMid, UInt32 timeHiAndVersion, UInt16 clockSeq, UInt8 node[]):
_timeLow(timeLow),
_timeMid(timeMid),
_timeHiAndVersion(timeHiAndVersion),
_clockSeq(clockSeq)
{
std::memcpy(_node, node, sizeof(_node));
}
UUID::UUID(const char* bytes, Version version)
{
UInt32 i32;
UInt16 i16;
std::memcpy(&i32, bytes, sizeof(i32));
_timeLow = ByteOrder::fromNetwork(i32);
bytes += sizeof(i32);
std::memcpy(&i16, bytes, sizeof(i16));
_timeMid = ByteOrder::fromNetwork(i16);
bytes += sizeof(i16);
std::memcpy(&i16, bytes, sizeof(i16));
_timeHiAndVersion = ByteOrder::fromNetwork(i16);
bytes += sizeof(i16);
std::memcpy(&i16, bytes, sizeof(i16));
_clockSeq = ByteOrder::fromNetwork(i16);
bytes += sizeof(i16);
std::memcpy(_node, bytes, sizeof(_node));
_timeHiAndVersion &= 0x0FFF;
_timeHiAndVersion |= (version << 12);
_clockSeq &= 0x3FFF;
_clockSeq |= 0x8000;
}
UUID::~UUID()
{
}
UUID& UUID::operator = (const UUID& uuid)
{
if (&uuid != this)
{
_timeLow = uuid._timeLow;
_timeMid = uuid._timeMid;
_timeHiAndVersion = uuid._timeHiAndVersion;
_clockSeq = uuid._clockSeq;
std::memcpy(_node, uuid._node, sizeof(_node));
}
return *this;
}
void UUID::swap(UUID& uuid) noexcept
{
std::swap(_timeLow, uuid._timeLow);
std::swap(_timeMid, uuid._timeMid);
std::swap(_timeHiAndVersion, uuid._timeHiAndVersion);
std::swap(_clockSeq, uuid._clockSeq);
std::swap_ranges(_node, _node + 6, &uuid._node[0]);
}
void UUID::parse(const std::string& uuid)
{
if (!tryParse(uuid))
throw SyntaxException(uuid);
}
bool UUID::tryParse(const std::string& uuid)
{
if (uuid.size() < 32)
return false;
bool haveHyphens = false;
if (uuid[8] == '-' && uuid[13] == '-' && uuid[18] == '-' && uuid[23] == '-')
{
if (uuid.size() >= 36)
haveHyphens = true;
else
return false;
}
UUID newUUID;
std::string::const_iterator it = uuid.begin();
newUUID._timeLow = 0;
for (int i = 0; i < 8; ++i)
{
Int16 n = nibble(*it++);
if (n < 0) return false;
newUUID._timeLow = (newUUID._timeLow << 4) | n;
}
if (haveHyphens) ++it;
newUUID._timeMid = 0;
for (int i = 0; i < 4; ++i)
{
Int16 n = nibble(*it++);
if (n < 0) return false;
newUUID._timeMid = (newUUID._timeMid << 4) | n;
}
if (haveHyphens) ++it;
newUUID._timeHiAndVersion = 0;
for (int i = 0; i < 4; ++i)
{
Int16 n = nibble(*it++);
if (n < 0) return false;
newUUID._timeHiAndVersion = (newUUID._timeHiAndVersion << 4) | n;
}
if (haveHyphens) ++it;
newUUID._clockSeq = 0;
for (int i = 0; i < 4; ++i)
{
Int16 n = nibble(*it++);
if (n < 0) return false;
newUUID._clockSeq = (newUUID._clockSeq << 4) | n;
}
if (haveHyphens) ++it;
for (int i = 0; i < 6; ++i)
{
Int16 n1 = nibble(*it++);
if (n1 < 0) return false;
Int16 n2 = nibble(*it++);
if (n2 < 0) return false;
newUUID._node[i] = (n1 << 4) | n2;
}
swap(newUUID);
return true;
}
std::string UUID::toString() const
{
std::string result;
result.reserve(36);
appendHex(result, _timeLow);
result += '-';
appendHex(result, _timeMid);
result += '-';
appendHex(result, _timeHiAndVersion);
result += '-';
appendHex(result, _clockSeq);
result += '-';
for (int i = 0; i < sizeof(_node); ++i)
appendHex(result, _node[i]);
return result;
}
void UUID::copyFrom(const char* buffer)
{
UInt32 i32;
UInt16 i16;
std::memcpy(&i32, buffer, sizeof(i32));
_timeLow = ByteOrder::fromNetwork(i32);
buffer += sizeof(i32);
std::memcpy(&i16, buffer, sizeof(i16));
_timeMid = ByteOrder::fromNetwork(i16);
buffer += sizeof(i16);
std::memcpy(&i16, buffer, sizeof(i16));
_timeHiAndVersion = ByteOrder::fromNetwork(i16);
buffer += sizeof(i16);
std::memcpy(&i16, buffer, sizeof(i16));
_clockSeq = ByteOrder::fromNetwork(i16);
buffer += sizeof(i16);
std::memcpy(_node, buffer, sizeof(_node));
}
void UUID::copyTo(char* buffer) const
{
UInt32 i32 = ByteOrder::toNetwork(_timeLow);
std::memcpy(buffer, &i32, sizeof(i32));
buffer += sizeof(i32);
UInt16 i16 = ByteOrder::toNetwork(_timeMid);
std::memcpy(buffer, &i16, sizeof(i16));
buffer += sizeof(i16);
i16 = ByteOrder::toNetwork(_timeHiAndVersion);
std::memcpy(buffer, &i16, sizeof(i16));
buffer += sizeof(i16);
i16 = ByteOrder::toNetwork(_clockSeq);
std::memcpy(buffer, &i16, sizeof(i16));
buffer += sizeof(i16);
std::memcpy(buffer, _node, sizeof(_node));
}
int UUID::variant() const
{
int v = _clockSeq >> 13;
if ((v & 6) == 6)
return v;
else if (v & 4)
return 2;
else
return 0;
}
int UUID::compare(const UUID& uuid) const
{
if (_timeLow != uuid._timeLow) return _timeLow < uuid._timeLow ? -1 : 1;
if (_timeMid != uuid._timeMid) return _timeMid < uuid._timeMid ? -1 : 1;
if (_timeHiAndVersion != uuid._timeHiAndVersion) return _timeHiAndVersion < uuid._timeHiAndVersion ? -1 : 1;
if (_clockSeq != uuid._clockSeq) return _clockSeq < uuid._clockSeq ? -1 : 1;
for (int i = 0; i < sizeof(_node); ++i)
{
if (_node[i] < uuid._node[i])
return -1;
else if (_node[i] > uuid._node[i])
return 1;
}
return 0;
}
void UUID::appendHex(std::string& str, UInt8 n)
{
static const char* digits = "0123456789abcdef";
str += digits[(n >> 4) & 0xF];
str += digits[n & 0xF];
}
void UUID::appendHex(std::string& str, UInt16 n)
{
appendHex(str, UInt8(n >> 8));
appendHex(str, UInt8(n & 0xFF));
}
void UUID::appendHex(std::string& str, UInt32 n)
{
appendHex(str, UInt16(n >> 16));
appendHex(str, UInt16(n & 0xFFFF));
}
Int16 UUID::nibble(char hex)
{
if (hex >= 'a' && hex <= 'f')
return hex - 'a' + 10;
else if (hex >= 'A' && hex <= 'F')
return hex - 'A' + 10;
else if (hex >= '0' && hex <= '9')
return hex - '0';
else
return -1;
}
void UUID::fromNetwork()
{
_timeLow = ByteOrder::fromNetwork(_timeLow);
_timeMid = ByteOrder::fromNetwork(_timeMid);
_timeHiAndVersion = ByteOrder::fromNetwork(_timeHiAndVersion);
_clockSeq = ByteOrder::fromNetwork(_clockSeq);
}
void UUID::toNetwork()
{
_timeLow = ByteOrder::toNetwork(_timeLow);
_timeMid = ByteOrder::toNetwork(_timeMid);
_timeHiAndVersion = ByteOrder::toNetwork(_timeHiAndVersion);
_clockSeq = ByteOrder::toNetwork(_clockSeq);
}
namespace
{
static UUID uuidNull;
static UUID uuidDNS("6ba7b810-9dad-11d1-80b4-00c04fd430c8");
static UUID uuidURI("6ba7b811-9dad-11d1-80b4-00c04fd430c8");
static UUID uuidOID("6ba7b812-9dad-11d1-80b4-00c04fd430c8");
static UUID uuidX500("6ba7b814-9dad-11d1-80b4-00c04fd430c8");
}
const UUID& UUID::null()
{
return uuidNull;
}
const UUID& UUID::dns()
{
return uuidDNS;
}
const UUID& UUID::uri()
{
return uuidURI;
}
const UUID& UUID::oid()
{
return uuidOID;
}
const UUID& UUID::x500()
{
return uuidX500;
}
} // namespace Poco