@chainlesschain/personal-data-hub 0.3.1 → 0.3.7

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.
Files changed (79) hide show
  1. package/__tests__/adapters/email-adapter-snapshot.test.js +237 -0
  2. package/__tests__/adapters/email-adapter.test.js +1 -1
  3. package/__tests__/adapters/email-pdf-extractor.test.js +1 -1
  4. package/__tests__/adapters/email-retry-progress.test.js +1 -1
  5. package/__tests__/adapters/email-templates.test.js +1 -1
  6. package/__tests__/adapters/social-bilibili-adb-api-client.test.js +721 -0
  7. package/__tests__/adapters/social-bilibili-adb-chromium-cookies-reader.test.js +346 -0
  8. package/__tests__/adapters/social-bilibili-adb-collector.test.js +284 -0
  9. package/__tests__/adapters/social-bilibili-adb-cookies-extension.test.js +343 -0
  10. package/__tests__/adapters/social-bilibili-adb-snapshot-builder.test.js +296 -0
  11. package/__tests__/adapters/social-douyin-adb-collector.test.js +254 -0
  12. package/__tests__/adapters/social-douyin-adb-im-db-parser.test.js +304 -0
  13. package/__tests__/adapters/social-douyin-adb-snapshot-builder.test.js +216 -0
  14. package/__tests__/adapters/social-kuaishou-adb-api-client.test.js +432 -0
  15. package/__tests__/adapters/social-kuaishou-adb-collector.test.js +276 -0
  16. package/__tests__/adapters/social-kuaishou-adb-cookies-extension.test.js +141 -0
  17. package/__tests__/adapters/social-kuaishou-adb-snapshot-builder.test.js +178 -0
  18. package/__tests__/adapters/social-toutiao-adb-api-client.test.js +537 -0
  19. package/__tests__/adapters/social-toutiao-adb-collector.test.js +285 -0
  20. package/__tests__/adapters/social-toutiao-adb-cookies-extension.test.js +163 -0
  21. package/__tests__/adapters/social-toutiao-adb-snapshot-builder.test.js +196 -0
  22. package/__tests__/adapters/social-weibo-adb-api-client.test.js +362 -0
  23. package/__tests__/adapters/social-weibo-adb-collector.test.js +201 -0
  24. package/__tests__/adapters/social-weibo-adb-snapshot-builder.test.js +189 -0
  25. package/__tests__/adapters/social-xiaohongshu-adb-collector.test.js +207 -0
  26. package/__tests__/adapters/social-xiaohongshu-adb-sign-provider-injection.test.js +351 -0
  27. package/__tests__/adapters/social-xiaohongshu-adb-sign.test.js +130 -0
  28. package/__tests__/adapters/system-data-android.test.js +32 -1
  29. package/__tests__/longtail-adapters.test.js +15 -2
  30. package/__tests__/shopping-adapters.test.js +96 -0
  31. package/__tests__/sign-providers.test.js +62 -0
  32. package/__tests__/travel-adapters.test.js +66 -0
  33. package/__tests__/whatsapp-adapter.test.js +5 -2
  34. package/lib/adapters/browser-history-chrome/chrome-db-reader.js +11 -1
  35. package/lib/adapters/email-imap/email-adapter.js +224 -17
  36. package/lib/adapters/messaging-telegram/index.js +15 -12
  37. package/lib/adapters/messaging-whatsapp/index.js +15 -12
  38. package/lib/adapters/shopping-taobao/index.js +161 -21
  39. package/lib/adapters/social-bilibili-adb/api-client.js +555 -0
  40. package/lib/adapters/social-bilibili-adb/chromium-cookies-reader.js +296 -0
  41. package/lib/adapters/social-bilibili-adb/collector.js +190 -0
  42. package/lib/adapters/social-bilibili-adb/cookies-extension.js +250 -0
  43. package/lib/adapters/social-bilibili-adb/index.js +51 -0
  44. package/lib/adapters/social-bilibili-adb/snapshot-builder.js +197 -0
  45. package/lib/adapters/social-douyin/index.js +4 -0
  46. package/lib/adapters/social-douyin-adb/collector.js +165 -0
  47. package/lib/adapters/social-douyin-adb/db-extension.js +281 -0
  48. package/lib/adapters/social-douyin-adb/im-db-parser.js +287 -0
  49. package/lib/adapters/social-douyin-adb/index.js +57 -0
  50. package/lib/adapters/social-douyin-adb/snapshot-builder.js +174 -0
  51. package/lib/adapters/social-kuaishou-adb/api-client.js +397 -0
  52. package/lib/adapters/social-kuaishou-adb/collector.js +196 -0
  53. package/lib/adapters/social-kuaishou-adb/cookies-extension.js +261 -0
  54. package/lib/adapters/social-kuaishou-adb/index.js +53 -0
  55. package/lib/adapters/social-kuaishou-adb/snapshot-builder.js +145 -0
  56. package/lib/adapters/social-toutiao-adb/api-client.js +377 -0
  57. package/lib/adapters/social-toutiao-adb/collector.js +200 -0
  58. package/lib/adapters/social-toutiao-adb/cookies-extension.js +266 -0
  59. package/lib/adapters/social-toutiao-adb/index.js +52 -0
  60. package/lib/adapters/social-toutiao-adb/snapshot-builder.js +148 -0
  61. package/lib/adapters/social-weibo-adb/api-client.js +281 -0
  62. package/lib/adapters/social-weibo-adb/collector.js +169 -0
  63. package/lib/adapters/social-weibo-adb/cookies-extension.js +251 -0
  64. package/lib/adapters/social-weibo-adb/index.js +55 -0
  65. package/lib/adapters/social-weibo-adb/snapshot-builder.js +145 -0
  66. package/lib/adapters/social-xiaohongshu-adb/api-client.js +309 -0
  67. package/lib/adapters/social-xiaohongshu-adb/collector.js +209 -0
  68. package/lib/adapters/social-xiaohongshu-adb/cookies-extension.js +211 -0
  69. package/lib/adapters/social-xiaohongshu-adb/index.js +50 -0
  70. package/lib/adapters/social-xiaohongshu-adb/sign.js +90 -0
  71. package/lib/adapters/social-xiaohongshu-adb/snapshot-builder.js +126 -0
  72. package/lib/adapters/system-data-android/adapter.js +77 -3
  73. package/lib/adapters/travel-amap/index.js +16 -10
  74. package/lib/adapters/travel-ctrip/index.js +25 -9
  75. package/lib/adapters/vscode/vscode-reader.js +7 -1
  76. package/lib/sign-providers/index.js +20 -0
  77. package/lib/sign-providers/interface.js +82 -0
  78. package/lib/sign-providers/null-sign-provider.js +30 -0
  79. package/package.json +10 -1
@@ -0,0 +1,276 @@
1
+ "use strict";
2
+
3
+ import { describe, it, expect, vi } from "vitest";
4
+ const os = require("node:os");
5
+
6
+ const {
7
+ collect,
8
+ collectAndSync,
9
+ } = require("../../lib/adapters/social-kuaishou-adb/collector");
10
+ const {
11
+ KuaishouApiClient,
12
+ } = require("../../lib/adapters/social-kuaishou-adb/api-client");
13
+
14
+ function makeFakeFetch(responses) {
15
+ const calls = [];
16
+ const fakeFetch = async (urlStr, opts) => {
17
+ calls.push({ url: urlStr, opts });
18
+ for (const [pattern, payload] of responses) {
19
+ if (urlStr.includes(pattern)) {
20
+ const resolved =
21
+ typeof payload === "function" ? await payload(urlStr, opts) : payload;
22
+ return {
23
+ ok: resolved.status == null || resolved.status === 200,
24
+ status: resolved.status || 200,
25
+ text: async () => resolved.body,
26
+ };
27
+ }
28
+ }
29
+ throw new Error("fake fetch: no response for " + urlStr);
30
+ };
31
+ return { fakeFetch, calls };
32
+ }
33
+
34
+ function gqlBody(opName, dataFn) {
35
+ return async (urlStr, opts) => {
36
+ const reqBody = JSON.parse(opts.body);
37
+ if (reqBody.operationName !== opName) {
38
+ return { body: JSON.stringify({ data: {} }) };
39
+ }
40
+ return { body: JSON.stringify({ data: dataFn() }) };
41
+ };
42
+ }
43
+
44
+ const HAPPY_RESPONSES = [
45
+ // Single graphql endpoint — switch on operationName via body inspection
46
+ [
47
+ "graphql",
48
+ async (urlStr, opts) => {
49
+ const reqBody = JSON.parse(opts.body);
50
+ let data = {};
51
+ if (reqBody.operationName === "visionFeedRecommend") {
52
+ data = {
53
+ visionFeedRecommend: {
54
+ feeds: [
55
+ {
56
+ photo: { id: "W1", caption: "watched", timestamp: 1700000000 },
57
+ author: { id: "AUTH", name: "Author" },
58
+ },
59
+ ],
60
+ },
61
+ };
62
+ } else if (reqBody.operationName === "visionProfilePhotoList") {
63
+ data = {
64
+ visionProfilePhotoList: {
65
+ feeds: [
66
+ {
67
+ photo: { id: "OWN1", caption: "my", timestamp: 1700001000 },
68
+ },
69
+ ],
70
+ },
71
+ };
72
+ } else if (reqBody.operationName === "visionSearchPhoto") {
73
+ data = {
74
+ visionSearchPhoto: {
75
+ recentSearchList: [{ keyword: "kw", time: 1700002000 }],
76
+ },
77
+ };
78
+ }
79
+ return { body: JSON.stringify({ data }) };
80
+ },
81
+ ],
82
+ ];
83
+
84
+ const apiPhPayload = encodeURIComponent(
85
+ JSON.stringify({
86
+ user_id: "12345",
87
+ user_name: "Alice",
88
+ }),
89
+ );
90
+
91
+ const COOKIE_PAYLOAD = {
92
+ cookie: `userId=12345; kuaishou.web.cp.api_ph=${apiPhPayload}`,
93
+ uid: "12345",
94
+ diagnostic: { cookieCount: 5, hadEncrypted: false, cookieNames: [] },
95
+ };
96
+
97
+ function makeBridge(invokeResult) {
98
+ return {
99
+ invoke: vi.fn(async () => invokeResult),
100
+ };
101
+ }
102
+
103
+ describe("collect — happy path with signProvider", () => {
104
+ it("warmUp → 3 signed endpoints → shutdown", async () => {
105
+ const { fakeFetch } = makeFakeFetch(HAPPY_RESPONSES);
106
+ const lifecycle = [];
107
+ const sign = {
108
+ warmUp: vi.fn(async (c) => lifecycle.push({ warmUp: c })),
109
+ signUrl: vi.fn(async (url) => {
110
+ const u = new URL(String(url));
111
+ u.searchParams.set("__NS_sig3", "BRIDGE_SIG");
112
+ return u;
113
+ }),
114
+ signedHeaders: vi.fn(async () => ({
115
+ kpf: "PC_WEB",
116
+ kpn: "KUAISHOU_VISION",
117
+ })),
118
+ shutdown: vi.fn(async () => lifecycle.push("shutdown")),
119
+ };
120
+ const client = new KuaishouApiClient({
121
+ fetch: fakeFetch,
122
+ signProvider: sign,
123
+ });
124
+ const r = await collect(makeBridge(COOKIE_PAYLOAD), {
125
+ apiClient: client,
126
+ signProvider: sign,
127
+ stagingDir: os.tmpdir(),
128
+ });
129
+ expect(sign.warmUp).toHaveBeenCalledWith(COOKIE_PAYLOAD.cookie);
130
+ expect(sign.shutdown).toHaveBeenCalledOnce();
131
+ expect(r.uid).toBe("12345");
132
+ expect(r.nickname).toBe("Alice");
133
+ expect(r.profileFetchFailed).toBe(false);
134
+ expect(r.eventCounts.profile).toBe(1);
135
+ expect(r.eventCounts.watch).toBe(1);
136
+ expect(r.eventCounts.collect).toBe(1);
137
+ expect(r.eventCounts.search).toBe(1);
138
+ expect(r.signProviderHits).toBe(3);
139
+ expect(r.signProviderFallbacks).toBe(0);
140
+ });
141
+ });
142
+
143
+ describe("collect — fallback path (no signProvider)", () => {
144
+ it("3 signed endpoints short-circuit; profile from cookie still emitted", async () => {
145
+ const { fakeFetch } = makeFakeFetch(HAPPY_RESPONSES);
146
+ const client = new KuaishouApiClient({ fetch: fakeFetch });
147
+ const r = await collect(makeBridge(COOKIE_PAYLOAD), {
148
+ apiClient: client,
149
+ stagingDir: os.tmpdir(),
150
+ });
151
+ expect(r.uid).toBe("12345");
152
+ expect(r.profileFetchFailed).toBe(false);
153
+ expect(r.eventCounts.profile).toBe(1); // pure cookie parse
154
+ expect(r.eventCounts.watch).toBe(0); // short-circuit
155
+ expect(r.eventCounts.collect).toBe(0);
156
+ expect(r.eventCounts.search).toBe(0);
157
+ expect(r.signProviderUsed).toBe("none");
158
+ expect(r.signProviderHits).toBe(0);
159
+ expect(r.signProviderFallbacks).toBe(3);
160
+ });
161
+ });
162
+
163
+ describe("collect — profile fetch fails (no api_ph)", () => {
164
+ it("emits empty snapshot with cookie-derived uid + profileFetchFailed=true", async () => {
165
+ const cookiePayload = {
166
+ cookie: "userId=12345", // no api_ph → fetchProfile fails
167
+ uid: "12345",
168
+ diagnostic: {},
169
+ };
170
+ const { fakeFetch } = makeFakeFetch([]);
171
+ const client = new KuaishouApiClient({ fetch: fakeFetch });
172
+ const r = await collect(makeBridge(cookiePayload), {
173
+ apiClient: client,
174
+ stagingDir: os.tmpdir(),
175
+ });
176
+ expect(r.profileFetchFailed).toBe(true);
177
+ expect(r.uid).toBe("12345"); // cookie pre-extract
178
+ expect(r.eventCounts.total).toBe(0);
179
+ expect(r.lastErrorCode).toBe(-8);
180
+ });
181
+ });
182
+
183
+ describe("collect — bridge warmUp failure", () => {
184
+ it("tolerates warmUp throw; signed endpoints fall through to short-circuit", async () => {
185
+ const { fakeFetch } = makeFakeFetch(HAPPY_RESPONSES);
186
+ const sign = {
187
+ warmUp: vi.fn(async () => {
188
+ throw new Error("kuaishou.com 403");
189
+ }),
190
+ signUrl: vi.fn(async () => null),
191
+ signedHeaders: vi.fn(async () => ({})),
192
+ shutdown: vi.fn(async () => {}),
193
+ };
194
+ const client = new KuaishouApiClient({
195
+ fetch: fakeFetch,
196
+ signProvider: sign,
197
+ });
198
+ const r = await collect(makeBridge(COOKIE_PAYLOAD), {
199
+ apiClient: client,
200
+ signProvider: sign,
201
+ stagingDir: os.tmpdir(),
202
+ });
203
+ expect(r.profileFetchFailed).toBe(false); // profile pure cookie parse
204
+ expect(r.eventCounts.watch).toBe(0);
205
+ expect(client._fallbackHits).toBe(3);
206
+ expect(sign.shutdown).toHaveBeenCalledOnce();
207
+ });
208
+ });
209
+
210
+ describe("collect — malformed bridge payload", () => {
211
+ it("throws when bridge returns no cookie", async () => {
212
+ const bridge = { invoke: vi.fn(async () => ({ uid: "1" })) };
213
+ await expect(
214
+ collect(bridge, { stagingDir: os.tmpdir() }),
215
+ ).rejects.toThrow(/malformed payload/);
216
+ });
217
+
218
+ it("throws when bridge missing invoke", async () => {
219
+ await expect(collect({}, {})).rejects.toThrow(
220
+ /bridge must expose invoke/,
221
+ );
222
+ });
223
+ });
224
+
225
+ describe("collect — signProviderUsed diagnostic", () => {
226
+ it("reports class name when bridge present", async () => {
227
+ const { fakeFetch } = makeFakeFetch(HAPPY_RESPONSES);
228
+ class KuaishouSignBridge {
229
+ constructor() {
230
+ this.warmUp = vi.fn(async () => {});
231
+ this.signUrl = vi.fn(async (url) => {
232
+ const u = new URL(String(url));
233
+ u.searchParams.set("__NS_sig3", "X");
234
+ return u;
235
+ });
236
+ this.signedHeaders = vi.fn(async () => ({
237
+ kpf: "PC_WEB",
238
+ kpn: "KUAISHOU_VISION",
239
+ }));
240
+ this.shutdown = vi.fn(async () => {});
241
+ }
242
+ }
243
+ const sign = new KuaishouSignBridge();
244
+ const client = new KuaishouApiClient({
245
+ fetch: fakeFetch,
246
+ signProvider: sign,
247
+ });
248
+ const r = await collect(makeBridge(COOKIE_PAYLOAD), {
249
+ apiClient: client,
250
+ signProvider: sign,
251
+ stagingDir: os.tmpdir(),
252
+ });
253
+ expect(r.signProviderUsed).toBe("KuaishouSignBridge");
254
+ });
255
+ });
256
+
257
+ describe("collectAndSync", () => {
258
+ it("orchestrates collect + registry.syncAdapter + cleanup", async () => {
259
+ const { fakeFetch } = makeFakeFetch(HAPPY_RESPONSES);
260
+ const client = new KuaishouApiClient({ fetch: fakeFetch });
261
+ const registry = {
262
+ syncAdapter: vi.fn(async (name) => ({ adapter: name, status: "ok" })),
263
+ };
264
+ const r = await collectAndSync(makeBridge(COOKIE_PAYLOAD), registry, {
265
+ apiClient: client,
266
+ stagingDir: os.tmpdir(),
267
+ });
268
+ expect(registry.syncAdapter).toHaveBeenCalledWith(
269
+ "social-kuaishou",
270
+ expect.objectContaining({ inputPath: expect.stringContaining(".json") }),
271
+ );
272
+ expect(r.adapter).toBe("social-kuaishou");
273
+ expect(r.kuaishou.uid).toBe("12345");
274
+ expect(r.kuaishou.eventCounts.profile).toBe(1);
275
+ });
276
+ });
@@ -0,0 +1,141 @@
1
+ "use strict";
2
+
3
+ import { describe, it, expect } from "vitest";
4
+
5
+ const {
6
+ createKuaishouCookiesExtension,
7
+ KUAISHOU_COOKIES_REMOTE_PATH,
8
+ KUAISHOU_COOKIE_HOST_DOMAIN,
9
+ KUAISHOU_LOGIN_COOKIES,
10
+ assembleKuaishouCookieHeader,
11
+ _internals,
12
+ } = require("../../lib/adapters/social-kuaishou-adb/cookies-extension");
13
+
14
+ describe("constants", () => {
15
+ it("path points to com.smile.gifmaker WebView cookies", () => {
16
+ expect(KUAISHOU_COOKIES_REMOTE_PATH).toBe(
17
+ "/data/data/com.smile.gifmaker/app_webview/Default/Cookies",
18
+ );
19
+ });
20
+
21
+ it("host domain is kuaishou.com", () => {
22
+ expect(KUAISHOU_COOKIE_HOST_DOMAIN).toBe("kuaishou.com");
23
+ });
24
+
25
+ it("login cookies enumerated (userId OR kuaishou.web.cp.api_ph)", () => {
26
+ expect([...KUAISHOU_LOGIN_COOKIES]).toEqual([
27
+ "userId",
28
+ "kuaishou.web.cp.api_ph",
29
+ ]);
30
+ });
31
+ });
32
+
33
+ describe("pickUidFromCookieMap", () => {
34
+ const make = (entries) =>
35
+ new Map(entries.map(([k, v]) => [k, { value: v }]));
36
+
37
+ it("prefers direct userId cookie", () => {
38
+ const map = make([
39
+ ["userId", "12345"],
40
+ ["kuaishou.web.cp.api_ph", encodeURIComponent('{"user_id":99999}')],
41
+ ]);
42
+ expect(_internals.pickUidFromCookieMap(map)).toBe("12345");
43
+ });
44
+
45
+ it("falls back to api_ph nested user_id", () => {
46
+ const map = make([
47
+ ["kuaishou.web.cp.api_ph", encodeURIComponent('{"user_id":"55555"}')],
48
+ ]);
49
+ expect(_internals.pickUidFromCookieMap(map)).toBe("55555");
50
+ });
51
+
52
+ it("falls back to api_ph nested uid alternate", () => {
53
+ const map = make([
54
+ ["kuaishou.web.cp.api_ph", encodeURIComponent('{"uid":"77777"}')],
55
+ ]);
56
+ expect(_internals.pickUidFromCookieMap(map)).toBe("77777");
57
+ });
58
+
59
+ it("handles api_ph not URL-encoded (raw)", () => {
60
+ const map = make([["kuaishou.web.cp.api_ph", '{"user_id":"33333"}']]);
61
+ expect(_internals.pickUidFromCookieMap(map)).toBe("33333");
62
+ });
63
+
64
+ it("returns null for userId=0 (guest sentinel)", () => {
65
+ const map = make([["userId", "0"]]);
66
+ expect(_internals.pickUidFromCookieMap(map)).toBe(null);
67
+ });
68
+
69
+ it("returns null when only anti-bot cookies present", () => {
70
+ const map = make([["did", "anonid"]]);
71
+ expect(_internals.pickUidFromCookieMap(map)).toBe(null);
72
+ });
73
+ });
74
+
75
+ describe("assembleKuaishouCookieHeader", () => {
76
+ const mkCookie = (name, value, hostKey = ".kuaishou.com") => ({
77
+ name,
78
+ value,
79
+ hostKey,
80
+ });
81
+
82
+ it("builds header from direct userId cookie", () => {
83
+ const cookies = [
84
+ mkCookie("userId", "12345"),
85
+ mkCookie("did", "anon"),
86
+ ];
87
+ const r = assembleKuaishouCookieHeader(cookies);
88
+ expect(r.header).toContain("userId=12345");
89
+ expect(r.uid).toBe("12345");
90
+ expect(r.missing).toEqual([]);
91
+ });
92
+
93
+ it("builds header from api_ph alone (no direct userId)", () => {
94
+ const cookies = [
95
+ mkCookie(
96
+ "kuaishou.web.cp.api_ph",
97
+ encodeURIComponent('{"user_id":"55555","user_name":"Alice"}'),
98
+ ),
99
+ ];
100
+ const r = assembleKuaishouCookieHeader(cookies);
101
+ expect(r.header).toContain("kuaishou.web.cp.api_ph=");
102
+ expect(r.uid).toBe("55555");
103
+ expect(r.missing).toEqual([]);
104
+ });
105
+
106
+ it("returns null header when no login cookie present", () => {
107
+ const cookies = [mkCookie("did", "anon"), mkCookie("ttwid", "x")];
108
+ const r = assembleKuaishouCookieHeader(cookies);
109
+ expect(r.header).toBe(null);
110
+ expect(r.uid).toBe(null);
111
+ expect(r.missing).toEqual(["userId", "kuaishou.web.cp.api_ph"]);
112
+ });
113
+
114
+ it("dedupes by longest hostKey when same name appears twice", () => {
115
+ const cookies = [
116
+ mkCookie("userId", "subdomain-val", "www.kuaishou.com"),
117
+ mkCookie("userId", "wildcard-val", ".kuaishou.com"),
118
+ ];
119
+ const r = assembleKuaishouCookieHeader(cookies);
120
+ expect(r.header).toContain("userId=subdomain-val");
121
+ });
122
+
123
+ it("throws TypeError on non-array input", () => {
124
+ expect(() => assembleKuaishouCookieHeader(null)).toThrow(TypeError);
125
+ expect(() => assembleKuaishouCookieHeader("x")).toThrow(TypeError);
126
+ });
127
+ });
128
+
129
+ describe("createKuaishouCookiesExtension", () => {
130
+ it("rejects when ctx missing required functions", async () => {
131
+ const ext = createKuaishouCookiesExtension();
132
+ await expect(ext({}, {})).rejects.toThrow(/ctx must provide/);
133
+ });
134
+
135
+ it("rejects when adb is not a function", async () => {
136
+ const ext = createKuaishouCookiesExtension();
137
+ await expect(
138
+ ext({}, { adb: null, pickDevice: () => "serial" }),
139
+ ).rejects.toThrow(/ctx must provide/);
140
+ });
141
+ });
@@ -0,0 +1,178 @@
1
+ "use strict";
2
+
3
+ import { describe, it, expect } from "vitest";
4
+ const fs = require("node:fs");
5
+ const path = require("node:path");
6
+ const os = require("node:os");
7
+
8
+ const {
9
+ buildSnapshot,
10
+ writeSnapshotJson,
11
+ cleanupSnapshotJson,
12
+ SNAPSHOT_SCHEMA_VERSION,
13
+ } = require("../../lib/adapters/social-kuaishou-adb/snapshot-builder");
14
+
15
+ describe("SNAPSHOT_SCHEMA_VERSION", () => {
16
+ it("is 1 (matches existing social-kuaishou adapter)", () => {
17
+ expect(SNAPSHOT_SCHEMA_VERSION).toBe(1);
18
+ });
19
+ });
20
+
21
+ describe("buildSnapshot", () => {
22
+ it("throws on missing uid", () => {
23
+ expect(() => buildSnapshot({})).toThrow(/uid must be a non-empty string/);
24
+ });
25
+
26
+ it("emits profile event from input.profile", () => {
27
+ const snap = buildSnapshot({
28
+ uid: "12345",
29
+ profile: {
30
+ nickname: "Alice",
31
+ kuaishouId: "alice_ks",
32
+ avatarUrl: "https://a/x.jpg",
33
+ sex: "F",
34
+ city: "Beijing",
35
+ },
36
+ snapshottedAt: 1716383021000,
37
+ });
38
+ expect(snap.schemaVersion).toBe(1);
39
+ expect(snap.account).toEqual({ uid: "12345", displayName: "" });
40
+ const prof = snap.events.find((e) => e.kind === "profile");
41
+ expect(prof).toMatchObject({
42
+ kind: "profile",
43
+ id: "profile-12345",
44
+ uid: "12345",
45
+ nickname: "Alice",
46
+ kuaishouId: "alice_ks",
47
+ city: "Beijing",
48
+ });
49
+ });
50
+
51
+ it("emits watch events with photo metadata", () => {
52
+ const snap = buildSnapshot({
53
+ uid: "1",
54
+ watch: [
55
+ {
56
+ photoId: "P1",
57
+ caption: "Funny vid",
58
+ duration: 30,
59
+ authorId: "A1",
60
+ authorName: "Auth1",
61
+ viewedAt: 1716000000000,
62
+ },
63
+ ],
64
+ });
65
+ const w = snap.events.filter((e) => e.kind === "watch");
66
+ expect(w).toHaveLength(1);
67
+ expect(w[0]).toMatchObject({
68
+ kind: "watch",
69
+ id: "photo-P1",
70
+ capturedAt: 1716000000000,
71
+ photoId: "P1",
72
+ caption: "Funny vid",
73
+ duration: 30,
74
+ authorId: "A1",
75
+ authorName: "Auth1",
76
+ });
77
+ });
78
+
79
+ it("emits collect events with self as author", () => {
80
+ const snap = buildSnapshot({
81
+ uid: "12345",
82
+ displayName: "Alice",
83
+ collect: [{ photoId: "C1", caption: "My vid", postedAt: 1716100000000 }],
84
+ });
85
+ const c = snap.events.filter((e) => e.kind === "collect");
86
+ expect(c).toHaveLength(1);
87
+ expect(c[0]).toMatchObject({
88
+ kind: "collect",
89
+ id: "collect-C1",
90
+ capturedAt: 1716100000000,
91
+ photoId: "C1",
92
+ caption: "My vid",
93
+ authorId: "12345", // self
94
+ authorName: "Alice",
95
+ });
96
+ });
97
+
98
+ it("emits search events with keyword:ts id", () => {
99
+ const snap = buildSnapshot({
100
+ uid: "1",
101
+ search: [{ keyword: "AI", searchedAt: 1716200000000 }],
102
+ });
103
+ const s = snap.events.find((e) => e.kind === "search");
104
+ expect(s).toMatchObject({
105
+ kind: "search",
106
+ id: "search-AI:1716200000000",
107
+ capturedAt: 1716200000000,
108
+ keyword: "AI",
109
+ searchAt: 1716200000000,
110
+ });
111
+ });
112
+
113
+ it("ignores entries missing photoId / keyword", () => {
114
+ const snap = buildSnapshot({
115
+ uid: "1",
116
+ watch: [{ caption: "no photoId" }, { photoId: "OK" }],
117
+ collect: [{ caption: "no photoId" }],
118
+ search: [{ keyword: "" }, { keyword: "good" }],
119
+ });
120
+ expect(snap.events.filter((e) => e.kind === "watch")).toHaveLength(1);
121
+ expect(snap.events.filter((e) => e.kind === "collect")).toHaveLength(0);
122
+ expect(snap.events.filter((e) => e.kind === "search")).toHaveLength(1);
123
+ });
124
+
125
+ it("falls back to snapshottedAt when event lacks timestamp", () => {
126
+ const snap = buildSnapshot({
127
+ uid: "1",
128
+ watch: [{ photoId: "P" }],
129
+ snapshottedAt: 1716000000000,
130
+ });
131
+ expect(snap.events[0].capturedAt).toBe(1716000000000);
132
+ });
133
+
134
+ it("emits all 4 kinds at once", () => {
135
+ const snap = buildSnapshot({
136
+ uid: "12345",
137
+ profile: { nickname: "Alice" },
138
+ watch: [{ photoId: "W1" }],
139
+ collect: [{ photoId: "C1" }],
140
+ search: [{ keyword: "kw1", searchedAt: 1 }],
141
+ });
142
+ expect(snap.events).toHaveLength(4);
143
+ expect(snap.events.map((e) => e.kind).sort()).toEqual([
144
+ "collect",
145
+ "profile",
146
+ "search",
147
+ "watch",
148
+ ]);
149
+ });
150
+ });
151
+
152
+ describe("writeSnapshotJson + cleanupSnapshotJson", () => {
153
+ it("writes valid JSON", () => {
154
+ const snap = buildSnapshot({ uid: "1", profile: { nickname: "A" } });
155
+ const full = writeSnapshotJson(snap);
156
+ try {
157
+ expect(fs.existsSync(full)).toBe(true);
158
+ const round = JSON.parse(fs.readFileSync(full, "utf-8"));
159
+ expect(round.account.uid).toBe("1");
160
+ } finally {
161
+ cleanupSnapshotJson(full);
162
+ }
163
+ });
164
+
165
+ it("rejects fileName with path separator", () => {
166
+ const snap = buildSnapshot({ uid: "1" });
167
+ expect(() =>
168
+ writeSnapshotJson(snap, { fileName: "../escape.json" }),
169
+ ).toThrow(/basename, not a path/);
170
+ });
171
+
172
+ it("cleanup tolerates missing file", () => {
173
+ expect(() => cleanupSnapshotJson(null)).not.toThrow();
174
+ expect(() =>
175
+ cleanupSnapshotJson(path.join(os.tmpdir(), "nonexistent-ks.json")),
176
+ ).not.toThrow();
177
+ });
178
+ });