@graffiti-garden/implementation-decentralized 0.0.1
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/LICENSE +674 -0
- package/dist/1-services/1-authorization.d.ts +37 -0
- package/dist/1-services/1-authorization.d.ts.map +1 -0
- package/dist/1-services/2-dids-tests.d.ts +2 -0
- package/dist/1-services/2-dids-tests.d.ts.map +1 -0
- package/dist/1-services/2-dids.d.ts +9 -0
- package/dist/1-services/2-dids.d.ts.map +1 -0
- package/dist/1-services/3-storage-buckets-tests.d.ts +2 -0
- package/dist/1-services/3-storage-buckets-tests.d.ts.map +1 -0
- package/dist/1-services/3-storage-buckets.d.ts +11 -0
- package/dist/1-services/3-storage-buckets.d.ts.map +1 -0
- package/dist/1-services/4-inboxes-tests.d.ts +2 -0
- package/dist/1-services/4-inboxes-tests.d.ts.map +1 -0
- package/dist/1-services/4-inboxes.d.ts +87 -0
- package/dist/1-services/4-inboxes.d.ts.map +1 -0
- package/dist/1-services/utilities.d.ts +7 -0
- package/dist/1-services/utilities.d.ts.map +1 -0
- package/dist/2-primitives/1-string-encoding-tests.d.ts +2 -0
- package/dist/2-primitives/1-string-encoding-tests.d.ts.map +1 -0
- package/dist/2-primitives/1-string-encoding.d.ts +6 -0
- package/dist/2-primitives/1-string-encoding.d.ts.map +1 -0
- package/dist/2-primitives/2-content-addresses-tests.d.ts +2 -0
- package/dist/2-primitives/2-content-addresses-tests.d.ts.map +1 -0
- package/dist/2-primitives/2-content-addresses.d.ts +8 -0
- package/dist/2-primitives/2-content-addresses.d.ts.map +1 -0
- package/dist/2-primitives/3-channel-attestations-tests.d.ts +2 -0
- package/dist/2-primitives/3-channel-attestations-tests.d.ts.map +1 -0
- package/dist/2-primitives/3-channel-attestations.d.ts +13 -0
- package/dist/2-primitives/3-channel-attestations.d.ts.map +1 -0
- package/dist/2-primitives/4-allowed-attestations-tests.d.ts +2 -0
- package/dist/2-primitives/4-allowed-attestations-tests.d.ts.map +1 -0
- package/dist/2-primitives/4-allowed-attestations.d.ts +9 -0
- package/dist/2-primitives/4-allowed-attestations.d.ts.map +1 -0
- package/dist/3-protocol/1-sessions.d.ts +81 -0
- package/dist/3-protocol/1-sessions.d.ts.map +1 -0
- package/dist/3-protocol/2-handles-tests.d.ts +2 -0
- package/dist/3-protocol/2-handles-tests.d.ts.map +1 -0
- package/dist/3-protocol/2-handles.d.ts +13 -0
- package/dist/3-protocol/2-handles.d.ts.map +1 -0
- package/dist/3-protocol/3-object-encoding-tests.d.ts +2 -0
- package/dist/3-protocol/3-object-encoding-tests.d.ts.map +1 -0
- package/dist/3-protocol/3-object-encoding.d.ts +43 -0
- package/dist/3-protocol/3-object-encoding.d.ts.map +1 -0
- package/dist/3-protocol/4-graffiti.d.ts +79 -0
- package/dist/3-protocol/4-graffiti.d.ts.map +1 -0
- package/dist/3-protocol/login-dialog.html.d.ts +2 -0
- package/dist/3-protocol/login-dialog.html.d.ts.map +1 -0
- package/dist/browser/ajv-QBSREQSI.js +9 -0
- package/dist/browser/ajv-QBSREQSI.js.map +7 -0
- package/dist/browser/build-BXWPS7VK.js +2 -0
- package/dist/browser/build-BXWPS7VK.js.map +7 -0
- package/dist/browser/chunk-RFBBAUMM.js +2 -0
- package/dist/browser/chunk-RFBBAUMM.js.map +7 -0
- package/dist/browser/graffiti-KV3G3O72-URO7SJIJ.js +2 -0
- package/dist/browser/graffiti-KV3G3O72-URO7SJIJ.js.map +7 -0
- package/dist/browser/index.js +16 -0
- package/dist/browser/index.js.map +7 -0
- package/dist/browser/login-dialog.html-XUWYDNNI.js +44 -0
- package/dist/browser/login-dialog.html-XUWYDNNI.js.map +7 -0
- package/dist/browser/rock-salt-LI7DAH66-KPFEBIBO.js +2 -0
- package/dist/browser/rock-salt-LI7DAH66-KPFEBIBO.js.map +7 -0
- package/dist/browser/style-YUTCEBZV-RWYJV575.js +287 -0
- package/dist/browser/style-YUTCEBZV-RWYJV575.js.map +7 -0
- package/dist/cjs/1-services/1-authorization.js +317 -0
- package/dist/cjs/1-services/1-authorization.js.map +7 -0
- package/dist/cjs/1-services/2-dids-tests.js +44 -0
- package/dist/cjs/1-services/2-dids-tests.js.map +7 -0
- package/dist/cjs/1-services/2-dids.js +47 -0
- package/dist/cjs/1-services/2-dids.js.map +7 -0
- package/dist/cjs/1-services/3-storage-buckets-tests.js +123 -0
- package/dist/cjs/1-services/3-storage-buckets-tests.js.map +7 -0
- package/dist/cjs/1-services/3-storage-buckets.js +148 -0
- package/dist/cjs/1-services/3-storage-buckets.js.map +7 -0
- package/dist/cjs/1-services/4-inboxes-tests.js +145 -0
- package/dist/cjs/1-services/4-inboxes-tests.js.map +7 -0
- package/dist/cjs/1-services/4-inboxes.js +539 -0
- package/dist/cjs/1-services/4-inboxes.js.map +7 -0
- package/dist/cjs/1-services/utilities.js +75 -0
- package/dist/cjs/1-services/utilities.js.map +7 -0
- package/dist/cjs/2-primitives/1-string-encoding-tests.js +50 -0
- package/dist/cjs/2-primitives/1-string-encoding-tests.js.map +7 -0
- package/dist/cjs/2-primitives/1-string-encoding.js +46 -0
- package/dist/cjs/2-primitives/1-string-encoding.js.map +7 -0
- package/dist/cjs/2-primitives/2-content-addresses-tests.js +62 -0
- package/dist/cjs/2-primitives/2-content-addresses-tests.js.map +7 -0
- package/dist/cjs/2-primitives/2-content-addresses.js +53 -0
- package/dist/cjs/2-primitives/2-content-addresses.js.map +7 -0
- package/dist/cjs/2-primitives/3-channel-attestations-tests.js +130 -0
- package/dist/cjs/2-primitives/3-channel-attestations-tests.js.map +7 -0
- package/dist/cjs/2-primitives/3-channel-attestations.js +84 -0
- package/dist/cjs/2-primitives/3-channel-attestations.js.map +7 -0
- package/dist/cjs/2-primitives/4-allowed-attestations-tests.js +96 -0
- package/dist/cjs/2-primitives/4-allowed-attestations-tests.js.map +7 -0
- package/dist/cjs/2-primitives/4-allowed-attestations.js +68 -0
- package/dist/cjs/2-primitives/4-allowed-attestations.js.map +7 -0
- package/dist/cjs/3-protocol/1-sessions.js +473 -0
- package/dist/cjs/3-protocol/1-sessions.js.map +7 -0
- package/dist/cjs/3-protocol/2-handles-tests.js +39 -0
- package/dist/cjs/3-protocol/2-handles-tests.js.map +7 -0
- package/dist/cjs/3-protocol/2-handles.js +65 -0
- package/dist/cjs/3-protocol/2-handles.js.map +7 -0
- package/dist/cjs/3-protocol/3-object-encoding-tests.js +253 -0
- package/dist/cjs/3-protocol/3-object-encoding-tests.js.map +7 -0
- package/dist/cjs/3-protocol/3-object-encoding.js +287 -0
- package/dist/cjs/3-protocol/3-object-encoding.js.map +7 -0
- package/dist/cjs/3-protocol/4-graffiti.js +937 -0
- package/dist/cjs/3-protocol/4-graffiti.js.map +7 -0
- package/dist/cjs/3-protocol/login-dialog.html.js +67 -0
- package/dist/cjs/3-protocol/login-dialog.html.js.map +7 -0
- package/dist/cjs/index.js +32 -0
- package/dist/cjs/index.js.map +7 -0
- package/dist/cjs/index.spec.js +130 -0
- package/dist/cjs/index.spec.js.map +7 -0
- package/dist/esm/1-services/1-authorization.js +304 -0
- package/dist/esm/1-services/1-authorization.js.map +7 -0
- package/dist/esm/1-services/2-dids-tests.js +24 -0
- package/dist/esm/1-services/2-dids-tests.js.map +7 -0
- package/dist/esm/1-services/2-dids.js +27 -0
- package/dist/esm/1-services/2-dids.js.map +7 -0
- package/dist/esm/1-services/3-storage-buckets-tests.js +103 -0
- package/dist/esm/1-services/3-storage-buckets-tests.js.map +7 -0
- package/dist/esm/1-services/3-storage-buckets.js +132 -0
- package/dist/esm/1-services/3-storage-buckets.js.map +7 -0
- package/dist/esm/1-services/4-inboxes-tests.js +125 -0
- package/dist/esm/1-services/4-inboxes-tests.js.map +7 -0
- package/dist/esm/1-services/4-inboxes.js +533 -0
- package/dist/esm/1-services/4-inboxes.js.map +7 -0
- package/dist/esm/1-services/utilities.js +60 -0
- package/dist/esm/1-services/utilities.js.map +7 -0
- package/dist/esm/2-primitives/1-string-encoding-tests.js +33 -0
- package/dist/esm/2-primitives/1-string-encoding-tests.js.map +7 -0
- package/dist/esm/2-primitives/1-string-encoding.js +26 -0
- package/dist/esm/2-primitives/1-string-encoding.js.map +7 -0
- package/dist/esm/2-primitives/2-content-addresses-tests.js +45 -0
- package/dist/esm/2-primitives/2-content-addresses-tests.js.map +7 -0
- package/dist/esm/2-primitives/2-content-addresses.js +33 -0
- package/dist/esm/2-primitives/2-content-addresses.js.map +7 -0
- package/dist/esm/2-primitives/3-channel-attestations-tests.js +116 -0
- package/dist/esm/2-primitives/3-channel-attestations-tests.js.map +7 -0
- package/dist/esm/2-primitives/3-channel-attestations.js +69 -0
- package/dist/esm/2-primitives/3-channel-attestations.js.map +7 -0
- package/dist/esm/2-primitives/4-allowed-attestations-tests.js +82 -0
- package/dist/esm/2-primitives/4-allowed-attestations-tests.js.map +7 -0
- package/dist/esm/2-primitives/4-allowed-attestations.js +51 -0
- package/dist/esm/2-primitives/4-allowed-attestations.js.map +7 -0
- package/dist/esm/3-protocol/1-sessions.js +465 -0
- package/dist/esm/3-protocol/1-sessions.js.map +7 -0
- package/dist/esm/3-protocol/2-handles-tests.js +19 -0
- package/dist/esm/3-protocol/2-handles-tests.js.map +7 -0
- package/dist/esm/3-protocol/2-handles.js +45 -0
- package/dist/esm/3-protocol/2-handles.js.map +7 -0
- package/dist/esm/3-protocol/3-object-encoding-tests.js +248 -0
- package/dist/esm/3-protocol/3-object-encoding-tests.js.map +7 -0
- package/dist/esm/3-protocol/3-object-encoding.js +280 -0
- package/dist/esm/3-protocol/3-object-encoding.js.map +7 -0
- package/dist/esm/3-protocol/4-graffiti.js +957 -0
- package/dist/esm/3-protocol/4-graffiti.js.map +7 -0
- package/dist/esm/3-protocol/login-dialog.html.js +47 -0
- package/dist/esm/3-protocol/login-dialog.html.js.map +7 -0
- package/dist/esm/index.js +14 -0
- package/dist/esm/index.js.map +7 -0
- package/dist/esm/index.spec.js +133 -0
- package/dist/esm/index.spec.js.map +7 -0
- package/dist/index.d.ts +10 -0
- package/dist/index.d.ts.map +1 -0
- package/dist/index.spec.d.ts +2 -0
- package/dist/index.spec.d.ts.map +1 -0
- package/package.json +62 -0
- package/src/1-services/1-authorization.ts +399 -0
- package/src/1-services/2-dids-tests.ts +24 -0
- package/src/1-services/2-dids.ts +30 -0
- package/src/1-services/3-storage-buckets-tests.ts +121 -0
- package/src/1-services/3-storage-buckets.ts +183 -0
- package/src/1-services/4-inboxes-tests.ts +154 -0
- package/src/1-services/4-inboxes.ts +722 -0
- package/src/1-services/utilities.ts +65 -0
- package/src/2-primitives/1-string-encoding-tests.ts +33 -0
- package/src/2-primitives/1-string-encoding.ts +33 -0
- package/src/2-primitives/2-content-addresses-tests.ts +46 -0
- package/src/2-primitives/2-content-addresses.ts +42 -0
- package/src/2-primitives/3-channel-attestations-tests.ts +125 -0
- package/src/2-primitives/3-channel-attestations.ts +95 -0
- package/src/2-primitives/4-allowed-attestations-tests.ts +86 -0
- package/src/2-primitives/4-allowed-attestations.ts +69 -0
- package/src/3-protocol/1-sessions.ts +601 -0
- package/src/3-protocol/2-handles-tests.ts +17 -0
- package/src/3-protocol/2-handles.ts +60 -0
- package/src/3-protocol/3-object-encoding-tests.ts +269 -0
- package/src/3-protocol/3-object-encoding.ts +396 -0
- package/src/3-protocol/4-graffiti.ts +1265 -0
- package/src/3-protocol/login-dialog.html.ts +43 -0
- package/src/index.spec.ts +158 -0
- package/src/index.ts +16 -0
|
@@ -0,0 +1,539 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
var __create = Object.create;
|
|
3
|
+
var __defProp = Object.defineProperty;
|
|
4
|
+
var __getOwnPropDesc = Object.getOwnPropertyDescriptor;
|
|
5
|
+
var __getOwnPropNames = Object.getOwnPropertyNames;
|
|
6
|
+
var __getProtoOf = Object.getPrototypeOf;
|
|
7
|
+
var __hasOwnProp = Object.prototype.hasOwnProperty;
|
|
8
|
+
var __export = (target, all) => {
|
|
9
|
+
for (var name in all)
|
|
10
|
+
__defProp(target, name, { get: all[name], enumerable: true });
|
|
11
|
+
};
|
|
12
|
+
var __copyProps = (to, from, except, desc) => {
|
|
13
|
+
if (from && typeof from === "object" || typeof from === "function") {
|
|
14
|
+
for (let key of __getOwnPropNames(from))
|
|
15
|
+
if (!__hasOwnProp.call(to, key) && key !== except)
|
|
16
|
+
__defProp(to, key, { get: () => from[key], enumerable: !(desc = __getOwnPropDesc(from, key)) || desc.enumerable });
|
|
17
|
+
}
|
|
18
|
+
return to;
|
|
19
|
+
};
|
|
20
|
+
var __toESM = (mod, isNodeMode, target) => (target = mod != null ? __create(__getProtoOf(mod)) : {}, __copyProps(
|
|
21
|
+
// If the importer is in node compatibility mode or this is not an ESM
|
|
22
|
+
// file that has been converted to a CommonJS file using a Babel-
|
|
23
|
+
// compatible transform (i.e. "__esModule" has not been set), then set
|
|
24
|
+
// "default" to the CommonJS "module.exports" for node compatibility.
|
|
25
|
+
isNodeMode || !mod || !mod.__esModule ? __defProp(target, "default", { value: mod, enumerable: true }) : target,
|
|
26
|
+
mod
|
|
27
|
+
));
|
|
28
|
+
var __toCommonJS = (mod) => __copyProps(__defProp({}, "__esModule", { value: true }), mod);
|
|
29
|
+
var inboxes_exports = {};
|
|
30
|
+
__export(inboxes_exports, {
|
|
31
|
+
Inboxes: () => Inboxes,
|
|
32
|
+
LABELED_MESSAGE_ID_KEY: () => LABELED_MESSAGE_ID_KEY,
|
|
33
|
+
LABELED_MESSAGE_LABEL_KEY: () => LABELED_MESSAGE_LABEL_KEY,
|
|
34
|
+
LABELED_MESSAGE_MESSAGE_KEY: () => LABELED_MESSAGE_MESSAGE_KEY,
|
|
35
|
+
LabeledMessageBaseSchema: () => LabeledMessageBaseSchema,
|
|
36
|
+
MESSAGE_METADATA_KEY: () => MESSAGE_METADATA_KEY,
|
|
37
|
+
MESSAGE_OBJECT_KEY: () => MESSAGE_OBJECT_KEY,
|
|
38
|
+
MESSAGE_TAGS_KEY: () => MESSAGE_TAGS_KEY,
|
|
39
|
+
MessageBaseSchema: () => MessageBaseSchema,
|
|
40
|
+
TagsSchema: () => TagsSchema,
|
|
41
|
+
Uint8ArraySchema: () => Uint8ArraySchema
|
|
42
|
+
});
|
|
43
|
+
module.exports = __toCommonJS(inboxes_exports);
|
|
44
|
+
var import_utilities = require("./utilities");
|
|
45
|
+
var import_api = require("@graffiti-garden/api");
|
|
46
|
+
var import_dag_cbor = require("@ipld/dag-cbor");
|
|
47
|
+
var import_mini = require("zod/mini");
|
|
48
|
+
class Inboxes {
|
|
49
|
+
getAuthorizationEndpoint = import_utilities.getAuthorizationEndpoint;
|
|
50
|
+
cache_ = null;
|
|
51
|
+
get cache() {
|
|
52
|
+
if (!this.cache_) {
|
|
53
|
+
this.cache_ = createCache();
|
|
54
|
+
}
|
|
55
|
+
return this.cache_;
|
|
56
|
+
}
|
|
57
|
+
async send(inboxUrl, message) {
|
|
58
|
+
(0, import_utilities.verifyHTTPSEndpoint)(inboxUrl);
|
|
59
|
+
const url2 = `${inboxUrl}/send`;
|
|
60
|
+
const response = await (0, import_utilities.fetchWithErrorHandling)(url2, {
|
|
61
|
+
method: "PUT",
|
|
62
|
+
headers: {
|
|
63
|
+
"Content-Type": "application/cbor"
|
|
64
|
+
},
|
|
65
|
+
body: new Uint8Array((0, import_dag_cbor.encode)({ m: message }))
|
|
66
|
+
});
|
|
67
|
+
const blob = await response.blob();
|
|
68
|
+
const cbor = (0, import_dag_cbor.decode)(await blob.arrayBuffer());
|
|
69
|
+
const parsed = SendResponseSchema.parse(cbor);
|
|
70
|
+
return parsed.id;
|
|
71
|
+
}
|
|
72
|
+
async get(inboxUrl, messageId, inboxToken) {
|
|
73
|
+
const messageCacheKey = getMessageCacheKey(inboxUrl, messageId);
|
|
74
|
+
const cache = await this.cache;
|
|
75
|
+
const cached = await cache.messages.get(messageCacheKey);
|
|
76
|
+
if (cached) return cached;
|
|
77
|
+
const url2 = `${inboxUrl}/message/${messageId}`;
|
|
78
|
+
const response = await (0, import_utilities.fetchWithErrorHandling)(url2, {
|
|
79
|
+
method: "GET",
|
|
80
|
+
headers: {
|
|
81
|
+
...inboxToken ? {
|
|
82
|
+
Authorization: `Bearer ${inboxToken}`
|
|
83
|
+
} : {}
|
|
84
|
+
}
|
|
85
|
+
});
|
|
86
|
+
const blob = await response.blob();
|
|
87
|
+
const cbor = (0, import_dag_cbor.decode)(await blob.arrayBuffer());
|
|
88
|
+
const parsed = LabeledMessageBaseSchema.parse(cbor);
|
|
89
|
+
await cache.messages.set(messageCacheKey, parsed);
|
|
90
|
+
return parsed;
|
|
91
|
+
}
|
|
92
|
+
async label(inboxUrl, messageId, label, inboxToken) {
|
|
93
|
+
(0, import_utilities.verifyHTTPSEndpoint)(inboxUrl);
|
|
94
|
+
if (inboxToken) {
|
|
95
|
+
const url2 = `${inboxUrl}/label/${messageId}`;
|
|
96
|
+
await (0, import_utilities.fetchWithErrorHandling)(url2, {
|
|
97
|
+
method: "PUT",
|
|
98
|
+
headers: {
|
|
99
|
+
"Content-Type": "application/cbor",
|
|
100
|
+
Authorization: `Bearer ${inboxToken}`
|
|
101
|
+
},
|
|
102
|
+
body: new Uint8Array((0, import_dag_cbor.encode)({ l: label }))
|
|
103
|
+
});
|
|
104
|
+
}
|
|
105
|
+
const cache = await this.cache;
|
|
106
|
+
const messageCacheKey = getMessageCacheKey(inboxUrl, messageId);
|
|
107
|
+
const result = await cache.messages.get(messageCacheKey);
|
|
108
|
+
if (result) {
|
|
109
|
+
await cache.messages.set(messageCacheKey, {
|
|
110
|
+
...result,
|
|
111
|
+
l: label
|
|
112
|
+
});
|
|
113
|
+
}
|
|
114
|
+
}
|
|
115
|
+
async fetchMessageBatch(inboxUrl, type, body, inboxToken, cursor) {
|
|
116
|
+
const response = await (0, import_utilities.fetchWithErrorHandling)(
|
|
117
|
+
`${inboxUrl}/${type}${cursor ? `?cursor=${encodeURIComponent(cursor)}` : ""}`,
|
|
118
|
+
{
|
|
119
|
+
method: "POST",
|
|
120
|
+
headers: {
|
|
121
|
+
"Content-Type": "application/cbor",
|
|
122
|
+
...inboxToken ? {
|
|
123
|
+
Authorization: `Bearer ${inboxToken}`
|
|
124
|
+
} : {}
|
|
125
|
+
},
|
|
126
|
+
body
|
|
127
|
+
}
|
|
128
|
+
);
|
|
129
|
+
const retryAfterHeader = response.headers.get("Retry-After");
|
|
130
|
+
const retryAfter = retryAfterHeader ? parseInt(retryAfterHeader) : void 0;
|
|
131
|
+
const waitTil = retryAfter && Number.isFinite(retryAfter) ? Date.now() + retryAfter * 1e3 : void 0;
|
|
132
|
+
return { response, waitTil };
|
|
133
|
+
}
|
|
134
|
+
async *yieldFromCache(cache, inboxUrl, messageIdsCacheKey, cachedMessageIds, cacheNumSeen = 0) {
|
|
135
|
+
const messageIds = cachedMessageIds.messageIds.slice(cacheNumSeen);
|
|
136
|
+
const messages = await Promise.all(
|
|
137
|
+
messageIds.map(async (id) => {
|
|
138
|
+
const message = await cache.messages.get(
|
|
139
|
+
getMessageCacheKey(inboxUrl, id)
|
|
140
|
+
);
|
|
141
|
+
if (!message) {
|
|
142
|
+
try {
|
|
143
|
+
await cache.messageIds.del(messageIdsCacheKey);
|
|
144
|
+
} catch {
|
|
145
|
+
}
|
|
146
|
+
throw new Error("Cache out of sync - perhaps clear browser storage");
|
|
147
|
+
}
|
|
148
|
+
return message;
|
|
149
|
+
})
|
|
150
|
+
);
|
|
151
|
+
yield* messages;
|
|
152
|
+
}
|
|
153
|
+
async *lockedMessageStreamer(...args) {
|
|
154
|
+
if (typeof window === "undefined" || !await canUseIDB()) {
|
|
155
|
+
const streamer = this.messageStreamer(...args);
|
|
156
|
+
while (true) {
|
|
157
|
+
const next = await streamer.next();
|
|
158
|
+
if (next.done) return next.value;
|
|
159
|
+
yield next.value;
|
|
160
|
+
}
|
|
161
|
+
}
|
|
162
|
+
const messageIdsCacheKey = await args[0];
|
|
163
|
+
const lockKey = `graffiti:inbox:${messageIdsCacheKey}`;
|
|
164
|
+
let releaseLock = () => {
|
|
165
|
+
};
|
|
166
|
+
let hasLock = false;
|
|
167
|
+
await new Promise((resolvehasLock) => {
|
|
168
|
+
window.navigator.locks.request(
|
|
169
|
+
lockKey,
|
|
170
|
+
{
|
|
171
|
+
mode: "exclusive",
|
|
172
|
+
ifAvailable: true
|
|
173
|
+
},
|
|
174
|
+
async (lock) => {
|
|
175
|
+
hasLock = !!lock;
|
|
176
|
+
resolvehasLock();
|
|
177
|
+
await new Promise((r) => releaseLock = r);
|
|
178
|
+
}
|
|
179
|
+
);
|
|
180
|
+
});
|
|
181
|
+
if (hasLock) {
|
|
182
|
+
try {
|
|
183
|
+
const streamer = this.messageStreamer(...args);
|
|
184
|
+
while (true) {
|
|
185
|
+
const next = await streamer.next();
|
|
186
|
+
if (next.done) return next.value;
|
|
187
|
+
yield next.value;
|
|
188
|
+
}
|
|
189
|
+
} finally {
|
|
190
|
+
releaseLock();
|
|
191
|
+
}
|
|
192
|
+
}
|
|
193
|
+
releaseLock();
|
|
194
|
+
await window.navigator.locks.request(lockKey, () => {
|
|
195
|
+
});
|
|
196
|
+
const inboxUrl = args[1];
|
|
197
|
+
const objectSchema = args[5] ?? {};
|
|
198
|
+
const cacheVersion = args[6];
|
|
199
|
+
const cacheNumSeen = args[7];
|
|
200
|
+
const cache = await this.cache;
|
|
201
|
+
const cachedMessageIds = await cache.messageIds.get(messageIdsCacheKey);
|
|
202
|
+
if (!cachedMessageIds) {
|
|
203
|
+
throw new Error("Cache not found");
|
|
204
|
+
}
|
|
205
|
+
if (cacheVersion !== void 0 && cacheVersion !== cachedMessageIds.version) {
|
|
206
|
+
throw new import_api.GraffitiErrorCursorExpired("Cursor is stale");
|
|
207
|
+
}
|
|
208
|
+
const iterator = this.yieldFromCache(
|
|
209
|
+
cache,
|
|
210
|
+
inboxUrl,
|
|
211
|
+
messageIdsCacheKey,
|
|
212
|
+
cachedMessageIds,
|
|
213
|
+
cacheNumSeen
|
|
214
|
+
);
|
|
215
|
+
for await (const m of iterator) yield m;
|
|
216
|
+
const outputCursor = {
|
|
217
|
+
numSeen: cachedMessageIds.messageIds.length,
|
|
218
|
+
version: cachedMessageIds.version,
|
|
219
|
+
messageIdsCacheKey,
|
|
220
|
+
objectSchema
|
|
221
|
+
};
|
|
222
|
+
return JSON.stringify(outputCursor);
|
|
223
|
+
}
|
|
224
|
+
async *messageStreamer(messageIdsCacheKey_, inboxUrl, type, body, inboxToken, objectSchema = {}, cacheVersion, cacheNumSeen = 0) {
|
|
225
|
+
const validator = await (0, import_api.compileGraffitiObjectSchema)(objectSchema);
|
|
226
|
+
const messageIdsCacheKey = await messageIdsCacheKey_;
|
|
227
|
+
const cache = await this.cache;
|
|
228
|
+
let cachedMessageIds = await cache.messageIds.get(messageIdsCacheKey);
|
|
229
|
+
if (cacheVersion !== void 0 && cacheVersion !== cachedMessageIds?.version) {
|
|
230
|
+
throw new import_api.GraffitiErrorCursorExpired("Cursor is stale");
|
|
231
|
+
}
|
|
232
|
+
let waitTil = cachedMessageIds?.waitTil;
|
|
233
|
+
await waitFor(waitTil);
|
|
234
|
+
const cachedCursor = cachedMessageIds?.cursor;
|
|
235
|
+
let firstResponse = void 0;
|
|
236
|
+
try {
|
|
237
|
+
const out = await this.fetchMessageBatch(
|
|
238
|
+
inboxUrl,
|
|
239
|
+
type,
|
|
240
|
+
body,
|
|
241
|
+
inboxToken,
|
|
242
|
+
cachedCursor
|
|
243
|
+
);
|
|
244
|
+
firstResponse = out.response;
|
|
245
|
+
waitTil = out.waitTil;
|
|
246
|
+
} catch (e) {
|
|
247
|
+
if (!(e instanceof import_api.GraffitiErrorCursorExpired && cachedCursor)) {
|
|
248
|
+
console.error(
|
|
249
|
+
"Unexpected error in stream, waiting 5 seconds before continuing..."
|
|
250
|
+
);
|
|
251
|
+
await new Promise((resolve) => setTimeout(resolve, 5e3));
|
|
252
|
+
throw e;
|
|
253
|
+
}
|
|
254
|
+
await cache.messageIds.del(messageIdsCacheKey);
|
|
255
|
+
if (cacheVersion === void 0) {
|
|
256
|
+
cachedMessageIds = void 0;
|
|
257
|
+
} else {
|
|
258
|
+
throw e;
|
|
259
|
+
}
|
|
260
|
+
}
|
|
261
|
+
if (firstResponse !== void 0 && cachedMessageIds) {
|
|
262
|
+
const iterator = this.yieldFromCache(
|
|
263
|
+
cache,
|
|
264
|
+
inboxUrl,
|
|
265
|
+
messageIdsCacheKey,
|
|
266
|
+
cachedMessageIds,
|
|
267
|
+
cacheNumSeen
|
|
268
|
+
);
|
|
269
|
+
for await (const m of iterator) yield m;
|
|
270
|
+
}
|
|
271
|
+
if (firstResponse === void 0) {
|
|
272
|
+
const out = await this.fetchMessageBatch(
|
|
273
|
+
inboxUrl,
|
|
274
|
+
type,
|
|
275
|
+
body,
|
|
276
|
+
inboxToken
|
|
277
|
+
);
|
|
278
|
+
firstResponse = out.response;
|
|
279
|
+
waitTil = out.waitTil;
|
|
280
|
+
}
|
|
281
|
+
let response = firstResponse;
|
|
282
|
+
let cursor;
|
|
283
|
+
const version = cachedMessageIds?.version ?? crypto.randomUUID();
|
|
284
|
+
let messageIds = cachedMessageIds?.messageIds ?? [];
|
|
285
|
+
while (true) {
|
|
286
|
+
const blob = await response.blob();
|
|
287
|
+
const decoded = (0, import_dag_cbor.decode)(await blob.arrayBuffer());
|
|
288
|
+
const {
|
|
289
|
+
results,
|
|
290
|
+
hasMore,
|
|
291
|
+
cursor: nextCursor
|
|
292
|
+
} = MessageResultSchema.parse(decoded);
|
|
293
|
+
cursor = nextCursor;
|
|
294
|
+
const labeledMessages = results.map(
|
|
295
|
+
(result) => {
|
|
296
|
+
const object = result[LABELED_MESSAGE_MESSAGE_KEY][MESSAGE_OBJECT_KEY];
|
|
297
|
+
if (!validator(object)) {
|
|
298
|
+
throw new Error("Server returned data that does not match schema");
|
|
299
|
+
}
|
|
300
|
+
return {
|
|
301
|
+
...result,
|
|
302
|
+
[LABELED_MESSAGE_MESSAGE_KEY]: {
|
|
303
|
+
...result[LABELED_MESSAGE_MESSAGE_KEY],
|
|
304
|
+
[MESSAGE_OBJECT_KEY]: object
|
|
305
|
+
}
|
|
306
|
+
};
|
|
307
|
+
}
|
|
308
|
+
);
|
|
309
|
+
await Promise.all(
|
|
310
|
+
labeledMessages.map(
|
|
311
|
+
(m) => cache.messages.set(
|
|
312
|
+
getMessageCacheKey(inboxUrl, m[LABELED_MESSAGE_ID_KEY]),
|
|
313
|
+
m
|
|
314
|
+
)
|
|
315
|
+
)
|
|
316
|
+
);
|
|
317
|
+
messageIds = [
|
|
318
|
+
...messageIds,
|
|
319
|
+
...labeledMessages.map(
|
|
320
|
+
(m) => m[LABELED_MESSAGE_ID_KEY]
|
|
321
|
+
)
|
|
322
|
+
];
|
|
323
|
+
await cache.messageIds.set(messageIdsCacheKey, {
|
|
324
|
+
cursor,
|
|
325
|
+
version,
|
|
326
|
+
messageIds,
|
|
327
|
+
waitTil
|
|
328
|
+
});
|
|
329
|
+
cacheNumSeen += labeledMessages.length;
|
|
330
|
+
for (const m of labeledMessages) yield m;
|
|
331
|
+
if (!hasMore) break;
|
|
332
|
+
await waitFor(waitTil);
|
|
333
|
+
const out = await this.fetchMessageBatch(
|
|
334
|
+
inboxUrl,
|
|
335
|
+
type,
|
|
336
|
+
void 0,
|
|
337
|
+
// Body is never past the first time
|
|
338
|
+
inboxToken,
|
|
339
|
+
cursor
|
|
340
|
+
);
|
|
341
|
+
response = out.response;
|
|
342
|
+
waitTil = out.waitTil;
|
|
343
|
+
}
|
|
344
|
+
const outputCursor = {
|
|
345
|
+
numSeen: cacheNumSeen,
|
|
346
|
+
version,
|
|
347
|
+
messageIdsCacheKey,
|
|
348
|
+
objectSchema
|
|
349
|
+
};
|
|
350
|
+
return JSON.stringify(outputCursor);
|
|
351
|
+
}
|
|
352
|
+
query(inboxUrl, tags, objectSchema, inboxToken) {
|
|
353
|
+
(0, import_utilities.verifyHTTPSEndpoint)(inboxUrl);
|
|
354
|
+
const body = (0, import_dag_cbor.encode)({
|
|
355
|
+
tags,
|
|
356
|
+
schema: objectSchema
|
|
357
|
+
});
|
|
358
|
+
const messageIdsCacheKey = getMessageIdsCacheKey(inboxUrl, "query", body);
|
|
359
|
+
return this.lockedMessageStreamer(
|
|
360
|
+
messageIdsCacheKey,
|
|
361
|
+
inboxUrl,
|
|
362
|
+
"query",
|
|
363
|
+
new Uint8Array(body),
|
|
364
|
+
inboxToken,
|
|
365
|
+
objectSchema
|
|
366
|
+
);
|
|
367
|
+
}
|
|
368
|
+
continueQuery(inboxUrl, cursor, inboxToken) {
|
|
369
|
+
(0, import_utilities.verifyHTTPSEndpoint)(inboxUrl);
|
|
370
|
+
const decodedCursor = JSON.parse(cursor);
|
|
371
|
+
const { messageIdsCacheKey, numSeen, objectSchema, version } = CursorSchema.parse(decodedCursor);
|
|
372
|
+
return this.lockedMessageStreamer(
|
|
373
|
+
Promise.resolve(messageIdsCacheKey),
|
|
374
|
+
inboxUrl,
|
|
375
|
+
"query",
|
|
376
|
+
void 0,
|
|
377
|
+
inboxToken,
|
|
378
|
+
objectSchema,
|
|
379
|
+
version,
|
|
380
|
+
numSeen
|
|
381
|
+
);
|
|
382
|
+
}
|
|
383
|
+
export(inboxUrl, inboxToken) {
|
|
384
|
+
(0, import_utilities.verifyHTTPSEndpoint)(inboxUrl);
|
|
385
|
+
const messageIdsCacheKey = getMessageIdsCacheKey(inboxUrl, "export");
|
|
386
|
+
return this.lockedMessageStreamer(
|
|
387
|
+
messageIdsCacheKey,
|
|
388
|
+
inboxUrl,
|
|
389
|
+
"export",
|
|
390
|
+
void 0,
|
|
391
|
+
inboxToken
|
|
392
|
+
);
|
|
393
|
+
}
|
|
394
|
+
}
|
|
395
|
+
const GraffitiObjectSchema = (0, import_mini.strictObject)({
|
|
396
|
+
value: (0, import_mini.looseObject)({}),
|
|
397
|
+
channels: (0, import_mini.array)((0, import_mini.string)()),
|
|
398
|
+
allowed: (0, import_mini.optional)((0, import_mini.nullable)((0, import_mini.array)((0, import_mini.url)()))),
|
|
399
|
+
url: (0, import_mini.url)(),
|
|
400
|
+
actor: (0, import_mini.url)()
|
|
401
|
+
});
|
|
402
|
+
const Uint8ArraySchema = (0, import_mini.custom)(
|
|
403
|
+
(v) => v instanceof Uint8Array
|
|
404
|
+
);
|
|
405
|
+
const TagsSchema = (0, import_mini.array)(Uint8ArraySchema);
|
|
406
|
+
const MESSAGE_TAGS_KEY = "t";
|
|
407
|
+
const MESSAGE_OBJECT_KEY = "o";
|
|
408
|
+
const MESSAGE_METADATA_KEY = "m";
|
|
409
|
+
const MessageBaseSchema = (0, import_mini.strictObject)({
|
|
410
|
+
[MESSAGE_TAGS_KEY]: TagsSchema,
|
|
411
|
+
[MESSAGE_OBJECT_KEY]: GraffitiObjectSchema,
|
|
412
|
+
[MESSAGE_METADATA_KEY]: Uint8ArraySchema
|
|
413
|
+
});
|
|
414
|
+
const LABELED_MESSAGE_ID_KEY = "id";
|
|
415
|
+
const LABELED_MESSAGE_MESSAGE_KEY = "m";
|
|
416
|
+
const LABELED_MESSAGE_LABEL_KEY = "l";
|
|
417
|
+
const LabeledMessageBaseSchema = (0, import_mini.strictObject)({
|
|
418
|
+
[LABELED_MESSAGE_ID_KEY]: (0, import_mini.string)(),
|
|
419
|
+
[LABELED_MESSAGE_MESSAGE_KEY]: MessageBaseSchema,
|
|
420
|
+
[LABELED_MESSAGE_LABEL_KEY]: (0, import_mini.number)()
|
|
421
|
+
});
|
|
422
|
+
const SendResponseSchema = (0, import_mini.strictObject)({ id: (0, import_mini.string)() });
|
|
423
|
+
const MessageResultSchema = (0, import_mini.strictObject)({
|
|
424
|
+
results: (0, import_mini.array)(LabeledMessageBaseSchema),
|
|
425
|
+
hasMore: (0, import_mini.boolean)(),
|
|
426
|
+
cursor: (0, import_mini.string)()
|
|
427
|
+
});
|
|
428
|
+
const CursorSchema = (0, import_mini.strictObject)({
|
|
429
|
+
messageIdsCacheKey: (0, import_mini.string)(),
|
|
430
|
+
version: (0, import_mini.string)(),
|
|
431
|
+
numSeen: (0, import_mini.int)().check((0, import_mini.nonnegative)()),
|
|
432
|
+
objectSchema: (0, import_mini.union)([(0, import_mini.looseObject)({}), (0, import_mini.boolean)()])
|
|
433
|
+
});
|
|
434
|
+
async function canUseIDB() {
|
|
435
|
+
try {
|
|
436
|
+
if (!globalThis.indexedDB) return false;
|
|
437
|
+
await new Promise((resolve, reject) => {
|
|
438
|
+
const req = indexedDB.open("__idb_probe__", 1);
|
|
439
|
+
req.onupgradeneeded = () => req.result.createObjectStore("k");
|
|
440
|
+
req.onsuccess = () => {
|
|
441
|
+
req.result.close();
|
|
442
|
+
resolve();
|
|
443
|
+
};
|
|
444
|
+
req.onerror = () => reject(req.error);
|
|
445
|
+
});
|
|
446
|
+
return true;
|
|
447
|
+
} catch {
|
|
448
|
+
return false;
|
|
449
|
+
}
|
|
450
|
+
}
|
|
451
|
+
function getMessageCacheKey(inboxUrl, messageId) {
|
|
452
|
+
return `${encodeURIComponent(inboxUrl)}:${encodeURIComponent(messageId)}`;
|
|
453
|
+
}
|
|
454
|
+
async function getMessageIdsCacheKey(inboxUrl, type, body) {
|
|
455
|
+
const cacheIdData = (0, import_dag_cbor.encode)({
|
|
456
|
+
inboxUrl,
|
|
457
|
+
type,
|
|
458
|
+
body: body ?? null
|
|
459
|
+
});
|
|
460
|
+
return crypto.subtle.digest("SHA-256", new Uint8Array(cacheIdData)).then(
|
|
461
|
+
(bytes) => Array.from(new Uint8Array(bytes)).map((byte) => byte.toString(16).padStart(2, "0")).join("")
|
|
462
|
+
);
|
|
463
|
+
}
|
|
464
|
+
async function resetCacheDB() {
|
|
465
|
+
await new Promise((resolve) => {
|
|
466
|
+
const req = indexedDB.deleteDatabase("graffiti-inbox-cache");
|
|
467
|
+
req.onsuccess = () => resolve();
|
|
468
|
+
req.onerror = () => resolve();
|
|
469
|
+
req.onblocked = () => resolve();
|
|
470
|
+
});
|
|
471
|
+
}
|
|
472
|
+
async function createCache() {
|
|
473
|
+
if (await canUseIDB()) {
|
|
474
|
+
const { openDB } = await import("idb");
|
|
475
|
+
const db = await openDB("graffiti-inbox-cache", 1, {
|
|
476
|
+
upgrade(db2) {
|
|
477
|
+
if (!db2.objectStoreNames.contains("m")) db2.createObjectStore("m");
|
|
478
|
+
if (!db2.objectStoreNames.contains("q")) db2.createObjectStore("q");
|
|
479
|
+
}
|
|
480
|
+
});
|
|
481
|
+
return {
|
|
482
|
+
messages: {
|
|
483
|
+
get: async (k) => {
|
|
484
|
+
try {
|
|
485
|
+
return db.get("m", k);
|
|
486
|
+
} catch (error) {
|
|
487
|
+
console.error("Error getting message from cache:", error);
|
|
488
|
+
console.error("resetting cache...");
|
|
489
|
+
await resetCacheDB();
|
|
490
|
+
return void 0;
|
|
491
|
+
}
|
|
492
|
+
},
|
|
493
|
+
set: async (k, v) => {
|
|
494
|
+
await db.put("m", v, k);
|
|
495
|
+
},
|
|
496
|
+
del: (k) => db.delete("m", k)
|
|
497
|
+
},
|
|
498
|
+
messageIds: {
|
|
499
|
+
get: async (k) => {
|
|
500
|
+
try {
|
|
501
|
+
return await db.get("q", k);
|
|
502
|
+
} catch (error) {
|
|
503
|
+
console.error("Error getting message IDs from cache:", error);
|
|
504
|
+
console.error("resetting cache...");
|
|
505
|
+
await resetCacheDB();
|
|
506
|
+
return void 0;
|
|
507
|
+
}
|
|
508
|
+
},
|
|
509
|
+
set: async (k, v) => {
|
|
510
|
+
await db.put("q", v, k);
|
|
511
|
+
},
|
|
512
|
+
del: (k) => db.delete("q", k)
|
|
513
|
+
}
|
|
514
|
+
};
|
|
515
|
+
}
|
|
516
|
+
const m = /* @__PURE__ */ new Map();
|
|
517
|
+
const q = /* @__PURE__ */ new Map();
|
|
518
|
+
return {
|
|
519
|
+
messages: {
|
|
520
|
+
get: async (k) => m.get(k),
|
|
521
|
+
set: async (k, v) => void m.set(k, v),
|
|
522
|
+
del: async (k) => void m.delete(k)
|
|
523
|
+
},
|
|
524
|
+
messageIds: {
|
|
525
|
+
get: async (k) => q.get(k),
|
|
526
|
+
set: async (k, v) => void q.set(k, v),
|
|
527
|
+
del: async (k) => void q.delete(k)
|
|
528
|
+
}
|
|
529
|
+
};
|
|
530
|
+
}
|
|
531
|
+
async function waitFor(waitTil) {
|
|
532
|
+
if (waitTil !== void 0) {
|
|
533
|
+
const waitFor2 = waitTil - Date.now();
|
|
534
|
+
if (waitFor2 > 0) {
|
|
535
|
+
await new Promise((resolve) => setTimeout(resolve, waitFor2));
|
|
536
|
+
}
|
|
537
|
+
}
|
|
538
|
+
}
|
|
539
|
+
//# sourceMappingURL=4-inboxes.js.map
|
|
@@ -0,0 +1,7 @@
|
|
|
1
|
+
{
|
|
2
|
+
"version": 3,
|
|
3
|
+
"sources": ["../../../src/1-services/4-inboxes.ts"],
|
|
4
|
+
"sourcesContent": ["import type { JSONSchema, GraffitiObject } from \"@graffiti-garden/api\";\nimport {\n getAuthorizationEndpoint,\n fetchWithErrorHandling,\n verifyHTTPSEndpoint,\n} from \"./utilities\";\nimport {\n compileGraffitiObjectSchema,\n GraffitiErrorCursorExpired,\n} from \"@graffiti-garden/api\";\nimport {\n encode as dagCborEncode,\n decode as dagCborDecode,\n} from \"@ipld/dag-cbor\";\nimport {\n type infer as infer_,\n string,\n url,\n array,\n optional,\n nullable,\n strictObject,\n looseObject,\n nonnegative,\n int,\n boolean,\n custom,\n number,\n union,\n} from \"zod/mini\";\n\nexport class Inboxes {\n getAuthorizationEndpoint = getAuthorizationEndpoint;\n protected cache_: Promise<Cache> | null = null;\n protected get cache() {\n if (!this.cache_) {\n this.cache_ = createCache();\n }\n return this.cache_;\n }\n\n async send(inboxUrl: string, message: Message<{}>): Promise<string> {\n verifyHTTPSEndpoint(inboxUrl);\n const url = `${inboxUrl}/send`;\n\n const response = await fetchWithErrorHandling(url, {\n method: \"PUT\",\n headers: {\n \"Content-Type\": \"application/cbor\",\n },\n body: new Uint8Array(dagCborEncode({ m: message })),\n });\n\n const blob = await response.blob();\n const cbor = dagCborDecode(await blob.arrayBuffer());\n const parsed = SendResponseSchema.parse(cbor);\n return parsed.id;\n }\n\n async get(\n inboxUrl: string,\n messageId: string,\n inboxToken?: string | null,\n ): Promise<LabeledMessageBase> {\n const messageCacheKey = getMessageCacheKey(inboxUrl, messageId);\n const cache = await this.cache;\n const cached = await cache.messages.get(messageCacheKey);\n if (cached) return cached;\n\n const url = `${inboxUrl}/message/${messageId}`;\n const response = await fetchWithErrorHandling(url, {\n method: \"GET\",\n headers: {\n ...(inboxToken\n ? {\n Authorization: `Bearer ${inboxToken}`,\n }\n : {}),\n },\n });\n\n const blob = await response.blob();\n const cbor = dagCborDecode(await blob.arrayBuffer());\n const parsed = LabeledMessageBaseSchema.parse(cbor);\n\n await cache.messages.set(messageCacheKey, parsed);\n return parsed;\n }\n\n async label(\n inboxUrl: string,\n messageId: string,\n label: number,\n inboxToken?: string | null,\n ): Promise<void> {\n verifyHTTPSEndpoint(inboxUrl);\n\n if (inboxToken) {\n const url = `${inboxUrl}/label/${messageId}`;\n\n await fetchWithErrorHandling(url, {\n method: \"PUT\",\n headers: {\n \"Content-Type\": \"application/cbor\",\n Authorization: `Bearer ${inboxToken}`,\n },\n body: new Uint8Array(dagCborEncode({ l: label })),\n });\n }\n\n // Update the cache, even if no token.\n // Therefore people not logged in do not need to\n // repeatedly re-validate objects.\n const cache = await this.cache;\n const messageCacheKey = getMessageCacheKey(inboxUrl, messageId);\n const result = await cache.messages.get(messageCacheKey);\n if (result) {\n await cache.messages.set(messageCacheKey, {\n ...result,\n l: label,\n });\n }\n }\n\n protected async fetchMessageBatch(\n inboxUrl: string,\n type: \"query\" | \"export\",\n body: Uint8Array<ArrayBuffer> | undefined,\n inboxToken?: string | null,\n cursor?: string,\n ) {\n const response = await fetchWithErrorHandling(\n `${inboxUrl}/${type}${cursor ? `?cursor=${encodeURIComponent(cursor)}` : \"\"}`,\n {\n method: \"POST\",\n headers: {\n \"Content-Type\": \"application/cbor\",\n ...(inboxToken\n ? {\n Authorization: `Bearer ${inboxToken}`,\n }\n : {}),\n },\n body,\n },\n );\n const retryAfterHeader = response.headers.get(\"Retry-After\");\n const retryAfter = retryAfterHeader\n ? parseInt(retryAfterHeader)\n : undefined;\n\n const waitTil =\n retryAfter && Number.isFinite(retryAfter)\n ? Date.now() + retryAfter * 1000\n : undefined;\n\n return { response, waitTil };\n }\n\n protected async *yieldFromCache(\n cache: Cache,\n inboxUrl: string,\n messageIdsCacheKey: string,\n cachedMessageIds: CacheQueryValue,\n cacheNumSeen: number = 0,\n ): AsyncGenerator<LabeledMessageBase> {\n // Filter out all messageIds before\n // the number already seen\n const messageIds = cachedMessageIds.messageIds.slice(cacheNumSeen);\n\n // Get all the messages pointed to in the cache\n const messages = await Promise.all(\n messageIds.map(async (id) => {\n const message = await cache.messages.get(\n getMessageCacheKey(inboxUrl, id),\n );\n if (!message) {\n // Something is very wrong with the cache,\n // it refers to message IDs that are not cached\n try {\n await cache.messageIds.del(messageIdsCacheKey);\n } catch {}\n throw new Error(\"Cache out of sync - perhaps clear browser storage\");\n }\n return message;\n }),\n );\n\n yield* messages;\n }\n\n protected async *lockedMessageStreamer<Schema extends JSONSchema>(\n ...args: Parameters<typeof this.messageStreamer<Schema>>\n ): MessageStream<Schema> {\n if (typeof window === \"undefined\" || !(await canUseIDB())) {\n // TODO: implement locking in node as well, but not\n // high priority since most use will be in browser\n const streamer = this.messageStreamer<Schema>(...args);\n while (true) {\n const next = await streamer.next();\n if (next.done) return next.value;\n yield next.value;\n }\n }\n\n // Request the lock\n const messageIdsCacheKey = await args[0];\n const lockKey = `graffiti:inbox:${messageIdsCacheKey}`;\n let releaseLock = () => {};\n let hasLock: boolean = false;\n await new Promise<void>((resolvehasLock) => {\n window.navigator.locks.request(\n lockKey,\n {\n mode: \"exclusive\",\n ifAvailable: true,\n },\n async (lock) => {\n // Immediately return whether we\n // acquired the lock or not\n hasLock = !!lock;\n resolvehasLock();\n\n // Then wait for the release to be called\n await new Promise<void>((r) => (releaseLock = r));\n },\n );\n });\n if (hasLock) {\n // If we have the lock, simply proceed with the regular streamer\n try {\n const streamer = this.messageStreamer<Schema>(...args);\n while (true) {\n const next = await streamer.next();\n if (next.done) return next.value;\n yield next.value;\n }\n } finally {\n // Release the lock when all done\n releaseLock();\n }\n }\n\n // Someone else has the lock,\n // so wait until the lock is released,\n // then just return from the cache\n releaseLock();\n await window.navigator.locks.request(lockKey, () => {});\n\n // TODO: the arguments here are brittle\n // at some point, refactor things\n const inboxUrl = args[1];\n const objectSchema = args[5] ?? {};\n const cacheVersion = args[6];\n const cacheNumSeen = args[7];\n\n const cache = await this.cache;\n const cachedMessageIds = await cache.messageIds.get(messageIdsCacheKey);\n if (!cachedMessageIds) {\n throw new Error(\"Cache not found\");\n }\n if (\n cacheVersion !== undefined &&\n cacheVersion !== cachedMessageIds.version\n ) {\n throw new GraffitiErrorCursorExpired(\"Cursor is stale\");\n }\n\n const iterator = this.yieldFromCache(\n cache,\n inboxUrl,\n messageIdsCacheKey,\n cachedMessageIds,\n cacheNumSeen,\n );\n for await (const m of iterator) yield m as LabeledMessage<Schema>;\n\n const outputCursor: infer_<typeof CursorSchema> = {\n numSeen: cachedMessageIds.messageIds.length,\n version: cachedMessageIds.version,\n messageIdsCacheKey,\n objectSchema,\n };\n\n return JSON.stringify(outputCursor);\n }\n\n protected async *messageStreamer<Schema extends JSONSchema>(\n messageIdsCacheKey_: Promise<string>,\n inboxUrl: string,\n type: \"export\" | \"query\",\n body: Uint8Array<ArrayBuffer> | undefined,\n inboxToken?: string | null,\n objectSchema: Schema = {} as Schema,\n cacheVersion?: string,\n cacheNumSeen: number = 0,\n ): MessageStream<Schema> {\n const validator = await compileGraffitiObjectSchema(objectSchema);\n const messageIdsCacheKey = await messageIdsCacheKey_;\n const cache = await this.cache;\n\n let cachedMessageIds = await cache.messageIds.get(messageIdsCacheKey);\n if (\n cacheVersion !== undefined &&\n cacheVersion !== cachedMessageIds?.version\n ) {\n throw new GraffitiErrorCursorExpired(\"Cursor is stale\");\n }\n\n // If we are rate-limited, wait\n let waitTil = cachedMessageIds?.waitTil;\n await waitFor(waitTil);\n\n // See if the cursor is still active by\n // requesting an initial batch of messages\n const cachedCursor = cachedMessageIds?.cursor;\n let firstResponse: Response | undefined = undefined;\n try {\n const out = await this.fetchMessageBatch(\n inboxUrl,\n type,\n body,\n inboxToken,\n cachedCursor,\n );\n firstResponse = out.response;\n waitTil = out.waitTil;\n } catch (e) {\n if (!(e instanceof GraffitiErrorCursorExpired && cachedCursor)) {\n console.error(\n \"Unexpected error in stream, waiting 5 seconds before continuing...\",\n );\n await new Promise((resolve) => setTimeout(resolve, 5000));\n throw e;\n }\n\n // The cursor is stale\n await cache.messageIds.del(messageIdsCacheKey);\n if (cacheVersion === undefined) {\n // The query is not a continuation\n // so we can effectively ignore the error\n cachedMessageIds = undefined;\n } else {\n // Otherwise propogate it up so the\n // consumer can clear their message history\n throw e;\n }\n }\n\n if (firstResponse !== undefined && cachedMessageIds) {\n // Cursor is valid! Yield from the cache\n const iterator = this.yieldFromCache(\n cache,\n inboxUrl,\n messageIdsCacheKey,\n cachedMessageIds,\n cacheNumSeen,\n );\n for await (const m of iterator) yield m as LabeledMessage<Schema>;\n }\n\n if (firstResponse === undefined) {\n // The cursor was stale: try again\n const out = await this.fetchMessageBatch(\n inboxUrl,\n type,\n body,\n inboxToken,\n );\n firstResponse = out.response;\n waitTil = out.waitTil;\n }\n\n // Continue streaming results\n let response = firstResponse;\n let cursor: string;\n const version = cachedMessageIds?.version ?? crypto.randomUUID();\n let messageIds = cachedMessageIds?.messageIds ?? [];\n while (true) {\n const blob = await response.blob();\n const decoded = dagCborDecode(await blob.arrayBuffer());\n const {\n results,\n hasMore,\n cursor: nextCursor,\n } = MessageResultSchema.parse(decoded);\n cursor = nextCursor;\n\n const labeledMessages: LabeledMessage<Schema>[] = results.map(\n (result) => {\n const object =\n result[LABELED_MESSAGE_MESSAGE_KEY][MESSAGE_OBJECT_KEY];\n if (!validator(object)) {\n throw new Error(\"Server returned data that does not match schema\");\n }\n return {\n ...result,\n [LABELED_MESSAGE_MESSAGE_KEY]: {\n ...result[LABELED_MESSAGE_MESSAGE_KEY],\n [MESSAGE_OBJECT_KEY]: object,\n },\n };\n },\n );\n\n // First cache the messages with their labels\n await Promise.all(\n labeledMessages.map((m: LabeledMessageBase) =>\n cache.messages.set(\n getMessageCacheKey(inboxUrl, m[LABELED_MESSAGE_ID_KEY]),\n m,\n ),\n ),\n );\n // Then store all the messageids\n messageIds = [\n ...messageIds,\n ...labeledMessages.map(\n (m: LabeledMessageBase) => m[LABELED_MESSAGE_ID_KEY],\n ),\n ];\n await cache.messageIds.set(messageIdsCacheKey, {\n cursor,\n version,\n messageIds,\n waitTil,\n });\n\n // Update how many we've seen\n cacheNumSeen += labeledMessages.length;\n\n // Return the values\n for (const m of labeledMessages) yield m;\n\n if (!hasMore) break;\n\n // Otherwise get another response (after waiting for rate-limit)\n await waitFor(waitTil);\n const out = await this.fetchMessageBatch(\n inboxUrl,\n type,\n undefined, // Body is never past the first time\n inboxToken,\n cursor,\n );\n response = out.response;\n waitTil = out.waitTil;\n }\n\n const outputCursor: infer_<typeof CursorSchema> = {\n numSeen: cacheNumSeen,\n version,\n messageIdsCacheKey,\n objectSchema,\n };\n\n return JSON.stringify(outputCursor);\n }\n\n query<Schema extends JSONSchema>(\n inboxUrl: string,\n tags: Uint8Array[],\n objectSchema: Schema,\n inboxToken?: string | null,\n ): MessageStream<Schema> {\n verifyHTTPSEndpoint(inboxUrl);\n\n const body = dagCborEncode({\n tags,\n schema: objectSchema,\n });\n\n const messageIdsCacheKey = getMessageIdsCacheKey(inboxUrl, \"query\", body);\n return this.lockedMessageStreamer<Schema>(\n messageIdsCacheKey,\n inboxUrl,\n \"query\",\n new Uint8Array(body),\n inboxToken,\n objectSchema,\n );\n }\n\n continueQuery(\n inboxUrl: string,\n cursor: string,\n inboxToken?: string | null,\n ): MessageStream<{}> {\n verifyHTTPSEndpoint(inboxUrl);\n\n const decodedCursor = JSON.parse(cursor);\n const { messageIdsCacheKey, numSeen, objectSchema, version } =\n CursorSchema.parse(decodedCursor);\n\n return this.lockedMessageStreamer<{}>(\n Promise.resolve(messageIdsCacheKey),\n inboxUrl,\n \"query\",\n undefined,\n inboxToken,\n objectSchema,\n version,\n numSeen,\n );\n }\n\n export(inboxUrl: string, inboxToken: string): MessageStream<{}> {\n verifyHTTPSEndpoint(inboxUrl);\n const messageIdsCacheKey = getMessageIdsCacheKey(inboxUrl, \"export\");\n return this.lockedMessageStreamer<{}>(\n messageIdsCacheKey,\n inboxUrl,\n \"export\",\n undefined,\n inboxToken,\n );\n }\n}\n\nconst GraffitiObjectSchema = strictObject({\n value: looseObject({}),\n channels: array(string()),\n allowed: optional(nullable(array(url()))),\n url: url(),\n actor: url(),\n});\nexport const Uint8ArraySchema = custom<Uint8Array>(\n (v): v is Uint8Array => v instanceof Uint8Array,\n);\nexport const TagsSchema = array(Uint8ArraySchema);\n\nexport const MESSAGE_TAGS_KEY = \"t\";\nexport const MESSAGE_OBJECT_KEY = \"o\";\nexport const MESSAGE_METADATA_KEY = \"m\";\nexport const MessageBaseSchema = strictObject({\n [MESSAGE_TAGS_KEY]: TagsSchema,\n [MESSAGE_OBJECT_KEY]: GraffitiObjectSchema,\n [MESSAGE_METADATA_KEY]: Uint8ArraySchema,\n});\ntype MessageBase = infer_<typeof MessageBaseSchema>;\n\nexport const LABELED_MESSAGE_ID_KEY = \"id\";\nexport const LABELED_MESSAGE_MESSAGE_KEY = \"m\";\nexport const LABELED_MESSAGE_LABEL_KEY = \"l\";\nexport const LabeledMessageBaseSchema = strictObject({\n [LABELED_MESSAGE_ID_KEY]: string(),\n [LABELED_MESSAGE_MESSAGE_KEY]: MessageBaseSchema,\n [LABELED_MESSAGE_LABEL_KEY]: number(),\n});\ntype LabeledMessageBase = infer_<typeof LabeledMessageBaseSchema>;\n\nexport type Message<Schema extends JSONSchema> = MessageBase & {\n [MESSAGE_OBJECT_KEY]: GraffitiObject<Schema>;\n};\nexport type LabeledMessage<Schema extends JSONSchema> = LabeledMessageBase & {\n [LABELED_MESSAGE_MESSAGE_KEY]: {\n [MESSAGE_OBJECT_KEY]: GraffitiObject<Schema>;\n };\n};\n\nconst SendResponseSchema = strictObject({ id: string() });\n\nconst MessageResultSchema = strictObject({\n results: array(LabeledMessageBaseSchema),\n hasMore: boolean(),\n cursor: string(),\n});\n\nconst CursorSchema = strictObject({\n messageIdsCacheKey: string(),\n version: string(),\n numSeen: int().check(nonnegative()),\n objectSchema: union([looseObject({}), boolean()]),\n});\n\nexport interface MessageStream<\n Schema extends JSONSchema,\n> extends AsyncGenerator<LabeledMessage<Schema>, string> {}\n\ntype CacheQueryValue = {\n cursor: string;\n version: string;\n messageIds: string[];\n waitTil?: number;\n};\n\nasync function canUseIDB(): Promise<boolean> {\n try {\n if (!globalThis.indexedDB) return false;\n\n // Small probe database\n await new Promise<void>((resolve, reject) => {\n const req = indexedDB.open(\"__idb_probe__\", 1);\n req.onupgradeneeded = () => req.result.createObjectStore(\"k\");\n req.onsuccess = () => {\n req.result.close();\n resolve();\n };\n req.onerror = () => reject(req.error);\n });\n\n return true;\n } catch {\n return false;\n }\n}\n\ntype Cache = {\n messages: {\n get(k: string): Promise<LabeledMessageBase | undefined>;\n set(k: string, value: LabeledMessageBase): Promise<void>;\n del(k: string): Promise<void>;\n };\n messageIds: {\n get(k: string): Promise<CacheQueryValue | undefined>;\n set(k: string, value: CacheQueryValue): Promise<void>;\n del(k: string): Promise<void>;\n };\n};\n\nfunction getMessageCacheKey(inboxUrl: string, messageId: string) {\n return `${encodeURIComponent(inboxUrl)}:${encodeURIComponent(messageId)}`;\n}\nasync function getMessageIdsCacheKey(\n inboxUrl: string,\n type: \"query\" | \"export\",\n body?: Uint8Array,\n): Promise<string> {\n const cacheIdData = dagCborEncode({\n inboxUrl,\n type,\n body: body ?? null,\n });\n return crypto.subtle\n .digest(\"SHA-256\", new Uint8Array(cacheIdData))\n .then((bytes) =>\n Array.from(new Uint8Array(bytes))\n .map((byte) => byte.toString(16).padStart(2, \"0\"))\n .join(\"\"),\n );\n}\n\nasync function resetCacheDB() {\n await new Promise<void>((resolve) => {\n const req = indexedDB.deleteDatabase(\"graffiti-inbox-cache\");\n req.onsuccess = () => resolve();\n req.onerror = () => resolve(); // best effort\n req.onblocked = () => resolve(); // best effort\n });\n}\n\nasync function createCache(): Promise<Cache> {\n if (await canUseIDB()) {\n const { openDB } = await import(\"idb\");\n const db = await openDB(\"graffiti-inbox-cache\", 1, {\n upgrade(db) {\n if (!db.objectStoreNames.contains(\"m\")) db.createObjectStore(\"m\");\n if (!db.objectStoreNames.contains(\"q\")) db.createObjectStore(\"q\");\n },\n });\n\n return {\n messages: {\n get: async (k) => {\n try {\n return db.get(\"m\", k);\n } catch (error) {\n console.error(\"Error getting message from cache:\", error);\n console.error(\"resetting cache...\");\n await resetCacheDB();\n return undefined;\n }\n },\n set: async (k, v) => {\n await db.put(\"m\", v, k);\n },\n del: (k) => db.delete(\"m\", k),\n },\n messageIds: {\n get: async (k) => {\n try {\n return await db.get(\"q\", k);\n } catch (error) {\n console.error(\"Error getting message IDs from cache:\", error);\n console.error(\"resetting cache...\");\n await resetCacheDB();\n return undefined;\n }\n },\n set: async (k, v) => {\n await db.put(\"q\", v, k);\n },\n del: (k) => db.delete(\"q\", k),\n },\n };\n }\n\n const m = new Map<string, LabeledMessageBase>();\n const q = new Map<string, CacheQueryValue>();\n\n return {\n messages: {\n get: async (k) => m.get(k),\n set: async (k, v) => void m.set(k, v),\n del: async (k) => void m.delete(k),\n },\n messageIds: {\n get: async (k) => q.get(k),\n set: async (k, v) => void q.set(k, v),\n del: async (k) => void q.delete(k),\n },\n };\n}\n\nasync function waitFor(waitTil?: number) {\n if (waitTil !== undefined) {\n const waitFor = waitTil - Date.now();\n if (waitFor > 0) {\n await new Promise((resolve) => setTimeout(resolve, waitFor));\n }\n }\n}\n"],
|
|
5
|
+
"mappings": "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",
|
|
6
|
+
"names": ["url", "dagCborEncode", "dagCborDecode", "db", "waitFor"]
|
|
7
|
+
}
|
|
@@ -0,0 +1,75 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
var __defProp = Object.defineProperty;
|
|
3
|
+
var __getOwnPropDesc = Object.getOwnPropertyDescriptor;
|
|
4
|
+
var __getOwnPropNames = Object.getOwnPropertyNames;
|
|
5
|
+
var __hasOwnProp = Object.prototype.hasOwnProperty;
|
|
6
|
+
var __export = (target, all) => {
|
|
7
|
+
for (var name in all)
|
|
8
|
+
__defProp(target, name, { get: all[name], enumerable: true });
|
|
9
|
+
};
|
|
10
|
+
var __copyProps = (to, from, except, desc) => {
|
|
11
|
+
if (from && typeof from === "object" || typeof from === "function") {
|
|
12
|
+
for (let key of __getOwnPropNames(from))
|
|
13
|
+
if (!__hasOwnProp.call(to, key) && key !== except)
|
|
14
|
+
__defProp(to, key, { get: () => from[key], enumerable: !(desc = __getOwnPropDesc(from, key)) || desc.enumerable });
|
|
15
|
+
}
|
|
16
|
+
return to;
|
|
17
|
+
};
|
|
18
|
+
var __toCommonJS = (mod) => __copyProps(__defProp({}, "__esModule", { value: true }), mod);
|
|
19
|
+
var utilities_exports = {};
|
|
20
|
+
__export(utilities_exports, {
|
|
21
|
+
GraffitiErrorUnauthorized: () => GraffitiErrorUnauthorized,
|
|
22
|
+
fetchWithErrorHandling: () => fetchWithErrorHandling,
|
|
23
|
+
getAuthorizationEndpoint: () => getAuthorizationEndpoint,
|
|
24
|
+
verifyHTTPSEndpoint: () => verifyHTTPSEndpoint
|
|
25
|
+
});
|
|
26
|
+
module.exports = __toCommonJS(utilities_exports);
|
|
27
|
+
var import_api = require("@graffiti-garden/api");
|
|
28
|
+
const SERVICE_ENDPOINT_PREFIX_HTTPS = "https://";
|
|
29
|
+
function verifyHTTPSEndpoint(endpoint) {
|
|
30
|
+
if (!endpoint.startsWith(SERVICE_ENDPOINT_PREFIX_HTTPS)) {
|
|
31
|
+
throw new Error("Unrecognized storage bucket endpoint type");
|
|
32
|
+
}
|
|
33
|
+
}
|
|
34
|
+
async function getAuthorizationEndpoint(serviceEndpoint) {
|
|
35
|
+
verifyHTTPSEndpoint(serviceEndpoint);
|
|
36
|
+
const authUrl = `${serviceEndpoint}/auth`;
|
|
37
|
+
const response = await fetch(authUrl);
|
|
38
|
+
if (!response.ok) {
|
|
39
|
+
throw new Error("Failed to get storage bucket authorization endpoint");
|
|
40
|
+
}
|
|
41
|
+
return await response.text();
|
|
42
|
+
}
|
|
43
|
+
class GraffitiErrorUnauthorized extends Error {
|
|
44
|
+
constructor(message) {
|
|
45
|
+
super(message);
|
|
46
|
+
this.name = "GraffitiErrorUnauthorized";
|
|
47
|
+
Object.setPrototypeOf(this, GraffitiErrorUnauthorized.prototype);
|
|
48
|
+
}
|
|
49
|
+
}
|
|
50
|
+
async function fetchWithErrorHandling(...args) {
|
|
51
|
+
const response = await fetch(...args);
|
|
52
|
+
if (!response.ok) {
|
|
53
|
+
let errorText;
|
|
54
|
+
try {
|
|
55
|
+
errorText = await response.text();
|
|
56
|
+
} catch {
|
|
57
|
+
errorText = response.statusText;
|
|
58
|
+
}
|
|
59
|
+
if (response.status === 401) {
|
|
60
|
+
throw new GraffitiErrorUnauthorized(errorText);
|
|
61
|
+
} else if (response.status === 403) {
|
|
62
|
+
throw new import_api.GraffitiErrorForbidden(errorText);
|
|
63
|
+
} else if (response.status === 404) {
|
|
64
|
+
throw new import_api.GraffitiErrorNotFound(errorText);
|
|
65
|
+
} else if (response.status === 410) {
|
|
66
|
+
throw new import_api.GraffitiErrorCursorExpired(errorText);
|
|
67
|
+
} else if (response.status === 413) {
|
|
68
|
+
throw new import_api.GraffitiErrorTooLarge(errorText);
|
|
69
|
+
} else {
|
|
70
|
+
throw new Error(errorText);
|
|
71
|
+
}
|
|
72
|
+
}
|
|
73
|
+
return response;
|
|
74
|
+
}
|
|
75
|
+
//# sourceMappingURL=utilities.js.map
|