package coreapi import ( "context" "errors" "strings" "sync" "time" cid "github.com/ipfs/go-cid" coreiface "github.com/ipfs/interface-go-ipfs-core" caopts "github.com/ipfs/interface-go-ipfs-core/options" p2phost "github.com/libp2p/go-libp2p-core/host" peer "github.com/libp2p/go-libp2p-core/peer" routing "github.com/libp2p/go-libp2p-core/routing" pubsub "github.com/libp2p/go-libp2p-pubsub" ) type PubSubAPI CoreAPI type pubSubSubscription struct { cancel context.CancelFunc subscription *pubsub.Subscription } type pubSubMessage struct { msg *pubsub.Message } func (api *PubSubAPI) Ls(ctx context.Context) ([]string, error) { _, err := api.checkNode() if err != nil { return nil, err } return api.pubSub.GetTopics(), nil } func (api *PubSubAPI) Peers(ctx context.Context, opts ...caopts.PubSubPeersOption) ([]peer.ID, error) { _, err := api.checkNode() if err != nil { return nil, err } settings, err := caopts.PubSubPeersOptions(opts...) if err != nil { return nil, err } return api.pubSub.ListPeers(settings.Topic), nil } func (api *PubSubAPI) Publish(ctx context.Context, topic string, data []byte) error { _, err := api.checkNode() if err != nil { return err } return api.pubSub.Publish(topic, data) } func (api *PubSubAPI) Subscribe(ctx context.Context, topic string, opts ...caopts.PubSubSubscribeOption) (coreiface.PubSubSubscription, error) { options, err := caopts.PubSubSubscribeOptions(opts...) if err != nil { return nil, err } r, err := api.checkNode() if err != nil { return nil, err } sub, err := api.pubSub.Subscribe(topic) if err != nil { return nil, err } pubctx, cancel := context.WithCancel(api.nctx) if options.Discover { go func() { blk, err := api.core().Block().Put(pubctx, strings.NewReader("floodsub:"+topic)) if err != nil { log.Error("pubsub discovery: ", err) return } connectToPubSubPeers(pubctx, r, api.peerHost, blk.Path().Cid()) }() } return &pubSubSubscription{cancel, sub}, nil } func connectToPubSubPeers(ctx context.Context, r routing.Routing, ph p2phost.Host, cid cid.Cid) { ctx, cancel := context.WithCancel(ctx) defer cancel() provs := r.FindProvidersAsync(ctx, cid, 10) var wg sync.WaitGroup for p := range provs { wg.Add(1) go func(pi peer.AddrInfo) { defer wg.Done() ctx, cancel := context.WithTimeout(ctx, time.Second*10) defer cancel() err := ph.Connect(ctx, pi) if err != nil { log.Info("pubsub discover: ", err) return } log.Info("connected to pubsub peer:", pi.ID) }(p) } wg.Wait() } func (api *PubSubAPI) checkNode() (routing.Routing, error) { if api.pubSub == nil { return nil, errors.New("experimental pubsub feature not enabled. Run daemon with --enable-pubsub-experiment to use.") } err := api.checkOnline(false) if err != nil { return nil, err } return api.routing, nil } func (sub *pubSubSubscription) Close() error { sub.cancel() sub.subscription.Cancel() return nil } func (sub *pubSubSubscription) Next(ctx context.Context) (coreiface.PubSubMessage, error) { msg, err := sub.subscription.Next(ctx) if err != nil { return nil, err } return &pubSubMessage{msg}, nil } func (msg *pubSubMessage) From() peer.ID { return peer.ID(msg.msg.From) } func (msg *pubSubMessage) Data() []byte { return msg.msg.Data } func (msg *pubSubMessage) Seq() []byte { return msg.msg.Seqno } func (msg *pubSubMessage) Topics() []string { return msg.msg.TopicIDs } func (api *PubSubAPI) core() coreiface.CoreAPI { return (*CoreAPI)(api) }