nwaku/waku/waku_enr/sharding.nim
2024-09-10 15:07:12 -06:00

256 lines
7.3 KiB
Nim

{.push raises: [].}
import
std/[options, bitops, sequtils, net],
stew/endians2,
results,
chronicles,
eth/keys,
libp2p/[multiaddress, multicodec],
libp2p/crypto/crypto
import ../common/enr, ../waku_core
logScope:
topics = "waku enr sharding"
const MaxShardIndex*: uint16 = 1023
const
ShardingIndicesListEnrField* = "rs"
ShardingIndicesListMaxLength* = 64
ShardingBitVectorEnrField* = "rsv"
type RelayShards* = object
clusterId*: uint16
shardIds*: seq[uint16]
func topics*(rs: RelayShards): seq[RelayShard] =
rs.shardIds.mapIt(RelayShard(clusterId: rs.clusterId, shardId: it))
func init*(T: type RelayShards, clusterId, shardId: uint16): Result[T, string] =
if shardId > MaxShardIndex:
return err("invalid shard Id")
ok(RelayShards(clusterId: clusterId, shardIds: @[shardId]))
func init*(
T: type RelayShards, clusterId: uint16, shardIds: varargs[uint16]
): Result[T, string] =
if toSeq(shardIds).anyIt(it > MaxShardIndex):
return err("invalid shard")
let indicesSeq = deduplicate(@shardIds)
if shardIds.len < 1:
return err("invalid shard count")
ok(RelayShards(clusterId: clusterId, shardIds: indicesSeq))
func init*(
T: type RelayShards, clusterId: uint16, shardIds: seq[uint16]
): Result[T, string] =
if shardIds.anyIt(it > MaxShardIndex):
return err("invalid shard")
let indicesSeq = deduplicate(shardIds)
if shardIds.len < 1:
return err("invalid shard count")
ok(RelayShards(clusterId: clusterId, shardIds: indicesSeq))
func topicsToRelayShards*(topics: seq[string]): Result[Option[RelayShards], string] =
if topics.len < 1:
return ok(none(RelayShards))
let parsedTopicsRes = topics.mapIt(RelayShard.parse(it))
for res in parsedTopicsRes:
if res.isErr():
return err("failed to parse topic: " & $res.error)
if parsedTopicsRes.anyIt(it.get().clusterId != parsedTopicsRes[0].get().clusterId):
return err("use shards with the same cluster Id.")
let relayShard =
?RelayShards.init(
parsedTopicsRes[0].get().clusterId, parsedTopicsRes.mapIt(it.get().shardId)
)
return ok(some(relayShard))
func contains*(rs: RelayShards, clusterId, shardId: uint16): bool =
return rs.clusterId == clusterId and rs.shardIds.contains(shardId)
func contains*(rs: RelayShards, shard: RelayShard): bool =
return rs.contains(shard.clusterId, shard.shardId)
func contains*(rs: RelayShards, topic: PubsubTopic): bool =
let parseRes = RelayShard.parse(topic)
if parseRes.isErr():
return false
rs.contains(parseRes.value)
# ENR builder extension
func toIndicesList*(rs: RelayShards): EnrResult[seq[byte]] =
if rs.shardIds.len > high(uint8).int:
return err("shards list too long")
var res: seq[byte]
res.add(rs.clusterId.toBytesBE())
res.add(rs.shardIds.len.uint8)
for shardId in rs.shardIds:
res.add(shardId.toBytesBE())
ok(res)
func fromIndicesList*(buf: seq[byte]): Result[RelayShards, string] =
if buf.len < 3:
return
err("insufficient data: expected at least 3 bytes, got " & $buf.len & " bytes")
let clusterId = uint16.fromBytesBE(buf[0 .. 1])
let length = int(buf[2])
if buf.len != 3 + 2 * length:
return err(
"invalid data: `length` field is " & $length & " but " & $buf.len &
" bytes were provided"
)
var shardIds: seq[uint16]
for i in 0 ..< length:
shardIds.add(uint16.fromBytesBE(buf[3 + 2 * i ..< 5 + 2 * i]))
ok(RelayShards(clusterId: clusterId, shardIds: shardIds))
func toBitVector*(rs: RelayShards): seq[byte] =
## The value is comprised of a two-byte cluster id in network byte
## order concatenated with a 128-byte wide bit vector. The bit vector
## indicates which shard ids of the respective cluster id the node is part
## of. The right-most bit in the bit vector represents shard id 0, the left-most
## bit represents shard id 1023.
var res: seq[byte]
res.add(rs.clusterId.toBytesBE())
var vec = newSeq[byte](128)
for shardId in rs.shardIds:
vec[shardId div 8].setBit(shardId mod 8)
res.add(vec)
res
func fromBitVector(buf: seq[byte]): EnrResult[RelayShards] =
if buf.len != 130:
return err("invalid data: expected 130 bytes")
let clusterId = uint16.fromBytesBE(buf[0 .. 1])
var shardIds: seq[uint16]
for i in 0u16 ..< 128u16:
for j in 0u16 ..< 8u16:
if not buf[2 + i].testBit(j):
continue
shardIds.add(j + 8 * i)
ok(RelayShards(clusterId: clusterId, shardIds: shardIds))
func withWakuRelayShardingIndicesList*(
builder: var EnrBuilder, rs: RelayShards
): EnrResult[void] =
let value = ?rs.toIndicesList()
builder.addFieldPair(ShardingIndicesListEnrField, value)
ok()
func withWakuRelayShardingBitVector*(
builder: var EnrBuilder, rs: RelayShards
): EnrResult[void] =
let value = rs.toBitVector()
builder.addFieldPair(ShardingBitVectorEnrField, value)
ok()
func withWakuRelaySharding*(builder: var EnrBuilder, rs: RelayShards): EnrResult[void] =
if rs.shardIds.len >= ShardingIndicesListMaxLength:
builder.withWakuRelayShardingBitVector(rs)
else:
builder.withWakuRelayShardingIndicesList(rs)
func withShardedTopics*(
builder: var EnrBuilder, topics: seq[string]
): Result[void, string] =
let relayShardOp = topicsToRelayShards(topics).valueOr:
return err("building ENR with relay sharding failed: " & $error)
let relayShard = relayShardOp.valueOr:
return ok()
builder.withWakuRelaySharding(relayShard).isOkOr:
return err($error)
return ok()
# ENR record accessors (e.g., Record, TypedRecord, etc.)
proc relayShardingIndicesList*(record: TypedRecord): Option[RelayShards] =
let field = record.tryGet(ShardingIndicesListEnrField, seq[byte]).valueOr:
return none(RelayShards)
let indexList = fromIndicesList(field).valueOr:
debug "invalid shards list", error = error
return none(RelayShards)
some(indexList)
proc relayShardingBitVector*(record: TypedRecord): Option[RelayShards] =
let field = record.tryGet(ShardingBitVectorEnrField, seq[byte]).valueOr:
return none(RelayShards)
let bitVector = fromBitVector(field).valueOr:
debug "invalid shards bit vector", error = error
return none(RelayShards)
some(bitVector)
proc relaySharding*(record: TypedRecord): Option[RelayShards] =
let indexList = record.relayShardingIndicesList().valueOr:
return record.relayShardingBitVector()
return some(indexList)
## Utils
proc containsShard*(r: Record, clusterId, shardId: uint16): bool =
if shardId > MaxShardIndex:
return false
let record = r.toTyped().valueOr:
debug "invalid ENR record", error = error
return false
let rs = record.relaySharding().valueOr:
return false
rs.contains(clusterId, shardId)
proc containsShard*(r: Record, shard: RelayShard): bool =
return containsShard(r, shard.clusterId, shard.shardId)
proc containsShard*(r: Record, topic: PubsubTopic): bool =
let parseRes = RelayShard.parse(topic)
if parseRes.isErr():
debug "invalid static sharding topic", topic = topic, error = parseRes.error
return false
containsShard(r, parseRes.value)
proc isClusterMismatched*(record: Record, clusterId: uint16): bool =
## Check the ENR sharding info for matching cluster id
if (let typedRecord = record.toTyped(); typedRecord.isOk()):
if (let relayShard = typedRecord.get().relaySharding(); relayShard.isSome()):
return relayShard.get().clusterId != clusterId
return false