2023-06-28 13:33:07 +00:00
|
|
|
# Copyright (c) 2018-2023 Status Research & Development GmbH
|
2022-07-21 16:54:07 +00:00
|
|
|
# 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.
|
|
|
|
|
2023-08-25 09:29:07 +00:00
|
|
|
{.push raises: [].}
|
2022-07-21 16:54:07 +00:00
|
|
|
|
|
|
|
import
|
2023-06-28 13:33:07 +00:00
|
|
|
chronos, chronicles, presto/client,
|
|
|
|
"."/[rest_types, eth2_rest_serialization, rest_common]
|
2022-07-21 16:54:07 +00:00
|
|
|
|
|
|
|
proc getValidatorsActivity*(epoch: Epoch,
|
|
|
|
body: seq[ValidatorIndex]
|
|
|
|
): RestPlainResponse {.
|
|
|
|
rest, endpoint: "/nimbus/v1/validator/activity/{epoch}",
|
|
|
|
meth: MethodPost.}
|
2023-06-28 13:33:07 +00:00
|
|
|
|
|
|
|
proc getTimesyncInifo*(body: RestNimbusTimestamp1): RestPlainResponse {.
|
|
|
|
rest, endpoint: "/nimbus/v1/timesync", meth: MethodPost.}
|
|
|
|
|
|
|
|
proc getTimeOffset*(client: RestClientRef,
|
|
|
|
delay: Duration): Future[int64] {.async.} =
|
|
|
|
let
|
|
|
|
timestamp1 = getTimestamp()
|
|
|
|
data = RestNimbusTimestamp1(timestamp1: timestamp1)
|
|
|
|
resp = await client.getTimesyncInifo(data)
|
|
|
|
timestamp4 = getTimestamp()
|
|
|
|
|
|
|
|
return
|
|
|
|
case resp.status
|
|
|
|
of 200:
|
|
|
|
if resp.contentType.isNone() or
|
|
|
|
isWildCard(resp.contentType.get().mediaType) or
|
|
|
|
resp.contentType.get().mediaType != ApplicationJsonMediaType:
|
|
|
|
raise newException(RestError, "Missing or incorrect Content-Type")
|
|
|
|
|
|
|
|
let stamps = decodeBytes(RestNimbusTimestamp2, resp.data,
|
|
|
|
resp.contentType).valueOr:
|
|
|
|
raise newException(RestError, $error)
|
|
|
|
|
|
|
|
trace "Time offset data",
|
|
|
|
timestamp1 = timestamp1,
|
|
|
|
timestamp2 = stamps.timestamp2,
|
|
|
|
timestamp3 = stamps.timestamp3,
|
|
|
|
timestamp4 = timestamp4,
|
|
|
|
delay14 = delay.nanoseconds,
|
|
|
|
delay23 = stamps.delay
|
|
|
|
|
|
|
|
# t1 - time when we sent request.
|
|
|
|
# t2 - time when remote server received request.
|
|
|
|
# t3 - time when remote server sent response.
|
|
|
|
# t4 - time when we received response.
|
|
|
|
# delay14 = validator client processing delay.
|
|
|
|
# delay23 = beacon node processing delay.
|
|
|
|
#
|
|
|
|
# Round-trip network delay `delta` = (t4 - t1) - (t3 - t2)
|
|
|
|
# but with delays this will be:
|
|
|
|
# `delta` = (t4 - t1 + delay14) - (t3 - t2 + delay23)
|
|
|
|
# Estimated server time is t3 + (delta div 2)
|
|
|
|
# Estimated clock skew `theta` = t3 + (delta div 2) - t4
|
|
|
|
let
|
|
|
|
delay14 = delay.nanoseconds
|
|
|
|
delay23 = int64(stamps.delay)
|
|
|
|
offset = (int64(stamps.timestamp2) - int64(timestamp1) +
|
|
|
|
int64(stamps.timestamp3) - int64(timestamp4) +
|
|
|
|
delay14 - delay23) div 2
|
|
|
|
offset
|
|
|
|
else:
|
|
|
|
let error = decodeBytes(RestErrorMessage, resp.data,
|
|
|
|
resp.contentType).valueOr:
|
|
|
|
let msg = "Incorrect response error format (" & $resp.status &
|
|
|
|
") [" & $error & "]"
|
|
|
|
raise (ref RestResponseError)(msg: msg, status: resp.status)
|
|
|
|
let msg = "Error response (" & $resp.status & ") [" & error.message & "]"
|
|
|
|
raise (ref RestResponseError)(
|
|
|
|
msg: msg, status: error.code, message: error.message)
|