2019-12-10 20:49:54 +00:00
|
|
|
#
|
|
|
|
# Waku Mail Client & Server
|
|
|
|
# (c) Copyright 2019
|
|
|
|
# Status Research & Development GmbH
|
|
|
|
#
|
|
|
|
# Licensed under either of
|
|
|
|
# Apache License, version 2.0, (LICENSE-APACHEv2)
|
|
|
|
# MIT license (LICENSE-MIT)
|
|
|
|
#
|
|
|
|
import
|
|
|
|
chronos,
|
|
|
|
eth/[p2p, async_utils], eth/p2p/rlpx_protocols/waku_protocol
|
|
|
|
|
|
|
|
const
|
|
|
|
requestCompleteTimeout = chronos.seconds(5)
|
|
|
|
|
|
|
|
type
|
|
|
|
Cursor = Bytes
|
|
|
|
|
|
|
|
MailRequest* = object
|
|
|
|
lower*: uint32 ## Unix timestamp; oldest requested envelope's creation time
|
|
|
|
upper*: uint32 ## Unix timestamp; newest requested envelope's creation time
|
|
|
|
bloom*: Bytes ## Bloom filter to apply on the envelopes
|
|
|
|
limit*: uint32 ## Maximum amount of envelopes to return
|
|
|
|
cursor*: Cursor ## Optional cursor
|
|
|
|
|
|
|
|
proc requestMail*(node: EthereumNode, peerId: NodeId, request: MailRequest,
|
|
|
|
symKey: SymKey, requests = 10): Future[Option[Cursor]] {.async.} =
|
|
|
|
## Send p2p mail request and check request complete.
|
|
|
|
## If result is none, and error occured. If result is a none empty cursor,
|
|
|
|
## more envelopes are available.
|
|
|
|
# TODO: Perhaps don't go the recursive route or could use the actual response
|
|
|
|
# proc to implement this (via a handler) and store the necessary data in the
|
|
|
|
# WakuPeer object.
|
2019-12-12 13:43:05 +00:00
|
|
|
# TODO: Several requestMail calls in parallel can create issues with handling
|
|
|
|
# the wrong response to a request. Can additionaly check the requestId but
|
|
|
|
# that would only solve it half. Better to use the requestResponse mechanism.
|
2019-12-10 20:49:54 +00:00
|
|
|
|
|
|
|
# TODO: move this check out of requestMail?
|
|
|
|
let peer = node.getPeer(peerId, Waku)
|
|
|
|
if not peer.isSome():
|
|
|
|
error "Invalid peer"
|
|
|
|
return result
|
|
|
|
elif not peer.get().state(Waku).trusted:
|
|
|
|
return result
|
|
|
|
|
|
|
|
var writer = initRlpWriter()
|
|
|
|
writer.append(request)
|
|
|
|
let payload = writer.finish()
|
|
|
|
let data = encode(Payload(payload: payload, symKey: some(symKey)))
|
|
|
|
if not data.isSome():
|
|
|
|
error "Encoding of payload failed"
|
|
|
|
return result
|
|
|
|
|
|
|
|
# TODO: should this envelope be valid in terms of ttl, PoW, etc.?
|
|
|
|
let env = Envelope(expiry:0, ttl: 0, data: data.get(), nonce: 0)
|
|
|
|
# Send the request
|
|
|
|
traceAsyncErrors peer.get().p2pRequest(env)
|
|
|
|
|
|
|
|
# Wait for the Request Complete packet
|
|
|
|
var f = peer.get().nextMsg(Waku.p2pRequestComplete)
|
|
|
|
if await f.withTimeout(requestCompleteTimeout):
|
|
|
|
let response = f.read()
|
|
|
|
# TODO: I guess the idea is to check requestId (Hash) also?
|
|
|
|
let requests = requests - 1
|
|
|
|
# If there is cursor data, do another request
|
2019-12-19 22:23:06 +00:00
|
|
|
if response.cursor.len > 0 and requests > 0:
|
2019-12-10 20:49:54 +00:00
|
|
|
var newRequest = request
|
2019-12-19 22:23:06 +00:00
|
|
|
newRequest.cursor = response.cursor
|
2019-12-10 20:49:54 +00:00
|
|
|
return await requestMail(node, peerId, newRequest, symKey, requests)
|
|
|
|
else:
|
2019-12-19 22:23:06 +00:00
|
|
|
return some(response.cursor)
|
2019-12-10 20:49:54 +00:00
|
|
|
else:
|
|
|
|
error "p2pRequestComplete timeout"
|
|
|
|
return result
|
|
|
|
|
|
|
|
proc p2pRequestHandler(peer: Peer, envelope: Envelope) =
|
|
|
|
# Mail server p2p request implementation
|
|
|
|
discard
|
|
|
|
|
|
|
|
proc enableMailServer*(node: EthereumNode, customHandler: P2PRequestHandler) =
|
|
|
|
node.protocolState(Waku).p2pRequestHandler = customHandler
|
|
|
|
|
|
|
|
proc enableMailServer*(node: EthereumNode) =
|
|
|
|
node.protocolState(Waku).p2pRequestHandler = p2pRequestHandler
|