mirror of
https://github.com/status-im/nimbus-eth1.git
synced 2025-01-11 21:04:11 +00:00
parent
156f4f0140
commit
69a1a988b0
@ -60,7 +60,7 @@ proc propagateEpochAccumulator*(
|
|||||||
encodedAccumulator
|
encodedAccumulator
|
||||||
)
|
)
|
||||||
discard await p.neighborhoodGossip(
|
discard await p.neighborhoodGossip(
|
||||||
ContentKeysList(@[encKey]), @[encodedAccumulator])
|
Opt.none(NodeId), ContentKeysList(@[encKey]), @[encodedAccumulator])
|
||||||
|
|
||||||
return ok()
|
return ok()
|
||||||
|
|
||||||
@ -87,14 +87,14 @@ proc historyPropagate*(
|
|||||||
const concurrentGossips = 20
|
const concurrentGossips = 20
|
||||||
|
|
||||||
var gossipQueue =
|
var gossipQueue =
|
||||||
newAsyncQueue[(ContentKeysList, seq[byte])](concurrentGossips)
|
newAsyncQueue[(Opt[NodeId], ContentKeysList, seq[byte])](concurrentGossips)
|
||||||
var gossipWorkers: seq[Future[void]]
|
var gossipWorkers: seq[Future[void]]
|
||||||
|
|
||||||
proc gossipWorker(p: PortalProtocol) {.async.} =
|
proc gossipWorker(p: PortalProtocol) {.async.} =
|
||||||
while true:
|
while true:
|
||||||
let (keys, content) = await gossipQueue.popFirst()
|
let (srcNodeId, keys, content) = await gossipQueue.popFirst()
|
||||||
|
|
||||||
discard await p.neighborhoodGossip(keys, @[content])
|
discard await p.neighborhoodGossip(srcNodeId, keys, @[content])
|
||||||
|
|
||||||
for i in 0 ..< concurrentGossips:
|
for i in 0 ..< concurrentGossips:
|
||||||
gossipWorkers.add(gossipWorker(p))
|
gossipWorkers.add(gossipWorker(p))
|
||||||
@ -121,7 +121,7 @@ proc historyPropagate*(
|
|||||||
p.storeContent(encKey, contentId, value[1])
|
p.storeContent(encKey, contentId, value[1])
|
||||||
|
|
||||||
await gossipQueue.addLast(
|
await gossipQueue.addLast(
|
||||||
(ContentKeysList(@[encode(value[0])]), value[1]))
|
(Opt.none(NodeId), ContentKeysList(@[encode(value[0])]), value[1]))
|
||||||
|
|
||||||
return ok()
|
return ok()
|
||||||
else:
|
else:
|
||||||
@ -152,7 +152,7 @@ proc historyPropagateBlock*(
|
|||||||
contentId = history_content.toContentId(encKey)
|
contentId = history_content.toContentId(encKey)
|
||||||
p.storeContent(encKey, contentId, value[1])
|
p.storeContent(encKey, contentId, value[1])
|
||||||
|
|
||||||
discard await p.neighborhoodGossip(ContentKeysList(@[encode(value[0])]), @[value[1]])
|
discard await p.neighborhoodGossip(Opt.none(NodeId), ContentKeysList(@[encode(value[0])]), @[value[1]])
|
||||||
|
|
||||||
return ok()
|
return ok()
|
||||||
else:
|
else:
|
||||||
@ -190,7 +190,7 @@ proc historyPropagateHeadersWithProof*(
|
|||||||
p.storeContent(encKey, contentId, encodedContent)
|
p.storeContent(encKey, contentId, encodedContent)
|
||||||
|
|
||||||
let keys = ContentKeysList(@[encode(contentKey)])
|
let keys = ContentKeysList(@[encode(contentKey)])
|
||||||
discard await p.neighborhoodGossip(keys, @[encodedContent])
|
discard await p.neighborhoodGossip(Opt.none(NodeId), keys, @[encodedContent])
|
||||||
|
|
||||||
return ok()
|
return ok()
|
||||||
|
|
||||||
@ -229,7 +229,7 @@ proc historyPropagateHeaders*(
|
|||||||
while true:
|
while true:
|
||||||
let (keys, content) = await gossipQueue.popFirst()
|
let (keys, content) = await gossipQueue.popFirst()
|
||||||
|
|
||||||
discard await p.neighborhoodGossip(keys, @[content])
|
discard await p.neighborhoodGossip(Opt.none(NodeId), keys, @[content])
|
||||||
|
|
||||||
for i in 0 ..< concurrentGossips:
|
for i in 0 ..< concurrentGossips:
|
||||||
gossipWorkers.add(gossipWorker(p))
|
gossipWorkers.add(gossipWorker(p))
|
||||||
|
@ -29,7 +29,7 @@ type
|
|||||||
LightClientNetwork* = ref object
|
LightClientNetwork* = ref object
|
||||||
portalProtocol*: PortalProtocol
|
portalProtocol*: PortalProtocol
|
||||||
lightClientDb*: LightClientDb
|
lightClientDb*: LightClientDb
|
||||||
contentQueue*: AsyncQueue[(ContentKeysList, seq[seq[byte]])]
|
contentQueue*: AsyncQueue[(Opt[NodeId], ContentKeysList, seq[seq[byte]])]
|
||||||
forkDigests*: ForkDigests
|
forkDigests*: ForkDigests
|
||||||
processContentLoop: Future[void]
|
processContentLoop: Future[void]
|
||||||
|
|
||||||
@ -173,7 +173,7 @@ proc new*(
|
|||||||
bootstrapRecords: openArray[Record] = [],
|
bootstrapRecords: openArray[Record] = [],
|
||||||
portalConfig: PortalProtocolConfig = defaultPortalProtocolConfig): T =
|
portalConfig: PortalProtocolConfig = defaultPortalProtocolConfig): T =
|
||||||
let
|
let
|
||||||
contentQueue = newAsyncQueue[(ContentKeysList, seq[seq[byte]])](50)
|
contentQueue = newAsyncQueue[(Opt[NodeId], ContentKeysList, seq[seq[byte]])](50)
|
||||||
|
|
||||||
stream = streamManager.registerNewStream(contentQueue)
|
stream = streamManager.registerNewStream(contentQueue)
|
||||||
|
|
||||||
@ -227,14 +227,15 @@ proc validateContent(
|
|||||||
|
|
||||||
proc neighborhoodGossipDiscardPeers(
|
proc neighborhoodGossipDiscardPeers(
|
||||||
p: PortalProtocol,
|
p: PortalProtocol,
|
||||||
|
srcNodeId: Opt[NodeId],
|
||||||
contentKeys: ContentKeysList,
|
contentKeys: ContentKeysList,
|
||||||
content: seq[seq[byte]]): Future[void] {.async.} =
|
content: seq[seq[byte]]): Future[void] {.async.} =
|
||||||
discard await p.neighborhoodGossip(contentKeys, content)
|
discard await p.neighborhoodGossip(srcNodeId, contentKeys, content)
|
||||||
|
|
||||||
proc processContentLoop(n: LightClientNetwork) {.async.} =
|
proc processContentLoop(n: LightClientNetwork) {.async.} =
|
||||||
try:
|
try:
|
||||||
while true:
|
while true:
|
||||||
let (contentKeys, contentItems) =
|
let (srcNodeId, contentKeys, contentItems) =
|
||||||
await n.contentQueue.popFirst()
|
await n.contentQueue.popFirst()
|
||||||
|
|
||||||
# When there is one invalid content item, all other content items are
|
# When there is one invalid content item, all other content items are
|
||||||
@ -243,7 +244,7 @@ proc processContentLoop(n: LightClientNetwork) {.async.} =
|
|||||||
# due to missing network data for validation.
|
# due to missing network data for validation.
|
||||||
if await n.validateContent(contentKeys, contentItems):
|
if await n.validateContent(contentKeys, contentItems):
|
||||||
asyncSpawn n.portalProtocol.neighborhoodGossipDiscardPeers(
|
asyncSpawn n.portalProtocol.neighborhoodGossipDiscardPeers(
|
||||||
contentKeys, contentItems
|
srcNodeId, contentKeys, contentItems
|
||||||
)
|
)
|
||||||
|
|
||||||
except CancelledError:
|
except CancelledError:
|
||||||
|
@ -55,7 +55,7 @@ type
|
|||||||
HistoryNetwork* = ref object
|
HistoryNetwork* = ref object
|
||||||
portalProtocol*: PortalProtocol
|
portalProtocol*: PortalProtocol
|
||||||
contentDB*: ContentDB
|
contentDB*: ContentDB
|
||||||
contentQueue*: AsyncQueue[(ContentKeysList, seq[seq[byte]])]
|
contentQueue*: AsyncQueue[(Opt[NodeId], ContentKeysList, seq[seq[byte]])]
|
||||||
accumulator*: FinishedAccumulator
|
accumulator*: FinishedAccumulator
|
||||||
processContentLoop: Future[void]
|
processContentLoop: Future[void]
|
||||||
statusLogLoop: Future[void]
|
statusLogLoop: Future[void]
|
||||||
@ -707,7 +707,7 @@ proc new*(
|
|||||||
bootstrapRecords: openArray[Record] = [],
|
bootstrapRecords: openArray[Record] = [],
|
||||||
portalConfig: PortalProtocolConfig = defaultPortalProtocolConfig): T =
|
portalConfig: PortalProtocolConfig = defaultPortalProtocolConfig): T =
|
||||||
let
|
let
|
||||||
contentQueue = newAsyncQueue[(ContentKeysList, seq[seq[byte]])](50)
|
contentQueue = newAsyncQueue[(Opt[NodeId], ContentKeysList, seq[seq[byte]])](50)
|
||||||
|
|
||||||
stream = streamManager.registerNewStream(contentQueue)
|
stream = streamManager.registerNewStream(contentQueue)
|
||||||
|
|
||||||
@ -748,14 +748,15 @@ proc validateContent(
|
|||||||
|
|
||||||
proc neighborhoodGossipDiscardPeers(
|
proc neighborhoodGossipDiscardPeers(
|
||||||
p: PortalProtocol,
|
p: PortalProtocol,
|
||||||
|
srcNodeId: Opt[NodeId],
|
||||||
contentKeys: ContentKeysList,
|
contentKeys: ContentKeysList,
|
||||||
content: seq[seq[byte]]): Future[void] {.async.} =
|
content: seq[seq[byte]]): Future[void] {.async.} =
|
||||||
discard await p.neighborhoodGossip(contentKeys, content)
|
discard await p.neighborhoodGossip(srcNodeId, contentKeys, content)
|
||||||
|
|
||||||
proc processContentLoop(n: HistoryNetwork) {.async.} =
|
proc processContentLoop(n: HistoryNetwork) {.async.} =
|
||||||
try:
|
try:
|
||||||
while true:
|
while true:
|
||||||
let (contentKeys, contentItems) =
|
let (srcNodeId, contentKeys, contentItems) =
|
||||||
await n.contentQueue.popFirst()
|
await n.contentQueue.popFirst()
|
||||||
|
|
||||||
# When there is one invalid content item, all other content items are
|
# When there is one invalid content item, all other content items are
|
||||||
@ -764,7 +765,7 @@ proc processContentLoop(n: HistoryNetwork) {.async.} =
|
|||||||
# due to missing network data for validation.
|
# due to missing network data for validation.
|
||||||
if await n.validateContent(contentKeys, contentItems):
|
if await n.validateContent(contentKeys, contentItems):
|
||||||
asyncSpawn n.portalProtocol.neighborhoodGossipDiscardPeers(
|
asyncSpawn n.portalProtocol.neighborhoodGossipDiscardPeers(
|
||||||
contentKeys, contentItems
|
srcNodeId, contentKeys, contentItems
|
||||||
)
|
)
|
||||||
|
|
||||||
except CancelledError:
|
except CancelledError:
|
||||||
|
@ -132,13 +132,14 @@ proc depthContentPropagate*(
|
|||||||
return ok()
|
return ok()
|
||||||
|
|
||||||
func contentDataToKeys(
|
func contentDataToKeys(
|
||||||
contentData: seq[ContentDataDist]): (ContentKeysList, seq[seq[byte]]) =
|
contentData: seq[ContentDataDist]):
|
||||||
|
(Opt[NodeId], ContentKeysList, seq[seq[byte]]) =
|
||||||
var contentKeys: seq[ByteList]
|
var contentKeys: seq[ByteList]
|
||||||
var content: seq[seq[byte]]
|
var content: seq[seq[byte]]
|
||||||
for cd in contentData:
|
for cd in contentData:
|
||||||
contentKeys.add(ByteList.init(cd.contentKey))
|
contentKeys.add(ByteList.init(cd.contentKey))
|
||||||
content.add(cd.content)
|
content.add(cd.content)
|
||||||
return (ContentKeysList(contentKeys), content)
|
return (Opt.none(NodeId), ContentKeysList(contentKeys), content)
|
||||||
|
|
||||||
proc breadthContentPropagate*(
|
proc breadthContentPropagate*(
|
||||||
p: PortalProtocol, seedDbPath: string):
|
p: PortalProtocol, seedDbPath: string):
|
||||||
@ -152,15 +153,15 @@ proc breadthContentPropagate*(
|
|||||||
const gossipsPerBatch = 5
|
const gossipsPerBatch = 5
|
||||||
|
|
||||||
var gossipQueue =
|
var gossipQueue =
|
||||||
newAsyncQueue[(ContentKeysList, seq[seq[byte]])](concurrentGossips)
|
newAsyncQueue[(Opt[NodeId], ContentKeysList, seq[seq[byte]])](concurrentGossips)
|
||||||
|
|
||||||
var gossipWorkers: seq[Future[void]]
|
var gossipWorkers: seq[Future[void]]
|
||||||
|
|
||||||
proc gossipWorker(p: PortalProtocol) {.async.} =
|
proc gossipWorker(p: PortalProtocol) {.async.} =
|
||||||
while true:
|
while true:
|
||||||
let (keys, content) = await gossipQueue.popFirst()
|
let (srcNodeId, keys, content) = await gossipQueue.popFirst()
|
||||||
|
|
||||||
discard await p.neighborhoodGossip(keys, content)
|
discard await p.neighborhoodGossip(srcNodeId, keys, content)
|
||||||
|
|
||||||
for i in 0 ..< concurrentGossips:
|
for i in 0 ..< concurrentGossips:
|
||||||
gossipWorkers.add(gossipWorker(p))
|
gossipWorkers.add(gossipWorker(p))
|
||||||
|
@ -22,7 +22,7 @@ const
|
|||||||
type StateNetwork* = ref object
|
type StateNetwork* = ref object
|
||||||
portalProtocol*: PortalProtocol
|
portalProtocol*: PortalProtocol
|
||||||
contentDB*: ContentDB
|
contentDB*: ContentDB
|
||||||
contentQueue*: AsyncQueue[(ContentKeysList, seq[seq[byte]])]
|
contentQueue*: AsyncQueue[(Opt[NodeId], ContentKeysList, seq[seq[byte]])]
|
||||||
processContentLoop: Future[void]
|
processContentLoop: Future[void]
|
||||||
|
|
||||||
func toContentIdHandler(contentKey: ByteList): results.Opt[ContentId] =
|
func toContentIdHandler(contentKey: ByteList): results.Opt[ContentId] =
|
||||||
@ -70,7 +70,7 @@ proc new*(
|
|||||||
bootstrapRecords: openArray[Record] = [],
|
bootstrapRecords: openArray[Record] = [],
|
||||||
portalConfig: PortalProtocolConfig = defaultPortalProtocolConfig): T =
|
portalConfig: PortalProtocolConfig = defaultPortalProtocolConfig): T =
|
||||||
|
|
||||||
let cq = newAsyncQueue[(ContentKeysList, seq[seq[byte]])](50)
|
let cq = newAsyncQueue[(Opt[NodeId], ContentKeysList, seq[seq[byte]])](50)
|
||||||
|
|
||||||
let s = streamManager.registerNewStream(cq)
|
let s = streamManager.registerNewStream(cq)
|
||||||
|
|
||||||
|
@ -20,7 +20,7 @@ import
|
|||||||
"."/[portal_stream, portal_protocol_config],
|
"."/[portal_stream, portal_protocol_config],
|
||||||
./messages
|
./messages
|
||||||
|
|
||||||
export messages, routing_table
|
export messages, routing_table, protocol
|
||||||
|
|
||||||
declareCounter portal_message_requests_incoming,
|
declareCounter portal_message_requests_incoming,
|
||||||
"Portal wire protocol incoming message requests",
|
"Portal wire protocol incoming message requests",
|
||||||
@ -1128,10 +1128,10 @@ proc getNClosestNodesWithRadius*(
|
|||||||
|
|
||||||
proc neighborhoodGossip*(
|
proc neighborhoodGossip*(
|
||||||
p: PortalProtocol,
|
p: PortalProtocol,
|
||||||
|
srcNodeId: Opt[NodeId],
|
||||||
contentKeys: ContentKeysList,
|
contentKeys: ContentKeysList,
|
||||||
content: seq[seq[byte]]): Future[int] {.async.} =
|
content: seq[seq[byte]]): Future[int] {.async.} =
|
||||||
## Returns number of peers to which content was gossiped
|
## Returns number of peers to which content was gossiped
|
||||||
|
|
||||||
if content.len() == 0:
|
if content.len() == 0:
|
||||||
return 0
|
return 0
|
||||||
|
|
||||||
@ -1170,7 +1170,10 @@ proc neighborhoodGossip*(
|
|||||||
let radius = p.radiusCache.get(node.id)
|
let radius = p.radiusCache.get(node.id)
|
||||||
if radius.isSome():
|
if radius.isSome():
|
||||||
if p.inRange(node.id, radius.unsafeGet(), contentId):
|
if p.inRange(node.id, radius.unsafeGet(), contentId):
|
||||||
gossipNodes.add(node)
|
if srcNodeId.isNone:
|
||||||
|
gossipNodes.add(node)
|
||||||
|
elif node.id != srcNodeId.get():
|
||||||
|
gossipNodes.add(node)
|
||||||
|
|
||||||
if gossipNodes.len >= 8: # use local nodes for gossip
|
if gossipNodes.len >= 8: # use local nodes for gossip
|
||||||
portal_gossip_without_lookup.inc(labelValues = [$p.protocolId])
|
portal_gossip_without_lookup.inc(labelValues = [$p.protocolId])
|
||||||
|
@ -67,7 +67,7 @@ type
|
|||||||
connectionTimeout: Duration
|
connectionTimeout: Duration
|
||||||
contentReadTimeout*: Duration
|
contentReadTimeout*: Duration
|
||||||
rng: ref HmacDrbgContext
|
rng: ref HmacDrbgContext
|
||||||
contentQueue*: AsyncQueue[(ContentKeysList, seq[seq[byte]])]
|
contentQueue*: AsyncQueue[(Opt[NodeId], ContentKeysList, seq[seq[byte]])]
|
||||||
|
|
||||||
StreamManager* = ref object
|
StreamManager* = ref object
|
||||||
transport: UtpDiscv5Protocol
|
transport: UtpDiscv5Protocol
|
||||||
@ -248,12 +248,12 @@ proc readContentOffer(
|
|||||||
# as `AcceptConnectionCallback` is `asyncSpawn`'ed and there are no limits
|
# as `AcceptConnectionCallback` is `asyncSpawn`'ed and there are no limits
|
||||||
# on the `contentOffers`. Might move the queue to before the reading of the
|
# on the `contentOffers`. Might move the queue to before the reading of the
|
||||||
# socket, and let the specific networks handle that.
|
# socket, and let the specific networks handle that.
|
||||||
await stream.contentQueue.put((offer.contentKeys, contentItems))
|
await stream.contentQueue.put((Opt.some(offer.nodeId), offer.contentKeys, contentItems))
|
||||||
|
|
||||||
proc new(
|
proc new(
|
||||||
T: type PortalStream,
|
T: type PortalStream,
|
||||||
transport: UtpDiscv5Protocol,
|
transport: UtpDiscv5Protocol,
|
||||||
contentQueue: AsyncQueue[(ContentKeysList, seq[seq[byte]])],
|
contentQueue: AsyncQueue[(Opt[NodeId], ContentKeysList, seq[seq[byte]])],
|
||||||
connectionTimeout: Duration,
|
connectionTimeout: Duration,
|
||||||
contentReadTimeout: Duration,
|
contentReadTimeout: Duration,
|
||||||
rng: ref HmacDrbgContext): T =
|
rng: ref HmacDrbgContext): T =
|
||||||
@ -340,7 +340,7 @@ proc new*(T: type StreamManager, d: protocol.Protocol): T =
|
|||||||
|
|
||||||
proc registerNewStream*(
|
proc registerNewStream*(
|
||||||
m : StreamManager,
|
m : StreamManager,
|
||||||
contentQueue: AsyncQueue[(ContentKeysList, seq[seq[byte]])],
|
contentQueue: AsyncQueue[(Opt[NodeId], ContentKeysList, seq[seq[byte]])],
|
||||||
connectionTimeout = defaultConnectionTimeout,
|
connectionTimeout = defaultConnectionTimeout,
|
||||||
contentReadTimeout = defaultContentReadTimeout): PortalStream =
|
contentReadTimeout = defaultContentReadTimeout): PortalStream =
|
||||||
|
|
||||||
|
@ -202,6 +202,6 @@ proc installPortalApiHandlers*(
|
|||||||
key = hexToSeqByte(contentKey)
|
key = hexToSeqByte(contentKey)
|
||||||
content = hexToSeqByte(contentValue)
|
content = hexToSeqByte(contentValue)
|
||||||
contentKeys = ContentKeysList(@[ByteList.init(key)])
|
contentKeys = ContentKeysList(@[ByteList.init(key)])
|
||||||
numberOfPeers = await p.neighborhoodGossip(contentKeys, @[content])
|
numberOfPeers = await p.neighborhoodGossip(Opt.none(NodeId), contentKeys, @[content])
|
||||||
|
|
||||||
return numberOfPeers
|
return numberOfPeers
|
||||||
|
@ -35,7 +35,7 @@ proc initPortalProtocol(
|
|||||||
d = initDiscoveryNode(rng, privKey, address, bootstrapRecords)
|
d = initDiscoveryNode(rng, privKey, address, bootstrapRecords)
|
||||||
db = ContentDB.new("", uint32.high, inMemory = true)
|
db = ContentDB.new("", uint32.high, inMemory = true)
|
||||||
manager = StreamManager.new(d)
|
manager = StreamManager.new(d)
|
||||||
q = newAsyncQueue[(ContentKeysList, seq[seq[byte]])](50)
|
q = newAsyncQueue[(Opt[NodeId], ContentKeysList, seq[seq[byte]])](50)
|
||||||
stream = manager.registerNewStream(q)
|
stream = manager.registerNewStream(q)
|
||||||
|
|
||||||
proto = PortalProtocol.new(
|
proto = PortalProtocol.new(
|
||||||
@ -171,7 +171,7 @@ procSuite "Portal Wire Protocol Tests":
|
|||||||
|
|
||||||
check res.isOk()
|
check res.isOk()
|
||||||
|
|
||||||
let (contentKeys, contentItems) =
|
let (srcNodeId, contentKeys, contentItems) =
|
||||||
await proto2.stream.contentQueue.popFirst()
|
await proto2.stream.contentQueue.popFirst()
|
||||||
|
|
||||||
check contentItems.len() == content.len()
|
check contentItems.len() == content.len()
|
||||||
@ -332,7 +332,7 @@ procSuite "Portal Wire Protocol Tests":
|
|||||||
dbLimit = 100_000'u32
|
dbLimit = 100_000'u32
|
||||||
db = ContentDB.new("", dbLimit, inMemory = true)
|
db = ContentDB.new("", dbLimit, inMemory = true)
|
||||||
m = StreamManager.new(node1)
|
m = StreamManager.new(node1)
|
||||||
q = newAsyncQueue[(ContentKeysList, seq[seq[byte]])](50)
|
q = newAsyncQueue[(Opt[NodeId], ContentKeysList, seq[seq[byte]])](50)
|
||||||
stream = m.registerNewStream(q)
|
stream = m.registerNewStream(q)
|
||||||
|
|
||||||
proto1 = PortalProtocol.new(
|
proto1 = PortalProtocol.new(
|
||||||
|
@ -224,7 +224,7 @@ proc run(config: PortalCliConf) =
|
|||||||
let
|
let
|
||||||
db = ContentDB.new("", config.storageSize, inMemory = true)
|
db = ContentDB.new("", config.storageSize, inMemory = true)
|
||||||
sm = StreamManager.new(d)
|
sm = StreamManager.new(d)
|
||||||
cq = newAsyncQueue[(ContentKeysList, seq[seq[byte]])](50)
|
cq = newAsyncQueue[(Opt[NodeId], ContentKeysList, seq[seq[byte]])](50)
|
||||||
stream = sm.registerNewStream(cq)
|
stream = sm.registerNewStream(cq)
|
||||||
portal = PortalProtocol.new(d, config.protocolId,
|
portal = PortalProtocol.new(d, config.protocolId,
|
||||||
testContentIdHandler, createGetHandler(db), stream,
|
testContentIdHandler, createGetHandler(db), stream,
|
||||||
|
Loading…
x
Reference in New Issue
Block a user