package sync import ( "reflect" "strings" pubsub "github.com/libp2p/go-libp2p-pubsub" "github.com/pkg/errors" ssz "github.com/prysmaticlabs/fastssz" "github.com/prysmaticlabs/prysm/v4/beacon-chain/p2p" ethpb "github.com/prysmaticlabs/prysm/v4/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) (ssz.Unmarshaler, error) { if msg == nil || msg.Topic == nil || *msg.Topic == "" { return nil, errNilPubsubMessage } topic := *msg.Topic fDigest, err := p2p.ExtractGossipDigest(topic) if err != nil { return nil, errors.Wrapf(err, "extraction failed for topic: %s", 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, ok := proto.Clone(base).(ssz.Unmarshaler) if !ok { return nil, errors.Errorf("message of %T does not support marshaller interface", base) } // Handle different message types across forks. if topic == p2p.BlockSubnetTopicFormat { m, err = extractBlockDataType(fDigest[:], s.cfg.clock) if err != nil { return nil, err } } 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 (_ *Service) replaceForkDigest(topic string) (string, error) { subStrings := strings.Split(topic, "/") if len(subStrings) != 4 { return "", errInvalidTopic } subStrings[2] = "%x" return strings.Join(subStrings, "/"), nil }