package rest import ( "context" "encoding/base64" "fmt" "net/http" "strconv" "strings" "time" "github.com/go-chi/chi/v5" "github.com/multiformats/go-multiaddr" "github.com/waku-org/go-waku/waku/v2/node" "github.com/waku-org/go-waku/waku/v2/peerstore" "github.com/waku-org/go-waku/waku/v2/protocol/store" "github.com/waku-org/go-waku/waku/v2/protocol/store/pb" "github.com/waku-org/go-waku/waku/v2/utils" ) type StoreService struct { node *node.WakuNode mux *chi.Mux } type StoreResponse struct { Messages []StoreWakuMessage `json:"messages"` Cursor *HistoryCursor `json:"cursor,omitempty"` ErrorMessage string `json:"error_message,omitempty"` } type HistoryCursor struct { PubsubTopic string `json:"pubsub_topic"` SenderTime string `json:"sender_time"` StoreTime string `json:"store_time"` Digest []byte `json:"digest"` } type StoreWakuMessage struct { Payload []byte `json:"payload"` ContentTopic string `json:"content_topic"` Version int32 `json:"version"` Timestamp int64 `json:"timestamp"` Meta []byte `json:"meta"` } const routeStoreMessagesV1 = "/store/v1/messages" func NewStoreService(node *node.WakuNode, m *chi.Mux) *StoreService { s := &StoreService{ node: node, mux: m, } m.Get(routeStoreMessagesV1, s.getV1Messages) return s } func getStoreParams(r *http.Request) (multiaddr.Multiaddr, *store.Query, []store.HistoryRequestOption, error) { query := &store.Query{} var options []store.HistoryRequestOption peerAddrStr := r.URL.Query().Get("peerAddr") m, err := multiaddr.NewMultiaddr(peerAddrStr) if err != nil { return nil, nil, nil, err } peerID, err := utils.GetPeerID(m) if err != nil { return nil, nil, nil, err } options = append(options, store.WithPeer(peerID)) query.Topic = r.URL.Query().Get("pubsubTopic") contentTopics := r.URL.Query().Get("contentTopics") if contentTopics != "" { query.ContentTopics = strings.Split(contentTopics, ",") } startTimeStr := r.URL.Query().Get("startTime") if startTimeStr != "" { query.StartTime, err = strconv.ParseInt(startTimeStr, 10, 64) if err != nil { return nil, nil, nil, err } } endTimeStr := r.URL.Query().Get("endTime") if endTimeStr != "" { query.EndTime, err = strconv.ParseInt(endTimeStr, 10, 64) if err != nil { return nil, nil, nil, err } } var cursor *pb.Index senderTimeStr := r.URL.Query().Get("senderTime") storeTimeStr := r.URL.Query().Get("storeTime") digestStr := r.URL.Query().Get("digest") if senderTimeStr != "" || storeTimeStr != "" || digestStr != "" { cursor = &pb.Index{} if senderTimeStr != "" { cursor.SenderTime, err = strconv.ParseInt(senderTimeStr, 10, 64) if err != nil { return nil, nil, nil, err } } if storeTimeStr != "" { cursor.ReceiverTime, err = strconv.ParseInt(storeTimeStr, 10, 64) if err != nil { return nil, nil, nil, err } } if digestStr != "" { cursor.Digest, err = base64.URLEncoding.DecodeString(digestStr) if err != nil { return nil, nil, nil, err } } cursor.PubsubTopic = query.Topic options = append(options, store.WithCursor(cursor)) } pageSizeStr := r.URL.Query().Get("pageSize") ascendingStr := r.URL.Query().Get("ascending") if ascendingStr != "" || pageSizeStr != "" { ascending := true pageSize := uint64(store.MaxPageSize) if ascendingStr != "" { ascending, err = strconv.ParseBool(ascendingStr) if err != nil { return nil, nil, nil, err } } if pageSizeStr != "" { pageSize, err = strconv.ParseUint(pageSizeStr, 10, 64) if err != nil { return nil, nil, nil, err } } options = append(options, store.WithPaging(ascending, pageSize)) } return m, query, options, nil } func writeStoreError(w http.ResponseWriter, code int, err error) { writeResponse(w, StoreResponse{ErrorMessage: err.Error()}, code) } func toStoreResponse(result *store.Result) StoreResponse { response := StoreResponse{} cursor := result.Cursor() if cursor != nil { response.Cursor = &HistoryCursor{ PubsubTopic: cursor.PubsubTopic, SenderTime: fmt.Sprintf("%d", cursor.SenderTime), StoreTime: fmt.Sprintf("%d", cursor.ReceiverTime), Digest: cursor.Digest, } } for _, m := range result.Messages { response.Messages = append(response.Messages, StoreWakuMessage{ Payload: m.Payload, ContentTopic: m.ContentTopic, Version: int32(m.Version), Timestamp: m.Timestamp, Meta: m.Meta, }) } return response } func (d *StoreService) getV1Messages(w http.ResponseWriter, r *http.Request) { peerAddr, query, options, err := getStoreParams(r) if err != nil { writeStoreError(w, http.StatusBadRequest, err) return } ctx, cancel := context.WithTimeout(r.Context(), 5*time.Second) defer cancel() _, err = d.node.AddPeer(peerAddr, peerstore.Static) if err != nil { writeStoreError(w, http.StatusInternalServerError, err) return } result, err := d.node.Store().Query(ctx, *query, options...) if err != nil { writeStoreError(w, http.StatusInternalServerError, err) return } writeErrOrResponse(w, nil, toStoreResponse(result)) }