Cleanup RPC pubkey handling (#3489)

This commit is contained in:
Mamy Ratsimbazafy 2022-03-13 08:12:45 +01:00 committed by GitHub
parent 89ac586bd4
commit 9fd7305e26
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
2 changed files with 17 additions and 22 deletions

View File

@ -26,6 +26,14 @@ type
RpcServer = RpcHttpServer RpcServer = RpcHttpServer
ValidatorQuery = object ValidatorQuery = object
# Security note / threat model:
# - The validator pubkey are stored in their raw bytes representation
# in the `keyset`.
# - While the input is from unknown source (as far as the beacon node is concerned),
# users are asked to not expose their
# RPC endpoints to untrusted network or use a reverse proxy in front.
# - At usage time, keys in the keyset are compared to keys registered
# in the Ethereum BeaconState which are valid
keyset: HashSet[ValidatorPubKey] keyset: HashSet[ValidatorPubKey]
ids: seq[uint64] ids: seq[uint64]
@ -35,18 +43,7 @@ type
template unimplemented() = template unimplemented() =
raise (ref CatchableError)(msg: "Unimplemented") raise (ref CatchableError)(msg: "Unimplemented")
proc parsePubkey(str: string): ValidatorPubKey {.raises: [Defect, ValueError].} = proc createIdQuery(ids: openArray[string]): Result[ValidatorQuery, cstring] =
const expectedLen = RawPubKeySize * 2 + 2
if str.len != expectedLen: # +2 because of the `0x` prefix
raise newException(ValueError,
"A hex public key should be exactly " & $expectedLen & " characters. " &
$str.len & " provided")
let pubkeyRes = fromHex(ValidatorPubKey, str)
if pubkeyRes.isErr:
raise newException(ValueError, "Not a valid public key")
return pubkeyRes[]
proc createIdQuery(ids: openArray[string]): Result[ValidatorQuery, string] =
# validatorIds array should have maximum 30 items, and all items should be # validatorIds array should have maximum 30 items, and all items should be
# unique. # unique.
if len(ids) > 30: if len(ids) > 30:
@ -63,19 +60,15 @@ proc createIdQuery(ids: openArray[string]): Result[ValidatorQuery, string] =
for item in ids: for item in ids:
if item.startsWith("0x"): if item.startsWith("0x"):
if len(item) != RawPubKeySize * 2 + 2: let pubkey = ? ValidatorPubkey.fromHex(item)
return err("Incorrect hexadecimal key") res.keyset.incl(pubkey)
let pubkeyRes = ValidatorPubKey.fromHex(item)
if pubkeyRes.isErr:
return err("Incorrect public key")
res.keyset.incl(pubkeyRes.get())
else: else:
var tmp: uint64 var tmp: uint64
try: try:
if parseBiggestUInt(item, tmp) != len(item): if parseBiggestUInt(item, tmp) != len(item):
return err("Incorrect index value") return err("Incorrect index value")
except ValueError: except ValueError:
return err("Cannot parse index value: " & item) return err("Cannot parse index value")
res.ids.add(tmp) res.ids.add(tmp)
ok(res) ok(res)
@ -233,7 +226,7 @@ proc installBeaconApiHandlers*(rpcServer: RpcServer, node: BeaconNode) {.
if validatorIds.isSome: if validatorIds.isSome:
let vqres = createIdQuery(validatorIds.get()) let vqres = createIdQuery(validatorIds.get())
if vqres.isErr: if vqres.isErr:
raise newException(CatchableError, vqres.error) raise newException(CatchableError, $vqres.error)
vquery = vqres.get() vquery = vqres.get()
if validatorIds.isNone(): if validatorIds.isNone():
@ -283,7 +276,7 @@ proc installBeaconApiHandlers*(rpcServer: RpcServer, node: BeaconNode) {.
let current_epoch = getStateField(node.dag.headState.data, slot).epoch let current_epoch = getStateField(node.dag.headState.data, slot).epoch
let vqres = createIdQuery([validatorId]) let vqres = createIdQuery([validatorId])
if vqres.isErr: if vqres.isErr:
raise newException(CatchableError, vqres.error) raise newException(CatchableError, $vqres.error)
let vquery = vqres.get() let vquery = vqres.get()
withStateForStateId(stateId): withStateForStateId(stateId):
@ -321,7 +314,7 @@ proc installBeaconApiHandlers*(rpcServer: RpcServer, node: BeaconNode) {.
else: else:
let vqres = createIdQuery(validatorsId.get()) let vqres = createIdQuery(validatorsId.get())
if vqres.isErr: if vqres.isErr:
raise newException(CatchableError, vqres.error) raise newException(CatchableError, $vqres.error)
var vquery = vqres.get() var vquery = vqres.get()
for index in vquery.ids: for index in vquery.ids:

View File

@ -360,6 +360,8 @@ func fromHex*(T: type BlsCurveType, hexStr: string): BlsResult[T] {.inline.} =
func `==`*(a, b: ValidatorPubKey | ValidatorSig): bool = func `==`*(a, b: ValidatorPubKey | ValidatorSig): bool =
equalMem(unsafeAddr a.blob[0], unsafeAddr b.blob[0], sizeof(a.blob)) equalMem(unsafeAddr a.blob[0], unsafeAddr b.blob[0], sizeof(a.blob))
func `==`*(a, b: ValidatorPrivKey): bool {.error: "Secret keys should stay secret".}
# Hashing # Hashing
# ---------------------------------------------------------------------- # ----------------------------------------------------------------------