package sync import ( "reflect" "strings" pubsub "github.com/libp2p/go-libp2p-pubsub" "github.com/pkg/errors" "github.com/prysmaticlabs/prysm/beacon-chain/p2p" ethpb "github.com/prysmaticlabs/prysm/proto/prysm/v1alpha1" "google.golang.org/protobuf/proto" ) var errNilPubsubMessage = errors.New("nil pubsub message") var errInvalidTopic = errors.New("invalid topic format") func (s *Service) decodePubsubMessage(msg *pubsub.Message) (proto.Message, error) { if msg == nil || msg.Topic == nil || *msg.Topic == "" { return nil, errNilPubsubMessage } topic := *msg.Topic topic = strings.TrimSuffix(topic, s.cfg.P2P.Encoding().ProtocolSuffix()) topic, err := s.replaceForkDigest(topic) if err != nil { return nil, err } // Specially handle subnet messages. switch { case strings.Contains(topic, p2p.GossipAttestationMessage): topic = p2p.GossipTypeMapping[reflect.TypeOf(ðpb.Attestation{})] // Given that both sync message related subnets have the same message name, we have to // differentiate them below. case strings.Contains(topic, p2p.GossipSyncCommitteeMessage) && !strings.Contains(topic, p2p.SyncContributionAndProofSubnetTopicFormat): topic = p2p.GossipTypeMapping[reflect.TypeOf(ðpb.SyncCommitteeMessage{})] } base := p2p.GossipTopicMappings(topic, 0) if base == nil { return nil, p2p.ErrMessageNotMapped } m := proto.Clone(base) if err := s.cfg.P2P.Encoding().DecodeGossip(msg.Data, m); err != nil { return nil, err } return m, nil } // Replaces our fork digest with the formatter. func (s *Service) replaceForkDigest(topic string) (string, error) { subStrings := strings.Split(topic, "/") if len(subStrings) != 4 { return "", errInvalidTopic } subStrings[2] = "%x" return strings.Join(subStrings, "/"), nil }