2017-05-26 04:48:44 +08:00
|
|
|
/*
|
|
|
|
* pubsub.actor.cpp
|
|
|
|
*
|
|
|
|
* This source file is part of the FoundationDB open source project
|
|
|
|
*
|
2022-03-22 04:36:23 +08:00
|
|
|
* Copyright 2013-2022 Apple Inc. and the FoundationDB project authors
|
2018-02-22 02:25:11 +08:00
|
|
|
*
|
2017-05-26 04:48:44 +08:00
|
|
|
* Licensed under the Apache License, Version 2.0 (the "License");
|
|
|
|
* you may not use this file except in compliance with the License.
|
|
|
|
* You may obtain a copy of the License at
|
2018-02-22 02:25:11 +08:00
|
|
|
*
|
2017-05-26 04:48:44 +08:00
|
|
|
* http://www.apache.org/licenses/LICENSE-2.0
|
2018-02-22 02:25:11 +08:00
|
|
|
*
|
2017-05-26 04:48:44 +08:00
|
|
|
* Unless required by applicable law or agreed to in writing, software
|
|
|
|
* distributed under the License is distributed on an "AS IS" BASIS,
|
|
|
|
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
|
|
* See the License for the specific language governing permissions and
|
|
|
|
* limitations under the License.
|
|
|
|
*/
|
|
|
|
|
2019-05-05 01:52:02 +08:00
|
|
|
#include <cinttypes>
|
2019-02-18 07:41:16 +08:00
|
|
|
#include "fdbclient/NativeAPI.actor.h"
|
2018-10-20 01:30:13 +08:00
|
|
|
#include "fdbserver/pubsub.h"
|
2021-03-11 02:06:03 +08:00
|
|
|
#include "flow/actorcompiler.h" // This must be the last #include.
|
2017-05-26 04:48:44 +08:00
|
|
|
|
2021-03-11 02:06:03 +08:00
|
|
|
Value uInt64ToValue(uint64_t v) {
|
2017-05-26 04:48:44 +08:00
|
|
|
return StringRef(format("%016llx", v));
|
|
|
|
}
|
2021-03-11 02:06:03 +08:00
|
|
|
uint64_t valueToUInt64(const StringRef& v) {
|
2017-05-26 04:48:44 +08:00
|
|
|
uint64_t x = 0;
|
2021-03-11 02:06:03 +08:00
|
|
|
sscanf(v.toString().c_str(), "%" SCNx64, &x);
|
2017-05-26 04:48:44 +08:00
|
|
|
return x;
|
|
|
|
}
|
|
|
|
|
|
|
|
Key keyForInbox(uint64_t inbox) {
|
|
|
|
return StringRef(format("i/%016llx", inbox));
|
|
|
|
}
|
2021-09-04 05:16:22 +08:00
|
|
|
Key keyForInboxSubscription(uint64_t inbox, uint64_t feed) {
|
2017-05-26 04:48:44 +08:00
|
|
|
return StringRef(format("i/%016llx/subs/%016llx", inbox, feed));
|
|
|
|
}
|
2021-09-04 05:16:22 +08:00
|
|
|
Key keyForInboxSubscriptionCount(uint64_t inbox) {
|
2017-05-26 04:48:44 +08:00
|
|
|
return StringRef(format("i/%016llx/subsCnt", inbox));
|
|
|
|
}
|
|
|
|
Key keyForInboxStalePrefix(uint64_t inbox) {
|
|
|
|
return StringRef(format("i/%016llx/stale/", inbox));
|
|
|
|
}
|
|
|
|
Key keyForInboxStaleFeed(uint64_t inbox, uint64_t feed) {
|
|
|
|
return StringRef(format("i/%016llx/stale/%016llx", inbox, feed));
|
|
|
|
}
|
|
|
|
Key keyForInboxCacheByIDPrefix(uint64_t inbox) {
|
|
|
|
return StringRef(format("i/%016llx/cid/", inbox));
|
|
|
|
}
|
|
|
|
Key keyForInboxCacheByID(uint64_t inbox, uint64_t messageId) {
|
|
|
|
return StringRef(format("i/%016llx/cid/%016llx", inbox, messageId));
|
|
|
|
}
|
|
|
|
Key keyForInboxCacheByFeedPrefix(uint64_t inbox) {
|
|
|
|
return StringRef(format("i/%016llx/cf/", inbox));
|
|
|
|
}
|
|
|
|
Key keyForInboxCacheByFeed(uint64_t inbox, uint64_t feed) {
|
|
|
|
return StringRef(format("i/%016llx/cf/%016llx", inbox, feed));
|
|
|
|
}
|
|
|
|
|
|
|
|
Key keyForFeed(uint64_t feed) {
|
|
|
|
return StringRef(format("f/%016llx", feed));
|
|
|
|
}
|
|
|
|
Key keyForFeedSubcriber(uint64_t feed, uint64_t inbox) {
|
|
|
|
return StringRef(format("f/%016llx/subs/%016llx", feed, inbox));
|
|
|
|
}
|
|
|
|
Key keyForFeedSubcriberCount(uint64_t feed) {
|
|
|
|
return StringRef(format("f/%016llx/subscCnt", feed));
|
|
|
|
}
|
|
|
|
Key keyForFeedMessage(uint64_t feed, uint64_t message) {
|
|
|
|
return StringRef(format("f/%016llx/m/%016llx", feed, message));
|
|
|
|
}
|
|
|
|
Key keyForFeedMessagePrefix(uint64_t feed) {
|
|
|
|
return StringRef(format("f/%016llx/m/", feed));
|
|
|
|
}
|
|
|
|
Key keyForFeedMessageCount(uint64_t feed) {
|
|
|
|
return StringRef(format("f/%016llx/messCount", feed));
|
|
|
|
}
|
|
|
|
// the following should go at some point: change over to range query of count 1 from feed message list
|
|
|
|
Key keyForFeedLatestMessage(uint64_t feed) {
|
|
|
|
return StringRef(format("f/%016llx/latestMessID", feed));
|
|
|
|
}
|
|
|
|
Key keyForFeedWatcherPrefix(uint64_t feed) {
|
|
|
|
return StringRef(format("f/%016llx/watchers/", feed));
|
|
|
|
}
|
|
|
|
Key keyForFeedWatcher(uint64_t feed, uint64_t inbox) {
|
|
|
|
return StringRef(format("f/%016llx/watchers/%016llx", feed, inbox));
|
|
|
|
}
|
|
|
|
|
|
|
|
Standalone<StringRef> messagePrefix(LiteralStringRef("m/"));
|
|
|
|
|
|
|
|
Key keyForMessage(uint64_t message) {
|
|
|
|
return StringRef(format("m/%016llx", message));
|
|
|
|
}
|
|
|
|
|
|
|
|
Key keyForDisptchEntry(uint64_t message) {
|
|
|
|
return StringRef(format("d/%016llx", message));
|
|
|
|
}
|
|
|
|
|
2021-03-11 02:06:03 +08:00
|
|
|
PubSub::PubSub(Database _cx) : cx(_cx) {}
|
2017-05-26 04:48:44 +08:00
|
|
|
|
|
|
|
ACTOR Future<uint64_t> _createFeed(Database cx, Standalone<StringRef> metadata) {
|
2021-03-11 02:06:03 +08:00
|
|
|
state uint64_t id(deterministicRandom()->randomUniqueID().first()); // SOMEDAY: this should be an atomic increment
|
2017-05-26 04:48:44 +08:00
|
|
|
TraceEvent("PubSubCreateFeed").detail("Feed", id);
|
|
|
|
state Transaction tr(cx);
|
|
|
|
loop {
|
|
|
|
try {
|
|
|
|
state Optional<Value> val = wait(tr.get(keyForFeed(id)));
|
2021-03-11 02:06:03 +08:00
|
|
|
while (val.present()) {
|
2019-05-11 05:01:52 +08:00
|
|
|
id = id + deterministicRandom()->randomInt(1, 100);
|
2017-05-26 04:48:44 +08:00
|
|
|
Optional<Value> v = wait(tr.get(keyForFeed(id)));
|
|
|
|
val = v;
|
|
|
|
}
|
|
|
|
tr.set(keyForFeed(id), metadata);
|
|
|
|
tr.set(keyForFeedSubcriberCount(id), uInt64ToValue(0));
|
|
|
|
tr.set(keyForFeedMessageCount(id), uInt64ToValue(0));
|
2018-08-11 04:57:10 +08:00
|
|
|
wait(tr.commit());
|
2017-05-26 04:48:44 +08:00
|
|
|
break;
|
2021-03-11 02:06:03 +08:00
|
|
|
} catch (Error& e) {
|
|
|
|
wait(tr.onError(e));
|
2017-05-26 04:48:44 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
return id;
|
|
|
|
}
|
|
|
|
|
|
|
|
Future<uint64_t> PubSub::createFeed(Standalone<StringRef> metadata) {
|
|
|
|
return _createFeed(cx, metadata);
|
|
|
|
}
|
|
|
|
|
|
|
|
ACTOR Future<uint64_t> _createInbox(Database cx, Standalone<StringRef> metadata) {
|
2019-05-11 05:01:52 +08:00
|
|
|
state uint64_t id = deterministicRandom()->randomUniqueID().first();
|
2017-05-26 04:48:44 +08:00
|
|
|
TraceEvent("PubSubCreateInbox").detail("Inbox", id);
|
|
|
|
state Transaction tr(cx);
|
|
|
|
loop {
|
|
|
|
try {
|
|
|
|
state Optional<Value> val = wait(tr.get(keyForInbox(id)));
|
2021-03-11 02:06:03 +08:00
|
|
|
while (val.present()) {
|
2019-05-11 05:01:52 +08:00
|
|
|
id += deterministicRandom()->randomInt(1, 100);
|
2017-05-26 04:48:44 +08:00
|
|
|
Optional<Value> v = wait(tr.get(keyForFeed(id)));
|
|
|
|
val = v;
|
|
|
|
}
|
|
|
|
tr.set(keyForInbox(id), metadata);
|
2021-09-04 05:16:22 +08:00
|
|
|
tr.set(keyForInboxSubscriptionCount(id), uInt64ToValue(0));
|
2021-03-11 02:06:03 +08:00
|
|
|
wait(tr.commit());
|
2017-05-26 04:48:44 +08:00
|
|
|
break;
|
2021-03-11 02:06:03 +08:00
|
|
|
} catch (Error& e) {
|
|
|
|
wait(tr.onError(e));
|
2017-05-26 04:48:44 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
return id;
|
|
|
|
}
|
|
|
|
|
|
|
|
Future<uint64_t> PubSub::createInbox(Standalone<StringRef> metadata) {
|
|
|
|
return _createInbox(cx, metadata);
|
|
|
|
}
|
|
|
|
|
2021-09-04 05:16:22 +08:00
|
|
|
ACTOR Future<bool> _createSubscription(Database cx, uint64_t feed, uint64_t inbox) {
|
2017-05-26 04:48:44 +08:00
|
|
|
state Transaction tr(cx);
|
|
|
|
TraceEvent("PubSubCreateSubscription").detail("Feed", feed).detail("Inbox", inbox);
|
|
|
|
loop {
|
|
|
|
try {
|
2021-09-04 05:16:22 +08:00
|
|
|
Optional<Value> subscription = wait(tr.get(keyForInboxSubscription(inbox, feed)));
|
|
|
|
if (subscription.present()) {
|
2017-05-26 04:48:44 +08:00
|
|
|
// For idempotency, this could exist from a previous transaction from us that succeeded
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
Optional<Value> inboxVal = wait(tr.get(keyForInbox(inbox)));
|
2021-03-11 02:06:03 +08:00
|
|
|
if (!inboxVal.present()) {
|
2017-05-26 04:48:44 +08:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
Optional<Value> feedVal = wait(tr.get(keyForFeed(feed)));
|
2021-03-11 02:06:03 +08:00
|
|
|
if (!feedVal.present()) {
|
2017-05-26 04:48:44 +08:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Update the subscriptions of the inbox
|
2021-09-04 05:16:22 +08:00
|
|
|
Optional<Value> subscriptionCountVal = wait(tr.get(keyForInboxSubscriptionCount(inbox)));
|
|
|
|
uint64_t subscriptionCount = valueToUInt64(subscriptionCountVal.get()); // throws if count not present
|
|
|
|
tr.set(keyForInboxSubscription(inbox, feed), StringRef());
|
|
|
|
tr.set(keyForInboxSubscriptionCount(inbox), uInt64ToValue(subscriptionCount + 1));
|
2017-05-26 04:48:44 +08:00
|
|
|
|
|
|
|
// Update the subcribers of the feed
|
|
|
|
Optional<Value> subcriberCountVal = wait(tr.get(keyForFeedSubcriberCount(feed)));
|
|
|
|
uint64_t subcriberCount = valueToUInt64(subcriberCountVal.get()); // throws if count not present
|
|
|
|
tr.set(keyForFeedSubcriber(feed, inbox), StringRef());
|
|
|
|
tr.set(keyForFeedSubcriberCount(inbox), uInt64ToValue(subcriberCount + 1));
|
|
|
|
|
|
|
|
// Add inbox as watcher of feed.
|
|
|
|
tr.set(keyForFeedWatcher(feed, inbox), StringRef());
|
|
|
|
|
2021-03-11 02:06:03 +08:00
|
|
|
wait(tr.commit());
|
2017-05-26 04:48:44 +08:00
|
|
|
break;
|
2021-03-11 02:06:03 +08:00
|
|
|
} catch (Error& e) {
|
|
|
|
wait(tr.onError(e));
|
2017-05-26 04:48:44 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2021-09-04 05:16:22 +08:00
|
|
|
Future<bool> PubSub::createSubscription(uint64_t feed, uint64_t inbox) {
|
|
|
|
return _createSubscription(cx, feed, inbox);
|
2017-05-26 04:48:44 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
// Since we are not relying on "read-your-own-writes", we need to keep track of
|
|
|
|
// the highest-numbered inbox that we've cleared from the watchers list and
|
|
|
|
// make sure that further requests start after this inbox.
|
2021-03-11 02:06:03 +08:00
|
|
|
ACTOR Future<Void> updateFeedWatchers(Transaction* tr, uint64_t feed) {
|
2017-05-26 04:48:44 +08:00
|
|
|
state StringRef watcherPrefix = keyForFeedWatcherPrefix(feed);
|
|
|
|
state uint64_t highestInbox;
|
|
|
|
state bool first = true;
|
|
|
|
loop {
|
|
|
|
// Grab watching inboxes in swaths of 100
|
2021-05-04 04:14:16 +08:00
|
|
|
state RangeResult watchingInboxes =
|
2021-03-11 02:06:03 +08:00
|
|
|
wait((*tr).getRange(firstGreaterOrEqual(keyForFeedWatcher(feed, first ? 0 : highestInbox + 1)),
|
|
|
|
firstGreaterOrEqual(keyForFeedWatcher(feed, UINT64_MAX)),
|
|
|
|
100)); // REVIEW: does 100 make sense?
|
|
|
|
if (!watchingInboxes.size())
|
2017-05-26 04:48:44 +08:00
|
|
|
// If there are no watchers, return.
|
|
|
|
return Void();
|
|
|
|
first = false;
|
|
|
|
state int idx = 0;
|
2021-03-11 02:06:03 +08:00
|
|
|
for (; idx < watchingInboxes.size(); idx++) {
|
2017-05-26 04:48:44 +08:00
|
|
|
KeyRef key = watchingInboxes[idx].key;
|
|
|
|
StringRef inboxStr = key.removePrefix(watcherPrefix);
|
|
|
|
uint64_t inbox = valueToUInt64(inboxStr);
|
|
|
|
// add this feed to the stale list of inbox
|
|
|
|
(*tr).set(keyForInboxStaleFeed(inbox, feed), StringRef());
|
|
|
|
// remove the inbox from the list of watchers on this feed
|
|
|
|
(*tr).clear(key);
|
|
|
|
highestInbox = inbox;
|
|
|
|
}
|
2021-03-11 02:06:03 +08:00
|
|
|
if (watchingInboxes.size() < 100)
|
2017-05-26 04:48:44 +08:00
|
|
|
// If there were fewer watchers returned that we asked for, we're done.
|
|
|
|
return Void();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Posts a message to a feed. This updates the list of stale feeds to all watchers.
|
|
|
|
* Return: a per-feed (non-global) message ID.
|
|
|
|
*
|
|
|
|
* This needs many additions to make it "real"
|
|
|
|
* SOMEDAY: create better global message table to enforce cross-feed ordering.
|
|
|
|
* SOMEDAY: create a global "dispatching" list for feeds that have yet to fully update inboxes.
|
|
|
|
* Move feed in and remove watchers in one transaction, possibly
|
|
|
|
* SOMEDAY: create a global list of the most-subscribed-to feeds that all inbox reads check
|
|
|
|
*/
|
|
|
|
ACTOR Future<uint64_t> _postMessage(Database cx, uint64_t feed, Standalone<StringRef> data) {
|
|
|
|
state Transaction tr(cx);
|
|
|
|
state uint64_t messageId = UINT64_MAX - (uint64_t)now();
|
|
|
|
TraceEvent("PubSubPost").detail("Feed", feed).detail("Message", messageId);
|
|
|
|
loop {
|
|
|
|
try {
|
|
|
|
Optional<Value> feedValue = wait(tr.get(keyForFeed(feed)));
|
2021-03-11 02:06:03 +08:00
|
|
|
if (!feedValue.present()) {
|
2017-05-26 04:48:44 +08:00
|
|
|
// No such feed!!
|
|
|
|
return uint64_t(0);
|
|
|
|
}
|
|
|
|
|
|
|
|
// Get globally latest message, set our ID to that less one
|
2021-05-04 04:14:16 +08:00
|
|
|
state RangeResult latestMessage = wait(
|
2021-03-11 02:06:03 +08:00
|
|
|
tr.getRange(firstGreaterOrEqual(keyForMessage(0)), firstGreaterOrEqual(keyForMessage(UINT64_MAX)), 1));
|
|
|
|
if (!latestMessage.size()) {
|
2017-05-26 04:48:44 +08:00
|
|
|
messageId = UINT64_MAX - 1;
|
|
|
|
} else {
|
|
|
|
StringRef messageStr = latestMessage[0].key.removePrefix(messagePrefix);
|
|
|
|
messageId = valueToUInt64(messageStr) - 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
tr.set(keyForMessage(messageId), StringRef());
|
|
|
|
tr.set(keyForDisptchEntry(messageId), StringRef());
|
|
|
|
|
2018-08-11 04:57:10 +08:00
|
|
|
wait(tr.commit());
|
2017-05-26 04:48:44 +08:00
|
|
|
break;
|
2021-03-11 02:06:03 +08:00
|
|
|
} catch (Error& e) {
|
|
|
|
wait(tr.onError(e));
|
2017-05-26 04:48:44 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
tr = Transaction(cx);
|
|
|
|
loop {
|
|
|
|
try {
|
|
|
|
// Record this ID as the "latest message" for a feed
|
|
|
|
tr.set(keyForFeedLatestMessage(feed), uInt64ToValue(messageId));
|
|
|
|
|
|
|
|
// Store message in list of feed's messages
|
|
|
|
tr.set(keyForFeedMessage(feed, messageId), StringRef());
|
|
|
|
|
|
|
|
// Update the count of message that this feed has published
|
|
|
|
Optional<Value> cntValue = wait(tr.get(keyForFeedMessageCount(feed)));
|
|
|
|
uint64_t messageCount(valueToUInt64(cntValue.get()) + 1);
|
|
|
|
tr.set(keyForFeedMessageCount(feed), uInt64ToValue(messageCount));
|
|
|
|
|
|
|
|
// Go through the list of watching inboxes
|
2018-08-11 04:57:10 +08:00
|
|
|
wait(updateFeedWatchers(&tr, feed));
|
2017-05-26 04:48:44 +08:00
|
|
|
|
|
|
|
// Post the real message data; clear the "dispatching" entry
|
|
|
|
tr.set(keyForMessage(messageId), data);
|
|
|
|
tr.clear(keyForDisptchEntry(messageId));
|
|
|
|
|
2018-08-11 04:57:10 +08:00
|
|
|
wait(tr.commit());
|
2017-05-26 04:48:44 +08:00
|
|
|
break;
|
2021-03-11 02:06:03 +08:00
|
|
|
} catch (Error& e) {
|
|
|
|
wait(tr.onError(e));
|
2017-05-26 04:48:44 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
return messageId;
|
|
|
|
}
|
|
|
|
|
|
|
|
Future<uint64_t> PubSub::postMessage(uint64_t feed, Standalone<StringRef> data) {
|
|
|
|
return _postMessage(cx, feed, data);
|
|
|
|
}
|
|
|
|
|
|
|
|
ACTOR Future<int> singlePassInboxCacheUpdate(Database cx, uint64_t inbox, int swath) {
|
|
|
|
state Transaction tr(cx);
|
|
|
|
loop {
|
|
|
|
try {
|
|
|
|
// For each stale feed, update cache with latest message id
|
2021-05-04 04:14:16 +08:00
|
|
|
state RangeResult staleFeeds =
|
2021-03-11 02:06:03 +08:00
|
|
|
wait(tr.getRange(firstGreaterOrEqual(keyForInboxStaleFeed(inbox, 0)),
|
|
|
|
firstGreaterOrEqual(keyForInboxStaleFeed(inbox, UINT64_MAX)),
|
|
|
|
swath)); // REVIEW: does 100 make sense?
|
|
|
|
// printf(" --> stale feeds list size: %d\n", staleFeeds.size());
|
|
|
|
if (!staleFeeds.size())
|
2017-05-26 04:48:44 +08:00
|
|
|
// If there are no stale feeds, return.
|
|
|
|
return 0;
|
|
|
|
state StringRef stalePrefix = keyForInboxStalePrefix(inbox);
|
|
|
|
state int idx = 0;
|
2021-03-11 02:06:03 +08:00
|
|
|
for (; idx < staleFeeds.size(); idx++) {
|
2017-05-26 04:48:44 +08:00
|
|
|
StringRef feedStr = staleFeeds[idx].key.removePrefix(stalePrefix);
|
2021-03-11 02:06:03 +08:00
|
|
|
// printf(" --> clearing stale entry: %s\n", feedStr.toString().c_str());
|
2017-05-26 04:48:44 +08:00
|
|
|
state uint64_t feed = valueToUInt64(feedStr);
|
|
|
|
|
|
|
|
// SOMEDAY: change this to be a range query for the highest #'ed message
|
|
|
|
Optional<Value> v = wait(tr.get(keyForFeedLatestMessage(feed)));
|
|
|
|
state Value latestMessageValue = v.get();
|
2021-03-11 02:06:03 +08:00
|
|
|
// printf(" --> latest message from feed: %s\n", latestMessageValue.toString().c_str());
|
2017-05-26 04:48:44 +08:00
|
|
|
|
|
|
|
// find the messageID which is currently cached for this feed
|
|
|
|
Optional<Value> lastCachedValue = wait(tr.get(keyForInboxCacheByFeed(inbox, feed)));
|
2021-03-11 02:06:03 +08:00
|
|
|
if (lastCachedValue.present()) {
|
2017-05-26 04:48:44 +08:00
|
|
|
uint64_t lastCachedId = valueToUInt64(lastCachedValue.get());
|
|
|
|
// clear out the cache entry in the "by-ID" list for this feed
|
|
|
|
// SOMEDAY: should we leave this in there in some way, or should we pull a better/more recent cache?
|
|
|
|
tr.clear(keyForInboxCacheByID(inbox, lastCachedId));
|
|
|
|
}
|
2021-03-11 02:06:03 +08:00
|
|
|
// printf(" --> caching message by ID: %s\n", keyForInboxCacheByID(inbox,
|
|
|
|
// valueToUInt64(latestMessageValue)).toString().c_str());
|
2017-05-26 04:48:44 +08:00
|
|
|
tr.set(keyForInboxCacheByID(inbox, valueToUInt64(latestMessageValue)), uInt64ToValue(feed));
|
|
|
|
|
|
|
|
// set the latest message
|
|
|
|
tr.set(keyForInboxCacheByFeed(inbox, feed), latestMessageValue);
|
|
|
|
tr.clear(staleFeeds[idx].key);
|
|
|
|
// place watch back on feed
|
|
|
|
tr.set(keyForFeedWatcher(feed, inbox), StringRef());
|
2021-03-11 02:06:03 +08:00
|
|
|
// printf(" --> adding watch to feed: %s\n", keyForFeedWatcher(feed, inbox).toString().c_str());
|
2017-05-26 04:48:44 +08:00
|
|
|
}
|
2018-08-11 04:57:10 +08:00
|
|
|
wait(tr.commit());
|
2017-05-26 04:48:44 +08:00
|
|
|
return staleFeeds.size();
|
2021-03-11 02:06:03 +08:00
|
|
|
} catch (Error& e) {
|
|
|
|
wait(tr.onError(e));
|
2017-05-26 04:48:44 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// SOMEDAY: evaluate if this could lead to painful loop if there's one frequent feed.
|
|
|
|
ACTOR Future<Void> updateInboxCache(Database cx, uint64_t inbox) {
|
|
|
|
state int swath = 100;
|
|
|
|
state int updatedEntries = swath;
|
2021-03-11 02:06:03 +08:00
|
|
|
while (updatedEntries >= swath) {
|
2017-05-26 04:48:44 +08:00
|
|
|
int retVal = wait(singlePassInboxCacheUpdate(cx, inbox, swath));
|
|
|
|
updatedEntries = retVal;
|
|
|
|
}
|
|
|
|
return Void();
|
|
|
|
}
|
|
|
|
|
2021-03-11 02:06:03 +08:00
|
|
|
ACTOR Future<MessageId> getFeedLatestAtOrAfter(Transaction* tr, Feed feed, MessageId position) {
|
2021-05-04 04:14:16 +08:00
|
|
|
state RangeResult lastMessageRange = wait((*tr).getRange(firstGreaterOrEqual(keyForFeedMessage(feed, position)),
|
|
|
|
firstGreaterOrEqual(keyForFeedMessage(feed, UINT64_MAX)),
|
|
|
|
1));
|
2021-03-11 02:06:03 +08:00
|
|
|
if (!lastMessageRange.size())
|
2017-05-26 04:48:44 +08:00
|
|
|
return uint64_t(0);
|
|
|
|
KeyValueRef m = lastMessageRange[0];
|
|
|
|
StringRef prefix = keyForFeedMessagePrefix(feed);
|
|
|
|
StringRef mIdStr = m.key.removePrefix(prefix);
|
|
|
|
return valueToUInt64(mIdStr);
|
|
|
|
}
|
|
|
|
|
2021-03-11 02:06:03 +08:00
|
|
|
ACTOR Future<Message> getMessage(Transaction* tr, Feed feed, MessageId id) {
|
2017-05-26 04:48:44 +08:00
|
|
|
state Message m;
|
|
|
|
m.originatorFeed = feed;
|
|
|
|
m.messageId = id;
|
|
|
|
Optional<Value> data = wait(tr->get(keyForMessage(id)));
|
|
|
|
m.data = data.get();
|
|
|
|
return m;
|
|
|
|
}
|
|
|
|
|
2019-02-13 07:38:15 +08:00
|
|
|
ACTOR Future<std::vector<Message>> _listInboxMessages(Database cx, uint64_t inbox, int count, uint64_t cursor);
|
2017-05-26 04:48:44 +08:00
|
|
|
|
|
|
|
// inboxes with MANY fast feeds may be punished by the following checks
|
|
|
|
// SOMEDAY: add a check on global lists (or on dispatching list)
|
2021-03-11 02:06:03 +08:00
|
|
|
ACTOR Future<std::vector<Message>> _listInboxMessages(Database cx, uint64_t inbox, int count, uint64_t cursor) {
|
2017-05-26 04:48:44 +08:00
|
|
|
TraceEvent("PubSubListInbox").detail("Inbox", inbox).detail("Count", count).detail("Cursor", cursor);
|
2018-08-11 04:57:10 +08:00
|
|
|
wait(updateInboxCache(cx, inbox));
|
2017-05-26 04:48:44 +08:00
|
|
|
state StringRef perIdPrefix = keyForInboxCacheByIDPrefix(inbox);
|
|
|
|
loop {
|
|
|
|
state Transaction tr(cx);
|
|
|
|
state std::vector<Message> messages;
|
|
|
|
state std::map<MessageId, Feed> feedLatest;
|
|
|
|
try {
|
|
|
|
// Fetch all cached entries for all the feeds to which we are subscribed
|
2021-09-04 05:16:22 +08:00
|
|
|
Optional<Value> cntValue = wait(tr.get(keyForInboxSubscriptionCount(inbox)));
|
2017-05-26 04:48:44 +08:00
|
|
|
uint64_t subscriptions = valueToUInt64(cntValue.get());
|
2021-05-04 04:14:16 +08:00
|
|
|
state RangeResult feeds = wait(tr.getRange(firstGreaterOrEqual(keyForInboxCacheByID(inbox, 0)),
|
|
|
|
firstGreaterOrEqual(keyForInboxCacheByID(inbox, UINT64_MAX)),
|
|
|
|
subscriptions));
|
2021-03-11 02:06:03 +08:00
|
|
|
if (!feeds.size())
|
2017-05-26 04:48:44 +08:00
|
|
|
return messages;
|
|
|
|
|
|
|
|
// read cache into map, replace entries newer than cursor with the newest older than cursor
|
|
|
|
state int idx = 0;
|
2021-03-11 02:06:03 +08:00
|
|
|
for (; idx < feeds.size(); idx++) {
|
2017-05-26 04:48:44 +08:00
|
|
|
StringRef mIdStr = feeds[idx].key.removePrefix(perIdPrefix);
|
|
|
|
MessageId messageId = valueToUInt64(mIdStr);
|
|
|
|
state Feed feed = valueToUInt64(feeds[idx].value);
|
2021-03-11 02:06:03 +08:00
|
|
|
// printf(" -> cached message %016llx from feed %016llx\n", messageId, feed);
|
|
|
|
if (messageId >= cursor) {
|
|
|
|
// printf(" -> entering message %016llx from feed %016llx\n", messageId, feed);
|
2019-05-17 04:54:06 +08:00
|
|
|
feedLatest.emplace(messageId, feed);
|
2017-05-26 04:48:44 +08:00
|
|
|
} else {
|
|
|
|
// replace this with the first message older than the cursor
|
|
|
|
MessageId mId = wait(getFeedLatestAtOrAfter(&tr, feed, cursor));
|
2021-03-11 02:06:03 +08:00
|
|
|
if (mId) {
|
2019-05-17 04:54:06 +08:00
|
|
|
feedLatest.emplace(mId, feed);
|
2017-05-26 04:48:44 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
// There were some cached feeds, but none with messages older than "cursor"
|
2021-03-11 02:06:03 +08:00
|
|
|
if (!feedLatest.size())
|
2017-05-26 04:48:44 +08:00
|
|
|
return messages;
|
|
|
|
|
|
|
|
// Check the list of dispatching messages to make sure there are no older ones than ours
|
|
|
|
state MessageId earliestMessage = feedLatest.begin()->first;
|
2021-05-04 04:14:16 +08:00
|
|
|
RangeResult dispatching = wait(tr.getRange(firstGreaterOrEqual(keyForDisptchEntry(earliestMessage)),
|
|
|
|
firstGreaterOrEqual(keyForDisptchEntry(UINT64_MAX)),
|
|
|
|
1));
|
2017-05-26 04:48:44 +08:00
|
|
|
// If there are messages "older" than ours, try this again
|
|
|
|
// (with a new transaction and a flush of the "stale" feeds
|
2021-03-11 02:06:03 +08:00
|
|
|
if (dispatching.size()) {
|
|
|
|
std::vector<Message> r = wait(_listInboxMessages(cx, inbox, count, earliestMessage));
|
2017-05-26 04:48:44 +08:00
|
|
|
return r;
|
|
|
|
}
|
|
|
|
|
|
|
|
while (messages.size() < count && feedLatest.size() > 0) {
|
|
|
|
std::map<MessageId, Feed>::iterator latest = feedLatest.begin();
|
|
|
|
state MessageId id = latest->first;
|
|
|
|
state Feed f = latest->second;
|
|
|
|
feedLatest.erase(latest);
|
|
|
|
|
2021-03-11 02:06:03 +08:00
|
|
|
Message m = wait(getMessage(&tr, f, id));
|
|
|
|
messages.push_back(m);
|
2017-05-26 04:48:44 +08:00
|
|
|
|
|
|
|
MessageId nextMessage = wait(getFeedLatestAtOrAfter(&tr, f, id + 1));
|
2021-03-11 02:06:03 +08:00
|
|
|
if (nextMessage) {
|
2019-05-17 04:54:06 +08:00
|
|
|
feedLatest.emplace(nextMessage, f);
|
2017-05-26 04:48:44 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return messages;
|
2021-03-11 02:06:03 +08:00
|
|
|
} catch (Error& e) {
|
|
|
|
wait(tr.onError(e));
|
2017-05-26 04:48:44 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
Future<std::vector<Message>> PubSub::listInboxMessages(uint64_t inbox, int count, uint64_t cursor) {
|
|
|
|
return _listInboxMessages(cx, inbox, count, cursor);
|
|
|
|
}
|
|
|
|
|
|
|
|
ACTOR Future<std::vector<Message>> _listFeedMessages(Database cx, Feed feed, int count, uint64_t cursor) {
|
|
|
|
state std::vector<Message> messages;
|
|
|
|
state Transaction tr(cx);
|
|
|
|
TraceEvent("PubSubListFeed").detail("Feed", feed).detail("Count", count).detail("Cursor", cursor);
|
|
|
|
loop {
|
|
|
|
try {
|
2021-05-04 04:14:16 +08:00
|
|
|
state RangeResult messageIds = wait(tr.getRange(firstGreaterOrEqual(keyForFeedMessage(feed, cursor)),
|
|
|
|
firstGreaterOrEqual(keyForFeedMessage(feed, UINT64_MAX)),
|
|
|
|
count));
|
2021-03-11 02:06:03 +08:00
|
|
|
if (!messageIds.size())
|
2017-05-26 04:48:44 +08:00
|
|
|
return messages;
|
|
|
|
|
|
|
|
state int idx = 0;
|
2021-03-11 02:06:03 +08:00
|
|
|
for (; idx < messageIds.size(); idx++) {
|
2017-05-26 04:48:44 +08:00
|
|
|
StringRef mIdStr = messageIds[idx].key.removePrefix(keyForFeedMessagePrefix(feed));
|
|
|
|
MessageId messageId = valueToUInt64(mIdStr);
|
2021-03-11 02:06:03 +08:00
|
|
|
Message m = wait(getMessage(&tr, feed, messageId));
|
|
|
|
messages.push_back(m);
|
2017-05-26 04:48:44 +08:00
|
|
|
}
|
|
|
|
return messages;
|
2021-03-11 02:06:03 +08:00
|
|
|
} catch (Error& e) {
|
|
|
|
wait(tr.onError(e));
|
2017-05-26 04:48:44 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
Future<std::vector<Message>> PubSub::listFeedMessages(Feed feed, int count, uint64_t cursor) {
|
|
|
|
return _listFeedMessages(cx, feed, count, cursor);
|
|
|
|
}
|