nim-eth/eth/p2p/discoveryv5/encoding.nim

317 lines
9.7 KiB
Nim
Raw Normal View History

import
std/tables, nimcrypto, stint, chronicles,
types, node, enr, hkdf, ../enode, eth/[rlp, keys]
2019-12-16 21:38:45 +02:00
const
idNoncePrefix = "discovery-id-nonce"
keyAgreementPrefix = "discovery v5 key agreement"
2020-02-17 18:04:29 +01:00
authSchemeName* = "gcm"
gcmNonceSize* = 12
gcmTagSize = 16
aesKeySize* = 128 div 8
tagSize* = 32 ## size of the tag where each message (except whoareyou) starts
## with
2019-12-16 21:38:45 +02:00
type
AesKey = array[aesKeySize, byte]
PacketTag = array[tagSize, byte]
2019-12-16 21:38:45 +02:00
AuthResponse = object
version: int
signature: array[64, byte]
record: Record
Codec* = object
localNode*: Node
privKey*: PrivateKey
db*: Database
handshakes*: Table[string, Whoareyou] # TODO: Implement hash for NodeID
HandshakeSecrets = object
writeKey: AesKey
readKey: AesKey
authRespKey: AesKey
2019-12-16 21:38:45 +02:00
2020-02-17 18:04:29 +01:00
AuthHeader* = object
auth*: AuthTag
idNonce*: IdNonce
2020-02-17 18:04:29 +01:00
scheme*: string
ephemeralKey*: array[64, byte]
response*: seq[byte]
2019-12-16 21:38:45 +02:00
RandomSourceDepleted* = object of CatchableError
2019-12-16 21:38:45 +02:00
DecodeStatus* = enum
Success,
HandshakeError,
PacketError
proc randomBytes*(v: var openarray[byte]) =
2019-12-16 21:38:45 +02:00
if nimcrypto.randomBytes(v) != v.len:
raise newException(RandomSourceDepleted, "Could not randomize bytes")
2019-12-16 21:38:45 +02:00
proc idNonceHash(nonce, ephkey: openarray[byte]): array[32, byte] =
var ctx: sha256
ctx.init()
ctx.update(idNoncePrefix)
ctx.update(nonce)
ctx.update(ephkey)
ctx.finish().data
proc signIDNonce*(c: Codec, idNonce, ephKey: openarray[byte]): SignatureNR =
2019-12-16 21:38:45 +02:00
if signRawMessage(idNonceHash(idNonce, ephKey), c.privKey, result) != EthKeysStatus.Success:
raise newException(EthKeysException, "Could not sign idNonce")
2019-12-16 21:38:45 +02:00
proc deriveKeys(n1, n2: NodeID, priv: PrivateKey, pub: PublicKey,
idNonce: openarray[byte], result: var HandshakeSecrets) =
2019-12-16 21:38:45 +02:00
var eph: SharedSecretFull
if ecdhAgree(priv, pub, eph) != EthKeysStatus.Success:
raise newException(EthKeysException, "ecdhAgree failed")
2019-12-16 21:38:45 +02:00
# TODO: Unneeded allocation here
var info = newSeqOfCap[byte](idNoncePrefix.len + 32 * 2)
for i, c in keyAgreementPrefix: info.add(byte(c))
info.add(n1.toByteArrayBE())
info.add(n2.toByteArrayBE())
# echo "EPH: ", eph.data.toHex, " idNonce: ", challenge.idNonce.toHex, "info: ", info.toHex
static: assert(sizeof(result) == aesKeySize * 3)
2019-12-16 21:38:45 +02:00
var res = cast[ptr UncheckedArray[byte]](addr result)
hkdf(sha256, eph.data, idNonce, info, toOpenArray(res, 0, sizeof(result) - 1))
2019-12-16 21:38:45 +02:00
proc encryptGCM*(key, nonce, pt, authData: openarray[byte]): seq[byte] =
2019-12-16 21:38:45 +02:00
var ectx: GCM[aes128]
ectx.init(key, nonce, authData)
result = newSeq[byte](pt.len + gcmTagSize)
ectx.encrypt(pt, result)
ectx.getTag(result.toOpenArray(pt.len, result.high))
ectx.clear()
proc makeAuthHeader(c: Codec, toNode: Node, nonce: array[gcmNonceSize, byte],
handshakeSecrets: var HandshakeSecrets,
challenge: Whoareyou): seq[byte] =
2019-12-16 21:38:45 +02:00
var resp = AuthResponse(version: 5)
let ln = c.localNode
if challenge.recordSeq < ln.record.seqNum:
2019-12-16 21:38:45 +02:00
resp.record = ln.record
let ephKey = newPrivateKey()
let ephPubkey = ephKey.getPublicKey().getRaw
resp.signature = c.signIDNonce(challenge.idNonce, ephPubkey).getRaw
deriveKeys(ln.id, toNode.id, ephKey, toNode.node.pubKey, challenge.idNonce,
handshakeSecrets)
2019-12-16 21:38:45 +02:00
let respRlp = rlp.encode(resp)
var zeroNonce: array[gcmNonceSize, byte]
let respEnc = encryptGCM(handshakeSecrets.authRespKey, zeroNonce, respRLP, [])
2019-12-16 21:38:45 +02:00
let header = AuthHeader(auth: nonce, idNonce: challenge.idNonce,
scheme: authSchemeName, ephemeralKey: ephPubkey, response: respEnc)
2019-12-16 21:38:45 +02:00
rlp.encode(header)
proc `xor`[N: static[int], T](a, b: array[N, T]): array[N, T] =
for i in 0 .. a.high:
result[i] = a[i] xor b[i]
proc packetTag(destNode, srcNode: NodeID): PacketTag =
2019-12-16 21:38:45 +02:00
let destId = destNode.toByteArrayBE()
let srcId = srcNode.toByteArrayBE()
let destidHash = sha256.digest(destId)
result = srcId xor destidHash.data
proc encodeEncrypted*(c: Codec,
toNode: Node,
packetData: seq[byte],
challenge: Whoareyou):
(seq[byte], array[gcmNonceSize, byte]) =
2019-12-16 21:38:45 +02:00
var nonce: array[gcmNonceSize, byte]
randomBytes(nonce)
var headEnc: seq[byte]
var writeKey: AesKey
2019-12-16 21:38:45 +02:00
if challenge.isNil:
headEnc = rlp.encode(nonce)
var readKey: AesKey
2019-12-16 21:38:45 +02:00
# We might not have the node's keys if the handshake hasn't been performed
# yet. That's fine, we will be responded with whoareyou.
discard c.db.loadKeys(toNode.id, toNode.address, readKey, writeKey)
2019-12-16 21:38:45 +02:00
else:
var sec: HandshakeSecrets
headEnc = c.makeAuthHeader(toNode, nonce, sec, challenge)
writeKey = sec.writeKey
# TODO: is it safe to ignore the error here?
discard c.db.storeKeys(toNode.id, toNode.address, sec.readKey, sec.writeKey)
2019-12-16 21:38:45 +02:00
var body = packetData
let tag = packetTag(toNode.id, c.localNode.id)
var headBuf = newSeqOfCap[byte](tag.len + headEnc.len)
headBuf.add(tag)
headBuf.add(headEnc)
headBuf.add(encryptGCM(writeKey, nonce, body, tag))
return (headBuf, nonce)
proc decryptGCM(key: AesKey, nonce, ct, authData: openarray[byte]): seq[byte] =
2019-12-16 21:38:45 +02:00
var dctx: GCM[aes128]
dctx.init(key, nonce, authData)
result = newSeq[byte](ct.len - gcmTagSize)
var tag: array[gcmTagSize, byte]
dctx.decrypt(ct.toOpenArray(0, ct.high - gcmTagSize), result)
dctx.getTag(tag)
if tag != ct.toOpenArray(ct.len - gcmTagSize, ct.high):
result = @[]
dctx.clear()
2020-02-27 20:09:05 +02:00
type
DecodePacketResult = enum
decodingSuccessful
invalidPacketPayload
invalidPacketType
unsupportedPacketType
proc decodePacketBody(typ: byte,
body: openarray[byte],
res: var Packet): DecodePacketResult =
if typ < PacketKind.low.byte or typ > PacketKind.high.byte:
return invalidPacketType
let kind = cast[PacketKind](typ)
res = Packet(kind: kind)
var rlp = rlpFromBytes(@body.toRange)
if rlp.enterList:
2019-12-16 21:38:45 +02:00
res.reqId = rlp.read(RequestId)
proc decode[T](rlp: var Rlp, v: var T) {.inline, nimcall.} =
for k, v in v.fieldPairs:
v = rlp.read(typeof(v))
2020-02-27 20:09:05 +02:00
case kind
of unused: return invalidPacketPayload
of ping: rlp.decode(res.ping)
of pong: rlp.decode(res.pong)
of findNode: rlp.decode(res.findNode)
of nodes: rlp.decode(res.nodes)
of regtopic, ticket, regconfirmation, topicquery:
# TODO Implement these packet types
return unsupportedPacketType
return decodingSuccessful
2019-12-16 21:38:45 +02:00
else:
2020-02-27 20:09:05 +02:00
return invalidPacketPayload
2019-12-16 21:38:45 +02:00
proc decodeAuthResp(c: Codec, fromId: NodeId, head: AuthHeader,
challenge: Whoareyou, secrets: var HandshakeSecrets, newNode: var Node): bool =
2019-12-16 21:38:45 +02:00
if head.scheme != authSchemeName:
warn "Unknown auth scheme"
2019-12-16 21:38:45 +02:00
return false
var ephKey: PublicKey
if recoverPublicKey(head.ephemeralKey, ephKey) != EthKeysStatus.Success:
return false
deriveKeys(fromId, c.localNode.id, c.privKey, ephKey, challenge.idNonce,
secrets)
2019-12-16 21:38:45 +02:00
var zeroNonce: array[gcmNonceSize, byte]
let respData = decryptGCM(secrets.authRespKey, zeroNonce, head.response, [])
let authResp = rlp.decode(respData, AuthResponse)
newNode = newNode(authResp.record)
return true
2020-02-22 20:49:14 +02:00
proc decodeEncrypted*(c: var Codec,
fromId: NodeID,
fromAddr: Address,
input: seq[byte],
authTag: var AuthTag,
2020-02-22 20:49:14 +02:00
newNode: var Node,
packet: var Packet): DecodeStatus =
2019-12-16 21:38:45 +02:00
let input = input.toRange
var r = rlpFromBytes(input[tagSize .. ^1])
2019-12-16 21:38:45 +02:00
var auth: AuthHeader
var readKey: AesKey
2020-02-27 20:09:05 +02:00
logScope: sender = $fromAddr
2019-12-16 21:38:45 +02:00
if r.isList:
# Handshake - rlp list indicates auth-header
2019-12-16 21:38:45 +02:00
auth = r.read(AuthHeader)
authTag = auth.auth
let challenge = c.handshakes.getOrDefault($fromId)
if challenge.isNil:
2020-02-22 20:49:14 +02:00
trace "Decoding failed (no challenge)"
return HandshakeError
2019-12-16 21:38:45 +02:00
if auth.idNonce != challenge.idNonce:
2020-02-22 20:49:14 +02:00
trace "Decoding failed (different nonce)"
return HandshakeError
2019-12-16 21:38:45 +02:00
var sec: HandshakeSecrets
if not c.decodeAuthResp(fromId, auth, challenge, sec, newNode):
2020-02-22 20:49:14 +02:00
trace "Decoding failed (bad auth)"
return HandshakeError
2019-12-16 21:38:45 +02:00
c.handshakes.del($fromId)
# Swap keys to match remote
swap(sec.readKey, sec.writeKey)
# TODO: is it safe to ignore the error here?
discard c.db.storeKeys(fromId, fromAddr, sec.readKey, sec.writeKey)
2019-12-16 21:38:45 +02:00
readKey = sec.readKey
else:
# Message packet or random packet - rlp bytes (size 12) indicates auth-tag
authTag = r.read(AuthTag)
2019-12-16 21:38:45 +02:00
auth.auth = authTag
var writeKey: array[aesKeySize, byte]
2019-12-16 21:38:45 +02:00
if not c.db.loadKeys(fromId, fromAddr, readKey, writeKey):
2020-02-22 20:49:14 +02:00
trace "Decoding failed (no keys)"
return PacketError
2019-12-16 21:38:45 +02:00
# doAssert(false, "TODO: HANDLE ME!")
let headSize = tagSize + r.position
2019-12-16 21:38:45 +02:00
let bodyEnc = input[headSize .. ^1]
let body = decryptGCM(readKey, auth.auth, bodyEnc.toOpenArray,
input[0 .. tagSize - 1].toOpenArray)
2019-12-16 21:38:45 +02:00
if body.len > 1:
2020-02-27 20:09:05 +02:00
let status = decodePacketBody(body[0], body.toOpenArray(1, body.high), packet)
if status == decodingSuccessful:
return Success
2020-02-27 20:09:05 +02:00
else:
debug "Failed to decode discovery packet", reason = status
return PacketError
else:
return PacketError
2019-12-16 21:38:45 +02:00
proc newRequestId*(): RequestId =
2020-01-08 15:25:27 +02:00
if randomBytes(addr result, sizeof(result)) != sizeof(result):
raise newException(RandomSourceDepleted, "Could not randomize bytes")
2019-12-16 21:38:45 +02:00
proc numFields(T: typedesc): int =
for k, v in fieldPairs(default(T)): inc result
proc encodePacket*[T: SomePacket](p: T, reqId: RequestId): seq[byte] =
result = newSeqOfCap[byte](64)
result.add(packetKind(T).ord)
# result.add(rlp.encode(p))
const sz = numFields(T)
var writer = initRlpList(sz + 1)
writer.append(reqId)
for k, v in fieldPairs(p):
writer.append(v)
result.add(writer.finish())
proc encodePacket*[T: SomePacket](p: T): seq[byte] =
encodePacket(p, newRequestId())