hydrousdb 3.0.1 → 3.2.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/README.md +945 -493
- package/dist/index.cjs +539 -427
- package/dist/index.cjs.map +1 -1
- package/dist/index.d.cts +292 -325
- package/dist/index.d.ts +292 -325
- package/dist/{index.js → index.mjs} +541 -429
- package/dist/index.mjs.map +1 -0
- package/package.json +7 -6
- package/dist/index.js.map +0 -1
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"sources":["../src/utils/errors.ts","../src/utils/http.ts","../src/auth/client.ts","../src/utils/query.ts","../src/records/client.ts","../src/analytics/client.ts","../src/storage/manager.ts","../src/storage/scoped.ts","../src/client.ts"],"names":[],"mappings":"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","file":"index.mjs","sourcesContent":["export class HydrousError extends Error {\n readonly code: string;\n readonly status?: number;\n readonly requestId?: string;\n readonly details?: string[];\n\n constructor(\n message: string,\n code: string,\n status?: number,\n requestId?: string,\n details?: string[],\n ) {\n super(message);\n this.name = 'HydrousError';\n this.code = code;\n this.status = status;\n this.requestId = requestId;\n this.details = details;\n Object.setPrototypeOf(this, new.target.prototype);\n }\n\n toString(): string {\n return `HydrousError [${this.code}]: ${this.message}`;\n }\n}\n\nexport class AuthError extends HydrousError {\n constructor(message: string, code: string, status?: number, requestId?: string, details?: string[]) {\n super(message, code, status, requestId, details);\n this.name = 'AuthError';\n }\n}\n\nexport class RecordError extends HydrousError {\n constructor(message: string, code: string, status?: number, requestId?: string, details?: string[]) {\n super(message, code, status, requestId, details);\n this.name = 'RecordError';\n }\n}\n\nexport class StorageError extends HydrousError {\n constructor(message: string, code: string, status?: number, requestId?: string) {\n super(message, code, status, requestId);\n this.name = 'StorageError';\n }\n}\n\nexport class AnalyticsError extends HydrousError {\n constructor(message: string, code: string, status?: number, requestId?: string) {\n super(message, code, status, requestId);\n this.name = 'AnalyticsError';\n }\n}\n\nexport class ValidationError extends HydrousError {\n constructor(message: string, details?: string[]) {\n super(message, 'VALIDATION_ERROR', 400, undefined, details);\n this.name = 'ValidationError';\n }\n}\n\nexport class NetworkError extends HydrousError {\n readonly cause?: unknown;\n\n constructor(message: string, cause?: unknown) {\n super(message, 'NETWORK_ERROR');\n this.name = 'NetworkError';\n if (cause !== undefined) this.cause = cause;\n }\n}\n","import { HydrousError, NetworkError } from './errors.js';\n\nexport const DEFAULT_BASE_URL = 'https://db-api-82687684612.us-central1.run.app';\n\ninterface RequestOptions {\n method: string;\n body?: unknown;\n rawBody?: FormData | Uint8Array | ArrayBuffer;\n headers?: Record<string, string>;\n}\n\nexport class HttpClient {\n private readonly baseUrl: string;\n\n constructor(baseUrl: string) {\n this.baseUrl = baseUrl.replace(/\\/$/, '');\n }\n\n // ─── Core request method ──────────────────────────────────────────────────\n\n async request<T>(path: string, options: RequestOptions): Promise<T> {\n const url = `${this.baseUrl}${path}`;\n\n const headers: Record<string, string> = { ...options.headers };\n\n let body: BodyInit | undefined;\n\n if (options.rawBody !== undefined) {\n // FormData or binary — let the browser/node set Content-Type\n body = options.rawBody as BodyInit;\n } else if (options.body !== undefined) {\n headers['Content-Type'] = 'application/json';\n body = JSON.stringify(options.body);\n }\n\n let response: Response;\n try {\n response = await fetch(url, {\n method: options.method,\n headers,\n body,\n });\n } catch (err) {\n throw new NetworkError(\n `Network request failed: ${err instanceof Error ? err.message : String(err)}`,\n err,\n );\n }\n\n // Binary responses (downloads)\n const contentType = response.headers.get('content-type') ?? '';\n if (!contentType.includes('application/json')) {\n if (!response.ok) {\n throw new HydrousError(\n `Request failed with status ${response.status}`,\n 'REQUEST_FAILED',\n response.status,\n );\n }\n return response.arrayBuffer() as Promise<T>;\n }\n\n let data: Record<string, unknown>;\n try {\n data = await response.json() as Record<string, unknown>;\n } catch {\n throw new HydrousError(\n `Failed to parse response JSON`,\n 'PARSE_ERROR',\n response.status,\n );\n }\n\n if (!response.ok) {\n throw new HydrousError(\n (data['error'] as string) || (data['message'] as string) || `Request failed with status ${response.status}`,\n (data['code'] as string) || 'REQUEST_FAILED',\n response.status,\n data['requestId'] as string | undefined,\n data['details'] as string[] | undefined,\n );\n }\n\n return data as T;\n }\n\n // ─── Convenience wrappers ─────────────────────────────────────────────────\n\n get<T>(path: string, apiKey?: string, extraHeaders?: Record<string, string>): Promise<T> {\n return this.request<T>(path, {\n method: 'GET',\n headers: apiKey\n ? { 'X-Api-Key': apiKey, ...extraHeaders }\n : { ...extraHeaders },\n });\n }\n\n post<T>(path: string, apiKey: string, body?: unknown): Promise<T> {\n return this.request<T>(path, {\n method: 'POST',\n body,\n headers: { 'X-Api-Key': apiKey },\n });\n }\n\n put<T>(path: string, apiKey: string, body?: unknown): Promise<T> {\n return this.request<T>(path, {\n method: 'PUT',\n body,\n headers: { 'X-Api-Key': apiKey },\n });\n }\n\n patch<T>(path: string, apiKey: string, body?: unknown): Promise<T> {\n return this.request<T>(path, {\n method: 'PATCH',\n body,\n headers: { 'X-Api-Key': apiKey },\n });\n }\n\n delete<T>(path: string, apiKey: string, body?: unknown): Promise<T> {\n return this.request<T>(path, {\n method: 'DELETE',\n body,\n headers: { 'X-Api-Key': apiKey },\n });\n }\n\n /**\n * Upload directly to a signed GCS URL (no auth headers — signed URL is self-authenticating).\n * Supports optional progress tracking via XHR in browser environments.\n */\n async putToSignedUrl(\n signedUrl: string,\n data: Blob | Uint8Array | ArrayBuffer,\n mimeType: string,\n onProgress?: (percent: number) => void,\n ): Promise<void> {\n // Normalise to a type accepted by both XHR.send() and fetch body\n const body: Blob | ArrayBuffer =\n data instanceof Blob ? data :\n data instanceof Uint8Array ? data.buffer as ArrayBuffer :\n data;\n\n // Use XHR in browser for progress support; fall back to fetch in Node\n if (typeof XMLHttpRequest !== 'undefined' && typeof onProgress === 'function') {\n await new Promise<void>((resolve, reject) => {\n const xhr = new XMLHttpRequest();\n xhr.upload.onprogress = (e) => {\n if (e.lengthComputable) onProgress(Math.round((e.loaded / e.total) * 100));\n };\n xhr.onload = () => (xhr.status >= 200 && xhr.status < 300 ? resolve() : reject(new Error(`GCS upload failed: ${xhr.status}`)));\n xhr.onerror = () => reject(new Error('GCS upload network error'));\n xhr.open('PUT', signedUrl);\n xhr.setRequestHeader('Content-Type', mimeType);\n xhr.send(body);\n });\n } else {\n const res = await fetch(signedUrl, {\n method: 'PUT',\n headers: { 'Content-Type': mimeType },\n body,\n });\n if (!res.ok) {\n throw new HydrousError(`GCS upload failed: ${res.status}`, 'GCS_UPLOAD_FAILED', res.status);\n }\n }\n }\n}\n","import type { HttpClient } from '../utils/http.js';\nimport type {\n SignupOptions,\n LoginOptions,\n AuthResult,\n UserRecord,\n UpdateUserOptions,\n ChangePasswordOptions,\n ListUsersOptions,\n ListUsersResult,\n Session,\n} from '../types/index.js';\n\n// ─── Internal response shapes ─────────────────────────────────────────────────\n// These mirror exactly what each server route returns.\n\ninterface ApiAuthResponse {\n success: boolean;\n user: UserRecord;\n session: { sessionId: string; refreshToken: string; expiresAt: number };\n data?: UserRecord;\n requestId?: string;\n}\n\ninterface ApiUserResponse {\n success: boolean;\n data: UserRecord;\n requestId?: string;\n}\n\ninterface ApiSessionResponse {\n success: boolean;\n session: { sessionId: string; refreshToken: string; expiresAt: number };\n data?: UserRecord;\n requestId?: string;\n}\n\ninterface ApiMessageResponse {\n success: boolean;\n message: string;\n requestId?: string;\n}\n\ninterface ApiLockResponse {\n success: boolean;\n data: { lockedUntil: number; unlockTime: string };\n requestId?: string;\n}\n\ninterface ApiListUsersResponse {\n success: boolean;\n data?: UserRecord[];\n users?: UserRecord[];\n meta?: { hasMore: boolean; nextCursor: string | null };\n total?: number;\n limit?: number;\n offset?: number;\n hasMore?: boolean;\n nextCursor?: string | null;\n requestId?: string;\n}\n\ninterface ApiBulkDeleteResponse {\n success: boolean;\n message: string;\n meta: { succeeded: number; failed: number };\n requestId?: string;\n}\n\n/**\n * AuthClient — full user authentication for a single bucket.\n * Uses the `authKey` (`hk_auth_…`) sent via `X-Api-Key` header.\n *\n * All routes are under `/auth/:bucketKey/` — the bucketKey is the user bucket\n * name you passed when calling `db.auth('bucketKey')`.\n *\n * @example\n * ```ts\n * const db = createClient({ authKey: 'hk_auth_…', … });\n * const auth = db.auth('app-users');\n * const { user, session } = await auth.signup({ email: 'alice@example.com', password: 'hunter2' });\n * ```\n */\nexport class AuthClient {\n private readonly http: HttpClient;\n private readonly authKey: string;\n private readonly basePath: string;\n\n constructor(http: HttpClient, authKey: string, bucketKey: string) {\n this.http = http;\n this.authKey = authKey;\n this.basePath = `/auth/${bucketKey}`;\n }\n\n private post<T>(path: string, body?: unknown): Promise<T> {\n return this.http.post<T>(path, this.authKey, body);\n }\n private get<T>(path: string, query?: Record<string, string>): Promise<T> {\n const qs = query && Object.keys(query).length\n ? '?' + new URLSearchParams(query).toString()\n : '';\n return this.http.get<T>(`${path}${qs}`, this.authKey);\n }\n private patch<T>(path: string, body?: unknown): Promise<T> {\n return this.http.patch<T>(path, this.authKey, body);\n }\n private delete<T>(path: string, body?: unknown): Promise<T> {\n return this.http.delete<T>(path, this.authKey, body);\n }\n\n // ─── Registration & Login ─────────────────────────────────────────────────\n // Server: POST /auth/:bucket/signup → { user, session }\n // Server: POST /auth/:bucket/signin → { user, session }\n // Server: POST /auth/:bucket/signout → { success, message }\n\n async signup(options: SignupOptions): Promise<AuthResult> {\n const result = await this.post<ApiAuthResponse>(`${this.basePath}/signup`, options);\n const user = result.user ?? result.data!;\n return {\n user,\n session: {\n sessionId: result.session.sessionId,\n userId: user.id,\n bucketId: this.basePath.split('/')[2]!,\n createdAt: Date.now(),\n expiresAt: result.session.expiresAt,\n refreshToken: result.session.refreshToken,\n refreshExpiresAt: result.session.expiresAt,\n },\n };\n }\n\n async login(options: LoginOptions): Promise<AuthResult> {\n // Server route is /signin not /login\n const result = await this.post<ApiAuthResponse>(`${this.basePath}/signin`, options);\n const user = result.user ?? result.data!;\n return {\n user,\n session: {\n sessionId: result.session.sessionId,\n userId: user.id,\n bucketId: this.basePath.split('/')[2]!,\n createdAt: Date.now(),\n expiresAt: result.session.expiresAt,\n refreshToken: result.session.refreshToken,\n refreshExpiresAt: result.session.expiresAt,\n },\n };\n }\n\n async logout(options: { sessionId: string; allDevices?: boolean }): Promise<void> {\n // Server: POST /auth/:bucket/signout body: { sessionId, allDevices? }\n await this.post<ApiMessageResponse>(`${this.basePath}/signout`, options);\n }\n\n async refreshSession({ refreshToken }: { refreshToken: string }): Promise<Session> {\n // Server: POST /auth/:bucket/session/refresh body: { refreshToken }\n const result = await this.post<ApiSessionResponse>(\n `${this.basePath}/session/refresh`,\n { refreshToken },\n );\n const s = result.session;\n return {\n sessionId: s.sessionId,\n userId: result.data?.id ?? '',\n bucketId: this.basePath.split('/')[2]!,\n createdAt: Date.now(),\n expiresAt: s.expiresAt,\n refreshToken: s.refreshToken,\n refreshExpiresAt: s.expiresAt,\n };\n }\n\n async validateSession({ sessionId }: { sessionId: string }): Promise<{ user: UserRecord; session: { sessionId: string; expiresAt: number } }> {\n // Server: POST /auth/:bucket/session/validate body: { sessionId }\n const result = await this.post<{ success: boolean; data: UserRecord; session: { sessionId: string; expiresAt: number } }>(\n `${this.basePath}/session/validate`,\n { sessionId },\n );\n return { user: result.data, session: result.session };\n }\n\n // ─── User Profile ─────────────────────────────────────────────────────────\n // Server: GET /auth/:bucket/user?userId=... → { success, data: UserRecord }\n // Server: PATCH /auth/:bucket/user body: { sessionId, userId, updates: {...} }\n // Server: DELETE /auth/:bucket/user?userId=... header/body: sessionId\n\n async getUser({ userId }: { userId: string }): Promise<UserRecord> {\n // Server reads userId from query string, NOT path segment\n const result = await this.get<ApiUserResponse>(`${this.basePath}/user`, { userId });\n return result.data;\n }\n\n async updateUser(options: UpdateUserOptions): Promise<UserRecord> {\n const { sessionId, userId, updates } = options;\n // Server expects: { sessionId, userId, updates: { ...fields } }\n const result = await this.patch<ApiUserResponse>(\n `${this.basePath}/user`,\n { sessionId, userId, updates },\n );\n return result.data;\n }\n\n async deleteUser({ sessionId, userId }: { sessionId: string; userId: string }): Promise<void> {\n // Server reads userId from req.query and sessionId from body or x-session-id header\n await this.http.request<ApiMessageResponse>(`${this.basePath}/user?userId=${encodeURIComponent(userId)}`, {\n method: 'DELETE',\n body: { sessionId },\n headers: { 'X-Api-Key': this.authKey },\n });\n }\n\n // ─── Admin Operations ─────────────────────────────────────────────────────\n // Server: GET /auth/:bucket/users?limit=&cursor= → { data: UserRecord[], meta: { hasMore, nextCursor } }\n // Server: DELETE /auth/:bucket/users/bulk body: { userIds, hard?, sessionId }\n // Server: DELETE /auth/:bucket/user/hard?userId=... body: { sessionId }\n\n async listUsers(options: ListUsersOptions): Promise<ListUsersResult> {\n const { sessionId, limit = 50, cursor } = options;\n // Server: GET /auth/:bucket/users with query params\n // sessionId is sent as a header since GET bodies are unreliable\n const params: Record<string, string> = { limit: String(limit) };\n if (cursor) params['cursor'] = cursor;\n\n const result = await this.http.request<ApiListUsersResponse>(\n `${this.basePath}/users?${new URLSearchParams(params)}`,\n {\n method: 'GET',\n headers: { 'X-Api-Key': this.authKey, 'X-Session-Id': sessionId },\n },\n );\n return {\n users: result.data ?? result.users ?? [],\n hasMore: result.meta?.hasMore ?? result.hasMore ?? false,\n nextCursor: result.meta?.nextCursor ?? result.nextCursor ?? null,\n };\n }\n\n async hardDeleteUser({ sessionId, userId }: { sessionId: string; userId: string }): Promise<void> {\n // Server: DELETE /auth/:bucket/user/hard?userId=... body: { sessionId }\n await this.http.request<ApiMessageResponse>(\n `${this.basePath}/user/hard?userId=${encodeURIComponent(userId)}`,\n {\n method: 'DELETE',\n body: { sessionId },\n headers: { 'X-Api-Key': this.authKey },\n },\n );\n }\n\n async bulkDeleteUsers(options: { sessionId: string; userIds: string[]; hard?: boolean }): Promise<{ succeeded: number; failed: number }> {\n // Server: DELETE /auth/:bucket/users/bulk body: { userIds, hard?, sessionId }\n const result = await this.http.request<ApiBulkDeleteResponse>(\n `${this.basePath}/users/bulk`,\n {\n method: 'DELETE',\n body: { userIds: options.userIds, hard: options.hard ?? false, sessionId: options.sessionId },\n headers: { 'X-Api-Key': this.authKey },\n },\n );\n return { succeeded: result.meta.succeeded, failed: result.meta.failed };\n }\n\n async lockAccount(options: { sessionId: string; userId: string; duration?: number }): Promise<{ lockedUntil: number; unlockTime: string }> {\n // Server: POST /auth/:bucket/account/lock body: { sessionId, userId, duration? }\n const result = await this.post<ApiLockResponse>(\n `${this.basePath}/account/lock`,\n options,\n );\n return result.data;\n }\n\n async unlockAccount({ sessionId, userId }: { sessionId: string; userId: string }): Promise<void> {\n // Server: POST /auth/:bucket/account/unlock body: { sessionId, userId }\n await this.post<ApiMessageResponse>(`${this.basePath}/account/unlock`, { sessionId, userId });\n }\n\n // ─── Password Management ──────────────────────────────────────────────────\n // Server: POST /auth/:bucket/password/change body: { sessionId, userId, oldPassword, newPassword }\n // Server: POST /auth/:bucket/password/reset/request body: { email }\n // Server: POST /auth/:bucket/password/reset/confirm body: { resetToken, newPassword }\n\n async changePassword(options: ChangePasswordOptions): Promise<void> {\n const { sessionId, userId, currentPassword, newPassword } = options;\n // Server reads `oldPassword`, not `currentPassword`\n await this.post<ApiMessageResponse>(`${this.basePath}/password/change`, {\n sessionId,\n userId,\n oldPassword: currentPassword,\n newPassword,\n });\n }\n\n async requestPasswordReset({ email }: { email: string }): Promise<void> {\n await this.post<ApiMessageResponse>(`${this.basePath}/password/reset/request`, { email });\n }\n\n async confirmPasswordReset({ resetToken, newPassword }: { resetToken: string; newPassword: string }): Promise<void> {\n await this.post<ApiMessageResponse>(`${this.basePath}/password/reset/confirm`, { resetToken, newPassword });\n }\n\n // ─── Email Verification ───────────────────────────────────────────────────\n // Server: POST /auth/:bucket/email/verify/request body: { userId }\n // Server: POST /auth/:bucket/email/verify/confirm body: { verifyToken }\n\n async requestEmailVerification({ userId }: { userId: string }): Promise<void> {\n await this.post<ApiMessageResponse>(`${this.basePath}/email/verify/request`, { userId });\n }\n\n async confirmEmailVerification({ verifyToken }: { verifyToken: string }): Promise<void> {\n await this.post<ApiMessageResponse>(`${this.basePath}/email/verify/confirm`, { verifyToken });\n }\n}\n","import type { QueryFilter, QueryOptions } from '../types/index.js';\n\n/**\n * Build a query string from QueryOptions for GET /records/:bucket requests.\n */\nexport function buildQueryParams(options: QueryOptions = {}): string {\n const params = new URLSearchParams();\n\n if (options.limit !== undefined) params.set('limit', String(options.limit));\n if (options.offset !== undefined) params.set('offset', String(options.offset));\n if (options.orderBy !== undefined) params.set('orderBy', options.orderBy);\n if (options.order !== undefined) params.set('order', options.order);\n if (options.fields !== undefined) params.set('fields', options.fields);\n if (options.startAfter !== undefined) params.set('startAfter', options.startAfter);\n if (options.startAt !== undefined) params.set('startAt', options.startAt);\n if (options.endAt !== undefined) params.set('endAt', options.endAt);\n\n if (options.dateRange?.start !== undefined)\n params.set('startDate', new Date(options.dateRange.start).toISOString().split('T')[0]!);\n if (options.dateRange?.end !== undefined)\n params.set('endDate', new Date(options.dateRange.end).toISOString().split('T')[0]!);\n\n if (options.filters && options.filters.length > 0)\n params.set('filters', JSON.stringify(options.filters));\n\n const str = params.toString();\n return str ? `?${str}` : '';\n}\n\n/**\n * Normalise a filter shorthand into { field, op, value }.\n */\nexport function normaliseFilter(\n filter: QueryFilter | [string, QueryFilter['op'], QueryFilter['value']],\n): QueryFilter {\n if (Array.isArray(filter)) {\n const [field, op, value] = filter;\n return { field, op, value };\n }\n return filter;\n}\n\n/**\n * Detect a likely MIME type from a file extension.\n */\nexport function guessMimeType(filename: string): string {\n const ext = filename.split('.').pop()?.toLowerCase();\n const map: Record<string, string> = {\n jpg: 'image/jpeg', jpeg: 'image/jpeg', png: 'image/png',\n gif: 'image/gif', webp: 'image/webp', svg: 'image/svg+xml',\n pdf: 'application/pdf', mp4: 'video/mp4', webm: 'video/webm',\n mp3: 'audio/mpeg', wav: 'audio/wav', txt: 'text/plain',\n html: 'text/html', css: 'text/css', js: 'application/javascript',\n json: 'application/json', xml: 'application/xml',\n zip: 'application/zip', csv: 'text/csv',\n };\n return map[ext ?? ''] ?? 'application/octet-stream';\n}\n\n/**\n * Validate that a name contains only safe characters.\n * Must match the server-side SAFE_RE: /^[a-zA-Z_][a-zA-Z0-9_.]{0,200}$/\n * Also allows hyphens as the server's regex permits them in bucket names.\n */\nexport function assertSafeName(name: string, label = 'name'): void {\n if (!/^[a-zA-Z_][a-zA-Z0-9_.\\-]{0,200}$/.test(name)) {\n throw new Error(\n `Invalid ${label} \"${name}\". Must start with a letter or underscore and ` +\n `contain only letters, numbers, underscores, dots, or hyphens.`,\n );\n }\n}\n","import type { HttpClient } from '../utils/http.js';\nimport { buildQueryParams, assertSafeName } from '../utils/query.js';\nimport type {\n RecordData,\n RecordResult,\n QueryOptions,\n QueryResult,\n PatchRecordOptions,\n RecordHistoryEntry,\n CreateRecordOptions,\n BatchCreateOptions,\n} from '../types/index.js';\n\n// ─── Internal response shapes ─────────────────────────────────────────────────\n\ninterface ApiRecordResponse<T> {\n success: boolean;\n data?: T & RecordResult;\n record?: T & RecordResult;\n upserted?: boolean;\n meta?: { id: string; custom?: boolean };\n}\n\ninterface ApiQueryResponse<T> {\n success: boolean;\n data?: (T & RecordResult)[];\n meta?: { hasMore: boolean; nextCursor?: string | null; total?: number };\n records?: (T & RecordResult)[];\n total?: number;\n hasMore?: boolean;\n nextCursor?: string | null;\n}\n\ninterface ApiBatchCreateResponse<T> {\n success: boolean;\n data?: (T & RecordResult)[];\n meta?: { totalProcessed: number; successful: number; failed: number };\n records?: (T & RecordResult)[];\n errors?: unknown[];\n}\n\ninterface ApiBatchDeleteResponse {\n success: boolean;\n data?: { successful: number; failed: string[] };\n meta?: { count: number };\n deleted?: number;\n failed?: string[];\n}\n\ninterface ApiCountResponse {\n success: boolean;\n data?: { count: number };\n count?: number;\n}\n\ninterface ApiHistoryResponse {\n success: boolean;\n data?: RecordHistoryEntry[];\n history?: RecordHistoryEntry[];\n}\n\n/**\n * RecordsClient — typed CRUD + query + batch for a single bucket.\n * Uses the `bucketSecurityKey` (`hk_bucket_…`) sent via `X-Api-Key` header.\n *\n * @example\n * ```ts\n * interface Post { title: string; published: boolean }\n * const posts = db.records<Post>('blog-posts');\n * const post = await posts.create({ title: 'Hello', published: false });\n * ```\n */\nexport class RecordsClient<T extends RecordData = RecordData> {\n private readonly http: HttpClient;\n private readonly bucketKey: string;\n private readonly apiKey: string;\n private readonly basePath: string;\n\n constructor(http: HttpClient, bucketSecurityKey: string, bucketKey: string) {\n assertSafeName(bucketKey, 'bucketKey');\n this.http = http;\n this.bucketKey = bucketKey;\n this.apiKey = bucketSecurityKey;\n this.basePath = `/records/${bucketKey}`;\n }\n\n // ─── Single Record Operations ─────────────────────────────────────────────\n\n /**\n * Create a new record.\n *\n * @param data The record fields to store.\n * @param options Optional: queryableFields (for server-side filtering),\n * userEmail (audit trail), customRecordId (upsert by ID).\n *\n * @example\n * ```ts\n * const post = await posts.create(\n * { title: 'Hello', status: 'draft', authorId: 'u1' },\n * { queryableFields: ['status', 'authorId'], userEmail: 'alice@example.com' },\n * );\n * ```\n */\n async create(data: T, options: CreateRecordOptions = {}): Promise<T & RecordResult> {\n const { queryableFields, userEmail, customRecordId } = options;\n // Server: POST /records/:bucket\n // Body shape: { values: {...}, queryableFields?: [...], userEmail?: string, customRecordId?: string }\n const body: Record<string, unknown> = { values: data };\n if (queryableFields?.length) body['queryableFields'] = queryableFields;\n if (userEmail) body['userEmail'] = userEmail;\n if (customRecordId) body['customRecordId'] = customRecordId;\n\n const result = await this.http.post<ApiRecordResponse<T>>(this.basePath, this.apiKey, body);\n return (result.data ?? result.record)!;\n }\n\n /**\n * Get a single record by ID.\n */\n async get(id: string): Promise<T & RecordResult> {\n // Server: GET /records/:bucket/:id\n const result = await this.http.get<ApiRecordResponse<T>>(\n `${this.basePath}/${encodeURIComponent(id)}`,\n this.apiKey,\n );\n return (result.data ?? result.record)!;\n }\n\n /**\n * Fully replace a record (PUT semantics).\n */\n async set(id: string, data: T): Promise<T & RecordResult> {\n // Server: PUT /records/:bucket/:id\n const result = await this.http.put<ApiRecordResponse<T>>(\n `${this.basePath}/${encodeURIComponent(id)}`,\n this.apiKey,\n data,\n );\n return (result.data ?? result.record)!;\n }\n\n /**\n * Partially update a record (PATCH semantics).\n * By default merges with existing fields; set `merge: false` to replace.\n * Supports write-filter sentinels: `{ __op: 'increment', delta: 1 }` etc.\n */\n async patch(id: string, data: Partial<T>, options: PatchRecordOptions = {}): Promise<T & RecordResult> {\n const { merge = true } = options;\n // Server: PATCH /records/:bucket body: { recordId, values: { ...data, _merge } }\n const result = await this.http.patch<ApiRecordResponse<T>>(\n `${this.basePath}/${encodeURIComponent(id)}`,\n this.apiKey,\n { ...data, _merge: merge },\n );\n return (result.data ?? result.record)!;\n }\n\n /**\n * Delete a record permanently.\n */\n async delete(id: string): Promise<void> {\n // Server: DELETE /records/:bucket/:id\n await this.http.delete<{ success: boolean }>(\n `${this.basePath}/${encodeURIComponent(id)}`,\n this.apiKey,\n );\n }\n\n // ─── Batch Operations ─────────────────────────────────────────────────────\n\n /**\n * Create multiple records in one request (max 500).\n * Each record can include a `_customRecordId` field for upsert behaviour.\n *\n * @example\n * ```ts\n * const created = await posts.batchCreate(\n * [{ title: 'A' }, { title: 'B' }],\n * { queryableFields: ['title'], userEmail: 'alice@example.com' },\n * );\n * ```\n */\n async batchCreate(items: T[], options: BatchCreateOptions = {}): Promise<(T & RecordResult)[]> {\n const { queryableFields, userEmail } = options;\n // Server: POST /records/:bucket/batch/insert\n // Body: { records: [...], queryableFields?: [...], userEmail?: string }\n const body: Record<string, unknown> = { records: items };\n if (queryableFields?.length) body['queryableFields'] = queryableFields;\n if (userEmail) body['userEmail'] = userEmail;\n\n const result = await this.http.post<ApiBatchCreateResponse<T>>(\n `${this.basePath}/batch/insert`,\n this.apiKey,\n body,\n );\n return result.data ?? result.records ?? [];\n }\n\n /**\n * Delete multiple records in one request (max 500).\n */\n async batchDelete(ids: string[]): Promise<{ deleted: number; failed: string[] }> {\n // Server: POST /records/:bucket/batch/delete\n // Body: { recordIds: [...], userEmail?: string }\n const result = await this.http.post<ApiBatchDeleteResponse>(\n `${this.basePath}/batch/delete`,\n this.apiKey,\n { recordIds: ids },\n );\n return {\n deleted: result.data?.successful ?? result.deleted ?? 0,\n failed: result.data?.failed ?? result.failed ?? [],\n };\n }\n\n // ─── Querying ─────────────────────────────────────────────────────────────\n\n /**\n * Query records with optional filters, sorting, and pagination.\n *\n * @example\n * ```ts\n * const { records, hasMore, nextCursor } = await posts.query({\n * filters: [{ field: 'status', op: '==', value: 'published' }],\n * orderBy: 'createdAt',\n * order: 'desc',\n * limit: 20,\n * });\n * ```\n */\n async query(options: QueryOptions = {}): Promise<QueryResult<T>> {\n const qs = buildQueryParams(options);\n const result = await this.http.get<ApiQueryResponse<T>>(`${this.basePath}${qs}`, this.apiKey);\n const records = result.data ?? result.records ?? [];\n const hasMore = result.meta?.hasMore ?? result.hasMore ?? false;\n const total = result.meta?.total ?? result.total;\n const nextCursor = result.meta?.nextCursor ?? result.nextCursor ?? undefined;\n return { records, total, hasMore, nextCursor: nextCursor ?? undefined };\n }\n\n /**\n * Get all records matching the given options (no filter support — use `query()` for filters).\n */\n async getAll(options: Omit<QueryOptions, 'filters'> = {}): Promise<(T & RecordResult)[]> {\n const { records } = await this.query(options);\n return records;\n }\n\n /**\n * Count records matching the given filters.\n */\n async count(filters: QueryOptions['filters'] = []): Promise<number> {\n // Server: POST /records/:bucket/count body: { filters }\n const result = await this.http.post<ApiCountResponse>(\n `${this.basePath}/count`,\n this.apiKey,\n { filters },\n );\n return result.data?.count ?? result.count ?? 0;\n }\n\n // ─── Version History ──────────────────────────────────────────────────────\n\n /**\n * Get the version history of a record.\n */\n async getHistory(id: string): Promise<RecordHistoryEntry[]> {\n // Server: GET /records/:bucket/:id?showHistory=true\n const result = await this.http.get<{ success: boolean; history: RecordHistoryEntry[] }>(\n `${this.basePath}/${encodeURIComponent(id)}?showHistory=true`,\n this.apiKey,\n );\n return result.history;\n }\n\n /**\n * Restore a record to a previous version.\n */\n async restoreVersion(id: string, version: number): Promise<T & RecordResult> {\n // Server: POST /records/:bucket/:id/restore body: { version }\n const result = await this.http.post<ApiRecordResponse<T>>(\n `${this.basePath}/${encodeURIComponent(id)}/restore`,\n this.apiKey,\n { version },\n );\n return (result.data ?? result.record)!;\n }\n}\n","import type { HttpClient } from '../utils/http.js';\nimport { assertSafeName } from '../utils/query.js';\nimport type {\n AnalyticsQuery,\n AnalyticsResult,\n Aggregation,\n Granularity,\n SortOrder,\n DateRange,\n AnalyticsFilter,\n MetricDefinition,\n CountResult,\n DistributionRow,\n SumRow,\n TimeSeriesRow,\n FieldTimeSeriesRow,\n TopNRow,\n FieldStats,\n MultiMetricResult,\n StorageStatsResult,\n CrossBucketRow,\n RecordData,\n RecordResult,\n} from '../types/index.js';\n\ninterface ApiAnalyticsResult<T> {\n success: boolean;\n queryType: string;\n data: T;\n}\n\n/**\n * AnalyticsClient — BigQuery-powered aggregations for a single bucket.\n * Uses the `bucketSecurityKey` (`hk_bucket_…`) sent via `X-Api-Key` header.\n *\n * All methods POST a `queryType` body to `POST /analytics/:bucketKey`.\n * The `dateRange` is passed as `{ start, end }` ms timestamps — the server\n * converts them to ISO date strings internally.\n *\n * @example\n * ```ts\n * const analytics = db.analytics('orders');\n * const { count } = await analytics.count();\n * const top5 = await analytics.topN({ field: 'country', n: 5 });\n * ```\n */\nexport class AnalyticsClient {\n private readonly http: HttpClient;\n private readonly bucketSecurityKey: string;\n private readonly basePath: string;\n\n constructor(http: HttpClient, bucketSecurityKey: string, bucketKey: string) {\n assertSafeName(bucketKey, 'bucketKey');\n this.http = http;\n this.bucketSecurityKey = bucketSecurityKey;\n this.basePath = `/analytics/${bucketKey}`;\n }\n\n private async run<T>(query: AnalyticsQuery): Promise<T> {\n const result = await this.http.post<ApiAnalyticsResult<T>>(\n this.basePath,\n this.bucketSecurityKey,\n query,\n );\n return result.data;\n }\n\n // ─── Query methods ────────────────────────────────────────────────────────\n\n /** Count all records, optionally within a date range. */\n async count(opts: { dateRange?: DateRange } = {}): Promise<CountResult> {\n return this.run<CountResult>({ queryType: 'count', ...opts });\n }\n\n /** Get value distribution for a field (e.g. how many records per status). */\n async distribution(opts: {\n field: string;\n limit?: number;\n order?: SortOrder;\n dateRange?: DateRange;\n }): Promise<DistributionRow[]> {\n assertSafeName(opts.field, 'field');\n return this.run<DistributionRow[]>({ queryType: 'distribution', ...opts });\n }\n\n /** Sum a numeric field, optionally grouped by another field. */\n async sum(opts: {\n field: string;\n groupBy?: string;\n limit?: number;\n dateRange?: DateRange;\n }): Promise<SumRow[]> {\n assertSafeName(opts.field, 'field');\n if (opts.groupBy) assertSafeName(opts.groupBy, 'groupBy');\n return this.run<SumRow[]>({ queryType: 'sum', ...opts });\n }\n\n /** Count of records over time, bucketed by granularity. */\n async timeSeries(opts: {\n granularity?: Granularity;\n dateRange?: DateRange;\n } = {}): Promise<TimeSeriesRow[]> {\n return this.run<TimeSeriesRow[]>({ queryType: 'timeSeries', granularity: 'day', ...opts });\n }\n\n /** Aggregate a numeric field over time. */\n async fieldTimeSeries(opts: {\n field: string;\n aggregation?: Aggregation;\n granularity?: Granularity;\n dateRange?: DateRange;\n }): Promise<FieldTimeSeriesRow[]> {\n assertSafeName(opts.field, 'field');\n return this.run<FieldTimeSeriesRow[]>({\n queryType: 'fieldTimeSeries',\n aggregation: 'sum',\n granularity: 'day',\n ...opts,\n });\n }\n\n /** Top N values for a field by count. */\n async topN(opts: {\n field: string;\n n?: number;\n labelField?: string;\n order?: SortOrder;\n dateRange?: DateRange;\n }): Promise<TopNRow[]> {\n assertSafeName(opts.field, 'field');\n if (opts.labelField) assertSafeName(opts.labelField, 'labelField');\n return this.run<TopNRow[]>({ queryType: 'topN', n: 10, order: 'desc', ...opts });\n }\n\n /** Statistical summary (min, max, avg, sum, count, stddev) for a numeric field. */\n async stats(opts: {\n field: string;\n dateRange?: DateRange;\n }): Promise<FieldStats> {\n assertSafeName(opts.field, 'field');\n return this.run<FieldStats>({ queryType: 'stats', ...opts });\n }\n\n /** Fetch filtered records via the analytics engine (bypasses Firestore pagination). */\n async records<T extends RecordData = RecordData>(opts: {\n filters?: AnalyticsFilter[];\n selectFields?: string[];\n limit?: number;\n offset?: number;\n orderBy?: string;\n order?: SortOrder;\n dateRange?: DateRange;\n } = {}): Promise<(T & RecordResult)[]> {\n if (opts.orderBy) assertSafeName(opts.orderBy, 'orderBy');\n if (opts.selectFields) opts.selectFields.forEach(f => assertSafeName(f, 'selectField'));\n return this.run<(T & RecordResult)[]>({ queryType: 'records', limit: 100, order: 'desc', ...opts });\n }\n\n /** Compute multiple aggregations in a single request. */\n async multiMetric(opts: {\n metrics: MetricDefinition[];\n dateRange?: DateRange;\n }): Promise<MultiMetricResult> {\n opts.metrics.forEach(m => {\n assertSafeName(m.field, 'metric.field');\n assertSafeName(m.name, 'metric.name');\n });\n return this.run<MultiMetricResult>({ queryType: 'multiMetric', ...opts });\n }\n\n /** Storage usage stats for the bucket (record count, bytes, avg/min/max size). */\n async storageStats(opts: { dateRange?: DateRange } = {}): Promise<StorageStatsResult> {\n return this.run<StorageStatsResult>({ queryType: 'storageStats', ...opts });\n }\n\n /**\n * Compare a metric across multiple buckets in one query.\n * The caller's key must have read access to every bucket in `bucketKeys`.\n */\n async crossBucket(opts: {\n bucketKeys: string[];\n field: string;\n aggregation?: Aggregation;\n dateRange?: DateRange;\n }): Promise<CrossBucketRow[]> {\n assertSafeName(opts.field, 'field');\n opts.bucketKeys.forEach(k => assertSafeName(k, 'bucketKey'));\n return this.run<CrossBucketRow[]>({ queryType: 'crossBucket', aggregation: 'sum', ...opts });\n }\n\n /**\n * Raw query — use this when none of the typed helpers cover your use case.\n */\n async query<T = unknown>(query: AnalyticsQuery): Promise<AnalyticsResult<T>> {\n const data = await this.run<T>(query);\n return { queryType: query.queryType, data };\n }\n}\n","import type { HttpClient } from '../utils/http.js';\nimport { guessMimeType } from '../utils/query.js';\nimport type {\n UploadOptions,\n UploadResult,\n UploadUrlResult,\n ListOptions,\n ListResult,\n FileEntry,\n FileMetadata,\n SignedUrlResult,\n StorageStats,\n BatchUploadItem,\n BatchUploadUrlResult,\n BatchDownloadResult,\n} from '../types/index.js';\n\n// ─── Internal response shapes (mirror server exactly) ─────────────────────────\n\ninterface ApiUploadResult {\n success: boolean;\n path: string;\n mimeType: string;\n size: number;\n isPublic: boolean;\n publicUrl: string | null;\n downloadUrl: string | null;\n}\n\ninterface ApiUploadUrlResult {\n uploadUrl: string;\n path: string;\n mimeType: string;\n isPublic: boolean;\n size: number;\n expiresAt: string;\n expiresIn: number;\n}\n\n// Server returns { items: FileEntry[], pagination: { hasMore, nextCursor } }\n// Tests / legacy may return { files: FileEntry[], folders: string[], hasMore, nextCursor }\ninterface ApiListResult {\n success: boolean;\n items?: FileEntry[];\n pagination?: { hasMore: boolean; nextCursor?: string | null };\n files?: FileEntry[];\n folders?: string[];\n hasMore?: boolean;\n nextCursor?: string | null;\n}\n\ninterface ApiMetadataResult {\n success: boolean;\n path: string;\n name: string;\n size: number;\n mimeType: string;\n isPublic: boolean;\n publicUrl: string | null;\n downloadUrl: string | null;\n createdAt?: string;\n updatedAt?: string;\n}\n\ninterface ApiSignedUrlResult {\n success: boolean;\n signedUrl: string;\n expiresAt: string;\n expiresIn: number;\n path: string;\n}\n\ninterface ApiVisibilityResult {\n success: boolean;\n path: string;\n isPublic: boolean;\n publicUrl: string | null;\n downloadUrl: string | null;\n}\n\ninterface ApiStatsResult {\n success: boolean;\n stats: StorageStats;\n}\n\n// Server batch responses use { succeeded: [...], failed: [...] }\ninterface ApiBatchUploadUrlResult {\n succeeded: Array<ApiUploadUrlResult & { index: number }>;\n failed: Array<{ index: number; path: string; error: string; code?: string }>;\n}\n\ninterface ApiBatchConfirmResult {\n success: boolean;\n succeeded: ApiUploadResult[];\n failed: Array<{ path: string; error: string; code?: string }>;\n}\n\ninterface ApiBatchDownloadResult {\n success: boolean;\n succeeded: Array<{\n index: number;\n path: string;\n mimeType: string;\n size: number;\n isPublic: boolean;\n content: string; // base64\n }>;\n failed: Array<{ index: number; path: string; error: string; code?: string }>;\n}\n\nfunction toUploadResult(r: ApiUploadResult): UploadResult {\n return {\n path: r.path,\n mimeType: r.mimeType,\n size: r.size,\n isPublic: r.isPublic,\n publicUrl: r.publicUrl,\n downloadUrl: r.downloadUrl,\n };\n}\n\n/**\n * StorageManager — upload, download, list, move, copy, and delete files.\n * Uses an `X-Storage-Key` (`ssk_…`) header — separate from auth and bucket keys.\n *\n * Get a StorageManager via `db.storage('keyName')` where the name matches\n * one of the keys defined in `storageKeys` when calling `createClient`.\n *\n * @example\n * ```ts\n * const avatars = db.storage('avatars');\n * const result = await avatars.upload(file, 'alice.jpg', { isPublic: true });\n * console.log(result.publicUrl);\n * ```\n */\nexport class StorageManager {\n private readonly http: HttpClient;\n private readonly storageKey: string;\n readonly basePath = '/storage';\n\n constructor(http: HttpClient, storageKey: string) {\n this.http = http;\n this.storageKey = storageKey;\n }\n\n private get authHeaders(): Record<string, string> {\n return { 'X-Storage-Key': this.storageKey };\n }\n\n // ─── Upload: Simple (server-buffered) ────────────────────────────────────\n\n /**\n * Upload a file to storage in one step (server-buffered, up to 500 MB).\n * For files >10 MB or when upload progress is needed, use `getUploadUrl()`.\n *\n * @example\n * ```ts\n * const result = await storage.upload(file, 'avatars/alice.jpg', { isPublic: true });\n * console.log(result.publicUrl);\n * ```\n */\n async upload(\n data: Blob | Uint8Array | ArrayBuffer | Buffer,\n path: string,\n options: UploadOptions = {},\n ): Promise<UploadResult> {\n const { isPublic = false, overwrite = false, mimeType } = options;\n const mime = mimeType ?? guessMimeType(path);\n\n const formData = new FormData();\n const blob = data instanceof Blob\n ? data\n : new Blob([data as ArrayBuffer], { type: mime });\n\n formData.append('file', blob, path.split('/').pop() ?? 'file');\n formData.append('path', path);\n formData.append('mimeType', mime);\n formData.append('isPublic', String(isPublic));\n formData.append('overwrite', String(overwrite));\n\n const result = await this.http.request<ApiUploadResult>(`${this.basePath}/upload`, {\n method: 'POST',\n rawBody: formData,\n headers: this.authHeaders,\n });\n return toUploadResult(result);\n }\n\n /**\n * Upload raw JSON or plain-text data as a file.\n *\n * @example\n * ```ts\n * await storage.uploadRaw({ theme: 'dark' }, 'settings/config.json');\n * ```\n */\n async uploadRaw(\n data: unknown,\n path: string,\n options: UploadOptions = {},\n ): Promise<UploadResult> {\n const { isPublic = false, overwrite = false, mimeType = 'application/json' } = options;\n // Server reads `content`, not `data`\n const content = typeof data === 'string' ? data : JSON.stringify(data);\n\n const result = await this.http.request<ApiUploadResult>(`${this.basePath}/upload-raw`, {\n method: 'POST',\n body: { path, content, mimeType, isPublic, overwrite },\n headers: this.authHeaders,\n });\n return toUploadResult(result);\n }\n\n // ─── Upload: Direct-to-GCS (recommended for large files) ─────────────────\n\n /**\n * Step 1 — get a signed GCS URL to upload directly from the client.\n * Supports upload progress via `uploadToSignedUrl()`.\n *\n * @example\n * ```ts\n * const { uploadUrl, path: confirmedPath } = await storage.getUploadUrl({\n * path: 'videos/intro.mp4',\n * mimeType: 'video/mp4',\n * size: file.size,\n * isPublic: true,\n * });\n * await storage.uploadToSignedUrl(uploadUrl, file, 'video/mp4', pct => console.log(pct + '%'));\n * const result = await storage.confirmUpload({ path: confirmedPath, mimeType: 'video/mp4', isPublic: true });\n * ```\n */\n async getUploadUrl(opts: {\n path: string;\n mimeType: string;\n size: number;\n isPublic?: boolean;\n overwrite?: boolean;\n expiresInSeconds?: number;\n }): Promise<UploadUrlResult> {\n const { expiresInSeconds, ...rest } = opts;\n // Server reads `expiresIn` (not `expiresInSeconds`)\n const result = await this.http.request<ApiUploadUrlResult>(`${this.basePath}/upload-url`, {\n method: 'POST',\n body: { isPublic: false, overwrite: false, expiresIn: expiresInSeconds ?? 900, ...rest },\n headers: this.authHeaders,\n });\n return {\n uploadUrl: result.uploadUrl,\n path: result.path,\n mimeType: result.mimeType,\n expiresAt: result.expiresAt,\n expiresIn: result.expiresIn,\n };\n }\n\n /**\n * Step 2 — upload data directly to a signed GCS URL.\n * Supports progress tracking in browser environments.\n */\n async uploadToSignedUrl(\n signedUrl: string,\n data: Blob | Uint8Array | ArrayBuffer,\n mimeType: string,\n onProgress?: (percent: number) => void,\n ): Promise<void> {\n await this.http.putToSignedUrl(signedUrl, data, mimeType, onProgress);\n }\n\n /**\n * Step 3 — confirm a direct upload and register metadata on the server.\n */\n async confirmUpload(opts: {\n path: string;\n mimeType: string;\n isPublic?: boolean;\n }): Promise<UploadResult> {\n const result = await this.http.request<ApiUploadResult>(`${this.basePath}/confirm`, {\n method: 'POST',\n body: { isPublic: false, ...opts },\n headers: this.authHeaders,\n });\n return toUploadResult(result);\n }\n\n // ─── Batch Upload ─────────────────────────────────────────────────────────\n\n /**\n * Get signed upload URLs for multiple files at once (max 50).\n * Server returns `{ succeeded, failed }` — only succeeded items have URLs.\n *\n * @example\n * ```ts\n * const { files } = await storage.getBatchUploadUrls([\n * { path: 'images/a.jpg', mimeType: 'image/jpeg', size: 204800 },\n * { path: 'images/b.jpg', mimeType: 'image/jpeg', size: 153600 },\n * ]);\n * for (const f of files) {\n * await storage.uploadToSignedUrl(f.uploadUrl, blobs[f.index], f.mimeType);\n * await storage.confirmUpload({ path: f.path, mimeType: f.mimeType });\n * }\n * ```\n */\n async getBatchUploadUrls(files: BatchUploadItem[]): Promise<BatchUploadUrlResult> {\n // Server returns { succeeded: [...], failed: [...] }\n const result = await this.http.request<ApiBatchUploadUrlResult>(\n `${this.basePath}/batch-upload-urls`,\n { method: 'POST', body: { files }, headers: this.authHeaders },\n );\n return {\n files: result.succeeded.map(f => ({\n index: f.index,\n uploadUrl: f.uploadUrl,\n path: f.path,\n mimeType: f.mimeType,\n expiresAt: f.expiresAt,\n expiresIn: f.expiresIn,\n })),\n };\n }\n\n /**\n * Confirm multiple direct uploads at once.\n * Returns both succeeded and failed results.\n */\n async batchConfirmUploads(\n items: Array<{ path: string; mimeType: string; isPublic?: boolean }>,\n ): Promise<{ succeeded: UploadResult[]; failed: Array<{ path: string; error: string }> }> {\n // Server returns { succeeded: [...], failed: [...] }\n const result = await this.http.request<ApiBatchConfirmResult>(\n `${this.basePath}/batch-confirm`,\n { method: 'POST', body: { files: items }, headers: this.authHeaders },\n );\n return {\n succeeded: result.succeeded.map(toUploadResult),\n failed: result.failed,\n };\n }\n\n // ─── Download ─────────────────────────────────────────────────────────────\n\n /**\n * Download a private file as an ArrayBuffer.\n * For public files, use the `publicUrl` directly — no SDK needed.\n */\n async download(path: string): Promise<ArrayBuffer> {\n const encoded = path.split('/').map(encodeURIComponent).join('/');\n return this.http.request<ArrayBuffer>(`${this.basePath}/download/${encoded}`, {\n method: 'GET',\n headers: this.authHeaders,\n });\n }\n\n /**\n * Download multiple files at once (max 20).\n * Returns a structured result with succeeded and failed items.\n * Each succeeded item includes the file content as a base64 string.\n *\n * @example\n * ```ts\n * const { succeeded, failed } = await storage.batchDownload(['docs/a.pdf', 'docs/b.pdf']);\n * for (const f of succeeded) {\n * const bytes = Buffer.from(f.content, 'base64');\n * }\n * ```\n */\n async batchDownload(paths: string[]): Promise<BatchDownloadResult> {\n // Server returns { succeeded: [...], failed: [...] } — NOT { files: {...} }\n const result = await this.http.request<ApiBatchDownloadResult>(\n `${this.basePath}/batch-download`,\n { method: 'POST', body: { paths }, headers: this.authHeaders },\n );\n return {\n succeeded: result.succeeded.map(f => ({\n index: f.index,\n path: f.path,\n mimeType: f.mimeType,\n size: f.size,\n content: f.content,\n })),\n failed: result.failed.map(f => ({\n index: f.index,\n path: f.path,\n error: f.error,\n code: f.code,\n })),\n };\n }\n\n // ─── List ─────────────────────────────────────────────────────────────────\n\n /**\n * List files and folders at a given path prefix.\n * Returns separate `files` and `folders` arrays for easy consumption.\n *\n * @example\n * ```ts\n * const { files, folders, hasMore, nextCursor } = await storage.list({ prefix: 'avatars/', limit: 20 });\n * const page2 = await storage.list({ prefix: 'avatars/', cursor: nextCursor });\n * ```\n */\n async list(opts: ListOptions = {}): Promise<ListResult> {\n const params = new URLSearchParams();\n if (opts.prefix) params.set('prefix', opts.prefix);\n if (opts.limit) params.set('limit', String(opts.limit));\n if (opts.cursor) params.set('cursor', opts.cursor);\n const qs = params.toString() ? `?${params}` : '';\n\n // Server returns { items: FileEntry[], pagination: { hasMore, nextCursor } }\n // items has a `type` field: 'file' | 'folder'\n const result = await this.http.request<ApiListResult>(`${this.basePath}/list${qs}`, {\n method: 'GET',\n headers: this.authHeaders,\n });\n\n // Handle both server shape { items, pagination } and legacy/mock shape { files, folders, hasMore }\n if (result.items !== undefined) {\n const files: FileEntry[] = [];\n const folders: string[] = [];\n for (const item of result.items) {\n if (item.type === 'folder') {\n folders.push(item.path);\n } else {\n files.push(item);\n }\n }\n return {\n files,\n folders,\n hasMore: result.pagination?.hasMore ?? false,\n nextCursor: result.pagination?.nextCursor ?? undefined,\n };\n }\n\n // Legacy / mock shape\n return {\n files: result.files ?? [],\n folders: result.folders ?? [],\n hasMore: result.hasMore ?? false,\n nextCursor: result.nextCursor ?? undefined,\n };\n }\n\n // ─── Metadata ─────────────────────────────────────────────────────────────\n\n /**\n * Get metadata for a file: size, MIME type, visibility, URLs.\n */\n async getMetadata(path: string): Promise<FileMetadata> {\n const encoded = path.split('/').map(encodeURIComponent).join('/');\n const result = await this.http.request<ApiMetadataResult>(\n `${this.basePath}/metadata/${encoded}`,\n { method: 'GET', headers: this.authHeaders },\n );\n return {\n path: result.path,\n size: result.size,\n mimeType: result.mimeType,\n isPublic: result.isPublic,\n publicUrl: result.publicUrl,\n downloadUrl: result.downloadUrl,\n createdAt: result.createdAt,\n updatedAt: result.updatedAt,\n };\n }\n\n // ─── Signed URL ───────────────────────────────────────────────────────────\n\n /**\n * Generate a time-limited download URL for a private file.\n * Can be shared externally without requiring an `X-Storage-Key`.\n *\n * > Note: Downloads via signed URLs bypass the server — download stats are NOT tracked.\n *\n * @param path File path.\n * @param expiresIn URL lifetime in seconds (default 3600 = 1 hour).\n */\n async getSignedUrl(path: string, expiresIn = 3600): Promise<SignedUrlResult> {\n const result = await this.http.request<ApiSignedUrlResult>(`${this.basePath}/signed-url`, {\n method: 'POST',\n body: { path, expiresIn },\n headers: this.authHeaders,\n });\n return {\n signedUrl: result.signedUrl,\n expiresAt: result.expiresAt,\n expiresIn: result.expiresIn,\n path: result.path,\n };\n }\n\n // ─── Visibility ───────────────────────────────────────────────────────────\n\n /**\n * Change a file's visibility between public and private after upload.\n */\n async setVisibility(\n path: string,\n isPublic: boolean,\n ): Promise<{ path: string; isPublic: boolean; publicUrl: string | null; downloadUrl: string | null }> {\n const result = await this.http.request<ApiVisibilityResult>(`${this.basePath}/visibility`, {\n method: 'PATCH',\n body: { path, isPublic },\n headers: this.authHeaders,\n });\n return {\n path: result.path,\n isPublic: result.isPublic,\n publicUrl: result.publicUrl,\n downloadUrl: result.downloadUrl,\n };\n }\n\n // ─── Folder Operations ────────────────────────────────────────────────────\n\n async createFolder(path: string): Promise<{ path: string }> {\n const result = await this.http.request<{ success: boolean; path: string }>(\n `${this.basePath}/folder`,\n { method: 'POST', body: { path }, headers: this.authHeaders },\n );\n return { path: result.path };\n }\n\n // ─── File Operations ──────────────────────────────────────────────────────\n\n async deleteFile(path: string): Promise<void> {\n await this.http.request<{ success: boolean }>(`${this.basePath}/file`, {\n method: 'DELETE',\n body: { path },\n headers: this.authHeaders,\n });\n }\n\n async deleteFolder(path: string): Promise<void> {\n await this.http.request<{ success: boolean }>(`${this.basePath}/folder`, {\n method: 'DELETE',\n body: { path },\n headers: this.authHeaders,\n });\n }\n\n async move(from: string, to: string): Promise<{ from: string; to: string }> {\n const result = await this.http.request<{ success: boolean; from: string; to: string }>(\n `${this.basePath}/move`,\n { method: 'POST', body: { from, to }, headers: this.authHeaders },\n );\n return { from: result.from, to: result.to };\n }\n\n async copy(from: string, to: string): Promise<{ from: string; to: string }> {\n const result = await this.http.request<{ success: boolean; from: string; to: string }>(\n `${this.basePath}/copy`,\n { method: 'POST', body: { from, to }, headers: this.authHeaders },\n );\n return { from: result.from, to: result.to };\n }\n\n // ─── Stats ────────────────────────────────────────────────────────────────\n\n async getStats(): Promise<StorageStats> {\n const result = await this.http.request<ApiStatsResult>(`${this.basePath}/stats`, {\n method: 'GET',\n headers: this.authHeaders,\n });\n return result.stats;\n }\n\n // ─── Info (no auth) ───────────────────────────────────────────────────────\n\n async info(): Promise<{ ok: boolean; storageRoot: string }> {\n return this.http.request<{ ok: boolean; storageRoot: string }>(`${this.basePath}/info`, {\n method: 'GET',\n });\n }\n}\n","import type { StorageManager } from './manager.js';\nimport type {\n UploadOptions,\n UploadResult,\n UploadUrlResult,\n ListOptions,\n ListResult,\n FileMetadata,\n SignedUrlResult,\n StorageStats,\n BatchUploadItem,\n BatchUploadUrlResult,\n BatchDownloadResult,\n} from '../types/index.js';\n\n/**\n * ScopedStorage — a path-prefixed view over a StorageManager.\n * All paths you pass are automatically prefixed with the scope.\n *\n * Obtain via `db.storage('keyName').scope('prefix/')`.\n *\n * @example\n * ```ts\n * const userDocs = db.storage('documents').scope(`users/${userId}/`);\n *\n * // Uploads to: users/{userId}/contract.pdf\n * await userDocs.upload(pdfBuffer, 'contract.pdf');\n *\n * // Lists: users/{userId}/\n * const { files } = await userDocs.list();\n * ```\n */\nexport class ScopedStorage {\n private readonly manager: StorageManager;\n private readonly prefix: string;\n\n constructor(manager: StorageManager, prefix: string) {\n this.manager = manager;\n // Normalise: ensure exactly one trailing slash\n this.prefix = prefix.endsWith('/') ? prefix : `${prefix}/`;\n }\n\n private p(path: string): string {\n // Strip any leading slash the caller might add, then prepend scope\n return `${this.prefix}${path.replace(/^\\/+/, '')}`;\n }\n\n // ─── Upload ───────────────────────────────────────────────────────────────\n\n async upload(\n data: Blob | Uint8Array | ArrayBuffer | Buffer,\n path: string,\n options: UploadOptions = {},\n ): Promise<UploadResult> {\n return this.manager.upload(data, this.p(path), options);\n }\n\n async uploadRaw(\n data: unknown,\n path: string,\n options: UploadOptions = {},\n ): Promise<UploadResult> {\n return this.manager.uploadRaw(data, this.p(path), options);\n }\n\n async getUploadUrl(opts: {\n path: string;\n mimeType: string;\n size: number;\n isPublic?: boolean;\n overwrite?: boolean;\n expiresInSeconds?: number;\n }): Promise<UploadUrlResult> {\n return this.manager.getUploadUrl({ ...opts, path: this.p(opts.path) });\n }\n\n async uploadToSignedUrl(\n signedUrl: string,\n data: Blob | Uint8Array | ArrayBuffer,\n mimeType: string,\n onProgress?: (percent: number) => void,\n ): Promise<void> {\n return this.manager.uploadToSignedUrl(signedUrl, data, mimeType, onProgress);\n }\n\n async confirmUpload(opts: {\n path: string;\n mimeType: string;\n isPublic?: boolean;\n }): Promise<UploadResult> {\n return this.manager.confirmUpload({ ...opts, path: this.p(opts.path) });\n }\n\n async getBatchUploadUrls(files: BatchUploadItem[]): Promise<BatchUploadUrlResult> {\n return this.manager.getBatchUploadUrls(\n files.map(f => ({ ...f, path: this.p(f.path) })),\n );\n }\n\n async batchConfirmUploads(\n items: Array<{ path: string; mimeType: string; isPublic?: boolean }>,\n ): Promise<{ succeeded: UploadResult[]; failed: Array<{ path: string; error: string }> }> {\n return this.manager.batchConfirmUploads(\n items.map(i => ({ ...i, path: this.p(i.path) })),\n );\n }\n\n // ─── Download ─────────────────────────────────────────────────────────────\n\n async download(path: string): Promise<ArrayBuffer> {\n return this.manager.download(this.p(path));\n }\n\n async batchDownload(paths: string[]): Promise<BatchDownloadResult> {\n return this.manager.batchDownload(paths.map(p => this.p(p)));\n }\n\n // ─── List ─────────────────────────────────────────────────────────────────\n\n /**\n * List files within this scope. The `prefix` option is relative to the scope root.\n *\n * @example\n * ```ts\n * const userDocs = storage.scope('users/alice/');\n * // Lists users/alice/docs/\n * const { files } = await userDocs.list({ prefix: 'docs/' });\n * ```\n */\n async list(opts: ListOptions = {}): Promise<ListResult> {\n return this.manager.list({\n ...opts,\n prefix: this.p(opts.prefix ?? ''),\n });\n }\n\n // ─── Metadata & URLs ──────────────────────────────────────────────────────\n\n async getMetadata(path: string): Promise<FileMetadata> {\n return this.manager.getMetadata(this.p(path));\n }\n\n async getSignedUrl(path: string, expiresIn = 3600): Promise<SignedUrlResult> {\n return this.manager.getSignedUrl(this.p(path), expiresIn);\n }\n\n async setVisibility(\n path: string,\n isPublic: boolean,\n ): Promise<{ path: string; isPublic: boolean; publicUrl: string | null; downloadUrl: string | null }> {\n return this.manager.setVisibility(this.p(path), isPublic);\n }\n\n // ─── Folder Operations ────────────────────────────────────────────────────\n\n async createFolder(path: string): Promise<{ path: string }> {\n return this.manager.createFolder(this.p(path));\n }\n\n // ─── File Operations ──────────────────────────────────────────────────────\n\n async deleteFile(path: string): Promise<void> {\n return this.manager.deleteFile(this.p(path));\n }\n\n async deleteFolder(path: string): Promise<void> {\n return this.manager.deleteFolder(this.p(path));\n }\n\n async move(from: string, to: string): Promise<{ from: string; to: string }> {\n return this.manager.move(this.p(from), this.p(to));\n }\n\n async copy(from: string, to: string): Promise<{ from: string; to: string }> {\n return this.manager.copy(this.p(from), this.p(to));\n }\n\n // ─── Stats ────────────────────────────────────────────────────────────────\n\n async getStats(): Promise<StorageStats> {\n return this.manager.getStats();\n }\n\n // ─── Nesting ──────────────────────────────────────────────────────────────\n\n /**\n * Create a deeper scope within this one.\n *\n * @example\n * ```ts\n * const user = storage.scope('users/alice/');\n * const userDocs = user.scope('docs/');\n * // Effective prefix: users/alice/docs/\n * ```\n */\n scope(subPrefix: string): ScopedStorage {\n return new ScopedStorage(this.manager, this.p(subPrefix));\n }\n}\n","import { HttpClient, DEFAULT_BASE_URL } from './utils/http.js';\nimport { AuthClient } from './auth/client.js';\nimport { RecordsClient } from './records/client.js';\nimport { AnalyticsClient } from './analytics/client.js';\nimport { StorageManager } from './storage/manager.js';\nimport { ScopedStorage } from './storage/scoped.js';\nimport type { HydrousConfig, RecordData } from './types/index.js';\n\n/**\n * HydrousClient — the main entry point for the HydrousDB SDK.\n *\n * Each service uses its own dedicated API key:\n * - `authKey` (`hk_auth_…`) → `db.auth('bucket')`\n * - `bucketSecurityKey` (`hk_bucket_…`) → `db.records('bucket')` + `db.analytics('bucket')`\n * - `storageKeys` (`ssk_…`) → `db.storage('keyName')`\n *\n * Sub-clients are cached — calling `db.records('posts')` twice returns the same instance.\n *\n * @example\n * ```ts\n * import { createClient } from 'hydrousdb';\n *\n * const db = createClient({\n * authKey: process.env.HYDROUS_AUTH_KEY!,\n * bucketSecurityKey: process.env.HYDROUS_BUCKET_KEY!,\n * storageKeys: {\n * avatars: process.env.HYDROUS_STORAGE_AVATARS!,\n * documents: process.env.HYDROUS_STORAGE_DOCS!,\n * },\n * });\n *\n * const posts = db.records('blog-posts');\n * const auth = db.auth('app-users');\n * const analytics = db.analytics('orders');\n * const avatars = db.storage('avatars');\n * ```\n */\nexport class HydrousClient {\n private readonly http: HttpClient;\n private readonly authKey_: string;\n private readonly bucketSecurityKey_: string;\n private readonly storageKeys_: Record<string, string>;\n\n private readonly _recordsCache = new Map<string, RecordsClient<RecordData>>();\n private readonly _authCache = new Map<string, AuthClient>();\n private readonly _analyticsCache = new Map<string, AnalyticsClient>();\n private readonly _storageCache = new Map<string, StorageManager>();\n\n constructor(config: HydrousConfig) {\n if (!config.authKey) {\n throw new Error('[HydrousDB] authKey is required. Get yours from https://hydrousdb.com/dashboard.');\n }\n if (!config.bucketSecurityKey) {\n throw new Error('[HydrousDB] bucketSecurityKey is required. Get yours from https://hydrousdb.com/dashboard.');\n }\n if (!config.storageKeys || Object.keys(config.storageKeys).length === 0) {\n throw new Error('[HydrousDB] storageKeys is required. Define at least one storage key from https://hydrousdb.com/dashboard.');\n }\n\n this.http = new HttpClient(config.baseUrl ?? DEFAULT_BASE_URL);\n this.authKey_ = config.authKey;\n this.bucketSecurityKey_ = config.bucketSecurityKey;\n this.storageKeys_ = config.storageKeys;\n }\n\n // ─── Records ──────────────────────────────────────────────────────────────\n\n /**\n * Get a typed records client for the named bucket.\n *\n * @example\n * ```ts\n * interface Post { title: string; published: boolean }\n * const posts = db.records<Post>('blog-posts');\n * const post = await posts.create(\n * { title: 'Hello', published: false },\n * { queryableFields: ['published'] },\n * );\n * ```\n */\n records<T extends RecordData = RecordData>(bucketKey: string): RecordsClient<T> {\n if (!this._recordsCache.has(bucketKey)) {\n this._recordsCache.set(\n bucketKey,\n new RecordsClient<T>(this.http, this.bucketSecurityKey_, bucketKey) as RecordsClient<RecordData>,\n );\n }\n return this._recordsCache.get(bucketKey) as RecordsClient<T>;\n }\n\n // ─── Auth ─────────────────────────────────────────────────────────────────\n\n /**\n * Get an auth client for the named user bucket.\n *\n * @example\n * ```ts\n * const auth = db.auth('app-users');\n * const { user, session } = await auth.login({ email: '…', password: '…' });\n * ```\n */\n auth(bucketKey: string): AuthClient {\n if (!this._authCache.has(bucketKey)) {\n this._authCache.set(bucketKey, new AuthClient(this.http, this.authKey_, bucketKey));\n }\n return this._authCache.get(bucketKey)!;\n }\n\n // ─── Analytics ────────────────────────────────────────────────────────────\n\n /**\n * Get an analytics client for the named bucket.\n *\n * @example\n * ```ts\n * const analytics = db.analytics('orders');\n * const { count } = await analytics.count();\n * ```\n */\n analytics(bucketKey: string): AnalyticsClient {\n if (!this._analyticsCache.has(bucketKey)) {\n this._analyticsCache.set(\n bucketKey,\n new AnalyticsClient(this.http, this.bucketSecurityKey_, bucketKey),\n );\n }\n return this._analyticsCache.get(bucketKey)!;\n }\n\n // ─── Storage ──────────────────────────────────────────────────────────────\n\n /**\n * Get a storage manager for the named storage key.\n * The name must match a key defined in `storageKeys` when calling `createClient`.\n *\n * Attach `.scope(prefix)` to namespace all operations under a path prefix.\n *\n * @example\n * ```ts\n * const avatars = db.storage('avatars');\n * const userDocs = db.storage('documents').scope(`users/${userId}/`);\n *\n * await avatars.upload(file, `${userId}.jpg`, { isPublic: true });\n * await userDocs.upload(pdfBuffer, 'contract.pdf');\n * ```\n */\n storage(keyName: string): StorageManager & { scope: (prefix: string) => ScopedStorage } {\n const ssk = this.storageKeys_[keyName];\n if (!ssk) {\n const available = Object.keys(this.storageKeys_).join(', ');\n throw new Error(\n `[HydrousDB] Unknown storage key name \"${keyName}\". ` +\n `Available keys: ${available}. ` +\n `Add it to storageKeys in your createClient() config.`,\n );\n }\n\n if (!this._storageCache.has(keyName)) {\n this._storageCache.set(keyName, new StorageManager(this.http, ssk));\n }\n\n const mgr = this._storageCache.get(keyName)!;\n const extended = mgr as StorageManager & { scope: (prefix: string) => ScopedStorage };\n if (!extended.scope) {\n extended.scope = (prefix: string) => new ScopedStorage(mgr, prefix);\n }\n return extended;\n }\n}\n\n// ─── Factory ──────────────────────────────────────────────────────────────────\n\n/**\n * Create a new HydrousDB client.\n *\n * @example\n * ```ts\n * import { createClient } from 'hydrousdb';\n *\n * const db = createClient({\n * authKey: process.env.HYDROUS_AUTH_KEY!,\n * bucketSecurityKey: process.env.HYDROUS_BUCKET_KEY!,\n * storageKeys: { main: process.env.HYDROUS_STORAGE_MAIN! },\n * });\n * ```\n */\nexport function createClient(config: HydrousConfig): HydrousClient {\n return new HydrousClient(config);\n}\n"]}
|
package/package.json
CHANGED
|
@@ -1,16 +1,17 @@
|
|
|
1
1
|
{
|
|
2
2
|
"name": "hydrousdb",
|
|
3
|
-
"version": "3.0
|
|
3
|
+
"version": "3.2.0",
|
|
4
4
|
"type": "module",
|
|
5
|
-
"description": "Official JavaScript/TypeScript SDK for HydrousDB
|
|
6
|
-
"main": "./dist/index.
|
|
5
|
+
"description": "Official JavaScript/TypeScript SDK for HydrousDB \u2014 the backend-as-a-service platform for records, auth, storage, and analytics.",
|
|
6
|
+
"main": "./dist/index.cjs",
|
|
7
7
|
"module": "./dist/index.mjs",
|
|
8
8
|
"types": "./dist/index.d.ts",
|
|
9
9
|
"exports": {
|
|
10
10
|
".": {
|
|
11
|
-
|
|
12
|
-
"import": "./dist/index.mjs",
|
|
13
|
-
"require": "./dist/index.
|
|
11
|
+
"types": "./dist/index.d.ts",
|
|
12
|
+
"import": "./dist/index.mjs",
|
|
13
|
+
"require": "./dist/index.cjs",
|
|
14
|
+
"default": "./dist/index.mjs"
|
|
14
15
|
}
|
|
15
16
|
},
|
|
16
17
|
"files": [
|
package/dist/index.js.map
DELETED
|
@@ -1 +0,0 @@
|
|
|
1
|
-
{"version":3,"sources":["../src/utils/errors.ts","../src/utils/http.ts","../src/auth/client.ts","../src/utils/query.ts","../src/records/client.ts","../src/analytics/client.ts","../src/storage/manager.ts","../src/storage/scoped.ts","../src/client.ts"],"names":[],"mappings":"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","file":"index.js","sourcesContent":["export class HydrousError extends Error {\n /** Machine-readable error code returned by the API. */\n readonly code: string;\n /** HTTP status code (if applicable). */\n readonly status?: number;\n /** The original request ID for support tracing. */\n readonly requestId?: string;\n /** Additional validation details. */\n readonly details?: string[];\n\n constructor(\n message: string,\n code: string,\n status?: number,\n requestId?: string,\n details?: string[],\n ) {\n super(message);\n this.name = 'HydrousError';\n this.code = code;\n this.status = status;\n this.requestId = requestId;\n this.details = details;\n // Maintain proper prototype chain in transpiled environments\n Object.setPrototypeOf(this, new.target.prototype);\n }\n\n toString(): string {\n return `HydrousError [${this.code}]: ${this.message}`;\n }\n}\n\nexport class AuthError extends HydrousError {\n constructor(\n message: string,\n code: string,\n status?: number,\n requestId?: string,\n details?: string[],\n ) {\n super(message, code, status, requestId, details);\n this.name = 'AuthError';\n }\n}\n\nexport class RecordError extends HydrousError {\n constructor(\n message: string,\n code: string,\n status?: number,\n requestId?: string,\n details?: string[],\n ) {\n super(message, code, status, requestId, details);\n this.name = 'RecordError';\n }\n}\n\nexport class StorageError extends HydrousError {\n constructor(\n message: string,\n code: string,\n status?: number,\n requestId?: string,\n ) {\n super(message, code, status, requestId);\n this.name = 'StorageError';\n }\n}\n\nexport class AnalyticsError extends HydrousError {\n constructor(\n message: string,\n code: string,\n status?: number,\n requestId?: string,\n ) {\n super(message, code, status, requestId);\n this.name = 'AnalyticsError';\n }\n}\n\nexport class ValidationError extends HydrousError {\n constructor(message: string, details?: string[]) {\n super(message, 'VALIDATION_ERROR', 400, undefined, details);\n this.name = 'ValidationError';\n }\n}\n\nexport class NetworkError extends HydrousError {\n readonly cause?: unknown;\n\n constructor(message: string, cause?: unknown) {\n super(message, 'NETWORK_ERROR');\n this.name = 'NetworkError';\n if (cause) this.cause = cause;\n }\n}","import { HydrousError, NetworkError } from './errors.js';\n\nexport const DEFAULT_BASE_URL = 'https://db-api-82687684612.us-central1.run.app';\n\nexport interface RequestOptions {\n method?: 'GET' | 'POST' | 'PUT' | 'PATCH' | 'DELETE';\n body?: unknown;\n headers?: Record<string, string>;\n rawBody?: string | FormData | Uint8Array<ArrayBuffer>;\n contentType?: string;\n}\n\nexport class HttpClient {\n private readonly baseUrl: string;\n\n constructor(baseUrl: string) {\n this.baseUrl = baseUrl.replace(/\\/$/, '');\n }\n\n async request<T = unknown>(\n path: string,\n apiKeyOrOpts?: string | RequestOptions,\n opts: RequestOptions = {},\n ): Promise<T> {\n let apiKey: string | undefined;\n let resolvedOpts: RequestOptions;\n\n if (typeof apiKeyOrOpts === 'string') {\n apiKey = apiKeyOrOpts;\n resolvedOpts = opts;\n } else {\n apiKey = undefined;\n resolvedOpts = apiKeyOrOpts ?? opts;\n }\n\n const {\n method = 'GET',\n body,\n headers = {},\n rawBody,\n contentType = 'application/json',\n } = resolvedOpts;\n\n const url = `${this.baseUrl}${path}`;\n\n const reqHeaders: Record<string, string> = {\n ...(apiKey ? { 'X-Api-Key': apiKey } : {}),\n ...headers,\n };\n\n let reqBody: string | FormData | Uint8Array<ArrayBuffer> | null = null;\n\n if (rawBody !== undefined) {\n reqBody = rawBody;\n if (contentType) reqHeaders['Content-Type'] = contentType;\n } else if (body !== undefined) {\n reqBody = JSON.stringify(body);\n reqHeaders['Content-Type'] = 'application/json';\n }\n\n let response: Response;\n try {\n response = await fetch(url, {\n method,\n headers: reqHeaders,\n body: reqBody,\n });\n } catch (err) {\n throw new NetworkError(\n `Network request failed: ${err instanceof Error ? err.message : String(err)}`,\n err,\n );\n }\n\n const ct = response.headers.get('content-type') ?? '';\n\n if (!ct.includes('application/json')) {\n if (!response.ok) {\n throw new HydrousError(\n `Request failed with status ${response.status}`,\n `HTTP_${response.status}`,\n response.status,\n );\n }\n const buffer = await response.arrayBuffer();\n return buffer as T;\n }\n\n let responseData: unknown;\n try {\n responseData = await response.json() as unknown;\n } catch {\n throw new HydrousError(\n 'Failed to parse server response as JSON',\n 'PARSE_ERROR',\n response.status,\n );\n }\n\n if (!response.ok) {\n const errData = responseData as Record<string, unknown>;\n throw new HydrousError(\n (errData['error'] as string | undefined) ?? `Request failed with status ${response.status}`,\n (errData['code'] as string | undefined) ?? `HTTP_${response.status}`,\n response.status,\n errData['requestId'] as string | undefined,\n errData['details'] as string[] | undefined,\n );\n }\n\n return responseData as T;\n }\n\n get<T = unknown>(path: string, apiKey?: string, headers?: Record<string, string>): Promise<T> {\n return this.request<T>(path, apiKey, { method: 'GET', headers });\n }\n\n post<T = unknown>(path: string, apiKey?: string, body?: unknown, headers?: Record<string, string>): Promise<T> {\n return this.request<T>(path, apiKey, { method: 'POST', body, headers });\n }\n\n put<T = unknown>(path: string, apiKey?: string, body?: unknown, headers?: Record<string, string>): Promise<T> {\n return this.request<T>(path, apiKey, { method: 'PUT', body, headers });\n }\n\n patch<T = unknown>(path: string, apiKey?: string, body?: unknown, headers?: Record<string, string>): Promise<T> {\n return this.request<T>(path, apiKey, { method: 'PATCH', body, headers });\n }\n\n delete<T = unknown>(path: string, apiKey?: string, body?: unknown, headers?: Record<string, string>): Promise<T> {\n return this.request<T>(path, apiKey, { method: 'DELETE', body, headers });\n }\n\n async putToSignedUrl(\n signedUrl: string,\n data: Blob | Uint8Array<ArrayBuffer> | ArrayBuffer,\n mimeType: string,\n onProgress?: (percent: number) => void,\n ): Promise<void> {\n // Access XHR via globalThis to avoid requiring DOM lib types\n // eslint-disable-next-line @typescript-eslint/no-explicit-any\n const XHR = (typeof (globalThis as any)['XMLHttpRequest'] !== 'undefined')\n // eslint-disable-next-line @typescript-eslint/no-explicit-any\n ? (globalThis as any)['XMLHttpRequest'] as new () => any\n : undefined;\n\n if (XHR && onProgress) {\n return new Promise<void>((resolve, reject) => {\n const xhr = new XHR();\n\n xhr.upload.onprogress = (e: { lengthComputable: boolean; loaded: number; total: number }) => {\n if (e.lengthComputable) {\n onProgress(Math.round((e.loaded / e.total) * 100));\n }\n };\n\n xhr.onload = () => {\n if (xhr.status >= 200 && xhr.status < 300) {\n resolve();\n } else {\n reject(new Error(`Upload failed: ${xhr.status}`));\n }\n };\n\n xhr.onerror = () => reject(new Error('Upload network error'));\n xhr.open('PUT', signedUrl);\n xhr.setRequestHeader('Content-Type', mimeType);\n\n const payload: Blob = data instanceof Blob\n ? data\n : new Blob([data as ArrayBuffer], { type: mimeType });\n\n xhr.send(payload);\n });\n }\n\n const fetchBody: Blob = data instanceof Blob\n ? data\n : new Blob([data as ArrayBuffer], { type: mimeType });\n\n const response = await fetch(signedUrl, {\n method: 'PUT',\n headers: { 'Content-Type': mimeType },\n body: fetchBody,\n });\n\n if (!response.ok) {\n throw new NetworkError(`Signed URL upload failed with status ${response.status}`);\n }\n }\n}","import type { HttpClient } from '../utils/http.js';\nimport type {\n SignupOptions,\n LoginOptions,\n AuthResult,\n UserRecord,\n UpdateUserOptions,\n ChangePasswordOptions,\n ListUsersOptions,\n ListUsersResult,\n Session,\n} from '../types/index.js';\n\ninterface ApiAuthResult {\n success: boolean;\n user: UserRecord;\n session: Session;\n requestId?: string;\n}\n\ninterface ApiUserResult {\n success: boolean;\n user: UserRecord;\n requestId?: string;\n}\n\ninterface ApiSessionResult {\n success: boolean;\n session: Session;\n requestId?: string;\n}\n\ninterface ApiMessageResult {\n success: boolean;\n message: string;\n meta?: Record<string, unknown>;\n requestId?: string;\n}\n\n/**\n * AuthClient — full user authentication for a single bucket.\n * Uses the `authKey` (`hk_auth_…`) sent via `X-Api-Key` header.\n *\n * @example\n * ```ts\n * const db = createClient({ authKey: 'hk_auth_…', bucketSecurityKey: '…', storageKeys: { main: '…' } });\n * const auth = db.auth('my-app-users');\n * const { user, session } = await auth.signup({ email: 'alice@example.com', password: 'hunter2' });\n * ```\n */\nexport class AuthClient {\n private readonly http: HttpClient;\n private readonly authKey: string;\n private readonly basePath: string;\n\n constructor(http: HttpClient, authKey: string, bucketKey: string) {\n this.http = http;\n this.authKey = authKey;\n this.basePath = `/auth/${bucketKey}`;\n }\n\n private post<T>(path: string, body?: unknown): Promise<T> {\n return this.http.post<T>(path, this.authKey, body);\n }\n\n private get<T>(path: string): Promise<T> {\n return this.http.get<T>(path, this.authKey);\n }\n\n private patch<T>(path: string, body?: unknown): Promise<T> {\n return this.http.patch<T>(path, this.authKey, body);\n }\n\n private delete<T>(path: string, body?: unknown): Promise<T> {\n return this.http.delete<T>(path, this.authKey, body);\n }\n\n // ─── Registration & Login ─────────────────────────────────────────────────\n\n async signup(options: SignupOptions): Promise<AuthResult> {\n const result = await this.post<ApiAuthResult>(`${this.basePath}/signup`, options);\n return { user: result.user, session: result.session };\n }\n\n async login(options: LoginOptions): Promise<AuthResult> {\n const result = await this.post<ApiAuthResult>(`${this.basePath}/login`, options);\n return { user: result.user, session: result.session };\n }\n\n async logout({ sessionId }: { sessionId: string }): Promise<void> {\n await this.post<ApiMessageResult>(`${this.basePath}/logout`, { sessionId });\n }\n\n async refreshSession({ refreshToken }: { refreshToken: string }): Promise<Session> {\n const result = await this.post<ApiSessionResult>(`${this.basePath}/session/refresh`, { refreshToken });\n return result.session;\n }\n\n // ─── User Profile ─────────────────────────────────────────────────────────\n\n async getUser({ userId }: { userId: string }): Promise<UserRecord> {\n const result = await this.get<ApiUserResult>(`${this.basePath}/user/${userId}`);\n return result.user;\n }\n\n async updateUser(options: UpdateUserOptions): Promise<UserRecord> {\n const { sessionId, userId, data } = options;\n const result = await this.patch<ApiUserResult>(`${this.basePath}/user`, { sessionId, userId, ...data });\n return result.user;\n }\n\n async deleteUser({ sessionId, userId }: { sessionId: string; userId: string }): Promise<void> {\n await this.delete<ApiMessageResult>(`${this.basePath}/user`, { sessionId, userId });\n }\n\n // ─── Admin Operations ─────────────────────────────────────────────────────\n\n async listUsers(options: ListUsersOptions): Promise<ListUsersResult> {\n const { sessionId, limit = 50, offset = 0 } = options;\n const result = await this.post<{ success: boolean } & ListUsersResult>(\n `${this.basePath}/users/list`,\n { sessionId, limit, offset },\n );\n return { users: result.users, total: result.total, limit: result.limit, offset: result.offset };\n }\n\n async hardDeleteUser({ sessionId, userId }: { sessionId: string; userId: string }): Promise<void> {\n await this.delete<ApiMessageResult>(`${this.basePath}/user/hard`, { sessionId, userId });\n }\n\n async bulkDeleteUsers({ sessionId, userIds }: { sessionId: string; userIds: string[] }): Promise<{ deleted: number; failed: string[] }> {\n const result = await this.post<{ success: boolean; deleted: number; failed: string[] }>(\n `${this.basePath}/users/bulk-delete`,\n { sessionId, userIds },\n );\n return { deleted: result.deleted, failed: result.failed };\n }\n\n async lockAccount({ sessionId, userId, duration }: { sessionId: string; userId: string; duration?: number }): Promise<{ lockedUntil: number; unlockTime: string }> {\n const result = await this.post<{ success: boolean; data: { lockedUntil: number; unlockTime: string } }>(\n `${this.basePath}/account/lock`,\n { sessionId, userId, duration },\n );\n return result.data;\n }\n\n async unlockAccount({ sessionId, userId }: { sessionId: string; userId: string }): Promise<void> {\n await this.post<ApiMessageResult>(`${this.basePath}/account/unlock`, { sessionId, userId });\n }\n\n // ─── Password Management ──────────────────────────────────────────────────\n\n async changePassword(options: ChangePasswordOptions): Promise<void> {\n await this.post<ApiMessageResult>(`${this.basePath}/password/change`, options);\n }\n\n async requestPasswordReset({ email }: { email: string }): Promise<void> {\n await this.post<ApiMessageResult>(`${this.basePath}/password/reset/request`, { email });\n }\n\n async confirmPasswordReset({ resetToken, newPassword }: { resetToken: string; newPassword: string }): Promise<void> {\n await this.post<ApiMessageResult>(`${this.basePath}/password/reset/confirm`, { resetToken, newPassword });\n }\n\n // ─── Email Verification ───────────────────────────────────────────────────\n\n async requestEmailVerification({ userId }: { userId: string }): Promise<void> {\n await this.post<ApiMessageResult>(`${this.basePath}/email/verify/request`, { userId });\n }\n\n async confirmEmailVerification({ verifyToken }: { verifyToken: string }): Promise<void> {\n await this.post<ApiMessageResult>(`${this.basePath}/email/verify/confirm`, { verifyToken });\n }\n}\n","import type { QueryFilter, QueryOptions } from '../types/index.js';\n\n/**\n * Build query string parameters for GET /records/:bucket requests.\n */\nexport function buildQueryParams(options: QueryOptions = {}): string {\n const params = new URLSearchParams();\n\n if (options.limit !== undefined) params.set('limit', String(options.limit));\n if (options.offset !== undefined) params.set('offset', String(options.offset));\n if (options.orderBy !== undefined) params.set('orderBy', options.orderBy);\n if (options.order !== undefined) params.set('order', options.order);\n if (options.fields !== undefined) params.set('fields', options.fields);\n if (options.startAfter !== undefined) params.set('startAfter', options.startAfter);\n if (options.startAt !== undefined) params.set('startAt', options.startAt);\n if (options.endAt !== undefined) params.set('endAt', options.endAt);\n\n if (options.dateRange?.start !== undefined)\n params.set('startDate', new Date(options.dateRange.start).toISOString().split('T')[0]!);\n if (options.dateRange?.end !== undefined)\n params.set('endDate', new Date(options.dateRange.end).toISOString().split('T')[0]!);\n\n if (options.filters && options.filters.length > 0) {\n params.set('filters', JSON.stringify(options.filters));\n }\n\n const str = params.toString();\n return str ? `?${str}` : '';\n}\n\n/**\n * Normalise a filter shorthand into the standard { field, op, value } shape.\n * Accepts either a QueryFilter object or a 3-tuple [field, op, value].\n */\nexport function normaliseFilter(\n filter: QueryFilter | [string, QueryFilter['op'], QueryFilter['value']],\n): QueryFilter {\n if (Array.isArray(filter)) {\n const [field, op, value] = filter;\n return { field, op, value };\n }\n return filter;\n}\n\n/**\n * Detect a likely MIME type from a file extension.\n * Used when mimeType is not explicitly provided.\n */\nexport function guessMimeType(filename: string): string {\n const ext = filename.split('.').pop()?.toLowerCase();\n const map: Record<string, string> = {\n jpg: 'image/jpeg',\n jpeg: 'image/jpeg',\n png: 'image/png',\n gif: 'image/gif',\n webp: 'image/webp',\n svg: 'image/svg+xml',\n pdf: 'application/pdf',\n mp4: 'video/mp4',\n webm: 'video/webm',\n mp3: 'audio/mpeg',\n wav: 'audio/wav',\n txt: 'text/plain',\n html: 'text/html',\n css: 'text/css',\n js: 'application/javascript',\n json: 'application/json',\n xml: 'application/xml',\n zip: 'application/zip',\n csv: 'text/csv',\n };\n return map[ext ?? ''] ?? 'application/octet-stream';\n}\n\n/**\n * Validate that a bucket key contains only safe characters.\n * Must match the server-side SAFE_RE: /^[a-zA-Z_][a-zA-Z0-9_.]{0,200}$/\n */\nexport function assertSafeName(name: string, label = 'name'): void {\n if (!/^[a-zA-Z_][a-zA-Z0-9_.\\-]{0,200}$/.test(name)) {\n throw new Error(\n `Invalid ${label} \"${name}\". Must start with a letter or underscore and contain only letters, numbers, underscores, dots, or hyphens.`,\n );\n }\n}\n","import type { HttpClient } from '../utils/http.js';\nimport { buildQueryParams, assertSafeName } from '../utils/query.js';\nimport type {\n RecordData,\n RecordResult,\n QueryOptions,\n QueryResult,\n PatchRecordOptions,\n RecordHistoryEntry,\n} from '../types/index.js';\n\ninterface ApiRecordResult<T> {\n success: boolean;\n record?: T & RecordResult;\n data?: T & RecordResult;\n}\n\ninterface ApiQueryResult<T> {\n success: boolean;\n records: (T & RecordResult)[];\n total?: number;\n hasMore?: boolean;\n nextCursor?: string;\n}\n\ninterface ApiDeleteResult {\n success: boolean;\n deleted: boolean;\n id: string;\n}\n\ninterface ApiHistoryResult {\n success: boolean;\n history: RecordHistoryEntry[];\n}\n\n/**\n * RecordsClient — CRUD + query for a single bucket.\n * Uses the `bucketSecurityKey` (`hk_bucket_…`) sent via `X-Api-Key` header.\n *\n * @example\n * ```ts\n * const db = createClient({ authKey: '…', bucketSecurityKey: 'hk_bucket_…', storageKeys: { main: '…' } });\n * const posts = db.records('blog-posts');\n * const post = await posts.create({ title: 'Hello World', status: 'draft' });\n * ```\n */\nexport class RecordsClient<T extends RecordData = RecordData> {\n private readonly http: HttpClient;\n private readonly bucketKey: string;\n private readonly basePath: string;\n private readonly bucketKey_: string;\n\n constructor(http: HttpClient, bucketSecurityKey: string, bucketKey: string) {\n assertSafeName(bucketKey, 'bucketKey');\n this.http = http;\n this.bucketKey = bucketKey;\n this.bucketKey_ = bucketSecurityKey;\n this.basePath = `/records/${bucketKey}`;\n }\n\n private get key(): string { return this.bucketKey_; }\n\n // ─── Single Record Operations ─────────────────────────────────────────────\n\n async create(data: T): Promise<T & RecordResult> {\n const result = await this.http.post<ApiRecordResult<T>>(this.basePath, this.key, data);\n return (result.record ?? result.data)!;\n }\n\n async get(id: string): Promise<T & RecordResult> {\n const result = await this.http.get<ApiRecordResult<T>>(`${this.basePath}/${id}`, this.key);\n return (result.record ?? result.data)!;\n }\n\n async set(id: string, data: T): Promise<T & RecordResult> {\n const result = await this.http.put<ApiRecordResult<T>>(`${this.basePath}/${id}`, this.key, data);\n return (result.record ?? result.data)!;\n }\n\n async patch(id: string, data: Partial<T>, options: PatchRecordOptions = {}): Promise<T & RecordResult> {\n const { merge = true } = options;\n const result = await this.http.patch<ApiRecordResult<T>>(\n `${this.basePath}/${id}`,\n this.key,\n { ...data, _merge: merge },\n );\n return (result.record ?? result.data)!;\n }\n\n async delete(id: string): Promise<void> {\n await this.http.delete<ApiDeleteResult>(`${this.basePath}/${id}`, this.key);\n }\n\n // ─── Batch Operations ─────────────────────────────────────────────────────\n\n async batchCreate(items: T[]): Promise<(T & RecordResult)[]> {\n const result = await this.http.post<{ success: boolean; records: (T & RecordResult)[] }>(\n `${this.basePath}/batch`,\n this.key,\n { records: items },\n );\n return result.records;\n }\n\n async batchDelete(ids: string[]): Promise<{ deleted: number; failed: string[] }> {\n const result = await this.http.post<{ success: boolean; deleted: number; failed: string[] }>(\n `${this.basePath}/batch/delete`,\n this.key,\n { ids },\n );\n return { deleted: result.deleted, failed: result.failed };\n }\n\n // ─── Querying ─────────────────────────────────────────────────────────────\n\n async query(options: QueryOptions = {}): Promise<QueryResult<T>> {\n const qs = buildQueryParams(options);\n const result = await this.http.get<ApiQueryResult<T>>(`${this.basePath}${qs}`, this.key);\n return {\n records: result.records,\n total: result.total,\n hasMore: result.hasMore,\n nextCursor: result.nextCursor,\n };\n }\n\n async getAll(options: Omit<QueryOptions, 'filters'> = {}): Promise<(T & RecordResult)[]> {\n const { records } = await this.query(options);\n return records;\n }\n\n async count(filters: QueryOptions['filters'] = []): Promise<number> {\n const result = await this.http.post<{ success: boolean; count: number }>(\n `${this.basePath}/count`,\n this.key,\n { filters },\n );\n return result.count;\n }\n\n // ─── Version History ──────────────────────────────────────────────────────\n\n async getHistory(id: string): Promise<RecordHistoryEntry[]> {\n const result = await this.http.get<ApiHistoryResult>(`${this.basePath}/${id}/history`, this.key);\n return result.history;\n }\n\n async restoreVersion(id: string, version: number): Promise<T & RecordResult> {\n const result = await this.http.post<ApiRecordResult<T>>(\n `${this.basePath}/${id}/restore`,\n this.key,\n { version },\n );\n return (result.record ?? result.data)!;\n }\n}\n","import type { HttpClient } from '../utils/http.js';\nimport { assertSafeName } from '../utils/query.js';\nimport type {\n AnalyticsQuery,\n AnalyticsResult,\n Aggregation,\n Granularity,\n SortOrder,\n DateRange,\n AnalyticsFilter,\n MetricDefinition,\n CountResult,\n DistributionRow,\n SumRow,\n TimeSeriesRow,\n FieldTimeSeriesRow,\n TopNRow,\n FieldStats,\n MultiMetricResult,\n StorageStatsResult,\n CrossBucketRow,\n RecordData,\n RecordResult,\n} from '../types/index.js';\n\ninterface ApiAnalyticsResult<T> {\n success: boolean;\n queryType: string;\n data: T;\n}\n\n/**\n * AnalyticsClient — BigQuery-powered aggregations for a bucket.\n * Uses the `bucketSecurityKey` (`hk_bucket_…`) sent via `X-Api-Key` header.\n *\n * @example\n * ```ts\n * const db = createClient({ authKey: '…', bucketSecurityKey: 'hk_bucket_…', storageKeys: { main: '…' } });\n * const analytics = db.analytics('orders');\n * const { count } = await analytics.count();\n * ```\n */\nexport class AnalyticsClient {\n private readonly http: HttpClient;\n private readonly bucketSecurityKey: string;\n private readonly basePath: string;\n\n constructor(http: HttpClient, bucketSecurityKey: string, bucketKey: string) {\n assertSafeName(bucketKey, 'bucketKey');\n this.http = http;\n this.bucketSecurityKey = bucketSecurityKey;\n this.basePath = `/analytics/${bucketKey}`;\n }\n\n private async run<T>(query: AnalyticsQuery): Promise<T> {\n const result = await this.http.post<ApiAnalyticsResult<T>>(\n this.basePath,\n this.bucketSecurityKey,\n query,\n );\n return result.data;\n }\n\n async count(opts: { dateRange?: DateRange } = {}): Promise<CountResult> {\n return this.run<CountResult>({ queryType: 'count', ...opts });\n }\n\n async distribution(opts: { field: string; limit?: number; order?: SortOrder; dateRange?: DateRange }): Promise<DistributionRow[]> {\n assertSafeName(opts.field, 'field');\n return this.run<DistributionRow[]>({ queryType: 'distribution', ...opts });\n }\n\n async sum(opts: { field: string; groupBy?: string; limit?: number; dateRange?: DateRange }): Promise<SumRow[]> {\n assertSafeName(opts.field, 'field');\n if (opts.groupBy) assertSafeName(opts.groupBy, 'groupBy');\n return this.run<SumRow[]>({ queryType: 'sum', ...opts });\n }\n\n async timeSeries(opts: { granularity?: Granularity; dateRange?: DateRange } = {}): Promise<TimeSeriesRow[]> {\n return this.run<TimeSeriesRow[]>({ queryType: 'timeSeries', granularity: 'day', ...opts });\n }\n\n async fieldTimeSeries(opts: { field: string; aggregation?: Aggregation; granularity?: Granularity; dateRange?: DateRange }): Promise<FieldTimeSeriesRow[]> {\n assertSafeName(opts.field, 'field');\n return this.run<FieldTimeSeriesRow[]>({ queryType: 'fieldTimeSeries', aggregation: 'sum', granularity: 'day', ...opts });\n }\n\n async topN(opts: { field: string; n?: number; labelField?: string; order?: SortOrder; dateRange?: DateRange }): Promise<TopNRow[]> {\n assertSafeName(opts.field, 'field');\n if (opts.labelField) assertSafeName(opts.labelField, 'labelField');\n return this.run<TopNRow[]>({ queryType: 'topN', n: 10, order: 'desc', ...opts });\n }\n\n async stats(opts: { field: string; dateRange?: DateRange }): Promise<FieldStats> {\n assertSafeName(opts.field, 'field');\n return this.run<FieldStats>({ queryType: 'stats', ...opts });\n }\n\n async records<T extends RecordData = RecordData>(opts: { filters?: AnalyticsFilter[]; selectFields?: string[]; limit?: number; offset?: number; orderBy?: string; order?: SortOrder; dateRange?: DateRange } = {}): Promise<(T & RecordResult)[]> {\n if (opts.orderBy) assertSafeName(opts.orderBy, 'orderBy');\n if (opts.selectFields) opts.selectFields.forEach(f => assertSafeName(f, 'selectField'));\n return this.run<(T & RecordResult)[]>({ queryType: 'records', limit: 100, order: 'desc', ...opts });\n }\n\n async multiMetric(opts: { metrics: MetricDefinition[]; dateRange?: DateRange }): Promise<MultiMetricResult> {\n opts.metrics.forEach(m => { assertSafeName(m.field, 'metric.field'); assertSafeName(m.name, 'metric.name'); });\n return this.run<MultiMetricResult>({ queryType: 'multiMetric', ...opts });\n }\n\n async storageStats(opts: { dateRange?: DateRange } = {}): Promise<StorageStatsResult> {\n return this.run<StorageStatsResult>({ queryType: 'storageStats', ...opts });\n }\n\n async crossBucket(opts: { bucketKeys: string[]; field: string; aggregation?: Aggregation; dateRange?: DateRange }): Promise<CrossBucketRow[]> {\n assertSafeName(opts.field, 'field');\n opts.bucketKeys.forEach(k => assertSafeName(k, 'bucketKey'));\n return this.run<CrossBucketRow[]>({ queryType: 'crossBucket', aggregation: 'sum', ...opts });\n }\n\n async query<T = unknown>(query: AnalyticsQuery): Promise<AnalyticsResult<T>> {\n const data = await this.run<T>(query);\n return { queryType: query.queryType, data };\n }\n}\n","import type { HttpClient } from '../utils/http.js';\nimport { guessMimeType } from '../utils/query.js';\nimport { StorageError } from '../utils/errors.js';\nimport type {\n UploadOptions,\n UploadResult,\n UploadUrlResult,\n ListOptions,\n ListResult,\n FileEntry,\n FileMetadata,\n SignedUrlResult,\n StorageStats,\n BatchUploadItem,\n BatchUploadUrlResult,\n} from '../types/index.js';\n\ninterface ApiUploadUrlResult {\n uploadUrl: string;\n path: string;\n mimeType: string;\n expiresAt: string;\n expiresIn: number;\n}\n\ninterface ApiConfirmResult {\n success: boolean;\n path: string;\n mimeType: string;\n size: number;\n isPublic: boolean;\n publicUrl: string | null;\n downloadUrl: string | null;\n}\n\ninterface ApiListResult {\n success: boolean;\n files: FileEntry[];\n folders: string[];\n hasMore: boolean;\n nextCursor?: string;\n}\n\ninterface ApiMetadataResult {\n success: boolean;\n path: string;\n size: number;\n mimeType: string;\n isPublic: boolean;\n publicUrl: string | null;\n downloadUrl: string | null;\n createdAt?: string;\n updatedAt?: string;\n}\n\ninterface ApiSignedUrlResult {\n success: boolean;\n signedUrl: string;\n expiresAt: string;\n expiresIn: number;\n path: string;\n}\n\ninterface ApiVisibilityResult {\n success: boolean;\n path: string;\n isPublic: boolean;\n publicUrl: string | null;\n downloadUrl: string | null;\n}\n\ninterface ApiStatsResult {\n success: boolean;\n stats: StorageStats;\n}\n\n/**\n * StorageManager — upload, download, list, move, copy, and delete files.\n * Uses an `X-Storage-Key` (`ssk_…`) header — separate from auth and bucket keys.\n * Each key is scoped to a specific storage bucket and permission set.\n *\n * Get a StorageManager via `db.storage('keyName')` where the key name\n * matches one of the keys you defined in `storageKeys` when calling `createClient`.\n *\n * @example\n * ```ts\n * const db = createClient({ authKey: '…', bucketSecurityKey: '…', storageKeys: { avatars: 'ssk_avatars_…' } });\n * const avatars = db.storage('avatars');\n * const result = await avatars.upload(file, 'alice.jpg', { isPublic: true });\n * ```\n */\nexport class StorageManager {\n private readonly http: HttpClient;\n private readonly storageKey: string;\n private readonly basePath = '/storage';\n\n constructor(http: HttpClient, storageKey: string) {\n this.http = http;\n this.storageKey = storageKey;\n }\n\n /** Headers for all storage requests — uses X-Storage-Key, not X-Api-Key. */\n private get authHeaders(): Record<string, string> {\n return { 'X-Storage-Key': this.storageKey };\n }\n\n // ─── Upload: Simple (server-buffered) ────────────────────────────────────\n\n /**\n * Upload a file to storage in one step (server-buffered, up to 500 MB).\n * For files >10 MB or when you need upload progress, use `getUploadUrl()` instead.\n *\n * @param data File data as a Blob, Buffer, Uint8Array, or ArrayBuffer.\n * @param path Destination path in your storage (e.g. `\"avatars/alice.jpg\"`).\n * @param options Upload options: isPublic, overwrite, mimeType.\n *\n * @example\n * ```ts\n * // Upload a public avatar\n * const result = await storage.upload(file, 'avatars/alice.jpg', { isPublic: true });\n * console.log(result.publicUrl); // → https://...\n *\n * // Upload a private document\n * const result = await storage.upload(pdfBuffer, 'docs/contract.pdf');\n * console.log(result.downloadUrl); // → /storage/download/docs/contract.pdf\n * ```\n */\n async upload(\n data: Blob | Uint8Array<ArrayBuffer> | ArrayBuffer | Buffer,\n path: string,\n options: UploadOptions = {},\n ): Promise<UploadResult> {\n const { isPublic = false, overwrite = false, mimeType } = options;\n const mime = mimeType ?? guessMimeType(path);\n\n const formData = new FormData();\n const blob = data instanceof Blob\n ? data\n : new Blob([data as ArrayBuffer], { type: mime });\n\n formData.append('file', blob, path.split('/').pop() ?? 'file');\n formData.append('path', path);\n formData.append('mimeType', mime);\n formData.append('isPublic', String(isPublic));\n formData.append('overwrite', String(overwrite));\n\n const result = await this.http.request<ApiConfirmResult>(`${this.basePath}/upload`, {\n method: 'POST',\n rawBody: formData,\n headers: this.authHeaders,\n });\n\n return {\n path: result.path,\n mimeType: result.mimeType,\n size: result.size,\n isPublic: result.isPublic,\n publicUrl: result.publicUrl,\n downloadUrl: result.downloadUrl,\n };\n }\n\n /**\n * Upload raw JSON or plain text data as a file.\n *\n * @example\n * ```ts\n * const result = await storage.uploadRaw(\n * { config: { theme: 'dark' } },\n * 'settings/user-config.json',\n * { isPublic: false },\n * );\n * ```\n */\n async uploadRaw(\n data: unknown,\n path: string,\n options: UploadOptions = {},\n ): Promise<UploadResult> {\n const { isPublic = false, overwrite = false, mimeType = 'application/json' } = options;\n const body = typeof data === 'string' ? data : JSON.stringify(data);\n\n const result = await this.http.request<ApiConfirmResult>(`${this.basePath}/upload-raw`, {\n method: 'POST',\n body: { path, data: body, mimeType, isPublic, overwrite },\n headers: this.authHeaders,\n });\n\n return {\n path: result.path,\n mimeType: result.mimeType,\n size: result.size,\n isPublic: result.isPublic,\n publicUrl: result.publicUrl,\n downloadUrl: result.downloadUrl,\n };\n }\n\n // ─── Upload: Direct-to-GCS (recommended for large files) ─────────────────\n\n /**\n * Step 1 of the recommended upload flow.\n * Get a signed URL to upload directly to GCS from the client (supports progress).\n *\n * @example\n * ```ts\n * const { uploadUrl, path: confirmedPath } = await storage.getUploadUrl({\n * path: 'videos/intro.mp4',\n * mimeType: 'video/mp4',\n * size: file.size,\n * isPublic: true,\n * });\n *\n * // Upload using XHR for progress tracking\n * await storage.uploadToSignedUrl(uploadUrl, file, 'video/mp4', (pct) => {\n * console.log(`${pct}% uploaded`);\n * });\n *\n * // Step 3: confirm\n * const result = await storage.confirmUpload({ path: confirmedPath, mimeType: 'video/mp4', isPublic: true });\n * ```\n */\n async getUploadUrl(opts: {\n path: string;\n mimeType: string;\n size: number;\n isPublic?: boolean;\n overwrite?: boolean;\n expiresInSeconds?: number;\n }): Promise<UploadUrlResult> {\n const result = await this.http.request<ApiUploadUrlResult>(`${this.basePath}/upload-url`, {\n method: 'POST',\n body: { isPublic: false, overwrite: false, expiresInSeconds: 900, ...opts },\n headers: this.authHeaders,\n });\n return result;\n }\n\n /**\n * Upload data directly to a signed GCS URL (no auth headers needed).\n * Optionally tracks progress via a callback.\n *\n * @param signedUrl The URL returned by `getUploadUrl()`.\n * @param data File data.\n * @param mimeType Must match what was used in `getUploadUrl()`.\n * @param onProgress Optional callback called with 0–100 progress percentage.\n */\n async uploadToSignedUrl(\n signedUrl: string,\n data: Blob | Uint8Array<ArrayBuffer> | ArrayBuffer,\n mimeType: string,\n onProgress?: (percent: number) => void,\n ): Promise<void> {\n await this.http.putToSignedUrl(signedUrl, data, mimeType, onProgress);\n }\n\n /**\n * Step 3 of the recommended upload flow.\n * Confirm a direct upload and register metadata on the server.\n *\n * @example\n * ```ts\n * const result = await storage.confirmUpload({\n * path: 'videos/intro.mp4',\n * mimeType: 'video/mp4',\n * isPublic: true,\n * });\n * console.log(result.publicUrl);\n * ```\n */\n async confirmUpload(opts: {\n path: string;\n mimeType: string;\n isPublic?: boolean;\n }): Promise<UploadResult> {\n const result = await this.http.request<ApiConfirmResult>(`${this.basePath}/confirm`, {\n method: 'POST',\n body: { isPublic: false, ...opts },\n headers: this.authHeaders,\n });\n return {\n path: result.path,\n mimeType: result.mimeType,\n size: result.size,\n isPublic: result.isPublic,\n publicUrl: result.publicUrl,\n downloadUrl: result.downloadUrl,\n };\n }\n\n // ─── Batch Upload ─────────────────────────────────────────────────────────\n\n /**\n * Get signed upload URLs for multiple files at once.\n *\n * @example\n * ```ts\n * const { files } = await storage.getBatchUploadUrls([\n * { path: 'images/photo1.jpg', mimeType: 'image/jpeg', size: 204800 },\n * { path: 'images/photo2.jpg', mimeType: 'image/jpeg', size: 153600 },\n * ]);\n *\n * // Upload each file and confirm\n * for (const f of files) {\n * await storage.uploadToSignedUrl(f.uploadUrl, blobs[f.index], f.mimeType);\n * await storage.confirmUpload({ path: f.path, mimeType: f.mimeType });\n * }\n * ```\n */\n async getBatchUploadUrls(files: BatchUploadItem[]): Promise<BatchUploadUrlResult> {\n const result = await this.http.request<{ success: boolean; files: BatchUploadUrlResult['files'] }>(\n `${this.basePath}/batch-upload-urls`,\n { method: 'POST', body: { files }, headers: this.authHeaders },\n );\n return { files: result.files };\n }\n\n /**\n * Confirm multiple direct uploads at once.\n */\n async batchConfirmUploads(\n items: Array<{ path: string; mimeType: string; isPublic?: boolean }>,\n ): Promise<UploadResult[]> {\n const result = await this.http.request<{ success: boolean; files: ApiConfirmResult[] }>(\n `${this.basePath}/batch-confirm`,\n { method: 'POST', body: { files: items }, headers: this.authHeaders },\n );\n return result.files.map(r => ({\n path: r.path,\n mimeType: r.mimeType,\n size: r.size,\n isPublic: r.isPublic,\n publicUrl: r.publicUrl,\n downloadUrl: r.downloadUrl,\n }));\n }\n\n // ─── Download ─────────────────────────────────────────────────────────────\n\n /**\n * Download a private file as an ArrayBuffer.\n * For public files, use the `publicUrl` directly — no SDK needed.\n *\n * @example\n * ```ts\n * const buffer = await storage.download('docs/contract.pdf');\n * const blob = new Blob([buffer], { type: 'application/pdf' });\n * // Open in browser:\n * window.open(URL.createObjectURL(blob));\n * ```\n */\n async download(path: string): Promise<ArrayBuffer> {\n const encoded = path.split('/').map(encodeURIComponent).join('/');\n return this.http.request<ArrayBuffer>(`${this.basePath}/download/${encoded}`, {\n method: 'GET',\n headers: this.authHeaders,\n });\n }\n\n /**\n * Download multiple files at once, returned as a JSON map of `{ path: base64 }`.\n *\n * @example\n * ```ts\n * const files = await storage.batchDownload(['docs/a.pdf', 'docs/b.pdf']);\n * ```\n */\n async batchDownload(paths: string[]): Promise<Record<string, string>> {\n const result = await this.http.request<{ success: boolean; files: Record<string, string> }>(\n `${this.basePath}/batch-download`,\n { method: 'POST', body: { paths }, headers: this.authHeaders },\n );\n return result.files;\n }\n\n // ─── List ─────────────────────────────────────────────────────────────────\n\n /**\n * List files and folders at a given path prefix.\n *\n * @example\n * ```ts\n * // List everything in the root\n * const { files, folders } = await storage.list();\n *\n * // List a specific folder\n * const { files, folders, hasMore, nextCursor } = await storage.list({\n * prefix: 'avatars/',\n * limit: 20,\n * });\n *\n * // Next page\n * const page2 = await storage.list({ prefix: 'avatars/', cursor: nextCursor });\n * ```\n */\n async list(opts: ListOptions = {}): Promise<ListResult> {\n const params = new URLSearchParams();\n if (opts.prefix) params.set('prefix', opts.prefix);\n if (opts.limit) params.set('limit', String(opts.limit));\n if (opts.cursor) params.set('cursor', opts.cursor);\n if (opts.recursive) params.set('recursive', 'true');\n const qs = params.toString() ? `?${params}` : '';\n\n const result = await this.http.request<ApiListResult>(`${this.basePath}/list${qs}`, {\n method: 'GET',\n headers: this.authHeaders,\n });\n return {\n files: result.files,\n folders: result.folders,\n hasMore: result.hasMore,\n nextCursor: result.nextCursor,\n };\n }\n\n // ─── Metadata ─────────────────────────────────────────────────────────────\n\n /**\n * Get metadata for a file (size, MIME type, visibility, URLs).\n *\n * @example\n * ```ts\n * const meta = await storage.getMetadata('avatars/alice.jpg');\n * console.log(meta.size, meta.isPublic, meta.publicUrl);\n * ```\n */\n async getMetadata(path: string): Promise<FileMetadata> {\n const encoded = path.split('/').map(encodeURIComponent).join('/');\n const result = await this.http.request<ApiMetadataResult>(\n `${this.basePath}/metadata/${encoded}`,\n { method: 'GET', headers: this.authHeaders },\n );\n return {\n path: result.path,\n size: result.size,\n mimeType: result.mimeType,\n isPublic: result.isPublic,\n publicUrl: result.publicUrl,\n downloadUrl: result.downloadUrl,\n createdAt: result.createdAt,\n updatedAt: result.updatedAt,\n };\n }\n\n // ─── Signed URL (time-limited share) ─────────────────────────────────────\n\n /**\n * Generate a time-limited download URL for a private file.\n * The URL can be shared externally without requiring an `X-Storage-Key`.\n *\n * > **Note:** Downloads via signed URLs bypass the server, so download stats\n * > are NOT tracked. Use `downloadUrl` for tracked downloads.\n *\n * @param path Path to the file.\n * @param expiresIn URL lifetime in seconds (default 3600 = 1 hour).\n *\n * @example\n * ```ts\n * const { signedUrl, expiresAt } = await storage.getSignedUrl('docs/invoice.pdf', 1800);\n * // Share signedUrl with the recipient — it expires in 30 minutes.\n * ```\n */\n async getSignedUrl(path: string, expiresIn = 3600): Promise<SignedUrlResult> {\n const result = await this.http.request<ApiSignedUrlResult>(`${this.basePath}/signed-url`, {\n method: 'POST',\n body: { path, expiresIn },\n headers: this.authHeaders,\n });\n return {\n signedUrl: result.signedUrl,\n expiresAt: result.expiresAt,\n expiresIn: result.expiresIn,\n path: result.path,\n };\n }\n\n // ─── Visibility ───────────────────────────────────────────────────────────\n\n /**\n * Change a file's visibility between public and private after upload.\n *\n * @example\n * ```ts\n * // Make a file public\n * const result = await storage.setVisibility('avatars/alice.jpg', true);\n * console.log(result.publicUrl); // CDN URL\n *\n * // Make a file private\n * const result = await storage.setVisibility('avatars/alice.jpg', false);\n * console.log(result.downloadUrl); // Auth-required URL\n * ```\n */\n async setVisibility(\n path: string,\n isPublic: boolean,\n ): Promise<{ path: string; isPublic: boolean; publicUrl: string | null; downloadUrl: string | null }> {\n const result = await this.http.request<ApiVisibilityResult>(`${this.basePath}/visibility`, {\n method: 'PATCH',\n body: { path, isPublic },\n headers: this.authHeaders,\n });\n return {\n path: result.path,\n isPublic: result.isPublic,\n publicUrl: result.publicUrl,\n downloadUrl: result.downloadUrl,\n };\n }\n\n // ─── Folder Operations ────────────────────────────────────────────────────\n\n /**\n * Create a folder (a GCS prefix placeholder).\n *\n * @example\n * ```ts\n * await storage.createFolder('uploads/2025/');\n * ```\n */\n async createFolder(path: string): Promise<{ path: string }> {\n const result = await this.http.request<{ success: boolean; path: string }>(\n `${this.basePath}/folder`,\n { method: 'POST', body: { path }, headers: this.authHeaders },\n );\n return { path: result.path };\n }\n\n // ─── File Operations ──────────────────────────────────────────────────────\n\n /**\n * Delete a single file.\n *\n * @example\n * ```ts\n * await storage.deleteFile('avatars/old-avatar.jpg');\n * ```\n */\n async deleteFile(path: string): Promise<void> {\n await this.http.request<{ success: boolean }>(`${this.basePath}/file`, {\n method: 'DELETE',\n body: { path },\n headers: this.authHeaders,\n });\n }\n\n /**\n * Delete a folder and all its contents recursively.\n *\n * @example\n * ```ts\n * await storage.deleteFolder('temp/');\n * ```\n */\n async deleteFolder(path: string): Promise<void> {\n await this.http.request<{ success: boolean }>(`${this.basePath}/folder`, {\n method: 'DELETE',\n body: { path },\n headers: this.authHeaders,\n });\n }\n\n /**\n * Move or rename a file.\n *\n * @example\n * ```ts\n * // Rename\n * await storage.move('docs/draft.pdf', 'docs/final.pdf');\n * // Move to a different folder\n * await storage.move('inbox/report.xlsx', 'archive/2025/report.xlsx');\n * ```\n */\n async move(from: string, to: string): Promise<{ from: string; to: string }> {\n const result = await this.http.request<{ success: boolean; from: string; to: string }>(\n `${this.basePath}/move`,\n { method: 'POST', body: { from, to }, headers: this.authHeaders },\n );\n return { from: result.from, to: result.to };\n }\n\n /**\n * Copy a file to a new path.\n *\n * @example\n * ```ts\n * await storage.copy('templates/base.html', 'sites/my-site/index.html');\n * ```\n */\n async copy(from: string, to: string): Promise<{ from: string; to: string }> {\n const result = await this.http.request<{ success: boolean; from: string; to: string }>(\n `${this.basePath}/copy`,\n { method: 'POST', body: { from, to }, headers: this.authHeaders },\n );\n return { from: result.from, to: result.to };\n }\n\n // ─── Stats ────────────────────────────────────────────────────────────────\n\n /**\n * Get storage statistics for your key: total files, bytes, operation counts.\n *\n * @example\n * ```ts\n * const stats = await storage.getStats();\n * console.log(`${stats.totalFiles} files, ${(stats.totalBytes / 1e6).toFixed(1)} MB`);\n * ```\n */\n async getStats(): Promise<StorageStats> {\n const result = await this.http.request<ApiStatsResult>(`${this.basePath}/stats`, {\n method: 'GET',\n headers: this.authHeaders,\n });\n return result.stats;\n }\n\n // ─── Info (no auth) ───────────────────────────────────────────────────────\n\n /**\n * Ping the storage service. No authentication required.\n *\n * @example\n * ```ts\n * const info = await storage.info();\n * // → { ok: true, storageRoot: 'hydrous-storage' }\n * ```\n */\n async info(): Promise<{ ok: boolean; storageRoot: string }> {\n return this.http.get<{ ok: boolean; storageRoot: string }>(`${this.basePath}/info`);\n }\n}\n","import type { StorageManager } from './manager.js';\nimport type {\n UploadOptions,\n UploadResult,\n UploadUrlResult,\n ListOptions,\n ListResult,\n FileMetadata,\n SignedUrlResult,\n} from '../types/index.js';\n\n/**\n * ScopedStorage — a StorageManager pre-scoped to a specific folder prefix.\n *\n * All paths are automatically prepended with the scope, so you never\n * have to repeat the folder name.\n *\n * @example\n * ```ts\n * const db = createClient({ securityKey: 'sk_...' });\n *\n * // All operations in the \"avatars/\" folder\n * const avatars = db.storage.scope('avatars');\n *\n * await avatars.upload(file, 'alice.jpg'); // → \"avatars/alice.jpg\"\n * const list = await avatars.list(); // → files under \"avatars/\"\n * await avatars.deleteFile('alice.jpg'); // → deletes \"avatars/alice.jpg\"\n * ```\n */\nexport class ScopedStorage {\n private readonly manager: StorageManager;\n private readonly prefix: string;\n\n constructor(manager: StorageManager, prefix: string) {\n this.manager = manager;\n // Normalise: ensure the prefix ends with exactly one slash\n this.prefix = prefix.replace(/\\/+$/, '') + '/';\n }\n\n private scopedPath(userPath: string): string {\n return `${this.prefix}${userPath.replace(/^\\/+/, '')}`;\n }\n\n /** Upload a file within the scoped folder. */\n upload(\n data: Blob | Uint8Array<ArrayBuffer> | ArrayBuffer | Buffer,\n path: string,\n options?: UploadOptions,\n ): Promise<UploadResult> {\n return this.manager.upload(data, this.scopedPath(path), options);\n }\n\n /** Upload raw JSON or text within the scoped folder. */\n uploadRaw(data: unknown, path: string, options?: UploadOptions): Promise<UploadResult> {\n return this.manager.uploadRaw(data, this.scopedPath(path), options);\n }\n\n /** Get a signed upload URL for a file within the scoped folder. */\n getUploadUrl(opts: {\n path: string;\n mimeType: string;\n size: number;\n isPublic?: boolean;\n overwrite?: boolean;\n }): Promise<UploadUrlResult> {\n return this.manager.getUploadUrl({ ...opts, path: this.scopedPath(opts.path) });\n }\n\n /** Confirm a direct upload within the scoped folder. */\n confirmUpload(opts: {\n path: string;\n mimeType: string;\n isPublic?: boolean;\n }): Promise<UploadResult> {\n return this.manager.confirmUpload({ ...opts, path: this.scopedPath(opts.path) });\n }\n\n /** Download a file within the scoped folder. */\n download(path: string): Promise<ArrayBuffer> {\n return this.manager.download(this.scopedPath(path));\n }\n\n /**\n * List files within the scoped folder.\n * `prefix` in options is relative to the scope.\n */\n list(opts: ListOptions = {}): Promise<ListResult> {\n const scopedOpts = {\n ...opts,\n prefix: this.scopedPath(opts.prefix ?? ''),\n };\n return this.manager.list(scopedOpts);\n }\n\n /** Get metadata for a file within the scoped folder. */\n getMetadata(path: string): Promise<FileMetadata> {\n return this.manager.getMetadata(this.scopedPath(path));\n }\n\n /** Get a time-limited signed URL for a file within the scoped folder. */\n getSignedUrl(path: string, expiresIn?: number): Promise<SignedUrlResult> {\n return this.manager.getSignedUrl(this.scopedPath(path), expiresIn);\n }\n\n /** Change visibility of a file within the scoped folder. */\n setVisibility(\n path: string,\n isPublic: boolean,\n ): Promise<{ path: string; isPublic: boolean; publicUrl: string | null; downloadUrl: string | null }> {\n return this.manager.setVisibility(this.scopedPath(path), isPublic);\n }\n\n /** Delete a file within the scoped folder. */\n deleteFile(path: string): Promise<void> {\n return this.manager.deleteFile(this.scopedPath(path));\n }\n\n /** Delete a sub-folder within the scoped folder. */\n deleteFolder(path: string): Promise<void> {\n return this.manager.deleteFolder(this.scopedPath(path));\n }\n\n /** Move a file within the scoped folder. */\n move(from: string, to: string): Promise<{ from: string; to: string }> {\n return this.manager.move(this.scopedPath(from), this.scopedPath(to));\n }\n\n /** Copy a file within the scoped folder. */\n copy(from: string, to: string): Promise<{ from: string; to: string }> {\n return this.manager.copy(this.scopedPath(from), this.scopedPath(to));\n }\n\n /** Create a sub-folder within the scoped folder. */\n createFolder(path: string): Promise<{ path: string }> {\n return this.manager.createFolder(this.scopedPath(path));\n }\n\n /**\n * Create a further-scoped instance nested within this scope.\n *\n * @example\n * ```ts\n * const uploads = db.storage.scope('user-uploads');\n * const images = uploads.scope('images'); // → \"user-uploads/images/\"\n * ```\n */\n scope(subPrefix: string): ScopedStorage {\n return new ScopedStorage(this.manager, this.scopedPath(subPrefix));\n }\n}","import { HttpClient, DEFAULT_BASE_URL } from './utils/http.js';\nimport { AuthClient } from './auth/client.js';\nimport { RecordsClient } from './records/client.js';\nimport { AnalyticsClient } from './analytics/client.js';\nimport { StorageManager } from './storage/manager.js';\nimport { ScopedStorage } from './storage/scoped.js';\nimport type { HydrousConfig, RecordData } from './types/index.js';\n\n/**\n * HydrousClient — the main entry point for the HydrousDB SDK.\n *\n * Each service uses its own dedicated key:\n * - `authKey` (`hk_auth_…`) → `db.auth('bucket')`\n * - `bucketSecurityKey` (`hk_bucket_…`) → `db.records('bucket')` + `db.analytics('bucket')`\n * - `storageKeys` (`ssk_…`) → `db.storage('keyName')`\n *\n * @example\n * ```ts\n * import { createClient } from 'hydrousdb';\n *\n * const db = createClient({\n * authKey: 'hk_auth_…',\n * bucketSecurityKey: 'hk_bucket_…',\n * storageKeys: {\n * main: 'ssk_main_…',\n * avatars: 'ssk_avatars_…',\n * documents: 'ssk_docs_…',\n * },\n * });\n *\n * // Records & Analytics — use bucketSecurityKey automatically\n * const posts = db.records('blog-posts');\n * const analytics = db.analytics('orders');\n *\n * // Auth — uses authKey automatically\n * const auth = db.auth('app-users');\n *\n * // Storage — pick which key to use by name\n * const avatarStorage = db.storage('avatars');\n * const documentStorage = db.storage('documents');\n * ```\n */\nexport class HydrousClient {\n private readonly http: HttpClient;\n private readonly authKey_: string;\n private readonly bucketSecurityKey_: string;\n private readonly storageKeys_: Record<string, string>;\n\n private readonly _recordsCache = new Map<string, RecordsClient<RecordData>>();\n private readonly _authCache = new Map<string, AuthClient>();\n private readonly _analyticsCache = new Map<string, AnalyticsClient>();\n private readonly _storageCache = new Map<string, StorageManager>();\n\n constructor(config: HydrousConfig) {\n if (!config.authKey) {\n throw new Error('[HydrousDB] authKey is required. Get yours from https://hydrousdb.com/dashboard.');\n }\n if (!config.bucketSecurityKey) {\n throw new Error('[HydrousDB] bucketSecurityKey is required. Get yours from https://hydrousdb.com/dashboard.');\n }\n if (!config.storageKeys || Object.keys(config.storageKeys).length === 0) {\n throw new Error('[HydrousDB] storageKeys is required. Define at least one storage key from https://hydrousdb.com/dashboard.');\n }\n\n const baseUrl = config.baseUrl ?? DEFAULT_BASE_URL;\n this.http = new HttpClient(baseUrl);\n this.authKey_ = config.authKey;\n this.bucketSecurityKey_ = config.bucketSecurityKey;\n this.storageKeys_ = config.storageKeys;\n }\n\n // ─── Records ─────────────────────────────────────────────────────────────\n\n /**\n * Get a typed records client for the named bucket.\n * Uses your `bucketSecurityKey` automatically.\n *\n * @example\n * ```ts\n * interface Post { title: string; published: boolean }\n * const posts = db.records<Post>('blog-posts');\n * const post = await posts.create({ title: 'Hello', published: false });\n * ```\n */\n records<T extends RecordData = RecordData>(bucketKey: string): RecordsClient<T> {\n if (!this._recordsCache.has(bucketKey)) {\n this._recordsCache.set(\n bucketKey,\n new RecordsClient<T>(this.http, this.bucketSecurityKey_, bucketKey) as RecordsClient<RecordData>,\n );\n }\n return this._recordsCache.get(bucketKey) as RecordsClient<T>;\n }\n\n // ─── Auth ─────────────────────────────────────────────────────────────────\n\n /**\n * Get an auth client for the named user bucket.\n * Uses your `authKey` automatically.\n *\n * @example\n * ```ts\n * const auth = db.auth('app-users');\n * const { user, session } = await auth.login({ email: '…', password: '…' });\n * ```\n */\n auth(bucketKey: string): AuthClient {\n if (!this._authCache.has(bucketKey)) {\n this._authCache.set(bucketKey, new AuthClient(this.http, this.authKey_, bucketKey));\n }\n return this._authCache.get(bucketKey)!;\n }\n\n // ─── Analytics ────────────────────────────────────────────────────────────\n\n /**\n * Get an analytics client for the named bucket.\n * Uses your `bucketSecurityKey` automatically.\n *\n * @example\n * ```ts\n * const analytics = db.analytics('orders');\n * const { count } = await analytics.count();\n * ```\n */\n analytics(bucketKey: string): AnalyticsClient {\n if (!this._analyticsCache.has(bucketKey)) {\n this._analyticsCache.set(\n bucketKey,\n new AnalyticsClient(this.http, this.bucketSecurityKey_, bucketKey),\n );\n }\n return this._analyticsCache.get(bucketKey)!;\n }\n\n // ─── Storage ──────────────────────────────────────────────────────────────\n\n /**\n * Get a storage manager for the named storage key.\n * The name must match a key you defined in `storageKeys` when calling `createClient`.\n * Uses the corresponding `ssk_…` key automatically via `X-Storage-Key` header.\n *\n * @param keyName The name of the storage key (e.g. `\"avatars\"`, `\"documents\"`, `\"main\"`).\n *\n * @example\n * ```ts\n * const avatars = db.storage('avatars');\n * const documents = db.storage('documents');\n *\n * // Upload to avatars bucket\n * await avatars.upload(file, `${userId}.jpg`, { isPublic: true });\n *\n * // Scope to a sub-folder\n * const userDocs = db.storage('documents').scope(`users/${userId}`);\n * await userDocs.upload(pdfBuffer, 'contract.pdf');\n * ```\n */\n storage(keyName: string): StorageManager & { scope: (prefix: string) => ScopedStorage } {\n const ssk = this.storageKeys_[keyName];\n if (!ssk) {\n const available = Object.keys(this.storageKeys_).join(', ');\n throw new Error(\n `[HydrousDB] Unknown storage key name \"${keyName}\". ` +\n `Available keys: ${available}. ` +\n `Add it to storageKeys in your createClient() config.`,\n );\n }\n\n if (!this._storageCache.has(keyName)) {\n this._storageCache.set(keyName, new StorageManager(this.http, ssk));\n }\n\n const mgr = this._storageCache.get(keyName)!;\n const extended = mgr as StorageManager & { scope: (prefix: string) => ScopedStorage };\n if (!extended.scope) {\n extended.scope = (prefix: string) => new ScopedStorage(mgr, prefix);\n }\n return extended;\n }\n}\n\n// ─── Factory ──────────────────────────────────────────────────────────────────\n\n/**\n * Create a new HydrousDB client.\n *\n * @example\n * ```ts\n * import { createClient } from 'hydrousdb';\n *\n * const db = createClient({\n * authKey: process.env.HYDROUS_AUTH_KEY!,\n * bucketSecurityKey: process.env.HYDROUS_BUCKET_KEY!,\n * storageKeys: {\n * main: process.env.HYDROUS_STORAGE_MAIN!,\n * avatars: process.env.HYDROUS_STORAGE_AVATARS!,\n * documents: process.env.HYDROUS_STORAGE_DOCS!,\n * },\n * });\n * ```\n */\nexport function createClient(config: HydrousConfig): HydrousClient {\n return new HydrousClient(config);\n}\n"]}
|