2022-09-30 17:07:13 +00:00
|
|
|
package handlers
|
|
|
|
|
|
|
|
import (
|
|
|
|
"reflect"
|
|
|
|
"strings"
|
|
|
|
|
|
|
|
"github.com/ledgerwatch/erigon/cmd/lightclient/sentinel/proto"
|
|
|
|
"github.com/ledgerwatch/log/v3"
|
|
|
|
"github.com/libp2p/go-libp2p/core/network"
|
|
|
|
)
|
|
|
|
|
|
|
|
// curryStreamHandler converts a func(ctx *proto.StreamContext, dat proto.Packet) error to func(network.Stream)
|
|
|
|
// this allows us to write encoding non specific type safe handler without performance overhead
|
|
|
|
func curryStreamHandler[T proto.Packet](newcodec func(network.Stream) proto.StreamCodec, fn func(ctx *proto.StreamContext, v T) error) func(network.Stream) {
|
|
|
|
return func(s network.Stream) {
|
2022-09-30 22:25:17 +00:00
|
|
|
defer s.Close()
|
2022-09-30 17:07:13 +00:00
|
|
|
sd := newcodec(s)
|
|
|
|
var t T
|
|
|
|
val := t.Clone().(T)
|
|
|
|
ctx, err := sd.Decode(val)
|
|
|
|
if err != nil {
|
|
|
|
// the stream reset error is ignored, because
|
|
|
|
if !strings.Contains(err.Error(), "stream reset") {
|
|
|
|
log.Debug("fail to decode packet", "err", err, "path", ctx.Protocol, "pkt", reflect.TypeOf(val))
|
|
|
|
}
|
|
|
|
return
|
|
|
|
}
|
|
|
|
err = fn(ctx, val)
|
|
|
|
if err != nil {
|
|
|
|
log.Debug("failed handling packet", "err", err, "path", ctx.Protocol, "pkt", reflect.TypeOf(val))
|
|
|
|
return
|
|
|
|
}
|
|
|
|
log.Trace("[ReqResp] Req->Host", "from", ctx.Stream.ID(), "endpoint", ctx.Protocol, "msg", val)
|
|
|
|
}
|
|
|
|
}
|