2022-07-01 20:19:57 +02:00
|
|
|
# Nim-LibP2P
|
2023-01-20 15:47:40 +01:00
|
|
|
# Copyright (c) 2023 Status Research & Development GmbH
|
2022-07-01 20:19:57 +02:00
|
|
|
# Licensed under either of
|
|
|
|
# * Apache License, version 2.0, ([LICENSE-APACHE](LICENSE-APACHE))
|
|
|
|
# * MIT license ([LICENSE-MIT](LICENSE-MIT))
|
|
|
|
# at your option.
|
|
|
|
# This file may not be copied, modified, or distributed except according to
|
|
|
|
# those terms.
|
2020-03-23 15:03:36 +09:00
|
|
|
|
2023-06-07 13:12:49 +02:00
|
|
|
{.push raises: [].}
|
2021-03-09 13:22:52 +01:00
|
|
|
|
2023-06-28 16:44:58 +02:00
|
|
|
import std/options, std/macros
|
|
|
|
import stew/[byteutils, results]
|
|
|
|
|
|
|
|
export results
|
2020-03-23 15:03:36 +09:00
|
|
|
|
2022-07-01 20:19:57 +02:00
|
|
|
template public* {.pragma.}
|
|
|
|
|
2020-03-23 15:03:36 +09:00
|
|
|
const
|
|
|
|
ShortDumpMax = 12
|
|
|
|
|
2023-02-14 10:35:44 +01:00
|
|
|
template compilesOr*(a, b: untyped): untyped =
|
|
|
|
when compiles(a):
|
|
|
|
a
|
|
|
|
else:
|
|
|
|
b
|
|
|
|
|
2021-12-16 11:05:20 +01:00
|
|
|
func shortLog*(item: openArray[byte]): string =
|
2020-03-23 15:03:36 +09:00
|
|
|
if item.len <= ShortDumpMax:
|
|
|
|
result = item.toHex()
|
|
|
|
else:
|
|
|
|
const
|
|
|
|
split = ShortDumpMax div 2
|
|
|
|
dumpLen = (ShortDumpMax * 2) + 3
|
|
|
|
result = newStringOfCap(dumpLen)
|
|
|
|
result &= item.toOpenArray(0, split - 1).toHex()
|
|
|
|
result &= "..."
|
|
|
|
result &= item.toOpenArray(item.len - split, item.high).toHex()
|
|
|
|
|
|
|
|
func shortLog*(item: string): string =
|
|
|
|
if item.len <= ShortDumpMax:
|
|
|
|
result = item
|
|
|
|
else:
|
|
|
|
const
|
|
|
|
split = ShortDumpMax div 2
|
|
|
|
dumpLen = ShortDumpMax + 3
|
|
|
|
result = newStringOfCap(dumpLen)
|
|
|
|
result &= item[0..<split]
|
|
|
|
result &= "..."
|
|
|
|
result &= item[(item.len - split)..item.high]
|
2021-01-08 14:21:24 +09:00
|
|
|
|
|
|
|
when defined(libp2p_agents_metrics):
|
2021-03-15 01:42:47 +00:00
|
|
|
import strutils
|
|
|
|
export split
|
|
|
|
|
|
|
|
proc safeToLowerAscii*(s: string): Result[string, cstring] =
|
|
|
|
try:
|
|
|
|
ok(s.toLowerAscii())
|
2021-03-15 16:48:11 +09:00
|
|
|
except CatchableError:
|
2021-03-15 01:42:47 +00:00
|
|
|
err("toLowerAscii failed")
|
|
|
|
|
2021-01-08 14:21:24 +09:00
|
|
|
const
|
2023-04-03 11:05:01 +02:00
|
|
|
KnownLibP2PAgents* {.strdefine.} = "nim-libp2p"
|
2021-03-15 01:42:47 +00:00
|
|
|
KnownLibP2PAgentsSeq* = KnownLibP2PAgents.safeToLowerAscii().tryGet().split(",")
|
2023-05-18 10:24:17 +02:00
|
|
|
|
|
|
|
template safeConvert*[T: SomeInteger, S: Ordinal](value: S): T =
|
|
|
|
## Converts `value` from S to `T` iff `value` is guaranteed to be preserved.
|
|
|
|
when int64(T.low) <= int64(S.low()) and uint64(T.high) >= uint64(S.high):
|
|
|
|
T(value)
|
|
|
|
else:
|
|
|
|
{.error: "Source and target types have an incompatible range low..high".}
|
|
|
|
|
2023-07-31 11:13:51 +02:00
|
|
|
proc capLen*[T](s: var seq[T], length: Natural) =
|
|
|
|
if s.len > length:
|
|
|
|
s.setLen(length)
|
|
|
|
|
2023-05-18 10:24:17 +02:00
|
|
|
template exceptionToAssert*(body: untyped): untyped =
|
|
|
|
block:
|
|
|
|
var res: type(body)
|
|
|
|
when defined(nimHasWarnBareExcept):
|
|
|
|
{.push warning[BareExcept]:off.}
|
|
|
|
try:
|
|
|
|
res = body
|
|
|
|
except CatchableError as exc: raise exc
|
|
|
|
except Defect as exc: raise exc
|
|
|
|
except Exception as exc: raiseAssert exc.msg
|
|
|
|
when defined(nimHasWarnBareExcept):
|
|
|
|
{.pop.}
|
|
|
|
res
|
2023-06-28 16:44:58 +02:00
|
|
|
|
|
|
|
template withValue*[T](self: Opt[T] | Option[T], value, body: untyped): untyped =
|
|
|
|
if self.isSome:
|
|
|
|
let value {.inject.} = self.get()
|
|
|
|
body
|
|
|
|
|
|
|
|
macro withValue*[T](self: Opt[T] | Option[T], value, body, body2: untyped): untyped =
|
|
|
|
let elseBody = body2[0]
|
|
|
|
quote do:
|
|
|
|
if `self`.isSome:
|
|
|
|
let `value` {.inject.} = `self`.get()
|
|
|
|
`body`
|
|
|
|
else:
|
|
|
|
`elseBody`
|
|
|
|
|
|
|
|
template valueOr*[T](self: Option[T], body: untyped): untyped =
|
|
|
|
if self.isSome:
|
|
|
|
self.get()
|
|
|
|
else:
|
|
|
|
body
|
|
|
|
|
|
|
|
template toOpt*[T, E](self: Result[T, E]): Opt[T] =
|
|
|
|
if self.isOk:
|
|
|
|
when T is void: Result[void, void].ok()
|
|
|
|
else: Opt.some(self.unsafeGet())
|
|
|
|
else:
|
|
|
|
Opt.none(type(T))
|