@hatk/hatk 0.0.1-alpha.5 → 0.0.1-alpha.51
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/dist/adapter.d.ts +19 -0
- package/dist/adapter.d.ts.map +1 -0
- package/dist/adapter.js +107 -0
- package/dist/backfill.d.ts +60 -1
- package/dist/backfill.d.ts.map +1 -1
- package/dist/backfill.js +167 -33
- package/dist/car.d.ts +59 -1
- package/dist/car.d.ts.map +1 -1
- package/dist/car.js +179 -7
- package/dist/cbor.d.ts +37 -0
- package/dist/cbor.d.ts.map +1 -1
- package/dist/cbor.js +36 -3
- package/dist/cid.d.ts +37 -0
- package/dist/cid.d.ts.map +1 -1
- package/dist/cid.js +38 -3
- package/dist/cli.js +243 -996
- package/dist/config.d.ts +24 -1
- package/dist/config.d.ts.map +1 -1
- package/dist/config.js +37 -9
- package/dist/database/adapter-factory.d.ts +6 -0
- package/dist/database/adapter-factory.d.ts.map +1 -0
- package/dist/database/adapter-factory.js +20 -0
- package/dist/database/adapters/duckdb-search.d.ts +12 -0
- package/dist/database/adapters/duckdb-search.d.ts.map +1 -0
- package/dist/database/adapters/duckdb-search.js +27 -0
- package/dist/database/adapters/duckdb.d.ts +25 -0
- package/dist/database/adapters/duckdb.d.ts.map +1 -0
- package/dist/database/adapters/duckdb.js +161 -0
- package/dist/database/adapters/sqlite-search.d.ts +23 -0
- package/dist/database/adapters/sqlite-search.d.ts.map +1 -0
- package/dist/database/adapters/sqlite-search.js +74 -0
- package/dist/database/adapters/sqlite.d.ts +18 -0
- package/dist/database/adapters/sqlite.d.ts.map +1 -0
- package/dist/database/adapters/sqlite.js +88 -0
- package/dist/{db.d.ts → database/db.d.ts} +56 -6
- package/dist/database/db.d.ts.map +1 -0
- package/dist/{db.js → database/db.js} +727 -549
- package/dist/database/dialect.d.ts +45 -0
- package/dist/database/dialect.d.ts.map +1 -0
- package/dist/database/dialect.js +72 -0
- package/dist/{fts.d.ts → database/fts.d.ts} +7 -0
- package/dist/database/fts.d.ts.map +1 -0
- package/dist/{fts.js → database/fts.js} +116 -32
- package/dist/database/index.d.ts +7 -0
- package/dist/database/index.d.ts.map +1 -0
- package/dist/database/index.js +6 -0
- package/dist/database/ports.d.ts +50 -0
- package/dist/database/ports.d.ts.map +1 -0
- package/dist/database/ports.js +1 -0
- package/dist/{schema.d.ts → database/schema.d.ts} +14 -3
- package/dist/database/schema.d.ts.map +1 -0
- package/dist/{schema.js → database/schema.js} +81 -41
- package/dist/dev-entry.d.ts +8 -0
- package/dist/dev-entry.d.ts.map +1 -0
- package/dist/dev-entry.js +111 -0
- package/dist/feeds.d.ts +12 -8
- package/dist/feeds.d.ts.map +1 -1
- package/dist/feeds.js +45 -6
- package/dist/hooks.d.ts +85 -0
- package/dist/hooks.d.ts.map +1 -0
- package/dist/hooks.js +161 -0
- package/dist/hydrate.d.ts +6 -5
- package/dist/hydrate.d.ts.map +1 -1
- package/dist/hydrate.js +4 -16
- package/dist/indexer.d.ts +22 -0
- package/dist/indexer.d.ts.map +1 -1
- package/dist/indexer.js +96 -8
- package/dist/labels.d.ts +36 -0
- package/dist/labels.d.ts.map +1 -1
- package/dist/labels.js +71 -6
- package/dist/lexicon-resolve.d.ts.map +1 -1
- package/dist/lexicon-resolve.js +27 -112
- package/dist/lexicons/com/atproto/label/defs.json +75 -0
- package/dist/lexicons/com/atproto/moderation/defs.json +30 -0
- package/dist/lexicons/com/atproto/repo/strongRef.json +24 -0
- package/dist/lexicons/dev/hatk/createRecord.json +40 -0
- package/dist/lexicons/dev/hatk/createReport.json +48 -0
- package/dist/lexicons/dev/hatk/deleteRecord.json +25 -0
- package/dist/lexicons/dev/hatk/describeCollections.json +41 -0
- package/dist/lexicons/dev/hatk/describeFeeds.json +29 -0
- package/dist/lexicons/dev/hatk/describeLabels.json +45 -0
- package/dist/lexicons/dev/hatk/getFeed.json +30 -0
- package/dist/lexicons/dev/hatk/getPreferences.json +19 -0
- package/dist/lexicons/dev/hatk/getRecord.json +26 -0
- package/dist/lexicons/dev/hatk/getRecords.json +32 -0
- package/dist/lexicons/dev/hatk/putPreference.json +28 -0
- package/dist/lexicons/dev/hatk/putRecord.json +41 -0
- package/dist/lexicons/dev/hatk/searchRecords.json +32 -0
- package/dist/lexicons/dev/hatk/uploadBlob.json +23 -0
- package/dist/logger.d.ts +29 -0
- package/dist/logger.d.ts.map +1 -1
- package/dist/logger.js +29 -0
- package/dist/main.js +136 -67
- package/dist/mst.d.ts +18 -1
- package/dist/mst.d.ts.map +1 -1
- package/dist/mst.js +19 -8
- package/dist/oauth/db.d.ts +3 -1
- package/dist/oauth/db.d.ts.map +1 -1
- package/dist/oauth/db.js +48 -19
- package/dist/oauth/server.d.ts +24 -0
- package/dist/oauth/server.d.ts.map +1 -1
- package/dist/oauth/server.js +198 -22
- package/dist/oauth/session.d.ts +11 -0
- package/dist/oauth/session.d.ts.map +1 -0
- package/dist/oauth/session.js +65 -0
- package/dist/opengraph.d.ts +10 -0
- package/dist/opengraph.d.ts.map +1 -1
- package/dist/opengraph.js +73 -39
- package/dist/pds-proxy.d.ts +42 -0
- package/dist/pds-proxy.d.ts.map +1 -0
- package/dist/pds-proxy.js +207 -0
- package/dist/push.d.ts +33 -0
- package/dist/push.d.ts.map +1 -0
- package/dist/push.js +166 -0
- package/dist/renderer.d.ts +27 -0
- package/dist/renderer.d.ts.map +1 -0
- package/dist/renderer.js +46 -0
- package/dist/resolve-hatk.d.ts +6 -0
- package/dist/resolve-hatk.d.ts.map +1 -0
- package/dist/resolve-hatk.js +20 -0
- package/dist/response.d.ts +16 -0
- package/dist/response.d.ts.map +1 -0
- package/dist/response.js +69 -0
- package/dist/scanner.d.ts +21 -0
- package/dist/scanner.d.ts.map +1 -0
- package/dist/scanner.js +88 -0
- package/dist/seed.d.ts +19 -0
- package/dist/seed.d.ts.map +1 -1
- package/dist/seed.js +43 -4
- package/dist/server-init.d.ts +8 -0
- package/dist/server-init.d.ts.map +1 -0
- package/dist/server-init.js +62 -0
- package/dist/server.d.ts +26 -3
- package/dist/server.d.ts.map +1 -1
- package/dist/server.js +624 -635
- package/dist/setup.d.ts +28 -1
- package/dist/setup.d.ts.map +1 -1
- package/dist/setup.js +50 -3
- package/dist/templates/feed.tpl +14 -0
- package/dist/templates/hook.tpl +5 -0
- package/dist/templates/label.tpl +15 -0
- package/dist/templates/og.tpl +17 -0
- package/dist/templates/seed.tpl +11 -0
- package/dist/templates/setup.tpl +5 -0
- package/dist/templates/test-feed.tpl +19 -0
- package/dist/templates/test-xrpc.tpl +19 -0
- package/dist/templates/xrpc.tpl +41 -0
- package/dist/test.d.ts +1 -1
- package/dist/test.d.ts.map +1 -1
- package/dist/test.js +38 -32
- package/dist/views.js +1 -1
- package/dist/vite-plugin.d.ts +1 -1
- package/dist/vite-plugin.d.ts.map +1 -1
- package/dist/vite-plugin.js +254 -66
- package/dist/xrpc.d.ts +60 -10
- package/dist/xrpc.d.ts.map +1 -1
- package/dist/xrpc.js +155 -39
- package/package.json +15 -7
- package/public/admin.html +133 -54
- package/dist/db.d.ts.map +0 -1
- package/dist/fts.d.ts.map +0 -1
- package/dist/oauth/hooks.d.ts +0 -10
- package/dist/oauth/hooks.d.ts.map +0 -1
- package/dist/oauth/hooks.js +0 -40
- package/dist/schema.d.ts.map +0 -1
- package/dist/test-browser.d.ts +0 -14
- package/dist/test-browser.d.ts.map +0 -1
- package/dist/test-browser.js +0 -26
package/dist/opengraph.d.ts
CHANGED
|
@@ -28,7 +28,17 @@ export interface OpengraphResult {
|
|
|
28
28
|
description?: string;
|
|
29
29
|
};
|
|
30
30
|
}
|
|
31
|
+
export declare function defineOG(path: string, generate: (ctx: OpengraphContext) => Promise<OpengraphResult>): {
|
|
32
|
+
__type: "og";
|
|
33
|
+
path: string;
|
|
34
|
+
generate: (ctx: OpengraphContext) => Promise<OpengraphResult>;
|
|
35
|
+
};
|
|
31
36
|
export declare function initOpengraph(ogDir: string): Promise<void>;
|
|
37
|
+
/** Register a single OG handler from a scanned server/ module. */
|
|
38
|
+
export declare function registerOgHandler(ogMod: {
|
|
39
|
+
path: string;
|
|
40
|
+
generate: (ctx: OpengraphContext) => Promise<OpengraphResult>;
|
|
41
|
+
}): void;
|
|
32
42
|
export declare function handleOpengraphRequest(pathname: string): Promise<Buffer | null>;
|
|
33
43
|
export declare function buildOgMeta(pathname: string, origin: string): string | null;
|
|
34
44
|
//# sourceMappingURL=opengraph.d.ts.map
|
package/dist/opengraph.d.ts.map
CHANGED
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"opengraph.d.ts","sourceRoot":"","sources":["../src/opengraph.ts"],"names":[],"mappings":"
|
|
1
|
+
{"version":3,"file":"opengraph.d.ts","sourceRoot":"","sources":["../src/opengraph.ts"],"names":[],"mappings":"AAiBA,OAAO,KAAK,EAAE,WAAW,EAAE,MAAM,WAAW,CAAA;AAE5C,4CAA4C;AAC5C,MAAM,WAAW,UAAU;IACzB,IAAI,EAAE,MAAM,CAAA;IACZ,KAAK,EAAE;QACL,KAAK,CAAC,EAAE,MAAM,CAAC,MAAM,EAAE,GAAG,CAAC,CAAA;QAC3B,QAAQ,CAAC,EAAE,CAAC,UAAU,GAAG,MAAM,CAAC,EAAE,GAAG,MAAM,CAAA;QAC3C,GAAG,CAAC,EAAE,MAAM,CAAA;QACZ,KAAK,CAAC,EAAE,MAAM,CAAA;QACd,MAAM,CAAC,EAAE,MAAM,CAAA;QACf,CAAC,GAAG,EAAE,MAAM,GAAG,GAAG,CAAA;KACnB,CAAA;CACF;AAED,uDAAuD;AACvD,MAAM,WAAW,gBAAiB,SAAQ,WAAW;IACnD,UAAU,EAAE,CAAC,GAAG,EAAE,MAAM,KAAK,OAAO,CAAC,MAAM,GAAG,IAAI,CAAC,CAAA;CACpD;AAED,qDAAqD;AACrD,MAAM,WAAW,eAAe;IAC9B,OAAO,EAAE,UAAU,CAAA;IACnB,OAAO,CAAC,EAAE;QAAE,KAAK,CAAC,EAAE,MAAM,CAAC;QAAC,MAAM,CAAC,EAAE,MAAM,CAAC;QAAC,KAAK,CAAC,EAAE,GAAG,EAAE,CAAA;KAAE,CAAA;IAC5D,IAAI,CAAC,EAAE;QAAE,KAAK,CAAC,EAAE,MAAM,CAAC;QAAC,WAAW,CAAC,EAAE,MAAM,CAAA;KAAE,CAAA;CAChD;AAED,wBAAgB,QAAQ,CAAC,IAAI,EAAE,MAAM,EAAE,QAAQ,EAAE,CAAC,GAAG,EAAE,gBAAgB,KAAK,OAAO,CAAC,eAAe,CAAC;;;oBAA7C,gBAAgB,KAAK,OAAO,CAAC,eAAe,CAAC;EAEnG;AAkCD,wBAAsB,aAAa,CAAC,KAAK,EAAE,MAAM,GAAG,OAAO,CAAC,IAAI,CAAC,CAuEhE;AAED,kEAAkE;AAClE,wBAAgB,iBAAiB,CAAC,KAAK,EAAE;IACvC,IAAI,EAAE,MAAM,CAAA;IACZ,QAAQ,EAAE,CAAC,GAAG,EAAE,gBAAgB,KAAK,OAAO,CAAC,eAAe,CAAC,CAAA;CAC9D,GAAG,IAAI,CAmDP;AAED,wBAAsB,sBAAsB,CAAC,QAAQ,EAAE,MAAM,GAAG,OAAO,CAAC,MAAM,GAAG,IAAI,CAAC,CA+BrF;AAED,wBAAgB,WAAW,CAAC,QAAQ,EAAE,MAAM,EAAE,MAAM,EAAE,MAAM,GAAG,MAAM,GAAG,IAAI,CAyC3E"}
|
package/dist/opengraph.js
CHANGED
|
@@ -9,11 +9,23 @@ var __rewriteRelativeImportExtension = (this && this.__rewriteRelativeImportExte
|
|
|
9
9
|
import { resolve } from 'node:path';
|
|
10
10
|
import { readFileSync, readdirSync } from 'node:fs';
|
|
11
11
|
import { log } from "./logger.js";
|
|
12
|
-
|
|
13
|
-
|
|
14
|
-
|
|
15
|
-
|
|
16
|
-
|
|
12
|
+
// Lazy-imported to avoid CJS require() issues in Vite's module runner
|
|
13
|
+
let _satori = null;
|
|
14
|
+
let _Resvg = null;
|
|
15
|
+
async function getSatori() {
|
|
16
|
+
if (!_satori)
|
|
17
|
+
_satori = (await import('satori')).default;
|
|
18
|
+
return _satori;
|
|
19
|
+
}
|
|
20
|
+
async function getResvg() {
|
|
21
|
+
if (!_Resvg)
|
|
22
|
+
_Resvg = (await import('@resvg/resvg-js')).Resvg;
|
|
23
|
+
return _Resvg;
|
|
24
|
+
}
|
|
25
|
+
import { buildXrpcContext } from "./xrpc.js";
|
|
26
|
+
export function defineOG(path, generate) {
|
|
27
|
+
return { __type: 'og', path, generate };
|
|
28
|
+
}
|
|
17
29
|
const handlers = [];
|
|
18
30
|
const pageRoutes = [];
|
|
19
31
|
let defaultFont = null;
|
|
@@ -51,7 +63,7 @@ export async function initOpengraph(ogDir) {
|
|
|
51
63
|
for (const file of files) {
|
|
52
64
|
const name = file.replace(/\.(ts|js)$/, '');
|
|
53
65
|
const scriptPath = resolve(ogDir, file);
|
|
54
|
-
const mod = await import(__rewriteRelativeImportExtension(scriptPath));
|
|
66
|
+
const mod = await import(__rewriteRelativeImportExtension(/* @vite-ignore */ `${scriptPath}?t=${Date.now()}`));
|
|
55
67
|
const handler = mod.default;
|
|
56
68
|
if (!handler.path) {
|
|
57
69
|
console.warn(`[opengraph] ${file} missing 'path' export, skipping`);
|
|
@@ -64,38 +76,7 @@ export async function initOpengraph(ogDir) {
|
|
|
64
76
|
pattern,
|
|
65
77
|
paramNames,
|
|
66
78
|
execute: async (params) => {
|
|
67
|
-
const ctx =
|
|
68
|
-
db: { query: querySQL, run: runSQL },
|
|
69
|
-
params,
|
|
70
|
-
input: {},
|
|
71
|
-
limit: 1,
|
|
72
|
-
viewer: null,
|
|
73
|
-
packCursor,
|
|
74
|
-
unpackCursor,
|
|
75
|
-
isTakendown: isTakendownDid,
|
|
76
|
-
filterTakendownDids,
|
|
77
|
-
search: searchRecords,
|
|
78
|
-
resolve: resolveRecords,
|
|
79
|
-
lookup: async (collection, field, values) => {
|
|
80
|
-
if (values.length === 0)
|
|
81
|
-
return new Map();
|
|
82
|
-
const unique = [...new Set(values.filter(Boolean))];
|
|
83
|
-
return lookupByFieldBatch(collection, field, unique);
|
|
84
|
-
},
|
|
85
|
-
count: async (collection, field, values) => {
|
|
86
|
-
if (values.length === 0)
|
|
87
|
-
return new Map();
|
|
88
|
-
const unique = [...new Set(values.filter(Boolean))];
|
|
89
|
-
return countByFieldBatch(collection, field, unique);
|
|
90
|
-
},
|
|
91
|
-
exists: async (collection, filters) => {
|
|
92
|
-
const conditions = Object.entries(filters).map(([field, value]) => ({ field, value }));
|
|
93
|
-
const uri = await findUriByFields(collection, conditions);
|
|
94
|
-
return uri !== null;
|
|
95
|
-
},
|
|
96
|
-
labels: queryLabelsForUris,
|
|
97
|
-
blobUrl,
|
|
98
|
-
};
|
|
79
|
+
const ctx = buildXrpcContext(params, undefined, 1, null);
|
|
99
80
|
ctx.fetchImage = async (url) => {
|
|
100
81
|
try {
|
|
101
82
|
const resp = await fetch(url, { redirect: 'follow' });
|
|
@@ -117,7 +98,7 @@ export async function initOpengraph(ogDir) {
|
|
|
117
98
|
...result.options,
|
|
118
99
|
fonts: [...(defaultFont ? [defaultFont] : []), ...(result.options?.fonts || [])],
|
|
119
100
|
};
|
|
120
|
-
const svg = await
|
|
101
|
+
const svg = await (await getSatori())(element, options);
|
|
121
102
|
return { svg, meta: result.meta };
|
|
122
103
|
},
|
|
123
104
|
});
|
|
@@ -129,6 +110,58 @@ export async function initOpengraph(ogDir) {
|
|
|
129
110
|
}
|
|
130
111
|
}
|
|
131
112
|
}
|
|
113
|
+
/** Register a single OG handler from a scanned server/ module. */
|
|
114
|
+
export function registerOgHandler(ogMod) {
|
|
115
|
+
const { pattern, paramNames } = compilePath(ogMod.path);
|
|
116
|
+
const name = ogMod.path.replace(/^\//, '').replace(/\//g, '-').replace(/:/g, '');
|
|
117
|
+
// Load default font if not already loaded
|
|
118
|
+
if (!defaultFont) {
|
|
119
|
+
try {
|
|
120
|
+
const fontPath = resolve(import.meta.dirname, '..', 'fonts', 'Inter-Regular.woff');
|
|
121
|
+
const fontData = readFileSync(fontPath);
|
|
122
|
+
defaultFont = { name: 'Inter', data: fontData.buffer, weight: 400, style: 'normal' };
|
|
123
|
+
}
|
|
124
|
+
catch { }
|
|
125
|
+
}
|
|
126
|
+
handlers.push({
|
|
127
|
+
name,
|
|
128
|
+
path: ogMod.path,
|
|
129
|
+
pattern,
|
|
130
|
+
paramNames,
|
|
131
|
+
execute: async (params) => {
|
|
132
|
+
const ctx = buildXrpcContext(params, undefined, 1, null);
|
|
133
|
+
ctx.fetchImage = async (url) => {
|
|
134
|
+
try {
|
|
135
|
+
const resp = await fetch(url, { redirect: 'follow' });
|
|
136
|
+
if (!resp.ok)
|
|
137
|
+
return null;
|
|
138
|
+
const buf = Buffer.from(await resp.arrayBuffer());
|
|
139
|
+
const contentType = resp.headers.get('content-type') || 'image/jpeg';
|
|
140
|
+
return `data:${contentType};base64,${buf.toString('base64')}`;
|
|
141
|
+
}
|
|
142
|
+
catch {
|
|
143
|
+
return null;
|
|
144
|
+
}
|
|
145
|
+
};
|
|
146
|
+
const result = await ogMod.generate(ctx);
|
|
147
|
+
const element = result.element;
|
|
148
|
+
const options = {
|
|
149
|
+
width: 1200,
|
|
150
|
+
height: 630,
|
|
151
|
+
...result.options,
|
|
152
|
+
fonts: [...(defaultFont ? [defaultFont] : []), ...(result.options?.fonts || [])],
|
|
153
|
+
};
|
|
154
|
+
const svg = await (await getSatori())(element, options);
|
|
155
|
+
return { svg, meta: result.meta };
|
|
156
|
+
},
|
|
157
|
+
});
|
|
158
|
+
const pagePath = ogMod.path.replace(/^\/og/, '');
|
|
159
|
+
if (pagePath !== ogMod.path) {
|
|
160
|
+
const compiled = compilePath(pagePath);
|
|
161
|
+
pageRoutes.push({ ogPath: ogMod.path, pattern: compiled.pattern, paramNames: compiled.paramNames, name });
|
|
162
|
+
}
|
|
163
|
+
log(`[opengraph] registered: ${name} → ${ogMod.path}`);
|
|
164
|
+
}
|
|
132
165
|
export async function handleOpengraphRequest(pathname) {
|
|
133
166
|
const cached = cache.get(pathname);
|
|
134
167
|
if (cached && cached.expires > Date.now())
|
|
@@ -143,6 +176,7 @@ export async function handleOpengraphRequest(pathname) {
|
|
|
143
176
|
});
|
|
144
177
|
try {
|
|
145
178
|
const { svg, meta } = await handler.execute(params);
|
|
179
|
+
const Resvg = await getResvg();
|
|
146
180
|
const png = new Resvg(svg, { fitTo: { mode: 'width', value: 1200 } }).render().asPng();
|
|
147
181
|
if (cache.size >= CACHE_MAX) {
|
|
148
182
|
const oldest = cache.keys().next().value;
|
|
@@ -0,0 +1,42 @@
|
|
|
1
|
+
import type { OAuthConfig } from './config.ts';
|
|
2
|
+
export declare class ProxyError extends Error {
|
|
3
|
+
status: number;
|
|
4
|
+
constructor(status: number, message: string);
|
|
5
|
+
}
|
|
6
|
+
export declare class ScopeMissingProxyError extends ProxyError {
|
|
7
|
+
constructor();
|
|
8
|
+
}
|
|
9
|
+
export declare function pdsCreateRecord(oauthConfig: OAuthConfig, viewer: {
|
|
10
|
+
did: string;
|
|
11
|
+
}, input: {
|
|
12
|
+
collection: string;
|
|
13
|
+
repo?: string;
|
|
14
|
+
rkey?: string;
|
|
15
|
+
record: Record<string, unknown>;
|
|
16
|
+
}): Promise<{
|
|
17
|
+
uri?: string;
|
|
18
|
+
cid?: string;
|
|
19
|
+
}>;
|
|
20
|
+
export declare function pdsDeleteRecord(oauthConfig: OAuthConfig, viewer: {
|
|
21
|
+
did: string;
|
|
22
|
+
}, input: {
|
|
23
|
+
collection: string;
|
|
24
|
+
rkey: string;
|
|
25
|
+
}): Promise<Record<string, unknown>>;
|
|
26
|
+
export declare function pdsPutRecord(oauthConfig: OAuthConfig, viewer: {
|
|
27
|
+
did: string;
|
|
28
|
+
}, input: {
|
|
29
|
+
collection: string;
|
|
30
|
+
rkey: string;
|
|
31
|
+
record: Record<string, unknown>;
|
|
32
|
+
repo?: string;
|
|
33
|
+
}): Promise<{
|
|
34
|
+
uri?: string;
|
|
35
|
+
cid?: string;
|
|
36
|
+
}>;
|
|
37
|
+
export declare function pdsUploadBlob(oauthConfig: OAuthConfig, viewer: {
|
|
38
|
+
did: string;
|
|
39
|
+
}, body: Uint8Array, contentType: string): Promise<{
|
|
40
|
+
blob: unknown;
|
|
41
|
+
}>;
|
|
42
|
+
//# sourceMappingURL=pds-proxy.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"pds-proxy.d.ts","sourceRoot":"","sources":["../src/pds-proxy.ts"],"names":[],"mappings":"AAEA,OAAO,KAAK,EAAE,WAAW,EAAE,MAAM,aAAa,CAAA;AAU9C,qBAAa,UAAW,SAAQ,KAAK;IAE1B,MAAM,EAAE,MAAM;gBAAd,MAAM,EAAE,MAAM,EACrB,OAAO,EAAE,MAAM;CAIlB;AAED,qBAAa,sBAAuB,SAAQ,UAAU;;CAIrD;AAuHD,wBAAsB,eAAe,CACnC,WAAW,EAAE,WAAW,EACxB,MAAM,EAAE;IAAE,GAAG,EAAE,MAAM,CAAA;CAAE,EACvB,KAAK,EAAE;IAAE,UAAU,EAAE,MAAM,CAAC;IAAC,IAAI,CAAC,EAAE,MAAM,CAAC;IAAC,IAAI,CAAC,EAAE,MAAM,CAAC;IAAC,MAAM,EAAE,MAAM,CAAC,MAAM,EAAE,OAAO,CAAC,CAAA;CAAE,GAC3F,OAAO,CAAC;IAAE,GAAG,CAAC,EAAE,MAAM,CAAC;IAAC,GAAG,CAAC,EAAE,MAAM,CAAA;CAAE,CAAC,CAwCzC;AAED,wBAAsB,eAAe,CACnC,WAAW,EAAE,WAAW,EACxB,MAAM,EAAE;IAAE,GAAG,EAAE,MAAM,CAAA;CAAE,EACvB,KAAK,EAAE;IAAE,UAAU,EAAE,MAAM,CAAC;IAAC,IAAI,EAAE,MAAM,CAAA;CAAE,GAC1C,OAAO,CAAC,MAAM,CAAC,MAAM,EAAE,OAAO,CAAC,CAAC,CAyBlC;AAED,wBAAsB,YAAY,CAChC,WAAW,EAAE,WAAW,EACxB,MAAM,EAAE;IAAE,GAAG,EAAE,MAAM,CAAA;CAAE,EACvB,KAAK,EAAE;IAAE,UAAU,EAAE,MAAM,CAAC;IAAC,IAAI,EAAE,MAAM,CAAC;IAAC,MAAM,EAAE,MAAM,CAAC,MAAM,EAAE,OAAO,CAAC,CAAC;IAAC,IAAI,CAAC,EAAE,MAAM,CAAA;CAAE,GAC1F,OAAO,CAAC;IAAE,GAAG,CAAC,EAAE,MAAM,CAAC;IAAC,GAAG,CAAC,EAAE,MAAM,CAAA;CAAE,CAAC,CAqCzC;AAED,wBAAsB,aAAa,CACjC,WAAW,EAAE,WAAW,EACxB,MAAM,EAAE;IAAE,GAAG,EAAE,MAAM,CAAA;CAAE,EACvB,IAAI,EAAE,UAAU,EAChB,WAAW,EAAE,MAAM,GAClB,OAAO,CAAC;IAAE,IAAI,EAAE,OAAO,CAAA;CAAE,CAAC,CAS5B"}
|
|
@@ -0,0 +1,207 @@
|
|
|
1
|
+
// Shared PDS proxy functions — used by both HTTP route handlers and XRPC handlers.
|
|
2
|
+
import { getSession, getServerKey, deleteSession } from "./oauth/db.js";
|
|
3
|
+
import { createDpopProof } from "./oauth/dpop.js";
|
|
4
|
+
import { refreshPdsSession } from "./oauth/server.js";
|
|
5
|
+
import { validateRecord } from '@bigmoves/lexicon';
|
|
6
|
+
import { getLexiconArray } from "./database/schema.js";
|
|
7
|
+
import { insertRecord, deleteRecord as dbDeleteRecord } from "./database/db.js";
|
|
8
|
+
import { emit } from "./logger.js";
|
|
9
|
+
import { runLabelRules } from "./labels.js";
|
|
10
|
+
export class ProxyError extends Error {
|
|
11
|
+
status;
|
|
12
|
+
constructor(status, message) {
|
|
13
|
+
super(message);
|
|
14
|
+
this.status = status;
|
|
15
|
+
}
|
|
16
|
+
}
|
|
17
|
+
export class ScopeMissingProxyError extends ProxyError {
|
|
18
|
+
constructor() {
|
|
19
|
+
super(401, 'ScopeMissingError');
|
|
20
|
+
}
|
|
21
|
+
}
|
|
22
|
+
/** Shared retry logic: DPoP nonce handling + token refresh. */
|
|
23
|
+
async function withDpopRetry(oauthConfig, session, doFetch) {
|
|
24
|
+
let accessToken = session.access_token;
|
|
25
|
+
let result = await doFetch(accessToken);
|
|
26
|
+
if (result.ok)
|
|
27
|
+
return result;
|
|
28
|
+
let nonce;
|
|
29
|
+
// Step 1: handle DPoP nonce requirement
|
|
30
|
+
if (result.body.error === 'use_dpop_nonce') {
|
|
31
|
+
nonce = result.headers.get('DPoP-Nonce') || undefined;
|
|
32
|
+
if (nonce) {
|
|
33
|
+
result = await doFetch(accessToken, nonce);
|
|
34
|
+
if (result.ok)
|
|
35
|
+
return result;
|
|
36
|
+
}
|
|
37
|
+
}
|
|
38
|
+
// Step 2: handle insufficient scope — clear session so user re-authenticates with updated scopes
|
|
39
|
+
if (result.body.error === 'ScopeMissingError') {
|
|
40
|
+
await deleteSession(session.did);
|
|
41
|
+
throw new ScopeMissingProxyError();
|
|
42
|
+
}
|
|
43
|
+
// Step 3: handle expired PDS token — refresh and retry
|
|
44
|
+
// The PDS returns 'InvalidToken' or 'ExpiredToken' (AT Proto PascalCase convention)
|
|
45
|
+
// while the OAuth spec uses 'invalid_token' (RFC 6750 snake_case)
|
|
46
|
+
const err = result.body.error;
|
|
47
|
+
if (err === 'invalid_token' || err === 'InvalidToken' || err === 'ExpiredToken') {
|
|
48
|
+
const refreshed = await refreshPdsSession(oauthConfig, session);
|
|
49
|
+
if (refreshed) {
|
|
50
|
+
accessToken = refreshed.accessToken;
|
|
51
|
+
result = await doFetch(accessToken, nonce);
|
|
52
|
+
if (result.ok)
|
|
53
|
+
return result;
|
|
54
|
+
if (result.body.error === 'use_dpop_nonce') {
|
|
55
|
+
nonce = result.headers.get('DPoP-Nonce') || undefined;
|
|
56
|
+
if (nonce)
|
|
57
|
+
result = await doFetch(accessToken, nonce);
|
|
58
|
+
}
|
|
59
|
+
}
|
|
60
|
+
}
|
|
61
|
+
return result;
|
|
62
|
+
}
|
|
63
|
+
async function proxyToPds(oauthConfig, session, method, pdsUrl, body) {
|
|
64
|
+
const serverKey = await getServerKey('appview-oauth-key');
|
|
65
|
+
const privateJwk = JSON.parse(serverKey.privateKey);
|
|
66
|
+
const publicJwk = JSON.parse(serverKey.publicKey);
|
|
67
|
+
return withDpopRetry(oauthConfig, session, async (token, nonce) => {
|
|
68
|
+
const proof = await createDpopProof(privateJwk, publicJwk, method, pdsUrl, token, nonce);
|
|
69
|
+
const res = await fetch(pdsUrl, {
|
|
70
|
+
method,
|
|
71
|
+
headers: {
|
|
72
|
+
'Content-Type': 'application/json',
|
|
73
|
+
Authorization: `DPoP ${token}`,
|
|
74
|
+
DPoP: proof,
|
|
75
|
+
},
|
|
76
|
+
body: JSON.stringify(body),
|
|
77
|
+
});
|
|
78
|
+
const resBody = await res.json().catch(() => ({}));
|
|
79
|
+
return { ok: res.ok, status: res.status, body: resBody, headers: res.headers };
|
|
80
|
+
});
|
|
81
|
+
}
|
|
82
|
+
/** Proxy a raw binary request to the user's PDS with DPoP + nonce retry + token refresh. */
|
|
83
|
+
async function proxyToPdsRaw(oauthConfig, session, pdsUrl, body, contentType) {
|
|
84
|
+
const serverKey = await getServerKey('appview-oauth-key');
|
|
85
|
+
const privateJwk = JSON.parse(serverKey.privateKey);
|
|
86
|
+
const publicJwk = JSON.parse(serverKey.publicKey);
|
|
87
|
+
return withDpopRetry(oauthConfig, session, async (token, nonce) => {
|
|
88
|
+
const proof = await createDpopProof(privateJwk, publicJwk, 'POST', pdsUrl, token, nonce);
|
|
89
|
+
const res = await fetch(pdsUrl, {
|
|
90
|
+
method: 'POST',
|
|
91
|
+
headers: {
|
|
92
|
+
'Content-Type': contentType,
|
|
93
|
+
'Content-Length': String(body.length),
|
|
94
|
+
Authorization: `DPoP ${token}`,
|
|
95
|
+
DPoP: proof,
|
|
96
|
+
},
|
|
97
|
+
body: Buffer.from(body),
|
|
98
|
+
});
|
|
99
|
+
const resBody = await res.json().catch(() => ({}));
|
|
100
|
+
return { ok: res.ok, status: res.status, body: resBody, headers: res.headers };
|
|
101
|
+
});
|
|
102
|
+
}
|
|
103
|
+
// --- High-level proxy functions ---
|
|
104
|
+
export async function pdsCreateRecord(oauthConfig, viewer, input) {
|
|
105
|
+
const validationError = validateRecord(getLexiconArray(), input.collection, input.record);
|
|
106
|
+
if (validationError) {
|
|
107
|
+
throw new ProxyError(400, `InvalidRecord: ${validationError.path ? validationError.path + ': ' : ''}${validationError.message}`);
|
|
108
|
+
}
|
|
109
|
+
const session = await getSession(viewer.did);
|
|
110
|
+
if (!session)
|
|
111
|
+
throw new ProxyError(401, 'No PDS session for user');
|
|
112
|
+
const pdsUrl = `${session.pds_endpoint}/xrpc/com.atproto.repo.createRecord`;
|
|
113
|
+
const pdsBody = {
|
|
114
|
+
repo: viewer.did,
|
|
115
|
+
collection: input.collection,
|
|
116
|
+
rkey: input.rkey,
|
|
117
|
+
record: input.record,
|
|
118
|
+
};
|
|
119
|
+
const pdsRes = await proxyToPds(oauthConfig, session, 'POST', pdsUrl, pdsBody);
|
|
120
|
+
if (!pdsRes.ok)
|
|
121
|
+
throw new ProxyError(pdsRes.status, String(pdsRes.body.error || 'PDS write failed'));
|
|
122
|
+
try {
|
|
123
|
+
await insertRecord(input.collection, String(pdsRes.body.uri), String(pdsRes.body.cid), viewer.did, input.record);
|
|
124
|
+
await runLabelRules({
|
|
125
|
+
uri: String(pdsRes.body.uri),
|
|
126
|
+
cid: String(pdsRes.body.cid),
|
|
127
|
+
did: viewer.did,
|
|
128
|
+
collection: input.collection,
|
|
129
|
+
value: input.record,
|
|
130
|
+
});
|
|
131
|
+
}
|
|
132
|
+
catch (err) {
|
|
133
|
+
emit('pds-proxy', 'local_index_error', {
|
|
134
|
+
op: 'createRecord',
|
|
135
|
+
error: err instanceof Error ? err.message : String(err),
|
|
136
|
+
});
|
|
137
|
+
}
|
|
138
|
+
return pdsRes.body;
|
|
139
|
+
}
|
|
140
|
+
export async function pdsDeleteRecord(oauthConfig, viewer, input) {
|
|
141
|
+
const session = await getSession(viewer.did);
|
|
142
|
+
if (!session)
|
|
143
|
+
throw new ProxyError(401, 'No PDS session for user');
|
|
144
|
+
const pdsUrl = `${session.pds_endpoint}/xrpc/com.atproto.repo.deleteRecord`;
|
|
145
|
+
const pdsBody = {
|
|
146
|
+
repo: viewer.did,
|
|
147
|
+
collection: input.collection,
|
|
148
|
+
rkey: input.rkey,
|
|
149
|
+
};
|
|
150
|
+
const pdsRes = await proxyToPds(oauthConfig, session, 'POST', pdsUrl, pdsBody);
|
|
151
|
+
if (!pdsRes.ok)
|
|
152
|
+
throw new ProxyError(pdsRes.status, String(pdsRes.body.error || 'PDS delete failed'));
|
|
153
|
+
try {
|
|
154
|
+
const uri = `at://${viewer.did}/${input.collection}/${input.rkey}`;
|
|
155
|
+
await dbDeleteRecord(input.collection, uri);
|
|
156
|
+
}
|
|
157
|
+
catch (err) {
|
|
158
|
+
emit('pds-proxy', 'local_index_error', {
|
|
159
|
+
op: 'deleteRecord',
|
|
160
|
+
error: err instanceof Error ? err.message : String(err),
|
|
161
|
+
});
|
|
162
|
+
}
|
|
163
|
+
return pdsRes.body;
|
|
164
|
+
}
|
|
165
|
+
export async function pdsPutRecord(oauthConfig, viewer, input) {
|
|
166
|
+
const validationError = validateRecord(getLexiconArray(), input.collection, input.record);
|
|
167
|
+
if (validationError) {
|
|
168
|
+
throw new ProxyError(400, `InvalidRecord: ${validationError.path ? validationError.path + ': ' : ''}${validationError.message}`);
|
|
169
|
+
}
|
|
170
|
+
const session = await getSession(viewer.did);
|
|
171
|
+
if (!session)
|
|
172
|
+
throw new ProxyError(401, 'No PDS session for user');
|
|
173
|
+
const pdsUrl = `${session.pds_endpoint}/xrpc/com.atproto.repo.putRecord`;
|
|
174
|
+
const pdsBody = {
|
|
175
|
+
repo: viewer.did,
|
|
176
|
+
collection: input.collection,
|
|
177
|
+
rkey: input.rkey,
|
|
178
|
+
record: input.record,
|
|
179
|
+
};
|
|
180
|
+
const pdsRes = await proxyToPds(oauthConfig, session, 'POST', pdsUrl, pdsBody);
|
|
181
|
+
if (!pdsRes.ok)
|
|
182
|
+
throw new ProxyError(pdsRes.status, String(pdsRes.body.error || 'PDS write failed'));
|
|
183
|
+
try {
|
|
184
|
+
await insertRecord(input.collection, String(pdsRes.body.uri), String(pdsRes.body.cid), viewer.did, input.record);
|
|
185
|
+
await runLabelRules({
|
|
186
|
+
uri: String(pdsRes.body.uri),
|
|
187
|
+
cid: String(pdsRes.body.cid),
|
|
188
|
+
did: viewer.did,
|
|
189
|
+
collection: input.collection,
|
|
190
|
+
value: input.record,
|
|
191
|
+
});
|
|
192
|
+
}
|
|
193
|
+
catch (err) {
|
|
194
|
+
emit('pds-proxy', 'local_index_error', { op: 'putRecord', error: err instanceof Error ? err.message : String(err) });
|
|
195
|
+
}
|
|
196
|
+
return pdsRes.body;
|
|
197
|
+
}
|
|
198
|
+
export async function pdsUploadBlob(oauthConfig, viewer, body, contentType) {
|
|
199
|
+
const session = await getSession(viewer.did);
|
|
200
|
+
if (!session)
|
|
201
|
+
throw new ProxyError(401, 'No PDS session for user');
|
|
202
|
+
const pdsUrl = `${session.pds_endpoint}/xrpc/com.atproto.repo.uploadBlob`;
|
|
203
|
+
const pdsRes = await proxyToPdsRaw(oauthConfig, session, pdsUrl, body, contentType);
|
|
204
|
+
if (!pdsRes.ok)
|
|
205
|
+
throw new ProxyError(pdsRes.status, String(pdsRes.body.error || 'PDS upload failed'));
|
|
206
|
+
return pdsRes.body;
|
|
207
|
+
}
|
package/dist/push.d.ts
ADDED
|
@@ -0,0 +1,33 @@
|
|
|
1
|
+
export interface ApnsConfig {
|
|
2
|
+
keyFile: string;
|
|
3
|
+
keyId: string;
|
|
4
|
+
teamId: string;
|
|
5
|
+
bundleId: string;
|
|
6
|
+
production?: boolean;
|
|
7
|
+
}
|
|
8
|
+
export interface PushConfig {
|
|
9
|
+
apns: ApnsConfig;
|
|
10
|
+
}
|
|
11
|
+
export interface PushPayload {
|
|
12
|
+
did: string;
|
|
13
|
+
title: string;
|
|
14
|
+
body: string;
|
|
15
|
+
data?: Record<string, string>;
|
|
16
|
+
collapseId?: string;
|
|
17
|
+
}
|
|
18
|
+
export interface PushInterface {
|
|
19
|
+
send: (payload: PushPayload) => Promise<void>;
|
|
20
|
+
}
|
|
21
|
+
/** Initialize push with config. Must be called before send(). */
|
|
22
|
+
export declare function initPush(config: PushConfig, configDir: string): void;
|
|
23
|
+
/** Check if push is configured and available. */
|
|
24
|
+
export declare function isPushEnabled(): boolean;
|
|
25
|
+
/** Build the push interface injected into hook contexts. */
|
|
26
|
+
export declare function buildPushInterface(): PushInterface;
|
|
27
|
+
/** Register a push token for a DID. Upserts on conflict. */
|
|
28
|
+
export declare function registerToken(did: string, token: string, platform: string): Promise<void>;
|
|
29
|
+
/** Remove a push token. */
|
|
30
|
+
export declare function removeToken(token: string): Promise<void>;
|
|
31
|
+
/** Unregister a specific token for a DID. */
|
|
32
|
+
export declare function unregisterToken(did: string, token: string): Promise<void>;
|
|
33
|
+
//# sourceMappingURL=push.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"push.d.ts","sourceRoot":"","sources":["../src/push.ts"],"names":[],"mappings":"AAeA,MAAM,WAAW,UAAU;IACzB,OAAO,EAAE,MAAM,CAAA;IACf,KAAK,EAAE,MAAM,CAAA;IACb,MAAM,EAAE,MAAM,CAAA;IACd,QAAQ,EAAE,MAAM,CAAA;IAChB,UAAU,CAAC,EAAE,OAAO,CAAA;CACrB;AAED,MAAM,WAAW,UAAU;IACzB,IAAI,EAAE,UAAU,CAAA;CACjB;AAED,MAAM,WAAW,WAAW;IAC1B,GAAG,EAAE,MAAM,CAAA;IACX,KAAK,EAAE,MAAM,CAAA;IACb,IAAI,EAAE,MAAM,CAAA;IACZ,IAAI,CAAC,EAAE,MAAM,CAAC,MAAM,EAAE,MAAM,CAAC,CAAA;IAC7B,UAAU,CAAC,EAAE,MAAM,CAAA;CACpB;AAED,MAAM,WAAW,aAAa;IAC5B,IAAI,EAAE,CAAC,OAAO,EAAE,WAAW,KAAK,OAAO,CAAC,IAAI,CAAC,CAAA;CAC9C;AAOD,iEAAiE;AACjE,wBAAgB,QAAQ,CAAC,MAAM,EAAE,UAAU,EAAE,SAAS,EAAE,MAAM,GAAG,IAAI,CASpE;AAED,iDAAiD;AACjD,wBAAgB,aAAa,IAAI,OAAO,CAEvC;AAED,4DAA4D;AAC5D,wBAAgB,kBAAkB,IAAI,aAAa,CAElD;AAoID,4DAA4D;AAC5D,wBAAsB,aAAa,CAAC,GAAG,EAAE,MAAM,EAAE,KAAK,EAAE,MAAM,EAAE,QAAQ,EAAE,MAAM,GAAG,OAAO,CAAC,IAAI,CAAC,CAO/F;AAED,2BAA2B;AAC3B,wBAAsB,WAAW,CAAC,KAAK,EAAE,MAAM,GAAG,OAAO,CAAC,IAAI,CAAC,CAE9D;AAED,6CAA6C;AAC7C,wBAAsB,eAAe,CAAC,GAAG,EAAE,MAAM,EAAE,KAAK,EAAE,MAAM,GAAG,OAAO,CAAC,IAAI,CAAC,CAE/E"}
|
package/dist/push.js
ADDED
|
@@ -0,0 +1,166 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Push notification delivery via APNs HTTP/2.
|
|
3
|
+
*
|
|
4
|
+
* Provides `push.send()` for use in on-commit hook context. Looks up device
|
|
5
|
+
* tokens, builds APNs payloads, and sends via HTTP/2. Self-cleans invalid
|
|
6
|
+
* tokens on Apple 410 responses. Fire-and-forget — failures are logged via
|
|
7
|
+
* `emit()` but never throw.
|
|
8
|
+
*/
|
|
9
|
+
import { connect } from 'node:http2';
|
|
10
|
+
import { readFileSync } from 'node:fs';
|
|
11
|
+
import { createSign } from 'node:crypto';
|
|
12
|
+
import { resolve } from 'node:path';
|
|
13
|
+
import { emit } from "./logger.js";
|
|
14
|
+
import { runSQL, querySQL } from "./database/db.js";
|
|
15
|
+
let pushConfig = null;
|
|
16
|
+
let apnsKey = null;
|
|
17
|
+
let cachedJwt = null;
|
|
18
|
+
let http2Session = null;
|
|
19
|
+
/** Initialize push with config. Must be called before send(). */
|
|
20
|
+
export function initPush(config, configDir) {
|
|
21
|
+
pushConfig = config;
|
|
22
|
+
const keyPath = resolve(configDir, config.apns.keyFile);
|
|
23
|
+
try {
|
|
24
|
+
apnsKey = readFileSync(keyPath, 'utf8');
|
|
25
|
+
}
|
|
26
|
+
catch {
|
|
27
|
+
emit('push', 'init_error', { error: `APNs key file not found: ${keyPath}` });
|
|
28
|
+
pushConfig = null;
|
|
29
|
+
}
|
|
30
|
+
}
|
|
31
|
+
/** Check if push is configured and available. */
|
|
32
|
+
export function isPushEnabled() {
|
|
33
|
+
return pushConfig !== null && apnsKey !== null;
|
|
34
|
+
}
|
|
35
|
+
/** Build the push interface injected into hook contexts. */
|
|
36
|
+
export function buildPushInterface() {
|
|
37
|
+
return { send };
|
|
38
|
+
}
|
|
39
|
+
/** Create a JWT for APNs authentication (cached for 50 minutes). */
|
|
40
|
+
function getApnsJwt() {
|
|
41
|
+
if (cachedJwt && Date.now() < cachedJwt.expires)
|
|
42
|
+
return cachedJwt.token;
|
|
43
|
+
if (!pushConfig || !apnsKey)
|
|
44
|
+
throw new Error('Push not initialized');
|
|
45
|
+
const header = Buffer.from(JSON.stringify({
|
|
46
|
+
alg: 'ES256',
|
|
47
|
+
kid: pushConfig.apns.keyId,
|
|
48
|
+
})).toString('base64url');
|
|
49
|
+
const now = Math.floor(Date.now() / 1000);
|
|
50
|
+
const claims = Buffer.from(JSON.stringify({
|
|
51
|
+
iss: pushConfig.apns.teamId,
|
|
52
|
+
iat: now,
|
|
53
|
+
})).toString('base64url');
|
|
54
|
+
const signer = createSign('SHA256');
|
|
55
|
+
signer.update(`${header}.${claims}`);
|
|
56
|
+
const signature = signer.sign(apnsKey, 'base64url');
|
|
57
|
+
const token = `${header}.${claims}.${signature}`;
|
|
58
|
+
cachedJwt = { token, expires: Date.now() + 50 * 60 * 1000 };
|
|
59
|
+
return token;
|
|
60
|
+
}
|
|
61
|
+
/** Get or create an HTTP/2 connection to APNs. */
|
|
62
|
+
function getHttp2Session() {
|
|
63
|
+
if (http2Session && !http2Session.closed && !http2Session.destroyed) {
|
|
64
|
+
return http2Session;
|
|
65
|
+
}
|
|
66
|
+
const host = pushConfig?.apns.production !== false
|
|
67
|
+
? 'https://api.push.apple.com'
|
|
68
|
+
: 'https://api.sandbox.push.apple.com';
|
|
69
|
+
http2Session = connect(host);
|
|
70
|
+
http2Session.on('error', () => {
|
|
71
|
+
http2Session = null;
|
|
72
|
+
});
|
|
73
|
+
http2Session.on('close', () => {
|
|
74
|
+
http2Session = null;
|
|
75
|
+
});
|
|
76
|
+
return http2Session;
|
|
77
|
+
}
|
|
78
|
+
/** Send a push notification to all devices registered for a DID. */
|
|
79
|
+
async function send(payload) {
|
|
80
|
+
if (!pushConfig || !apnsKey)
|
|
81
|
+
return;
|
|
82
|
+
const tokens = await querySQL(`SELECT token, platform FROM _push_tokens WHERE did = $1`, [payload.did]);
|
|
83
|
+
if (tokens.length === 0)
|
|
84
|
+
return;
|
|
85
|
+
const jwt = getApnsJwt();
|
|
86
|
+
const apnsPayload = JSON.stringify({
|
|
87
|
+
aps: {
|
|
88
|
+
alert: { title: payload.title, body: payload.body },
|
|
89
|
+
sound: 'default',
|
|
90
|
+
},
|
|
91
|
+
...(payload.data || {}),
|
|
92
|
+
});
|
|
93
|
+
for (const { token, platform } of tokens) {
|
|
94
|
+
if (platform !== 'apns')
|
|
95
|
+
continue;
|
|
96
|
+
sendToApns(token, apnsPayload, jwt, payload).catch(() => { });
|
|
97
|
+
}
|
|
98
|
+
}
|
|
99
|
+
/** Send a single APNs push and handle the response. */
|
|
100
|
+
async function sendToApns(token, payload, jwt, original) {
|
|
101
|
+
const session = getHttp2Session();
|
|
102
|
+
const headers = {
|
|
103
|
+
':method': 'POST',
|
|
104
|
+
':path': `/3/device/${token}`,
|
|
105
|
+
'authorization': `bearer ${jwt}`,
|
|
106
|
+
'apns-topic': pushConfig.apns.bundleId,
|
|
107
|
+
'apns-push-type': 'alert',
|
|
108
|
+
};
|
|
109
|
+
if (original.collapseId) {
|
|
110
|
+
headers['apns-collapse-id'] = original.collapseId;
|
|
111
|
+
}
|
|
112
|
+
return new Promise((resolve) => {
|
|
113
|
+
const req = session.request(headers);
|
|
114
|
+
req.setTimeout(15_000, () => {
|
|
115
|
+
req.close();
|
|
116
|
+
emit('push', 'send_error', { did: original.did, error: 'APNs request timed out' });
|
|
117
|
+
resolve();
|
|
118
|
+
});
|
|
119
|
+
let status = 0;
|
|
120
|
+
let body = '';
|
|
121
|
+
req.on('response', (headers) => {
|
|
122
|
+
status = headers[':status'];
|
|
123
|
+
});
|
|
124
|
+
req.on('data', (chunk) => {
|
|
125
|
+
body += chunk.toString();
|
|
126
|
+
});
|
|
127
|
+
req.on('end', async () => {
|
|
128
|
+
if (status === 200) {
|
|
129
|
+
emit('push', 'sent', { did: original.did, token: token.slice(0, 8) + '...' });
|
|
130
|
+
}
|
|
131
|
+
else if (status === 410) {
|
|
132
|
+
// Token is no longer valid — remove it
|
|
133
|
+
await removeToken(token).catch(() => { });
|
|
134
|
+
emit('push', 'token_removed', { did: original.did, reason: 'expired' });
|
|
135
|
+
}
|
|
136
|
+
else {
|
|
137
|
+
emit('push', 'send_error', {
|
|
138
|
+
did: original.did,
|
|
139
|
+
status,
|
|
140
|
+
body: body.slice(0, 200),
|
|
141
|
+
});
|
|
142
|
+
}
|
|
143
|
+
resolve();
|
|
144
|
+
});
|
|
145
|
+
req.on('error', (err) => {
|
|
146
|
+
emit('push', 'send_error', { did: original.did, error: err.message });
|
|
147
|
+
resolve();
|
|
148
|
+
});
|
|
149
|
+
req.write(payload);
|
|
150
|
+
req.end();
|
|
151
|
+
});
|
|
152
|
+
}
|
|
153
|
+
/** Register a push token for a DID. Upserts on conflict. */
|
|
154
|
+
export async function registerToken(did, token, platform) {
|
|
155
|
+
await runSQL(`INSERT INTO _push_tokens (did, token, platform, created_at)
|
|
156
|
+
VALUES ($1, $2, $3, $4)
|
|
157
|
+
ON CONFLICT (did, token) DO UPDATE SET platform = excluded.platform`, [did, token, platform, new Date().toISOString()]);
|
|
158
|
+
}
|
|
159
|
+
/** Remove a push token. */
|
|
160
|
+
export async function removeToken(token) {
|
|
161
|
+
await runSQL(`DELETE FROM _push_tokens WHERE token = $1`, [token]);
|
|
162
|
+
}
|
|
163
|
+
/** Unregister a specific token for a DID. */
|
|
164
|
+
export async function unregisterToken(did, token) {
|
|
165
|
+
await runSQL(`DELETE FROM _push_tokens WHERE did = $1 AND token = $2`, [did, token]);
|
|
166
|
+
}
|
|
@@ -0,0 +1,27 @@
|
|
|
1
|
+
export interface SSRManifest {
|
|
2
|
+
getPreloadTags(url: string): string;
|
|
3
|
+
}
|
|
4
|
+
export interface RenderResult {
|
|
5
|
+
html: string;
|
|
6
|
+
head?: string;
|
|
7
|
+
}
|
|
8
|
+
export type RendererHandler = (request: Request, manifest: SSRManifest) => Promise<RenderResult>;
|
|
9
|
+
export declare function defineRenderer(handler: RendererHandler): {
|
|
10
|
+
__type: "renderer";
|
|
11
|
+
handler: RendererHandler;
|
|
12
|
+
};
|
|
13
|
+
export declare function registerRenderer(handler: RendererHandler): void;
|
|
14
|
+
export declare function setSSRManifest(manifest: SSRManifest): void;
|
|
15
|
+
export declare function getRenderer(): RendererHandler | null;
|
|
16
|
+
export declare function getSSRManifest(): SSRManifest | null;
|
|
17
|
+
/**
|
|
18
|
+
* Render an HTML page by calling the user's renderer and assembling the result
|
|
19
|
+
* into the index.html template.
|
|
20
|
+
*
|
|
21
|
+
* @param template - The index.html content (with <!--ssr-outlet--> placeholder)
|
|
22
|
+
* @param request - The incoming Request
|
|
23
|
+
* @param ogMeta - Optional OG meta tags to inject
|
|
24
|
+
* @returns Assembled HTML string, or null if no renderer is registered
|
|
25
|
+
*/
|
|
26
|
+
export declare function renderPage(template: string, request: Request, ogMeta?: string | null): Promise<string | null>;
|
|
27
|
+
//# sourceMappingURL=renderer.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"renderer.d.ts","sourceRoot":"","sources":["../src/renderer.ts"],"names":[],"mappings":"AAEA,MAAM,WAAW,WAAW;IAC1B,cAAc,CAAC,GAAG,EAAE,MAAM,GAAG,MAAM,CAAA;CACpC;AAED,MAAM,WAAW,YAAY;IAC3B,IAAI,EAAE,MAAM,CAAA;IACZ,IAAI,CAAC,EAAE,MAAM,CAAA;CACd;AAED,MAAM,MAAM,eAAe,GAAG,CAAC,OAAO,EAAE,OAAO,EAAE,QAAQ,EAAE,WAAW,KAAK,OAAO,CAAC,YAAY,CAAC,CAAA;AAKhG,wBAAgB,cAAc,CAAC,OAAO,EAAE,eAAe;;;EAEtD;AAED,wBAAgB,gBAAgB,CAAC,OAAO,EAAE,eAAe,GAAG,IAAI,CAG/D;AAED,wBAAgB,cAAc,CAAC,QAAQ,EAAE,WAAW,GAAG,IAAI,CAE1D;AAED,wBAAgB,WAAW,IAAI,eAAe,GAAG,IAAI,CAEpD;AAED,wBAAgB,cAAc,IAAI,WAAW,GAAG,IAAI,CAEnD;AAED;;;;;;;;GAQG;AACH,wBAAsB,UAAU,CAAC,QAAQ,EAAE,MAAM,EAAE,OAAO,EAAE,OAAO,EAAE,MAAM,CAAC,EAAE,MAAM,GAAG,IAAI,GAAG,OAAO,CAAC,MAAM,GAAG,IAAI,CAAC,CAsBnH"}
|