2021-03-24 00:50:18 +02:00
|
|
|
# Copyright (c) 2018-2020 Status Research & Development GmbH
|
|
|
|
# Licensed and distributed under either of
|
|
|
|
# * MIT license (license terms in the root directory or at https://opensource.org/licenses/MIT).
|
|
|
|
# * Apache v2 license (license terms in the root directory or at https://www.apache.org/licenses/LICENSE-2.0).
|
|
|
|
# at your option. This file may not be copied, modified, or distributed except according to those terms.
|
|
|
|
|
|
|
|
import
|
|
|
|
stew/results,
|
|
|
|
chronicles,
|
2021-08-03 17:17:11 +02:00
|
|
|
./rest_utils,
|
2021-10-19 16:09:26 +02:00
|
|
|
../beacon_node
|
2021-03-24 00:50:18 +02:00
|
|
|
|
2021-10-27 15:01:11 +03:00
|
|
|
export rest_utils
|
|
|
|
|
2021-03-24 00:50:18 +02:00
|
|
|
logScope: topics = "rest_eventapi"
|
|
|
|
|
|
|
|
proc validateEventTopics(events: seq[EventTopic]): Result[EventTopics,
|
|
|
|
cstring] =
|
|
|
|
const NonUniqueError = cstring("Event topics must be unique")
|
|
|
|
var res: set[EventTopic]
|
|
|
|
for item in events:
|
|
|
|
case item
|
|
|
|
of EventTopic.Head:
|
|
|
|
if EventTopic.Head in res:
|
|
|
|
return err(NonUniqueError)
|
|
|
|
res.incl(EventTopic.Head)
|
|
|
|
of EventTopic.Block:
|
|
|
|
if EventTopic.Block in res:
|
|
|
|
return err(NonUniqueError)
|
|
|
|
res.incl(EventTopic.Block)
|
|
|
|
of EventTopic.Attestation:
|
|
|
|
if EventTopic.Attestation in res:
|
|
|
|
return err(NonUniqueError)
|
|
|
|
res.incl(EventTopic.Attestation)
|
|
|
|
of EventTopic.VoluntaryExit:
|
|
|
|
if EventTopic.VoluntaryExit in res:
|
|
|
|
return err(NonUniqueError)
|
|
|
|
res.incl(EventTopic.VoluntaryExit)
|
|
|
|
of EventTopic.FinalizedCheckpoint:
|
|
|
|
if EventTopic.FinalizedCheckpoint in res:
|
|
|
|
return err(NonUniqueError)
|
|
|
|
res.incl(EventTopic.FinalizedCheckpoint)
|
|
|
|
of EventTopic.ChainReorg:
|
|
|
|
if EventTopic.ChainReorg in res:
|
|
|
|
return err(NonUniqueError)
|
|
|
|
res.incl(EventTopic.ChainReorg)
|
2021-09-22 15:17:15 +03:00
|
|
|
of EventTopic.ContributionAndProof:
|
|
|
|
if EventTopic.ContributionAndProof in res:
|
|
|
|
return err(NonUniqueError)
|
|
|
|
res.incl(EventTopic.ContributionAndProof)
|
2021-03-24 00:50:18 +02:00
|
|
|
if res == {}:
|
|
|
|
err("Empty topics list")
|
|
|
|
else:
|
|
|
|
ok(res)
|
|
|
|
|
2021-09-22 15:17:15 +03:00
|
|
|
proc eventHandler*(response: HttpResponseRef, node: BeaconNode,
|
|
|
|
T: typedesc, event: string,
|
|
|
|
serverEvent: string) {.async.} =
|
|
|
|
var fut = node.eventBus.waitEvent(T, event)
|
|
|
|
while true:
|
|
|
|
let jsonRes =
|
|
|
|
try:
|
|
|
|
let res = await fut
|
|
|
|
when T is ForkedTrustedSignedBeaconBlock:
|
|
|
|
let blockInfo = RestBlockInfo.init(res)
|
|
|
|
some(RestApiResponse.prepareJsonStringResponse(blockInfo))
|
|
|
|
else:
|
|
|
|
some(RestApiResponse.prepareJsonStringResponse(res))
|
|
|
|
except CancelledError:
|
|
|
|
none[string]()
|
|
|
|
if jsonRes.isNone() or (response.state != HttpResponseState.Sending):
|
|
|
|
# Cancellation happened or connection with remote peer has been lost.
|
|
|
|
break
|
|
|
|
# Initiating new event waiting to avoid race conditions and event misses.
|
|
|
|
fut = node.eventBus.waitEvent(T, event)
|
|
|
|
# Sending event and payload over wire.
|
|
|
|
let exitLoop =
|
|
|
|
try:
|
|
|
|
await response.sendEvent(serverEvent, jsonRes.get())
|
|
|
|
false
|
|
|
|
except CancelledError:
|
|
|
|
true
|
|
|
|
except HttpError as exc:
|
|
|
|
debug "Unable to deliver event to remote peer", error_name = $exc.name,
|
|
|
|
error_msg = $exc.msg
|
|
|
|
true
|
|
|
|
except CatchableError as exc:
|
|
|
|
debug "Unexpected error encountered", error_name = $exc.name,
|
|
|
|
error_msg = $exc.msg
|
|
|
|
true
|
|
|
|
if exitLoop:
|
|
|
|
if not(fut.finished()):
|
|
|
|
await fut.cancelAndWait()
|
|
|
|
break
|
|
|
|
|
2021-03-24 00:50:18 +02:00
|
|
|
proc installEventApiHandlers*(router: var RestRouter, node: BeaconNode) =
|
2021-08-23 13:41:48 +03:00
|
|
|
# https://ethereum.github.io/beacon-APIs/#/Events/eventstream
|
2022-01-06 08:38:40 +01:00
|
|
|
router.api(MethodGet, "/eth/v1/events") do (
|
2021-03-24 00:50:18 +02:00
|
|
|
topics: seq[EventTopic]) -> RestApiResponse:
|
|
|
|
let eventTopics =
|
|
|
|
block:
|
|
|
|
if topics.isErr():
|
|
|
|
return RestApiResponse.jsonError(Http400, "Invalid topics value",
|
|
|
|
$topics.error())
|
|
|
|
let res = validateEventTopics(topics.get())
|
|
|
|
if res.isErr():
|
|
|
|
return RestApiResponse.jsonError(Http400, "Invalid topics value",
|
|
|
|
$res.error())
|
|
|
|
res.get()
|
|
|
|
|
2022-01-21 18:52:34 +02:00
|
|
|
let res = preferredContentType(textEventStreamMediaType)
|
2021-09-22 15:17:15 +03:00
|
|
|
if res.isErr():
|
|
|
|
return RestApiResponse.jsonError(Http406, ContentNotAcceptableError)
|
2022-01-21 18:52:34 +02:00
|
|
|
if res.get() != textEventStreamMediaType:
|
2021-09-22 15:17:15 +03:00
|
|
|
return RestApiResponse.jsonError(Http500, InvalidAcceptError)
|
|
|
|
|
|
|
|
var response = request.getResponse()
|
|
|
|
response.keepAlive = false
|
|
|
|
try:
|
|
|
|
await response.prepareSSE()
|
|
|
|
except HttpError:
|
|
|
|
# It means that server failed to send HTTP response to the remote client
|
|
|
|
# so there no need to respond with HTTP error response.
|
|
|
|
return
|
|
|
|
|
|
|
|
let handlers =
|
|
|
|
block:
|
|
|
|
var res: seq[Future[void]]
|
|
|
|
if EventTopic.Head in eventTopics:
|
|
|
|
let handler = response.eventHandler(node, HeadChangeInfoObject,
|
|
|
|
"head-change", "head")
|
|
|
|
res.add(handler)
|
|
|
|
if EventTopic.Block in eventTopics:
|
|
|
|
let handler = response.eventHandler(node,
|
|
|
|
ForkedTrustedSignedBeaconBlock,
|
|
|
|
"signed-beacon-block", "block")
|
|
|
|
res.add(handler)
|
|
|
|
if EventTopic.Attestation in eventTopics:
|
|
|
|
let handler = response.eventHandler(node, Attestation,
|
|
|
|
"attestation-received",
|
|
|
|
"attestation")
|
|
|
|
res.add(handler)
|
|
|
|
if EventTopic.VoluntaryExit in eventTopics:
|
|
|
|
let handler = response.eventHandler(node, SignedVoluntaryExit,
|
|
|
|
"voluntary-exit",
|
|
|
|
"voluntary_exit")
|
|
|
|
res.add(handler)
|
|
|
|
if EventTopic.FinalizedCheckpoint in eventTopics:
|
|
|
|
let handler = response.eventHandler(node, FinalizationInfoObject,
|
|
|
|
"finalization",
|
|
|
|
"finalized_checkpoint")
|
|
|
|
res.add(handler)
|
|
|
|
if EventTopic.ChainReorg in eventTopics:
|
|
|
|
let handler = response.eventHandler(node, ReorgInfoObject,
|
|
|
|
"chain-reorg", "chain_reorg")
|
|
|
|
res.add(handler)
|
|
|
|
if EventTopic.ContributionAndProof in eventTopics:
|
|
|
|
let handler = response.eventHandler(node, SignedContributionAndProof,
|
|
|
|
"sync-contribution-and-proof",
|
|
|
|
"contribution_and_proof")
|
|
|
|
res.add(handler)
|
|
|
|
res
|
|
|
|
|
|
|
|
discard await one(handlers)
|
|
|
|
# One of the handlers finished, it means that connection has been droped, so
|
|
|
|
# we cancelling all other handlers.
|
|
|
|
let pending =
|
|
|
|
block:
|
|
|
|
var res: seq[Future[void]]
|
|
|
|
for fut in handlers:
|
|
|
|
if not(fut.finished()):
|
|
|
|
fut.cancel()
|
|
|
|
res.add(fut)
|
|
|
|
res
|
|
|
|
await allFutures(pending)
|
|
|
|
return
|
2021-04-13 13:19:31 +03:00
|
|
|
|
2022-01-06 08:38:40 +01:00
|
|
|
# Legacy URLS - Nimbus <= 1.5.5 used to expose the REST API with an additional
|
|
|
|
# `/api` path component
|
2021-04-13 13:19:31 +03:00
|
|
|
router.redirect(
|
|
|
|
MethodGet,
|
2022-01-06 08:38:40 +01:00
|
|
|
"/api/eth/v1/events",
|
|
|
|
"/eth/v1/events"
|
2021-04-13 13:19:31 +03:00
|
|
|
)
|