@neetru/sdk 1.2.0 → 2.1.0
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/CHANGELOG.md +284 -244
- package/README.md +194 -194
- package/dist/auth.cjs +3740 -345
- package/dist/auth.cjs.map +1 -1
- package/dist/auth.d.cts +5 -1
- package/dist/auth.d.ts +5 -1
- package/dist/auth.mjs +3740 -345
- package/dist/auth.mjs.map +1 -1
- package/dist/catalog.cjs.map +1 -1
- package/dist/catalog.d.cts +5 -1
- package/dist/catalog.d.ts +5 -1
- package/dist/catalog.mjs.map +1 -1
- package/dist/checkout.cjs.map +1 -1
- package/dist/checkout.d.cts +5 -1
- package/dist/checkout.d.ts +5 -1
- package/dist/checkout.mjs.map +1 -1
- package/dist/collection-ref-BBvTTXoG.d.cts +423 -0
- package/dist/collection-ref-BBvTTXoG.d.ts +423 -0
- package/dist/db-react.cjs +136 -0
- package/dist/db-react.cjs.map +1 -0
- package/dist/db-react.d.cts +99 -0
- package/dist/db-react.d.ts +99 -0
- package/dist/db-react.mjs +112 -0
- package/dist/db-react.mjs.map +1 -0
- package/dist/db.cjs +3599 -131
- package/dist/db.cjs.map +1 -1
- package/dist/db.d.cts +5 -8
- package/dist/db.d.ts +5 -8
- package/dist/db.mjs +3596 -131
- package/dist/db.mjs.map +1 -1
- package/dist/entitlements.cjs.map +1 -1
- package/dist/entitlements.d.cts +5 -1
- package/dist/entitlements.d.ts +5 -1
- package/dist/entitlements.mjs.map +1 -1
- package/dist/errors.cjs.map +1 -1
- package/dist/errors.mjs.map +1 -1
- package/dist/index.cjs +3957 -342
- package/dist/index.cjs.map +1 -1
- package/dist/index.d.cts +13 -6
- package/dist/index.d.ts +13 -6
- package/dist/index.mjs +3877 -263
- package/dist/index.mjs.map +1 -1
- package/dist/mocks.cjs +183 -7
- package/dist/mocks.cjs.map +1 -1
- package/dist/mocks.d.cts +18 -5
- package/dist/mocks.d.ts +18 -5
- package/dist/mocks.mjs +183 -7
- package/dist/mocks.mjs.map +1 -1
- package/dist/notifications.cjs.map +1 -1
- package/dist/notifications.d.cts +5 -1
- package/dist/notifications.d.ts +5 -1
- package/dist/notifications.mjs.map +1 -1
- package/dist/react.cjs.map +1 -1
- package/dist/react.d.cts +5 -1
- package/dist/react.d.ts +5 -1
- package/dist/react.mjs.map +1 -1
- package/dist/support.cjs.map +1 -1
- package/dist/support.d.cts +5 -1
- package/dist/support.d.ts +5 -1
- package/dist/support.mjs.map +1 -1
- package/dist/telemetry.cjs.map +1 -1
- package/dist/telemetry.d.cts +5 -1
- package/dist/telemetry.d.ts +5 -1
- package/dist/telemetry.mjs.map +1 -1
- package/dist/types-B1jylbMC.d.ts +1364 -0
- package/dist/types-Kmt4y1FQ.d.cts +1364 -0
- package/dist/usage.cjs.map +1 -1
- package/dist/usage.d.cts +5 -1
- package/dist/usage.d.ts +5 -1
- package/dist/usage.mjs.map +1 -1
- package/dist/webhooks.cjs.map +1 -1
- package/dist/webhooks.d.cts +5 -1
- package/dist/webhooks.d.ts +5 -1
- package/dist/webhooks.mjs.map +1 -1
- package/package.json +133 -111
- package/dist/types-CQAfwqUS.d.cts +0 -654
- package/dist/types-CQAfwqUS.d.ts +0 -654
package/dist/mocks.cjs
CHANGED
|
@@ -1,5 +1,49 @@
|
|
|
1
1
|
'use strict';
|
|
2
2
|
|
|
3
|
+
// src/errors.ts
|
|
4
|
+
var NeetruError = class _NeetruError extends Error {
|
|
5
|
+
code;
|
|
6
|
+
status;
|
|
7
|
+
requestId;
|
|
8
|
+
constructor(code, message, status, requestId) {
|
|
9
|
+
super(message);
|
|
10
|
+
this.name = "NeetruError";
|
|
11
|
+
this.code = code;
|
|
12
|
+
this.status = status;
|
|
13
|
+
this.requestId = requestId;
|
|
14
|
+
Object.setPrototypeOf(this, _NeetruError.prototype);
|
|
15
|
+
}
|
|
16
|
+
};
|
|
17
|
+
|
|
18
|
+
// src/db-errors.ts
|
|
19
|
+
var RETRYABLE_CODES = /* @__PURE__ */ new Set([
|
|
20
|
+
"db_unavailable",
|
|
21
|
+
"db_conflict",
|
|
22
|
+
"db_timeout"
|
|
23
|
+
]);
|
|
24
|
+
var NeetruDbError = class _NeetruDbError extends NeetruError {
|
|
25
|
+
/** Código de erro fechado — específico de DB. */
|
|
26
|
+
code;
|
|
27
|
+
/**
|
|
28
|
+
* `true` para erros transientes que o produto pode tentar novamente.
|
|
29
|
+
* São retryable: `db_unavailable`, `db_conflict`, `db_timeout`.
|
|
30
|
+
*/
|
|
31
|
+
retryable;
|
|
32
|
+
/**
|
|
33
|
+
* ID opaco do banco lógico — só para correlação com logs do Core.
|
|
34
|
+
* Nunca deve ser exibido ao usuário final.
|
|
35
|
+
*/
|
|
36
|
+
dbId;
|
|
37
|
+
constructor(code, message, dbId) {
|
|
38
|
+
super(code, message);
|
|
39
|
+
this.name = "NeetruDbError";
|
|
40
|
+
this.code = code;
|
|
41
|
+
this.retryable = RETRYABLE_CODES.has(code);
|
|
42
|
+
this.dbId = dbId;
|
|
43
|
+
Object.setPrototypeOf(this, _NeetruDbError.prototype);
|
|
44
|
+
}
|
|
45
|
+
};
|
|
46
|
+
|
|
3
47
|
// src/mocks.ts
|
|
4
48
|
var DEV_FIXTURE_USER = Object.freeze({
|
|
5
49
|
uid: "dev-fixture-uid-0001",
|
|
@@ -219,24 +263,61 @@ var MockDb = class {
|
|
|
219
263
|
return typeof v === "number" && typeof f.value === "number" && v >= f.value;
|
|
220
264
|
case "in":
|
|
221
265
|
return Array.isArray(f.value) && f.value.includes(v);
|
|
266
|
+
case "array-contains":
|
|
267
|
+
return Array.isArray(v) && v.includes(f.value);
|
|
268
|
+
case "not-in":
|
|
269
|
+
return Array.isArray(f.value) && !f.value.includes(v);
|
|
222
270
|
default:
|
|
223
271
|
return true;
|
|
224
272
|
}
|
|
225
273
|
};
|
|
226
274
|
let autoSeq = 0;
|
|
227
|
-
|
|
228
|
-
|
|
275
|
+
function buildListResult(items) {
|
|
276
|
+
return {
|
|
277
|
+
docs: items.map((d, i) => ({ id: d.id ?? String(i), data: d })),
|
|
278
|
+
nextCursor: null,
|
|
279
|
+
fromCache: false,
|
|
280
|
+
stale: false,
|
|
281
|
+
hasPendingWrites: false,
|
|
282
|
+
changes: []
|
|
283
|
+
};
|
|
284
|
+
}
|
|
285
|
+
function buildGetResult(data, id) {
|
|
286
|
+
return {
|
|
287
|
+
docs: [{ id, data }],
|
|
288
|
+
fromCache: false,
|
|
289
|
+
stale: false,
|
|
290
|
+
hasPendingWrites: false,
|
|
291
|
+
changes: []
|
|
292
|
+
};
|
|
293
|
+
}
|
|
294
|
+
const ref = {
|
|
295
|
+
async list(q) {
|
|
229
296
|
let items = Array.from(coll.values());
|
|
230
|
-
if (
|
|
297
|
+
if (q?.where && q.where.length > 0) {
|
|
231
298
|
items = items.filter(
|
|
232
|
-
(doc) =>
|
|
299
|
+
(doc) => q.where.every((f) => matchesFilter(doc, f))
|
|
233
300
|
);
|
|
234
301
|
}
|
|
235
|
-
if (
|
|
236
|
-
|
|
302
|
+
if (q?.orderBy) {
|
|
303
|
+
const { field, direction } = q.orderBy;
|
|
304
|
+
items = items.slice().sort((a, b) => {
|
|
305
|
+
const av = a[field];
|
|
306
|
+
const bv = b[field];
|
|
307
|
+
const avs = av ?? "";
|
|
308
|
+
const bvs = bv ?? "";
|
|
309
|
+
const cmp = avs < bvs ? -1 : avs > bvs ? 1 : 0;
|
|
310
|
+
return direction === "desc" ? -cmp : cmp;
|
|
311
|
+
});
|
|
312
|
+
}
|
|
313
|
+
const limit = q?.limit ?? 20;
|
|
314
|
+
items = items.slice(0, limit);
|
|
315
|
+
return buildListResult(items);
|
|
237
316
|
},
|
|
238
317
|
async get(id) {
|
|
239
|
-
|
|
318
|
+
const data = coll.get(id);
|
|
319
|
+
if (!data) return null;
|
|
320
|
+
return buildGetResult(data, id);
|
|
240
321
|
},
|
|
241
322
|
async add(data) {
|
|
242
323
|
const id = `mock-${++autoSeq}-${Math.random().toString(36).slice(2, 8)}`;
|
|
@@ -259,8 +340,103 @@ var MockDb = class {
|
|
|
259
340
|
async remove(id) {
|
|
260
341
|
coll.delete(id);
|
|
261
342
|
return { ok: true };
|
|
343
|
+
},
|
|
344
|
+
async batch(ops) {
|
|
345
|
+
for (const op of ops) {
|
|
346
|
+
const collMap = (() => {
|
|
347
|
+
if (!_store.has(op.collection)) _store.set(op.collection, /* @__PURE__ */ new Map());
|
|
348
|
+
return _store.get(op.collection);
|
|
349
|
+
})();
|
|
350
|
+
const id = op.id ?? `mock-batch-${++autoSeq}`;
|
|
351
|
+
if (op.op === "add") {
|
|
352
|
+
collMap.set(id, { ...op.data ?? {}, id });
|
|
353
|
+
} else if (op.op === "set") {
|
|
354
|
+
collMap.set(id, { ...op.data ?? {}, id });
|
|
355
|
+
} else if (op.op === "update") {
|
|
356
|
+
const cur = collMap.get(id);
|
|
357
|
+
collMap.set(id, { ...cur ?? {}, ...op.data ?? {}, id });
|
|
358
|
+
} else if (op.op === "remove") {
|
|
359
|
+
collMap.delete(id);
|
|
360
|
+
}
|
|
361
|
+
}
|
|
362
|
+
return { ok: true };
|
|
363
|
+
},
|
|
364
|
+
onDoc(id, cb) {
|
|
365
|
+
const data = coll.get(id);
|
|
366
|
+
Promise.resolve().then(() => cb(data ? data : null));
|
|
367
|
+
return () => {
|
|
368
|
+
};
|
|
369
|
+
},
|
|
370
|
+
onSnapshot(q, cb) {
|
|
371
|
+
Promise.resolve().then(async () => {
|
|
372
|
+
const snap = await ref.list(q);
|
|
373
|
+
cb(snap);
|
|
374
|
+
});
|
|
375
|
+
return () => {
|
|
376
|
+
};
|
|
377
|
+
},
|
|
378
|
+
doc(id) {
|
|
379
|
+
return {
|
|
380
|
+
async get() {
|
|
381
|
+
const data = coll.get(id);
|
|
382
|
+
if (!data) return null;
|
|
383
|
+
return buildGetResult(data, id);
|
|
384
|
+
},
|
|
385
|
+
async set(data) {
|
|
386
|
+
coll.set(id, { ...data, id });
|
|
387
|
+
return { ok: true };
|
|
388
|
+
},
|
|
389
|
+
async update(data) {
|
|
390
|
+
const cur = coll.get(id);
|
|
391
|
+
if (!cur) {
|
|
392
|
+
coll.set(id, { ...data, id });
|
|
393
|
+
} else {
|
|
394
|
+
coll.set(id, { ...cur, ...data });
|
|
395
|
+
}
|
|
396
|
+
return { ok: true };
|
|
397
|
+
},
|
|
398
|
+
async remove() {
|
|
399
|
+
coll.delete(id);
|
|
400
|
+
return { ok: true };
|
|
401
|
+
},
|
|
402
|
+
onSnapshot(cb) {
|
|
403
|
+
const data = coll.get(id);
|
|
404
|
+
Promise.resolve().then(
|
|
405
|
+
() => cb(data ? buildGetResult(data, id) : null)
|
|
406
|
+
);
|
|
407
|
+
return () => {
|
|
408
|
+
};
|
|
409
|
+
}
|
|
410
|
+
};
|
|
262
411
|
}
|
|
263
412
|
};
|
|
413
|
+
return ref;
|
|
414
|
+
}
|
|
415
|
+
async sql(_schema) {
|
|
416
|
+
throw new NeetruDbError(
|
|
417
|
+
"db_unavailable",
|
|
418
|
+
'[MockDb] sql() n\xE3o dispon\xEDvel no mock. Use createNeetruClient({ env: "dev" }) e `neetru dev` para o banco local.'
|
|
419
|
+
);
|
|
420
|
+
}
|
|
421
|
+
get syncState() {
|
|
422
|
+
return {
|
|
423
|
+
status: "idle",
|
|
424
|
+
pendingWrites: 0,
|
|
425
|
+
lastSyncedAt: null,
|
|
426
|
+
isLeaderTab: true
|
|
427
|
+
};
|
|
428
|
+
}
|
|
429
|
+
onSyncStateChanged(_cb) {
|
|
430
|
+
return () => {
|
|
431
|
+
};
|
|
432
|
+
}
|
|
433
|
+
async flush() {
|
|
434
|
+
}
|
|
435
|
+
async clearCache() {
|
|
436
|
+
this._store.clear();
|
|
437
|
+
}
|
|
438
|
+
async getConflicts() {
|
|
439
|
+
return [];
|
|
264
440
|
}
|
|
265
441
|
/** Test helper — substitui fixture inteira de uma collection. */
|
|
266
442
|
__setFixture(name, items) {
|
package/dist/mocks.cjs.map
CHANGED
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"sources":["../src/mocks.ts"],"names":[],"mappings":"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","file":"mocks.cjs","sourcesContent":["/**\n * Mocks determinísticos do SDK — overridáveis em tests do consumer ou\n * ativados automaticamente quando `NEETRU_ENV=dev`.\n *\n * Nada aqui faz I/O. Tudo é puro/sincrono/in-memory. Compat total com\n * runtimes browser/Node/Edge — sem `node:` builtins.\n *\n * Cada mock implementa o mesmo namespace interface das implementações reais\n * (`AuthNamespace`, `UsageNamespace`, `SupportNamespace`) — caller pode\n * substituir 1:1 sem branching.\n *\n * @example\n * ```ts\n * import { createNeetruClient, MockAuth } from '@neetru/sdk';\n *\n * const client = createNeetruClient({\n * mocks: { auth: new MockAuth({ uid: 'fake-user', email: 'x@x.com' }) },\n * });\n * ```\n */\n\nimport type {\n AuthNamespace,\n AuthStateListener,\n CreateTicketInput,\n DbCollectionRef,\n DbListOptions,\n DbNamespace,\n DbWhereFilter,\n EntitlementCheck,\n NeetruUser,\n SignInOptions,\n SupportNamespace,\n SupportTicket,\n UsageNamespace,\n UsageQuota,\n} from './types';\n\n// ─── User fixture default ──────────────────────────────────────────────────\n\n/**\n * User fixture determinístico — usado quando `NEETRU_ENV=dev` e nenhum mock\n * custom é injetado. Email/uid estáveis pra snapshot tests.\n */\nexport const DEV_FIXTURE_USER: NeetruUser = Object.freeze({\n uid: 'dev-fixture-uid-0001',\n email: 'dev@neetru.local',\n emailVerified: true,\n displayName: 'Dev Fixture',\n isCustomer: true,\n isStaff: false,\n}) as NeetruUser;\n\n// ─── MockAuth ──────────────────────────────────────────────────────────────\n\n/**\n * Implementação in-memory do `AuthNamespace`. signIn marca o user como\n * logado; signOut limpa. onAuthStateChanged dispara callback síncrono no\n * subscribe + a cada mudança.\n */\nexport class MockAuth implements AuthNamespace {\n private _user: NeetruUser | null;\n private _listeners: Set<AuthStateListener>;\n\n constructor(initialUser: NeetruUser | null = null) {\n this._user = initialUser;\n this._listeners = new Set();\n }\n\n async signIn(_options?: SignInOptions): Promise<NeetruUser> {\n // Mock: apenas marca user como logado (usa fixture se não havia).\n if (!this._user) this._user = { ...DEV_FIXTURE_USER };\n this._notify();\n return this._user;\n }\n\n async signOut(): Promise<void> {\n this._user = null;\n this._notify();\n }\n\n getUser(): NeetruUser | null {\n return this._user;\n }\n\n onAuthStateChanged(listener: AuthStateListener): () => void {\n this._listeners.add(listener);\n // Dispara imediatamente com estado atual (espelha API Firebase Auth).\n try {\n listener(this._user);\n } catch {\n // Callback do consumer não pode quebrar o subscribe.\n }\n return () => {\n this._listeners.delete(listener);\n };\n }\n\n /** Helper de tests — força um user state arbitrário. */\n __setUser(user: NeetruUser | null): void {\n this._user = user;\n this._notify();\n }\n\n private _notify(): void {\n for (const l of this._listeners) {\n try {\n l(this._user);\n } catch {\n // ignore consumer errors\n }\n }\n }\n}\n\n// ─── MockUsage ─────────────────────────────────────────────────────────────\n\ninterface UsageRecord {\n event: string;\n properties?: Record<string, string | number | boolean | null>;\n timestamp: string;\n}\n\n/**\n * In-memory `UsageNamespace`. `track()` apenas acumula no array\n * `__getRecords()` (útil pra asserts em test). `getQuota()` retorna fixture\n * com `used = trackedCount` da metric.\n */\nexport class MockUsage implements UsageNamespace {\n private _records: UsageRecord[] = [];\n private _quotas: Map<string, UsageQuota>;\n /** v0.3 — counters in-memory pra `report()` / `check()`. */\n private _counters: Map<string, number> = new Map();\n\n constructor(initialQuotas: Record<string, UsageQuota> = {}) {\n this._quotas = new Map(Object.entries(initialQuotas));\n }\n\n async track(\n event: string,\n properties?: Record<string, string | number | boolean | null>,\n ): Promise<{ ok: true }> {\n this._records.push({\n event,\n properties,\n timestamp: new Date().toISOString(),\n });\n return { ok: true };\n }\n\n async getQuota(metric: string): Promise<UsageQuota> {\n const existing = this._quotas.get(metric);\n if (existing) return existing;\n return {\n metric,\n used: this._records.filter((r) => r.event === metric).length,\n limit: -1, // unlimited em mock por default\n plan: 'mock',\n };\n }\n\n /** v0.3 — Mock report incrementa o counter local. */\n async report(\n resource: string,\n qty: number = 1,\n _options?: { productId?: string; tenantId?: string },\n ): Promise<{\n ok: true;\n counterId?: string;\n value?: number;\n limit?: number;\n remaining?: number;\n status?: string;\n }> {\n if (!resource) throw new Error('resource required');\n const safeQty = Number.isFinite(qty) && qty > 0 ? Math.floor(qty) : 1;\n const existing = this._counters.get(resource) ?? 0;\n const next = existing + safeQty;\n this._counters.set(resource, next);\n const limitFixture = this._quotas.get(resource);\n const limit = limitFixture?.limit ?? -1;\n return {\n ok: true,\n counterId: `mock_${resource}`,\n value: next,\n limit,\n remaining: limit < 0 ? -1 : Math.max(0, limit - next),\n status: limit > 0 && next >= limit ? 'read_only_overlimit' : 'ok',\n };\n }\n\n /** v0.3 — Mock check usa quotas + counters in-memory. */\n async check(\n resource: string,\n _options?: { productId?: string; tenantId?: string },\n ): Promise<{\n allowed: boolean;\n reason?: string;\n remaining?: number;\n limit?: number;\n planId?: string | null;\n planFeatures?: string[];\n }> {\n const quota = this._quotas.get(resource);\n const used = this._counters.get(resource) ?? 0;\n if (!quota) {\n return { allowed: true, reason: 'granted', limit: -1, remaining: -1 };\n }\n const remaining = quota.limit < 0 ? -1 : Math.max(0, quota.limit - used);\n if (remaining === 0) {\n return {\n allowed: false,\n reason: 'limit_exceeded',\n limit: quota.limit,\n remaining: 0,\n };\n }\n return {\n allowed: true,\n reason: 'granted',\n limit: quota.limit,\n remaining,\n };\n }\n\n /** Test helper. */\n __getRecords(): readonly UsageRecord[] {\n return [...this._records];\n }\n /** Test helper — substitui quota fixture. */\n __setQuota(metric: string, quota: UsageQuota): void {\n this._quotas.set(metric, quota);\n }\n /** Test helper — limpa estado. */\n __reset(): void {\n this._records = [];\n this._quotas.clear();\n this._counters.clear();\n }\n}\n\n// ─── MockSupport ───────────────────────────────────────────────────────────\n\n/**\n * In-memory `SupportNamespace`. createTicket gera ID `mock-ticket-{n}`;\n * listMyTickets retorna todos criados (FIFO). Sequência por instância\n * (evita poluição cross-test do estado de módulo).\n */\nexport class MockSupport implements SupportNamespace {\n private _tickets: SupportTicket[] = [];\n private _ticketSeq = 0;\n\n async createTicket(input: CreateTicketInput): Promise<SupportTicket> {\n if (!input?.subject) throw new Error('subject required');\n if (!input?.message) throw new Error('message required');\n const ticket: SupportTicket = {\n id: `mock-ticket-${++this._ticketSeq}`,\n subject: input.subject,\n message: input.message,\n severity: input.severity ?? 'normal',\n status: 'open',\n createdAt: new Date().toISOString(),\n productSlug: input.productSlug,\n };\n this._tickets.push(ticket);\n return ticket;\n }\n\n async listMyTickets(): Promise<SupportTicket[]> {\n return [...this._tickets];\n }\n\n /** Test helper. */\n __reset(): void {\n this._tickets = [];\n this._ticketSeq = 0;\n }\n}\n\n// ─── MockEntitlements ──────────────────────────────────────────────────────\n\n/**\n * In-memory entitlements. Por default `allowed=true` em tudo (dev convém).\n * Use `__deny(slug, feature)` em tests pra simular bloqueio.\n */\nexport class MockEntitlements {\n private _denied: Set<string> = new Set();\n\n async check(productSlug: string, feature: string): Promise<boolean> {\n return !this._denied.has(`${productSlug}:${feature}`);\n }\n\n async checkDetailed(productSlug: string, feature: string): Promise<EntitlementCheck> {\n const allowed = await this.check(productSlug, feature);\n return {\n allowed,\n productSlug,\n feature,\n reason: allowed ? 'granted' : 'mock_denied',\n };\n }\n\n /** Test helper. */\n __deny(productSlug: string, feature: string): void {\n this._denied.add(`${productSlug}:${feature}`);\n }\n /** Test helper. */\n __reset(): void {\n this._denied.clear();\n }\n}\n\n// ─── MockDb (v0.3) ─────────────────────────────────────────────────────────\n\n/**\n * In-memory `DbNamespace`. Cada `collection(name)` aponta para um Map; ops\n * são totalmente síncronas (mas API await-friendly).\n *\n * Útil para testar produtos consumidores sem rodar Core REST.\n */\nexport class MockDb implements DbNamespace {\n private _store: Map<string, Map<string, Record<string, unknown>>> = new Map();\n private _fixtures: Map<string, Record<string, Record<string, unknown>>>;\n\n constructor(initialFixtures: Record<string, Record<string, Record<string, unknown>>> = {}) {\n this._fixtures = new Map(Object.entries(initialFixtures));\n }\n\n collection(name: string): DbCollectionRef {\n const _store = this._store;\n const _fixtures = this._fixtures;\n if (!_store.has(name)) {\n const init = _fixtures.get(name);\n _store.set(\n name,\n init ? new Map(Object.entries(init)) : new Map(),\n );\n }\n const coll = _store.get(name)!;\n\n const matchesFilter = (\n doc: Record<string, unknown>,\n f: DbWhereFilter,\n ): boolean => {\n const v = doc[f.field];\n switch (f.op) {\n case '==':\n return v === f.value;\n case '!=':\n return v !== f.value;\n case '<':\n return typeof v === 'number' && typeof f.value === 'number' && v < f.value;\n case '<=':\n return typeof v === 'number' && typeof f.value === 'number' && v <= f.value;\n case '>':\n return typeof v === 'number' && typeof f.value === 'number' && v > f.value;\n case '>=':\n return typeof v === 'number' && typeof f.value === 'number' && v >= f.value;\n case 'in':\n return Array.isArray(f.value) && (f.value as unknown[]).includes(v);\n default:\n return true;\n }\n };\n\n let autoSeq = 0;\n\n return {\n async list<T = Record<string, unknown>>(opts?: DbListOptions): Promise<T[]> {\n let items = Array.from(coll.values());\n if (opts?.where && opts.where.length > 0) {\n items = items.filter((doc) =>\n (opts.where as DbWhereFilter[]).every((f) => matchesFilter(doc, f)),\n );\n }\n if (opts?.limit !== undefined) items = items.slice(0, opts.limit);\n return items as T[];\n },\n async get<T = Record<string, unknown>>(id: string): Promise<T | null> {\n return (coll.get(id) as T | undefined) ?? null;\n },\n async add(data: Record<string, unknown>): Promise<{ ok: true; id: string }> {\n const id = `mock-${++autoSeq}-${Math.random().toString(36).slice(2, 8)}`;\n coll.set(id, { ...data, id });\n return { ok: true, id };\n },\n async set(id: string, data: Record<string, unknown>): Promise<{ ok: true }> {\n coll.set(id, { ...data, id });\n return { ok: true };\n },\n async update(id: string, data: Record<string, unknown>): Promise<{ ok: true }> {\n const cur = coll.get(id);\n if (!cur) {\n // mock-friendly: cria se não existir (mimic Firestore set merge).\n coll.set(id, { ...data, id });\n } else {\n coll.set(id, { ...cur, ...data });\n }\n return { ok: true };\n },\n async remove(id: string): Promise<{ ok: true }> {\n coll.delete(id);\n return { ok: true };\n },\n };\n }\n\n /** Test helper — substitui fixture inteira de uma collection. */\n __setFixture(name: string, items: Record<string, Record<string, unknown>>): void {\n this._store.set(name, new Map(Object.entries(items)));\n }\n\n /** Test helper — reset total. */\n __reset(): void {\n this._store.clear();\n }\n}\n"]}
|
|
1
|
+
{"version":3,"sources":["../src/errors.ts","../src/db-errors.ts","../src/mocks.ts"],"names":[],"mappings":"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","file":"mocks.cjs","sourcesContent":["/**\r\n * Erros tipados do SDK.\r\n *\r\n * Todo erro lançado pelo SDK estende `NeetruError` — caller pode discriminar\r\n * por `.code` (string estável) sem parsing de message.\r\n */\r\n\r\n/** Códigos de erro estáveis do SDK. Adicionar aqui requer minor bump. */\r\nexport type NeetruErrorCode =\r\n | 'invalid_config'\r\n | 'missing_api_key'\r\n | 'unauthorized'\r\n | 'forbidden'\r\n | 'not_found'\r\n | 'rate_limited'\r\n | 'validation_failed'\r\n | 'network_error'\r\n | 'invalid_response'\r\n | 'server_error'\r\n | 'unknown';\r\n\r\n/**\r\n * Erro tipado padrão do SDK. Sempre lançado em vez de Error genérico.\r\n *\r\n * @example\r\n * ```ts\r\n * try { await client.catalog.list(); }\r\n * catch (e) {\r\n * if (e instanceof NeetruError && e.code === 'rate_limited') retry();\r\n * }\r\n * ```\r\n */\r\nexport class NeetruError extends Error {\r\n public readonly code: NeetruErrorCode | string;\r\n public readonly status?: number;\r\n public readonly requestId?: string;\r\n\r\n constructor(\r\n code: NeetruErrorCode | string,\r\n message: string,\r\n status?: number,\r\n requestId?: string,\r\n ) {\r\n super(message);\r\n this.name = 'NeetruError';\r\n this.code = code;\r\n this.status = status;\r\n this.requestId = requestId;\r\n // Preserva o prototype chain ao herdar de Error (downlevel quirk de TS).\r\n Object.setPrototypeOf(this, NeetruError.prototype);\r\n }\r\n}\r\n","/**\r\n * NeetruDbError — classe de erro especializada para o módulo `db`.\r\n *\r\n * Conforma com 02-sdk.md §3.5: `NeetruDbError extends NeetruError`,\r\n * campo `code` fechado em `NeetruDbErrorCode`, campo `retryable` booleano,\r\n * campo `dbId` opaco para correlação.\r\n *\r\n * Herda de `NeetruError` para que um `catch (e) { if (e instanceof NeetruError) }`\r\n * genérico continue pegando erros de DB. Quem quer granularidade usa\r\n * `instanceof NeetruDbError` e o `code` fechado.\r\n */\r\n\r\nimport { NeetruError } from './errors';\r\n\r\n/**\r\n * Códigos de erro específicos do módulo `db`.\r\n *\r\n * - `db_unavailable` — lifecycle: provisionando / falhou / arquivado.\r\n * NÃO significa \"sem rede\" — offline é transparente.\r\n * - `db_not_found` — documento/coleção não encontrado no servidor.\r\n * - `db_conflict` — conflito de escrita detectado.\r\n * - `db_quota_exceeded` — quota de escrita/leitura excedida no banco.\r\n * - `db_permission_denied` — sem permissão para a operação.\r\n * - `db_invalid_query` — query inválida (campo, operador, tipo).\r\n * - `db_timeout` — operação excedeu o timeout.\r\n * - `offline_quota_exceeded`— IndexedDB cheio; a escrita não pode ser enfileirada.\r\n * - `offline_unavailable` — IndexedDB indisponível; SDK degrado para cache RAM.\r\n */\r\nexport type NeetruDbErrorCode =\r\n | 'db_unavailable'\r\n | 'db_not_found'\r\n | 'db_conflict'\r\n | 'db_quota_exceeded'\r\n | 'db_permission_denied'\r\n | 'db_invalid_query'\r\n | 'db_timeout'\r\n | 'offline_quota_exceeded'\r\n | 'offline_unavailable';\r\n\r\n/**\r\n * Códigos que são retryable por natureza.\r\n */\r\nconst RETRYABLE_CODES = new Set<NeetruDbErrorCode>([\r\n 'db_unavailable',\r\n 'db_conflict',\r\n 'db_timeout',\r\n]);\r\n\r\n/**\r\n * Erro tipado do módulo `db` — estende `NeetruError` para compatibilidade.\r\n *\r\n * @example\r\n * ```ts\r\n * try {\r\n * await client.db.collection('orders').add({ ... });\r\n * } catch (e) {\r\n * if (e instanceof NeetruDbError && e.code === 'offline_quota_exceeded') {\r\n * // IndexedDB cheio — notificar usuário\r\n * }\r\n * }\r\n * ```\r\n */\r\nexport class NeetruDbError extends NeetruError {\r\n /** Código de erro fechado — específico de DB. */\r\n public override readonly code: NeetruDbErrorCode;\r\n\r\n /**\r\n * `true` para erros transientes que o produto pode tentar novamente.\r\n * São retryable: `db_unavailable`, `db_conflict`, `db_timeout`.\r\n */\r\n public readonly retryable: boolean;\r\n\r\n /**\r\n * ID opaco do banco lógico — só para correlação com logs do Core.\r\n * Nunca deve ser exibido ao usuário final.\r\n */\r\n public readonly dbId?: string;\r\n\r\n constructor(code: NeetruDbErrorCode, message: string, dbId?: string) {\r\n super(code, message);\r\n this.name = 'NeetruDbError';\r\n this.code = code;\r\n this.retryable = RETRYABLE_CODES.has(code);\r\n this.dbId = dbId;\r\n // Preserva o prototype chain (TypeScript + downlevel).\r\n Object.setPrototypeOf(this, NeetruDbError.prototype);\r\n }\r\n}\r\n","/**\r\n * Mocks determinísticos do SDK — overridáveis em tests do consumer ou\r\n * ativados automaticamente quando `NEETRU_ENV=dev`.\r\n *\r\n * Nada aqui faz I/O. Tudo é puro/sincrono/in-memory. Compat total com\r\n * runtimes browser/Node/Edge — sem `node:` builtins.\r\n *\r\n * Cada mock implementa o mesmo namespace interface das implementações reais\r\n * (`AuthNamespace`, `UsageNamespace`, `SupportNamespace`) — caller pode\r\n * substituir 1:1 sem branching.\r\n *\r\n * @example\r\n * ```ts\r\n * import { createNeetruClient, MockAuth } from '@neetru/sdk';\r\n *\r\n * const client = createNeetruClient({\r\n * mocks: { auth: new MockAuth({ uid: 'fake-user', email: 'x@x.com' }) },\r\n * });\r\n * ```\r\n */\r\n\r\nimport type {\r\n AuthNamespace,\r\n AuthStateListener,\r\n CreateTicketInput,\r\n EntitlementCheck,\r\n NeetruUser,\r\n SignInOptions,\r\n SupportNamespace,\r\n SupportTicket,\r\n UsageNamespace,\r\n UsageQuota,\r\n} from './types';\r\nimport type {\r\n NeetruDb,\r\n NeetruSqlClient,\r\n} from './db/client-db';\r\nimport type { DbCollectionRef, DbWhereFilter, DbListResult, DbGetResult, DbBatchOp, DbDocRef, DbQuery } from './db/collection-ref';\r\nimport type { SyncState, ConflictRecord, Unsubscribe } from './db/offline/types';\r\nimport { NeetruDbError } from './db-errors';\r\n\r\n// ─── User fixture default ──────────────────────────────────────────────────\r\n\r\n/**\r\n * User fixture determinístico — usado quando `NEETRU_ENV=dev` e nenhum mock\r\n * custom é injetado. Email/uid estáveis pra snapshot tests.\r\n */\r\nexport const DEV_FIXTURE_USER: NeetruUser = Object.freeze({\r\n uid: 'dev-fixture-uid-0001',\r\n email: 'dev@neetru.local',\r\n emailVerified: true,\r\n displayName: 'Dev Fixture',\r\n isCustomer: true,\r\n isStaff: false,\r\n}) as NeetruUser;\r\n\r\n// ─── MockAuth ──────────────────────────────────────────────────────────────\r\n\r\n/**\r\n * Implementação in-memory do `AuthNamespace`. signIn marca o user como\r\n * logado; signOut limpa. onAuthStateChanged dispara callback síncrono no\r\n * subscribe + a cada mudança.\r\n */\r\nexport class MockAuth implements AuthNamespace {\r\n private _user: NeetruUser | null;\r\n private _listeners: Set<AuthStateListener>;\r\n\r\n constructor(initialUser: NeetruUser | null = null) {\r\n this._user = initialUser;\r\n this._listeners = new Set();\r\n }\r\n\r\n async signIn(_options?: SignInOptions): Promise<NeetruUser> {\r\n // Mock: apenas marca user como logado (usa fixture se não havia).\r\n if (!this._user) this._user = { ...DEV_FIXTURE_USER };\r\n this._notify();\r\n return this._user;\r\n }\r\n\r\n async signOut(): Promise<void> {\r\n this._user = null;\r\n this._notify();\r\n }\r\n\r\n getUser(): NeetruUser | null {\r\n return this._user;\r\n }\r\n\r\n onAuthStateChanged(listener: AuthStateListener): () => void {\r\n this._listeners.add(listener);\r\n // Dispara imediatamente com estado atual (espelha API Firebase Auth).\r\n try {\r\n listener(this._user);\r\n } catch {\r\n // Callback do consumer não pode quebrar o subscribe.\r\n }\r\n return () => {\r\n this._listeners.delete(listener);\r\n };\r\n }\r\n\r\n /** Helper de tests — força um user state arbitrário. */\r\n __setUser(user: NeetruUser | null): void {\r\n this._user = user;\r\n this._notify();\r\n }\r\n\r\n private _notify(): void {\r\n for (const l of this._listeners) {\r\n try {\r\n l(this._user);\r\n } catch {\r\n // ignore consumer errors\r\n }\r\n }\r\n }\r\n}\r\n\r\n// ─── MockUsage ─────────────────────────────────────────────────────────────\r\n\r\ninterface UsageRecord {\r\n event: string;\r\n properties?: Record<string, string | number | boolean | null>;\r\n timestamp: string;\r\n}\r\n\r\n/**\r\n * In-memory `UsageNamespace`. `track()` apenas acumula no array\r\n * `__getRecords()` (útil pra asserts em test). `getQuota()` retorna fixture\r\n * com `used = trackedCount` da metric.\r\n */\r\nexport class MockUsage implements UsageNamespace {\r\n private _records: UsageRecord[] = [];\r\n private _quotas: Map<string, UsageQuota>;\r\n /** v0.3 — counters in-memory pra `report()` / `check()`. */\r\n private _counters: Map<string, number> = new Map();\r\n\r\n constructor(initialQuotas: Record<string, UsageQuota> = {}) {\r\n this._quotas = new Map(Object.entries(initialQuotas));\r\n }\r\n\r\n async track(\r\n event: string,\r\n properties?: Record<string, string | number | boolean | null>,\r\n ): Promise<{ ok: true }> {\r\n this._records.push({\r\n event,\r\n properties,\r\n timestamp: new Date().toISOString(),\r\n });\r\n return { ok: true };\r\n }\r\n\r\n async getQuota(metric: string): Promise<UsageQuota> {\r\n const existing = this._quotas.get(metric);\r\n if (existing) return existing;\r\n return {\r\n metric,\r\n used: this._records.filter((r) => r.event === metric).length,\r\n limit: -1, // unlimited em mock por default\r\n plan: 'mock',\r\n };\r\n }\r\n\r\n /** v0.3 — Mock report incrementa o counter local. */\r\n async report(\r\n resource: string,\r\n qty: number = 1,\r\n _options?: { productId?: string; tenantId?: string },\r\n ): Promise<{\r\n ok: true;\r\n counterId?: string;\r\n value?: number;\r\n limit?: number;\r\n remaining?: number;\r\n status?: string;\r\n }> {\r\n if (!resource) throw new Error('resource required');\r\n const safeQty = Number.isFinite(qty) && qty > 0 ? Math.floor(qty) : 1;\r\n const existing = this._counters.get(resource) ?? 0;\r\n const next = existing + safeQty;\r\n this._counters.set(resource, next);\r\n const limitFixture = this._quotas.get(resource);\r\n const limit = limitFixture?.limit ?? -1;\r\n return {\r\n ok: true,\r\n counterId: `mock_${resource}`,\r\n value: next,\r\n limit,\r\n remaining: limit < 0 ? -1 : Math.max(0, limit - next),\r\n status: limit > 0 && next >= limit ? 'read_only_overlimit' : 'ok',\r\n };\r\n }\r\n\r\n /** v0.3 — Mock check usa quotas + counters in-memory. */\r\n async check(\r\n resource: string,\r\n _options?: { productId?: string; tenantId?: string },\r\n ): Promise<{\r\n allowed: boolean;\r\n reason?: string;\r\n remaining?: number;\r\n limit?: number;\r\n planId?: string | null;\r\n planFeatures?: string[];\r\n }> {\r\n const quota = this._quotas.get(resource);\r\n const used = this._counters.get(resource) ?? 0;\r\n if (!quota) {\r\n return { allowed: true, reason: 'granted', limit: -1, remaining: -1 };\r\n }\r\n const remaining = quota.limit < 0 ? -1 : Math.max(0, quota.limit - used);\r\n if (remaining === 0) {\r\n return {\r\n allowed: false,\r\n reason: 'limit_exceeded',\r\n limit: quota.limit,\r\n remaining: 0,\r\n };\r\n }\r\n return {\r\n allowed: true,\r\n reason: 'granted',\r\n limit: quota.limit,\r\n remaining,\r\n };\r\n }\r\n\r\n /** Test helper. */\r\n __getRecords(): readonly UsageRecord[] {\r\n return [...this._records];\r\n }\r\n /** Test helper — substitui quota fixture. */\r\n __setQuota(metric: string, quota: UsageQuota): void {\r\n this._quotas.set(metric, quota);\r\n }\r\n /** Test helper — limpa estado. */\r\n __reset(): void {\r\n this._records = [];\r\n this._quotas.clear();\r\n this._counters.clear();\r\n }\r\n}\r\n\r\n// ─── MockSupport ───────────────────────────────────────────────────────────\r\n\r\n/**\r\n * In-memory `SupportNamespace`. createTicket gera ID `mock-ticket-{n}`;\r\n * listMyTickets retorna todos criados (FIFO). Sequência por instância\r\n * (evita poluição cross-test do estado de módulo).\r\n */\r\nexport class MockSupport implements SupportNamespace {\r\n private _tickets: SupportTicket[] = [];\r\n private _ticketSeq = 0;\r\n\r\n async createTicket(input: CreateTicketInput): Promise<SupportTicket> {\r\n if (!input?.subject) throw new Error('subject required');\r\n if (!input?.message) throw new Error('message required');\r\n const ticket: SupportTicket = {\r\n id: `mock-ticket-${++this._ticketSeq}`,\r\n subject: input.subject,\r\n message: input.message,\r\n severity: input.severity ?? 'normal',\r\n status: 'open',\r\n createdAt: new Date().toISOString(),\r\n productSlug: input.productSlug,\r\n };\r\n this._tickets.push(ticket);\r\n return ticket;\r\n }\r\n\r\n async listMyTickets(): Promise<SupportTicket[]> {\r\n return [...this._tickets];\r\n }\r\n\r\n /** Test helper. */\r\n __reset(): void {\r\n this._tickets = [];\r\n this._ticketSeq = 0;\r\n }\r\n}\r\n\r\n// ─── MockEntitlements ──────────────────────────────────────────────────────\r\n\r\n/**\r\n * In-memory entitlements. Por default `allowed=true` em tudo (dev convém).\r\n * Use `__deny(slug, feature)` em tests pra simular bloqueio.\r\n */\r\nexport class MockEntitlements {\r\n private _denied: Set<string> = new Set();\r\n\r\n async check(productSlug: string, feature: string): Promise<boolean> {\r\n return !this._denied.has(`${productSlug}:${feature}`);\r\n }\r\n\r\n async checkDetailed(productSlug: string, feature: string): Promise<EntitlementCheck> {\r\n const allowed = await this.check(productSlug, feature);\r\n return {\r\n allowed,\r\n productSlug,\r\n feature,\r\n reason: allowed ? 'granted' : 'mock_denied',\r\n };\r\n }\r\n\r\n /** Test helper. */\r\n __deny(productSlug: string, feature: string): void {\r\n this._denied.add(`${productSlug}:${feature}`);\r\n }\r\n /** Test helper. */\r\n __reset(): void {\r\n this._denied.clear();\r\n }\r\n}\r\n\r\n// ─── MockDb (v2.0) ─────────────────────────────────────────────────────────\r\n\r\n/**\r\n * In-memory `NeetruDb`. Cada `collection(name)` aponta para um Map; ops\r\n * são totalmente síncronas (mas API await-friendly).\r\n *\r\n * v2.0: implementa a interface `NeetruDb` completa (collection, sql, syncState,\r\n * flush, clearCache, getConflicts, onSyncStateChanged).\r\n *\r\n * Útil para testar produtos consumidores sem rodar Core REST nem IndexedDB.\r\n */\r\nexport class MockDb implements NeetruDb {\r\n private _store: Map<string, Map<string, Record<string, unknown>>> = new Map();\r\n private _fixtures: Map<string, Record<string, Record<string, unknown>>>;\r\n\r\n constructor(initialFixtures: Record<string, Record<string, Record<string, unknown>>> = {}) {\r\n this._fixtures = new Map(Object.entries(initialFixtures));\r\n }\r\n\r\n collection<T = Record<string, unknown>>(name: string): DbCollectionRef<T> {\r\n const _store = this._store;\r\n const _fixtures = this._fixtures;\r\n if (!_store.has(name)) {\r\n const init = _fixtures.get(name);\r\n _store.set(\r\n name,\r\n init ? new Map(Object.entries(init)) : new Map(),\r\n );\r\n }\r\n const coll = _store.get(name)!;\r\n\r\n const matchesFilter = (\r\n doc: Record<string, unknown>,\r\n f: DbWhereFilter,\r\n ): boolean => {\r\n const v = doc[f.field];\r\n switch (f.op) {\r\n case '==':\r\n return v === f.value;\r\n case '!=':\r\n return v !== f.value;\r\n case '<':\r\n return typeof v === 'number' && typeof f.value === 'number' && v < f.value;\r\n case '<=':\r\n return typeof v === 'number' && typeof f.value === 'number' && v <= f.value;\r\n case '>':\r\n return typeof v === 'number' && typeof f.value === 'number' && v > f.value;\r\n case '>=':\r\n return typeof v === 'number' && typeof f.value === 'number' && v >= f.value;\r\n case 'in':\r\n return Array.isArray(f.value) && (f.value as unknown[]).includes(v);\r\n case 'array-contains':\r\n return Array.isArray(v) && (v as unknown[]).includes(f.value);\r\n case 'not-in':\r\n return Array.isArray(f.value) && !(f.value as unknown[]).includes(v);\r\n default:\r\n return true;\r\n }\r\n };\r\n\r\n let autoSeq = 0;\r\n\r\n // Helper to build a minimal DbListResult from raw items\r\n function buildListResult<U>(items: Array<Record<string, unknown>>): DbListResult<U> {\r\n return {\r\n docs: items.map((d, i) => ({ id: (d.id as string) ?? String(i), data: d as U })),\r\n nextCursor: null,\r\n fromCache: false,\r\n stale: false,\r\n hasPendingWrites: false,\r\n changes: [],\r\n };\r\n }\r\n\r\n // Helper to build a DbGetResult for a single doc\r\n function buildGetResult<U>(data: Record<string, unknown>, id: string): DbGetResult<U> {\r\n return {\r\n docs: [{ id, data: data as U }],\r\n fromCache: false,\r\n stale: false,\r\n hasPendingWrites: false,\r\n changes: [],\r\n };\r\n }\r\n\r\n const ref: DbCollectionRef<T> = {\r\n async list(q?: DbQuery): Promise<DbListResult<T>> {\r\n let items = Array.from(coll.values());\r\n if (q?.where && q.where.length > 0) {\r\n items = items.filter((doc) =>\r\n (q.where as DbWhereFilter[]).every((f) => matchesFilter(doc, f)),\r\n );\r\n }\r\n if (q?.orderBy) {\r\n const { field, direction } = q.orderBy;\r\n items = items.slice().sort((a, b) => {\r\n const av = a[field];\r\n const bv = b[field];\r\n const avs = av ?? '';\r\n const bvs = bv ?? '';\r\n const cmp = (avs as string) < (bvs as string) ? -1 : (avs as string) > (bvs as string) ? 1 : 0;\r\n return direction === 'desc' ? -cmp : cmp;\r\n });\r\n }\r\n const limit = q?.limit ?? 20;\r\n items = items.slice(0, limit);\r\n return buildListResult<T>(items);\r\n },\r\n async get(id: string): Promise<DbGetResult<T> | null> {\r\n const data = coll.get(id);\r\n if (!data) return null;\r\n return buildGetResult<T>(data, id);\r\n },\r\n async add(data: Omit<T, 'id'>): Promise<{ ok: true; id: string }> {\r\n const id = `mock-${++autoSeq}-${Math.random().toString(36).slice(2, 8)}`;\r\n coll.set(id, { ...(data as Record<string, unknown>), id });\r\n return { ok: true, id };\r\n },\r\n async set(id: string, data: Omit<T, 'id'>): Promise<{ ok: true }> {\r\n coll.set(id, { ...(data as Record<string, unknown>), id });\r\n return { ok: true };\r\n },\r\n async update(id: string, data: Partial<Omit<T, 'id'>>): Promise<{ ok: true }> {\r\n const cur = coll.get(id);\r\n if (!cur) {\r\n coll.set(id, { ...(data as Record<string, unknown>), id });\r\n } else {\r\n coll.set(id, { ...cur, ...(data as Record<string, unknown>) });\r\n }\r\n return { ok: true };\r\n },\r\n async remove(id: string): Promise<{ ok: true }> {\r\n coll.delete(id);\r\n return { ok: true };\r\n },\r\n async batch(ops: DbBatchOp[]): Promise<{ ok: true }> {\r\n for (const op of ops) {\r\n const collMap = (() => {\r\n if (!_store.has(op.collection)) _store.set(op.collection, new Map());\r\n return _store.get(op.collection)!;\r\n })();\r\n const id = op.id ?? `mock-batch-${++autoSeq}`;\r\n if (op.op === 'add') {\r\n collMap.set(id, { ...(op.data ?? {}), id });\r\n } else if (op.op === 'set') {\r\n collMap.set(id, { ...(op.data ?? {}), id });\r\n } else if (op.op === 'update') {\r\n const cur = collMap.get(id);\r\n collMap.set(id, { ...(cur ?? {}), ...(op.data ?? {}), id });\r\n } else if (op.op === 'remove') {\r\n collMap.delete(id);\r\n }\r\n }\r\n return { ok: true };\r\n },\r\n onDoc(id: string, cb: (doc: T | null) => void): Unsubscribe {\r\n // Entrega snapshot inicial imediato (sync mock)\r\n const data = coll.get(id);\r\n Promise.resolve().then(() => cb(data ? (data as T) : null));\r\n // Mock: não há feed realtime — retorna no-op unsubscribe\r\n return () => {};\r\n },\r\n onSnapshot(q: DbQuery | undefined, cb: (snap: DbListResult<T>) => void): Unsubscribe {\r\n // Entrega snapshot inicial imediato (sync mock)\r\n Promise.resolve().then(async () => {\r\n const snap = await ref.list(q);\r\n cb(snap);\r\n });\r\n return () => {};\r\n },\r\n doc(id: string): DbDocRef<T> {\r\n return {\r\n async get() {\r\n const data = coll.get(id);\r\n if (!data) return null;\r\n return buildGetResult<T>(data, id);\r\n },\r\n async set(data: Omit<T, 'id'>) {\r\n coll.set(id, { ...(data as Record<string, unknown>), id });\r\n return { ok: true };\r\n },\r\n async update(data: Partial<Omit<T, 'id'>>) {\r\n const cur = coll.get(id);\r\n if (!cur) {\r\n coll.set(id, { ...(data as Record<string, unknown>), id });\r\n } else {\r\n coll.set(id, { ...cur, ...(data as Record<string, unknown>) });\r\n }\r\n return { ok: true };\r\n },\r\n async remove() {\r\n coll.delete(id);\r\n return { ok: true };\r\n },\r\n onSnapshot(cb: (snap: DbGetResult<T> | null) => void): Unsubscribe {\r\n const data = coll.get(id);\r\n Promise.resolve().then(() =>\r\n cb(data ? buildGetResult<T>(data, id) : null),\r\n );\r\n return () => {};\r\n },\r\n };\r\n },\r\n };\r\n\r\n return ref;\r\n }\r\n\r\n async sql<TSchema extends Record<string, unknown>>(\r\n _schema: TSchema,\r\n ): Promise<NeetruSqlClient<TSchema>> {\r\n throw new NeetruDbError(\r\n 'db_unavailable',\r\n '[MockDb] sql() não disponível no mock. Use createNeetruClient({ env: \"dev\" }) e `neetru dev` para o banco local.',\r\n );\r\n }\r\n\r\n get syncState(): SyncState {\r\n return {\r\n status: 'idle',\r\n pendingWrites: 0,\r\n lastSyncedAt: null,\r\n isLeaderTab: true,\r\n };\r\n }\r\n\r\n onSyncStateChanged(_cb: (s: SyncState) => void): Unsubscribe {\r\n return () => {};\r\n }\r\n\r\n async flush(): Promise<void> {\r\n // no-op in mock\r\n }\r\n\r\n async clearCache(): Promise<void> {\r\n this._store.clear();\r\n }\r\n\r\n async getConflicts(): Promise<ConflictRecord[]> {\r\n return [];\r\n }\r\n\r\n /** Test helper — substitui fixture inteira de uma collection. */\r\n __setFixture(name: string, items: Record<string, Record<string, unknown>>): void {\r\n this._store.set(name, new Map(Object.entries(items)));\r\n }\r\n\r\n /** Test helper — reset total. */\r\n __reset(): void {\r\n this._store.clear();\r\n }\r\n}\r\n"]}
|
package/dist/mocks.d.cts
CHANGED
|
@@ -1,4 +1,8 @@
|
|
|
1
|
-
import {
|
|
1
|
+
import { w as NeetruUser, A as AuthNamespace, H as SignInOptions, a as AuthStateListener, p as NeetruDb, v as NeetruSqlClient, E as EntitlementCheck, J as SupportNamespace, i as CreateTicketInput, Q as SupportTicket, Y as UsageNamespace, Z as UsageQuota } from './types-Kmt4y1FQ.cjs';
|
|
2
|
+
import { c as DbCollectionRef, S as SyncState, U as Unsubscribe, C as ConflictRecord } from './collection-ref-BBvTTXoG.cjs';
|
|
3
|
+
import '@neetru/realtime-protocol';
|
|
4
|
+
import 'drizzle-orm/node-postgres';
|
|
5
|
+
import './errors.cjs';
|
|
2
6
|
|
|
3
7
|
/**
|
|
4
8
|
* Mocks determinísticos do SDK — overridáveis em tests do consumer ou
|
|
@@ -121,16 +125,25 @@ declare class MockEntitlements {
|
|
|
121
125
|
__reset(): void;
|
|
122
126
|
}
|
|
123
127
|
/**
|
|
124
|
-
* In-memory `
|
|
128
|
+
* In-memory `NeetruDb`. Cada `collection(name)` aponta para um Map; ops
|
|
125
129
|
* são totalmente síncronas (mas API await-friendly).
|
|
126
130
|
*
|
|
127
|
-
*
|
|
131
|
+
* v2.0: implementa a interface `NeetruDb` completa (collection, sql, syncState,
|
|
132
|
+
* flush, clearCache, getConflicts, onSyncStateChanged).
|
|
133
|
+
*
|
|
134
|
+
* Útil para testar produtos consumidores sem rodar Core REST nem IndexedDB.
|
|
128
135
|
*/
|
|
129
|
-
declare class MockDb implements
|
|
136
|
+
declare class MockDb implements NeetruDb {
|
|
130
137
|
private _store;
|
|
131
138
|
private _fixtures;
|
|
132
139
|
constructor(initialFixtures?: Record<string, Record<string, Record<string, unknown>>>);
|
|
133
|
-
collection(name: string): DbCollectionRef
|
|
140
|
+
collection<T = Record<string, unknown>>(name: string): DbCollectionRef<T>;
|
|
141
|
+
sql<TSchema extends Record<string, unknown>>(_schema: TSchema): Promise<NeetruSqlClient<TSchema>>;
|
|
142
|
+
get syncState(): SyncState;
|
|
143
|
+
onSyncStateChanged(_cb: (s: SyncState) => void): Unsubscribe;
|
|
144
|
+
flush(): Promise<void>;
|
|
145
|
+
clearCache(): Promise<void>;
|
|
146
|
+
getConflicts(): Promise<ConflictRecord[]>;
|
|
134
147
|
/** Test helper — substitui fixture inteira de uma collection. */
|
|
135
148
|
__setFixture(name: string, items: Record<string, Record<string, unknown>>): void;
|
|
136
149
|
/** Test helper — reset total. */
|
package/dist/mocks.d.ts
CHANGED
|
@@ -1,4 +1,8 @@
|
|
|
1
|
-
import {
|
|
1
|
+
import { w as NeetruUser, A as AuthNamespace, H as SignInOptions, a as AuthStateListener, p as NeetruDb, v as NeetruSqlClient, E as EntitlementCheck, J as SupportNamespace, i as CreateTicketInput, Q as SupportTicket, Y as UsageNamespace, Z as UsageQuota } from './types-B1jylbMC.js';
|
|
2
|
+
import { c as DbCollectionRef, S as SyncState, U as Unsubscribe, C as ConflictRecord } from './collection-ref-BBvTTXoG.js';
|
|
3
|
+
import '@neetru/realtime-protocol';
|
|
4
|
+
import 'drizzle-orm/node-postgres';
|
|
5
|
+
import './errors.js';
|
|
2
6
|
|
|
3
7
|
/**
|
|
4
8
|
* Mocks determinísticos do SDK — overridáveis em tests do consumer ou
|
|
@@ -121,16 +125,25 @@ declare class MockEntitlements {
|
|
|
121
125
|
__reset(): void;
|
|
122
126
|
}
|
|
123
127
|
/**
|
|
124
|
-
* In-memory `
|
|
128
|
+
* In-memory `NeetruDb`. Cada `collection(name)` aponta para um Map; ops
|
|
125
129
|
* são totalmente síncronas (mas API await-friendly).
|
|
126
130
|
*
|
|
127
|
-
*
|
|
131
|
+
* v2.0: implementa a interface `NeetruDb` completa (collection, sql, syncState,
|
|
132
|
+
* flush, clearCache, getConflicts, onSyncStateChanged).
|
|
133
|
+
*
|
|
134
|
+
* Útil para testar produtos consumidores sem rodar Core REST nem IndexedDB.
|
|
128
135
|
*/
|
|
129
|
-
declare class MockDb implements
|
|
136
|
+
declare class MockDb implements NeetruDb {
|
|
130
137
|
private _store;
|
|
131
138
|
private _fixtures;
|
|
132
139
|
constructor(initialFixtures?: Record<string, Record<string, Record<string, unknown>>>);
|
|
133
|
-
collection(name: string): DbCollectionRef
|
|
140
|
+
collection<T = Record<string, unknown>>(name: string): DbCollectionRef<T>;
|
|
141
|
+
sql<TSchema extends Record<string, unknown>>(_schema: TSchema): Promise<NeetruSqlClient<TSchema>>;
|
|
142
|
+
get syncState(): SyncState;
|
|
143
|
+
onSyncStateChanged(_cb: (s: SyncState) => void): Unsubscribe;
|
|
144
|
+
flush(): Promise<void>;
|
|
145
|
+
clearCache(): Promise<void>;
|
|
146
|
+
getConflicts(): Promise<ConflictRecord[]>;
|
|
134
147
|
/** Test helper — substitui fixture inteira de uma collection. */
|
|
135
148
|
__setFixture(name: string, items: Record<string, Record<string, unknown>>): void;
|
|
136
149
|
/** Test helper — reset total. */
|
package/dist/mocks.mjs
CHANGED
|
@@ -1,3 +1,47 @@
|
|
|
1
|
+
// src/errors.ts
|
|
2
|
+
var NeetruError = class _NeetruError extends Error {
|
|
3
|
+
code;
|
|
4
|
+
status;
|
|
5
|
+
requestId;
|
|
6
|
+
constructor(code, message, status, requestId) {
|
|
7
|
+
super(message);
|
|
8
|
+
this.name = "NeetruError";
|
|
9
|
+
this.code = code;
|
|
10
|
+
this.status = status;
|
|
11
|
+
this.requestId = requestId;
|
|
12
|
+
Object.setPrototypeOf(this, _NeetruError.prototype);
|
|
13
|
+
}
|
|
14
|
+
};
|
|
15
|
+
|
|
16
|
+
// src/db-errors.ts
|
|
17
|
+
var RETRYABLE_CODES = /* @__PURE__ */ new Set([
|
|
18
|
+
"db_unavailable",
|
|
19
|
+
"db_conflict",
|
|
20
|
+
"db_timeout"
|
|
21
|
+
]);
|
|
22
|
+
var NeetruDbError = class _NeetruDbError extends NeetruError {
|
|
23
|
+
/** Código de erro fechado — específico de DB. */
|
|
24
|
+
code;
|
|
25
|
+
/**
|
|
26
|
+
* `true` para erros transientes que o produto pode tentar novamente.
|
|
27
|
+
* São retryable: `db_unavailable`, `db_conflict`, `db_timeout`.
|
|
28
|
+
*/
|
|
29
|
+
retryable;
|
|
30
|
+
/**
|
|
31
|
+
* ID opaco do banco lógico — só para correlação com logs do Core.
|
|
32
|
+
* Nunca deve ser exibido ao usuário final.
|
|
33
|
+
*/
|
|
34
|
+
dbId;
|
|
35
|
+
constructor(code, message, dbId) {
|
|
36
|
+
super(code, message);
|
|
37
|
+
this.name = "NeetruDbError";
|
|
38
|
+
this.code = code;
|
|
39
|
+
this.retryable = RETRYABLE_CODES.has(code);
|
|
40
|
+
this.dbId = dbId;
|
|
41
|
+
Object.setPrototypeOf(this, _NeetruDbError.prototype);
|
|
42
|
+
}
|
|
43
|
+
};
|
|
44
|
+
|
|
1
45
|
// src/mocks.ts
|
|
2
46
|
var DEV_FIXTURE_USER = Object.freeze({
|
|
3
47
|
uid: "dev-fixture-uid-0001",
|
|
@@ -217,24 +261,61 @@ var MockDb = class {
|
|
|
217
261
|
return typeof v === "number" && typeof f.value === "number" && v >= f.value;
|
|
218
262
|
case "in":
|
|
219
263
|
return Array.isArray(f.value) && f.value.includes(v);
|
|
264
|
+
case "array-contains":
|
|
265
|
+
return Array.isArray(v) && v.includes(f.value);
|
|
266
|
+
case "not-in":
|
|
267
|
+
return Array.isArray(f.value) && !f.value.includes(v);
|
|
220
268
|
default:
|
|
221
269
|
return true;
|
|
222
270
|
}
|
|
223
271
|
};
|
|
224
272
|
let autoSeq = 0;
|
|
225
|
-
|
|
226
|
-
|
|
273
|
+
function buildListResult(items) {
|
|
274
|
+
return {
|
|
275
|
+
docs: items.map((d, i) => ({ id: d.id ?? String(i), data: d })),
|
|
276
|
+
nextCursor: null,
|
|
277
|
+
fromCache: false,
|
|
278
|
+
stale: false,
|
|
279
|
+
hasPendingWrites: false,
|
|
280
|
+
changes: []
|
|
281
|
+
};
|
|
282
|
+
}
|
|
283
|
+
function buildGetResult(data, id) {
|
|
284
|
+
return {
|
|
285
|
+
docs: [{ id, data }],
|
|
286
|
+
fromCache: false,
|
|
287
|
+
stale: false,
|
|
288
|
+
hasPendingWrites: false,
|
|
289
|
+
changes: []
|
|
290
|
+
};
|
|
291
|
+
}
|
|
292
|
+
const ref = {
|
|
293
|
+
async list(q) {
|
|
227
294
|
let items = Array.from(coll.values());
|
|
228
|
-
if (
|
|
295
|
+
if (q?.where && q.where.length > 0) {
|
|
229
296
|
items = items.filter(
|
|
230
|
-
(doc) =>
|
|
297
|
+
(doc) => q.where.every((f) => matchesFilter(doc, f))
|
|
231
298
|
);
|
|
232
299
|
}
|
|
233
|
-
if (
|
|
234
|
-
|
|
300
|
+
if (q?.orderBy) {
|
|
301
|
+
const { field, direction } = q.orderBy;
|
|
302
|
+
items = items.slice().sort((a, b) => {
|
|
303
|
+
const av = a[field];
|
|
304
|
+
const bv = b[field];
|
|
305
|
+
const avs = av ?? "";
|
|
306
|
+
const bvs = bv ?? "";
|
|
307
|
+
const cmp = avs < bvs ? -1 : avs > bvs ? 1 : 0;
|
|
308
|
+
return direction === "desc" ? -cmp : cmp;
|
|
309
|
+
});
|
|
310
|
+
}
|
|
311
|
+
const limit = q?.limit ?? 20;
|
|
312
|
+
items = items.slice(0, limit);
|
|
313
|
+
return buildListResult(items);
|
|
235
314
|
},
|
|
236
315
|
async get(id) {
|
|
237
|
-
|
|
316
|
+
const data = coll.get(id);
|
|
317
|
+
if (!data) return null;
|
|
318
|
+
return buildGetResult(data, id);
|
|
238
319
|
},
|
|
239
320
|
async add(data) {
|
|
240
321
|
const id = `mock-${++autoSeq}-${Math.random().toString(36).slice(2, 8)}`;
|
|
@@ -257,8 +338,103 @@ var MockDb = class {
|
|
|
257
338
|
async remove(id) {
|
|
258
339
|
coll.delete(id);
|
|
259
340
|
return { ok: true };
|
|
341
|
+
},
|
|
342
|
+
async batch(ops) {
|
|
343
|
+
for (const op of ops) {
|
|
344
|
+
const collMap = (() => {
|
|
345
|
+
if (!_store.has(op.collection)) _store.set(op.collection, /* @__PURE__ */ new Map());
|
|
346
|
+
return _store.get(op.collection);
|
|
347
|
+
})();
|
|
348
|
+
const id = op.id ?? `mock-batch-${++autoSeq}`;
|
|
349
|
+
if (op.op === "add") {
|
|
350
|
+
collMap.set(id, { ...op.data ?? {}, id });
|
|
351
|
+
} else if (op.op === "set") {
|
|
352
|
+
collMap.set(id, { ...op.data ?? {}, id });
|
|
353
|
+
} else if (op.op === "update") {
|
|
354
|
+
const cur = collMap.get(id);
|
|
355
|
+
collMap.set(id, { ...cur ?? {}, ...op.data ?? {}, id });
|
|
356
|
+
} else if (op.op === "remove") {
|
|
357
|
+
collMap.delete(id);
|
|
358
|
+
}
|
|
359
|
+
}
|
|
360
|
+
return { ok: true };
|
|
361
|
+
},
|
|
362
|
+
onDoc(id, cb) {
|
|
363
|
+
const data = coll.get(id);
|
|
364
|
+
Promise.resolve().then(() => cb(data ? data : null));
|
|
365
|
+
return () => {
|
|
366
|
+
};
|
|
367
|
+
},
|
|
368
|
+
onSnapshot(q, cb) {
|
|
369
|
+
Promise.resolve().then(async () => {
|
|
370
|
+
const snap = await ref.list(q);
|
|
371
|
+
cb(snap);
|
|
372
|
+
});
|
|
373
|
+
return () => {
|
|
374
|
+
};
|
|
375
|
+
},
|
|
376
|
+
doc(id) {
|
|
377
|
+
return {
|
|
378
|
+
async get() {
|
|
379
|
+
const data = coll.get(id);
|
|
380
|
+
if (!data) return null;
|
|
381
|
+
return buildGetResult(data, id);
|
|
382
|
+
},
|
|
383
|
+
async set(data) {
|
|
384
|
+
coll.set(id, { ...data, id });
|
|
385
|
+
return { ok: true };
|
|
386
|
+
},
|
|
387
|
+
async update(data) {
|
|
388
|
+
const cur = coll.get(id);
|
|
389
|
+
if (!cur) {
|
|
390
|
+
coll.set(id, { ...data, id });
|
|
391
|
+
} else {
|
|
392
|
+
coll.set(id, { ...cur, ...data });
|
|
393
|
+
}
|
|
394
|
+
return { ok: true };
|
|
395
|
+
},
|
|
396
|
+
async remove() {
|
|
397
|
+
coll.delete(id);
|
|
398
|
+
return { ok: true };
|
|
399
|
+
},
|
|
400
|
+
onSnapshot(cb) {
|
|
401
|
+
const data = coll.get(id);
|
|
402
|
+
Promise.resolve().then(
|
|
403
|
+
() => cb(data ? buildGetResult(data, id) : null)
|
|
404
|
+
);
|
|
405
|
+
return () => {
|
|
406
|
+
};
|
|
407
|
+
}
|
|
408
|
+
};
|
|
260
409
|
}
|
|
261
410
|
};
|
|
411
|
+
return ref;
|
|
412
|
+
}
|
|
413
|
+
async sql(_schema) {
|
|
414
|
+
throw new NeetruDbError(
|
|
415
|
+
"db_unavailable",
|
|
416
|
+
'[MockDb] sql() n\xE3o dispon\xEDvel no mock. Use createNeetruClient({ env: "dev" }) e `neetru dev` para o banco local.'
|
|
417
|
+
);
|
|
418
|
+
}
|
|
419
|
+
get syncState() {
|
|
420
|
+
return {
|
|
421
|
+
status: "idle",
|
|
422
|
+
pendingWrites: 0,
|
|
423
|
+
lastSyncedAt: null,
|
|
424
|
+
isLeaderTab: true
|
|
425
|
+
};
|
|
426
|
+
}
|
|
427
|
+
onSyncStateChanged(_cb) {
|
|
428
|
+
return () => {
|
|
429
|
+
};
|
|
430
|
+
}
|
|
431
|
+
async flush() {
|
|
432
|
+
}
|
|
433
|
+
async clearCache() {
|
|
434
|
+
this._store.clear();
|
|
435
|
+
}
|
|
436
|
+
async getConflicts() {
|
|
437
|
+
return [];
|
|
262
438
|
}
|
|
263
439
|
/** Test helper — substitui fixture inteira de uma collection. */
|
|
264
440
|
__setFixture(name, items) {
|