nim-eth/eth/keyfile/uuid.nim

177 lines
5.1 KiB
Nim
Raw Normal View History

2019-02-05 12:45:09 +02:00
#
# Ethereum KeyFile
# (c) Copyright 2018
# Status Research & Development GmbH
#
# Licensed under either of
# Apache License, version 2.0, (LICENSE-APACHEv2)
# MIT license (LICENSE-MIT)
## This module implements interface to cross-platform UUID
## generator.
##
## - ``Windows`` - using rpcrt4.dll's `UuidCreate()`.
## - ``Linux`` and ``Android`` - using `/proc/sys/kernel/random/uuid`.
## - ``MacOS`` and ``iOS`` - using `uuid_generate_random()`.
## - ``FreeBSD``, ``OpenBSD``, ``NetBSD``,
## ``DragonflyBSD`` - using `uuid_create()`.
{.push raises: [Defect].}
2019-02-05 12:45:09 +02:00
2020-04-07 11:56:25 +02:00
import stew/[byteutils, endians2, results]
from nimcrypto import stripSpaces
2020-04-07 11:56:25 +02:00
export results
2019-02-05 12:45:09 +02:00
type
2019-02-05 12:45:09 +02:00
UUID* = object
## Represents UUID object
data*: array[16, byte]
UuidResult*[T] = Result[T, cstring]
2019-02-05 12:45:09 +02:00
proc uuidFromString*(s: string): UuidResult[UUID] =
2019-02-05 12:45:09 +02:00
## Convert string representation of UUID into UUID object.
if len(s) != 36:
return err("uuid: length must be 36 bytes")
2019-02-05 12:45:09 +02:00
for i in 0..<len(s):
if s[i] notin {'A'..'F', '0'..'9', 'a'..'f', '-'}:
return err("uuid: invalid characters")
try:
var d = hexToSeqByte(stripSpaces(s))
var
a = uint32.fromBytesBE(d.toOpenArray(0, 3))
b = uint16.fromBytesBE(d.toOpenArray(4, 5))
c = uint16.fromBytesBE(d.toOpenArray(6, 7))
var ret: UUID
copyMem(addr ret.data[0], addr a, 4)
copyMem(addr ret.data[4], addr b, 2)
copyMem(addr ret.data[6], addr c, 2)
copyMem(addr ret.data[8], addr d[8], 8)
ok(ret)
except CatchableError:
err("uuid: cannot parse hex string")
proc uuidToString*(u: UUID): string =
2019-02-05 12:45:09 +02:00
## Convert UUID object into string representation.
## UUID are lowercase, per RFC4122
2019-02-05 12:45:09 +02:00
result = newStringOfCap(38)
var d: array[8, byte]
2020-03-05 01:25:21 +01:00
var
a = uint32.fromBytesBE(u.data.toOpenArray(0, 3))
b = uint16.fromBytesBE(u.data.toOpenArray(4, 5))
c = uint16.fromBytesBE(u.data.toOpenArray(6, 7))
copyMem(addr d[0], addr a, 4)
copyMem(addr d[4], addr b, 2)
copyMem(addr d[6], addr c, 2)
result.add(toHex(toOpenArray(d, 0, 3)))
2019-02-05 12:45:09 +02:00
result.add("-")
result.add(toHex(toOpenArray(d, 4, 5)))
2019-02-05 12:45:09 +02:00
result.add("-")
result.add(toHex(toOpenArray(d, 6, 7)))
2019-02-05 12:45:09 +02:00
result.add("-")
result.add(toHex(toOpenArray(u.data, 8, 9)))
2019-02-05 12:45:09 +02:00
result.add("-")
result.add(toHex(toOpenArray(u.data, 10, 15)))
2019-02-05 12:45:09 +02:00
proc `$`*(u: UUID): string =
2019-02-05 12:45:09 +02:00
## Convert UUID object to lowercase string representation.
uuidToString(u)
2019-02-05 12:45:09 +02:00
when defined(nimdoc):
proc uuidGenerate*(): UuidResult[UUID]
2019-02-05 12:45:09 +02:00
## Generates new unique UUID and store it to `output`.
##
## Return 1 on success, and 0 on failure
when defined(posix):
when defined(macosx):
proc uuidGenerateRandom(a: pointer)
{.importc: "uuid_generate_random", header: "uuid/uuid.h".}
proc uuidGenerate*(): UuidResult[UUID] =
var output: UUID
2019-02-05 12:45:09 +02:00
uuidGenerateRandom(cast[pointer](addr output))
ok(output)
2019-02-05 12:45:09 +02:00
elif defined(freebsd) or defined(netbsd) or defined(openbsd) or
defined(dragonflybsd):
proc uuidCreate(a: pointer, s: ptr uint32)
{.importc: "uuid_create", header: "uuid.h".}
proc uuidGenerate*(): UuidResult[UUID] =
2019-02-05 12:45:09 +02:00
var status: uint32 = 0
var output: UUID
2019-02-05 12:45:09 +02:00
uuidCreate(cast[pointer](addr output), addr status)
if status == 0:
ok(output)
2019-02-05 12:45:09 +02:00
else:
err("uuid: uuid_create failed")
2019-02-05 12:45:09 +02:00
elif defined(linux) or defined(android):
import posix, os, nimcrypto/sysrand
proc uuidRead(bytes: var string, length: int): int =
result = -1
let fd = posix.open("/proc/sys/kernel/random/uuid", posix.O_RDONLY)
if fd != -1:
result = 0
while result < length:
var p = cast[pointer](cast[uint](addr bytes[0]) + uint(result))
var res = posix.read(fd, p, length - result)
if res > 0:
result += res
elif res == 0:
break
else:
if osLastError().int32 != EINTR:
result = -1
break
discard posix.close(fd)
proc uuidGenerate*(): UuidResult[UUID] =
2019-02-05 12:45:09 +02:00
var buffer = newString(37)
if uuidRead(buffer, 36) == 36:
buffer.setLen(36)
uuidFromString(buffer)
2019-02-05 12:45:09 +02:00
else:
var output: UUID
2019-02-05 12:45:09 +02:00
if randomBytes(output.data) == sizeof(output.data):
ok(output)
else:
err("uuid: cannot get random bytes")
2019-02-05 12:45:09 +02:00
else:
import nimcrypto/sysrand
proc uuidGenerate*(): UuidResult[UUID] =
var output: UUID
2019-02-05 12:45:09 +02:00
if randomBytes(output.data) == sizeof(output.data):
ok(output)
2019-02-05 12:45:09 +02:00
else:
err("uuid: cannot get random bytes")
2019-02-05 12:45:09 +02:00
elif defined(windows):
proc UuidCreate(p: pointer): int32
{.stdcall, dynlib: "rpcrt4", importc: "UuidCreate".}
proc uuidGenerate*(): UuidResult[UUID] =
var output: UUID
2019-02-05 12:45:09 +02:00
if UuidCreate(cast[pointer](addr output)) == 0:
ok(output)
2019-02-05 12:45:09 +02:00
else:
err("uuid: UuidCreate failed")
2019-02-05 12:45:09 +02:00
elif not defined(nimdoc):
import nimcrypto/sysrand
proc uuidGenerate*(): UuidResult[UUID] =
var output: UUID
2019-02-05 12:45:09 +02:00
if randomBytes(output.data) == sizeof(output.data):
ok(output)
2019-02-05 12:45:09 +02:00
else:
err("uuid: cannot get random bytes")