@objectstack/client 1.0.6 → 1.0.7

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
@@ -1,5 +1,5 @@
1
1
 
2
- > @objectstack/client@1.0.6 build /home/runner/work/spec/spec/packages/client
2
+ > @objectstack/client@1.0.7 build /home/runner/work/spec/spec/packages/client
3
3
  > tsup --config ../../tsup.config.ts
4
4
 
5
5
  CLI Building entry: src/index.ts
@@ -10,13 +10,13 @@
10
10
  CLI Cleaning output folder
11
11
  ESM Build start
12
12
  CJS Build start
13
- ESM dist/index.mjs 19.46 KB
14
- ESM dist/index.mjs.map 40.47 KB
15
- ESM ⚡️ Build success in 35ms
16
- CJS dist/index.js 20.62 KB
17
- CJS dist/index.js.map 40.53 KB
18
- CJS ⚡️ Build success in 42ms
13
+ CJS dist/index.js 21.61 KB
14
+ CJS dist/index.js.map 42.36 KB
15
+ CJS ⚡️ Build success in 41ms
16
+ ESM dist/index.mjs 20.45 KB
17
+ ESM dist/index.mjs.map 42.31 KB
18
+ ESM ⚡️ Build success in 43ms
19
19
  DTS Build start
20
- DTS ⚡️ Build success in 10222ms
20
+ DTS ⚡️ Build success in 3866ms
21
21
  DTS dist/index.d.mts 10.28 KB
22
22
  DTS dist/index.d.ts 10.28 KB
package/CHANGELOG.md CHANGED
@@ -1,5 +1,13 @@
1
1
  # @objectstack/client
2
2
 
3
+ ## 1.0.7
4
+
5
+ ### Patch Changes
6
+
7
+ - ebdf787: feat: implement standard service discovery via `/.well-known/objectstack`
8
+ - @objectstack/spec@1.0.7
9
+ - @objectstack/core@1.0.7
10
+
3
11
  ## 1.0.6
4
12
 
5
13
  ### Patch Changes
package/dist/index.js CHANGED
@@ -616,8 +616,36 @@ var ObjectStackClient = class {
616
616
  async connect() {
617
617
  this.logger.debug("Connecting to ObjectStack server", { baseUrl: this.baseUrl });
618
618
  try {
619
- const res = await this.fetch(`${this.baseUrl}/api/v1`);
620
- const data = await res.json();
619
+ let data;
620
+ try {
621
+ let wellKnownUrl;
622
+ try {
623
+ const url = new URL(this.baseUrl);
624
+ wellKnownUrl = `${url.origin}/.well-known/objectstack`;
625
+ } catch {
626
+ wellKnownUrl = "/.well-known/objectstack";
627
+ }
628
+ this.logger.debug("Probing .well-known discovery", { url: wellKnownUrl });
629
+ const res = await this.fetchImpl(wellKnownUrl);
630
+ if (res.ok) {
631
+ data = await res.json();
632
+ this.logger.debug("Discovered via .well-known");
633
+ }
634
+ } catch (e) {
635
+ this.logger.debug("Standard discovery probe failed", { error: e.message });
636
+ }
637
+ if (!data) {
638
+ const fallbackUrl = `${this.baseUrl}/api/v1`;
639
+ this.logger.debug("Falling back to legacy discovery", { url: fallbackUrl });
640
+ const res = await this.fetchImpl(fallbackUrl);
641
+ if (!res.ok) {
642
+ throw new Error(`Failed to connect to ${fallbackUrl}: ${res.statusText}`);
643
+ }
644
+ data = await res.json();
645
+ }
646
+ if (!data) {
647
+ throw new Error("Connection failed: No discovery data returned");
648
+ }
621
649
  this.discoveryInfo = data;
622
650
  this.logger.info("Connected to ObjectStack server", {
623
651
  version: data.version,
package/dist/index.js.map CHANGED
@@ -1 +1 @@
1
- {"version":3,"sources":["../src/index.ts","../src/query-builder.ts"],"sourcesContent":["import { QueryAST, SortNode, AggregationNode } from '@objectstack/spec/data';\nimport { \n BatchUpdateRequest, \n BatchUpdateResponse, \n UpdateManyRequest,\n DeleteManyRequest,\n BatchOptions,\n MetadataCacheRequest,\n MetadataCacheResponse,\n StandardErrorCode,\n ErrorCategory,\n GetDiscoveryResponse,\n GetMetaTypesResponse,\n GetMetaItemsResponse,\n LoginRequest,\n SessionResponse,\n GetPresignedUrlRequest,\n PresignedUrlResponse,\n CompleteUploadRequest,\n FileUploadResponse\n} from '@objectstack/spec/api';\nimport { Logger, createLogger } from '@objectstack/core';\n\nexport interface ClientConfig {\n baseUrl: string;\n token?: string;\n /**\n * Custom fetch implementation (e.g. node-fetch or for Next.js caching)\n */\n fetch?: (input: RequestInfo | URL, init?: RequestInit) => Promise<Response>;\n /**\n * Logger instance for debugging\n */\n logger?: Logger;\n /**\n * Enable debug logging\n */\n debug?: boolean;\n}\n\n/**\n * Discovery Result\n * Re-export from @objectstack/spec/api for convenience\n */\nexport type DiscoveryResult = GetDiscoveryResponse;\n\nexport interface QueryOptions {\n select?: string[]; // Simplified Selection\n filters?: Record<string, any>; // Map or AST\n sort?: string | string[] | SortNode[]; // 'name' or ['-created_at'] or AST\n top?: number;\n skip?: number;\n // Advanced features\n aggregations?: AggregationNode[];\n groupBy?: string[];\n}\n\nexport interface PaginatedResult<T = any> {\n value: T[];\n count: number;\n}\n\nexport interface StandardError {\n code: StandardErrorCode;\n message: string;\n category: ErrorCategory;\n httpStatus: number;\n retryable: boolean;\n details?: Record<string, any>;\n}\n\nexport class ObjectStackClient {\n private baseUrl: string;\n private token?: string;\n private fetchImpl: (input: RequestInfo | URL, init?: RequestInit) => Promise<Response>;\n private discoveryInfo?: DiscoveryResult;\n private logger: Logger;\n\n constructor(config: ClientConfig) {\n this.baseUrl = config.baseUrl.replace(/\\/$/, ''); // Remove trailing slash\n this.token = config.token;\n this.fetchImpl = config.fetch || globalThis.fetch.bind(globalThis);\n \n // Initialize logger\n this.logger = config.logger || createLogger({ \n level: config.debug ? 'debug' : 'info',\n format: 'pretty'\n });\n \n this.logger.debug('ObjectStack client created', { baseUrl: this.baseUrl });\n }\n\n /**\n * Initialize the client by discovering server capabilities.\n */\n async connect() {\n this.logger.debug('Connecting to ObjectStack server', { baseUrl: this.baseUrl });\n \n try {\n // Connect to the discovery endpoint at /api/v1\n const res = await this.fetch(`${this.baseUrl}/api/v1`);\n \n const data = await res.json();\n this.discoveryInfo = data;\n \n this.logger.info('Connected to ObjectStack server', { \n version: data.version,\n apiName: data.apiName,\n capabilities: data.capabilities \n });\n \n return data as DiscoveryResult;\n } catch (e) {\n this.logger.error('Failed to connect to ObjectStack server', e as Error, { baseUrl: this.baseUrl });\n throw e;\n }\n }\n\n /**\n * Metadata Operations\n */\n meta = {\n /**\n * Get all available metadata types\n * Returns types like 'object', 'plugin', 'view', etc.\n */\n getTypes: async (): Promise<GetMetaTypesResponse> => {\n const route = this.getRoute('metadata');\n const res = await this.fetch(`${this.baseUrl}${route}`);\n return res.json();\n },\n\n /**\n * Get all items of a specific metadata type\n * @param type - Metadata type name (e.g., 'object', 'plugin')\n */\n getItems: async (type: string): Promise<GetMetaItemsResponse> => {\n const route = this.getRoute('metadata');\n const res = await this.fetch(`${this.baseUrl}${route}/${type}`);\n return res.json();\n },\n\n /**\n * Get a specific object definition by name\n * @deprecated Use `getItem('object', name)` instead for consistency with spec protocol\n * @param name - Object name (snake_case identifier)\n */\n getObject: async (name: string) => {\n const route = this.getRoute('metadata');\n const res = await this.fetch(`${this.baseUrl}${route}/object/${name}`);\n return res.json();\n },\n\n /**\n * Get a specific metadata item by type and name\n * @param type - Metadata type (e.g., 'object', 'plugin')\n * @param name - Item name (snake_case identifier)\n */\n getItem: async (type: string, name: string) => {\n const route = this.getRoute('metadata');\n const res = await this.fetch(`${this.baseUrl}${route}/${type}/${name}`);\n return res.json();\n },\n\n /**\n * Save a metadata item\n * @param type - Metadata type (e.g., 'object', 'plugin')\n * @param name - Item name\n * @param item - The metadata content to save\n */\n saveItem: async (type: string, name: string, item: any) => {\n const route = this.getRoute('metadata');\n const res = await this.fetch(`${this.baseUrl}${route}/${type}/${name}`, {\n method: 'PUT',\n body: JSON.stringify(item)\n });\n return res.json();\n },\n \n /**\n * Get object metadata with cache support\n * Supports ETag-based conditional requests for efficient caching\n */\n getCached: async (name: string, cacheOptions?: MetadataCacheRequest): Promise<MetadataCacheResponse> => {\n const route = this.getRoute('metadata');\n const headers: Record<string, string> = {};\n \n if (cacheOptions?.ifNoneMatch) {\n headers['If-None-Match'] = cacheOptions.ifNoneMatch;\n }\n if (cacheOptions?.ifModifiedSince) {\n headers['If-Modified-Since'] = cacheOptions.ifModifiedSince;\n }\n \n const res = await this.fetch(`${this.baseUrl}${route}/object/${name}`, {\n headers\n });\n \n // Check for 304 Not Modified\n if (res.status === 304) {\n return {\n notModified: true,\n etag: cacheOptions?.ifNoneMatch ? { \n value: cacheOptions.ifNoneMatch.replace(/^W\\/|\"/g, ''),\n weak: cacheOptions.ifNoneMatch.startsWith('W/')\n } : undefined\n };\n }\n \n const data = await res.json();\n const etag = res.headers.get('ETag');\n const lastModified = res.headers.get('Last-Modified');\n \n return {\n data,\n etag: etag ? { \n value: etag.replace(/^W\\/|\"/g, ''), \n weak: etag.startsWith('W/') \n } : undefined,\n lastModified: lastModified || undefined,\n notModified: false\n };\n },\n \n getView: async (object: string, type: 'list' | 'form' = 'list') => {\n const route = this.getRoute('ui');\n const res = await this.fetch(`${this.baseUrl}${route}/view/${object}?type=${type}`);\n return res.json();\n }\n };\n\n /**\n * Analytics Services\n */\n analytics = {\n query: async (payload: any) => {\n const route = this.getRoute('analytics');\n const res = await this.fetch(`${this.baseUrl}${route}/query`, {\n method: 'POST',\n body: JSON.stringify(payload)\n });\n return res.json();\n },\n meta: async (cube: string) => {\n const route = this.getRoute('analytics');\n const res = await this.fetch(`${this.baseUrl}${route}/meta/${cube}`);\n return res.json();\n },\n explain: async (payload: any) => {\n const route = this.getRoute('analytics');\n const res = await this.fetch(`${this.baseUrl}${route}/explain`, {\n method: 'POST',\n body: JSON.stringify(payload)\n });\n return res.json();\n }\n };\n\n /**\n * Hub Management Services\n */\n hub = {\n spaces: {\n list: async () => {\n const route = this.getRoute('hub');\n const res = await this.fetch(`${this.baseUrl}${route}/spaces`);\n return res.json();\n },\n create: async (payload: any) => {\n const route = this.getRoute('hub');\n const res = await this.fetch(`${this.baseUrl}${route}/spaces`, {\n method: 'POST',\n body: JSON.stringify(payload)\n });\n return res.json();\n }\n },\n plugins: {\n install: async (pkg: string, version?: string) => {\n const route = this.getRoute('hub');\n const res = await this.fetch(`${this.baseUrl}${route}/plugins/install`, {\n method: 'POST',\n body: JSON.stringify({ pkg, version })\n });\n return res.json();\n }\n }\n };\n\n /**\n * Authentication Services\n */\n auth = {\n login: async (request: LoginRequest): Promise<SessionResponse> => {\n const route = this.getRoute('auth');\n const res = await this.fetch(`${this.baseUrl}${route}/login`, {\n method: 'POST',\n body: JSON.stringify(request)\n });\n const data = await res.json();\n // Auto-set token if present in response\n if (data.data?.token) {\n this.token = data.data.token;\n }\n return data;\n },\n \n logout: async () => {\n const route = this.getRoute('auth');\n await this.fetch(`${this.baseUrl}${route}/logout`, { method: 'POST' });\n this.token = undefined;\n },\n\n me: async (): Promise<SessionResponse> => {\n const route = this.getRoute('auth');\n const res = await this.fetch(`${this.baseUrl}${route}/me`);\n return res.json();\n }\n };\n\n /**\n * Storage Services\n */\n storage = {\n upload: async (file: any, scope: string = 'user'): Promise<FileUploadResponse> => {\n // 1. Get Presigned URL\n const presignedReq: GetPresignedUrlRequest = {\n filename: file.name,\n mimeType: file.type,\n size: file.size,\n scope\n };\n \n const route = this.getRoute('storage');\n const presignedRes = await this.fetch(`${this.baseUrl}${route}/upload/presigned`, {\n method: 'POST',\n body: JSON.stringify(presignedReq)\n });\n const { data: presigned } = await presignedRes.json() as { data: PresignedUrlResponse['data'] };\n\n // 2. Upload to Cloud directly (Bypass API Middleware to avoid Auth headers if using S3)\n // Use fetchImpl directly\n const uploadRes = await this.fetchImpl(presigned.uploadUrl, {\n method: presigned.method,\n headers: presigned.headers,\n body: file\n });\n\n if (!uploadRes.ok) {\n throw new Error(`Storage Upload Failed: ${uploadRes.statusText}`);\n }\n\n // 3. Complete Upload\n const completeReq: CompleteUploadRequest = {\n fileId: presigned.fileId\n };\n const completeRes = await this.fetch(`${this.baseUrl}${route}/upload/complete`, {\n method: 'POST',\n body: JSON.stringify(completeReq)\n });\n \n return completeRes.json();\n },\n \n getDownloadUrl: async (fileId: string): Promise<string> => {\n const route = this.getRoute('storage');\n const res = await this.fetch(`${this.baseUrl}${route}/files/${fileId}/url`);\n const data = await res.json();\n return data.url;\n }\n };\n\n /**\n * Automation Services\n */\n automation = {\n trigger: async (triggerName: string, payload: any) => {\n const route = this.getRoute('automation');\n const res = await this.fetch(`${this.baseUrl}${route}/trigger/${triggerName}`, {\n method: 'POST',\n body: JSON.stringify(payload)\n });\n return res.json();\n }\n };\n\n /**\n * Data Operations\n */\n data = {\n /**\n * Advanced Query using ObjectStack Query Protocol\n * Supports both simplified options and full AST\n */\n query: async <T = any>(object: string, query: Partial<QueryAST>): Promise<PaginatedResult<T>> => {\n const route = this.getRoute('data');\n // POST for complex query to avoid URL length limits and allow clean JSON AST\n // Convention: POST /api/v1/data/:object/query\n const res = await this.fetch(`${this.baseUrl}${route}/${object}/query`, {\n method: 'POST',\n body: JSON.stringify(query)\n });\n return res.json();\n },\n\n find: async <T = any>(object: string, options: QueryOptions = {}): Promise<PaginatedResult<T>> => {\n const route = this.getRoute('data');\n const queryParams = new URLSearchParams();\n \n // 1. Handle Pagination\n if (options.top) queryParams.set('top', options.top.toString());\n if (options.skip) queryParams.set('skip', options.skip.toString());\n\n // 2. Handle Sort\n if (options.sort) {\n // Check if it's AST \n if (Array.isArray(options.sort) && typeof options.sort[0] === 'object') {\n queryParams.set('sort', JSON.stringify(options.sort));\n } else {\n const sortVal = Array.isArray(options.sort) ? options.sort.join(',') : options.sort;\n queryParams.set('sort', sortVal as string);\n }\n }\n \n // 3. Handle Select\n if (options.select) {\n queryParams.set('select', options.select.join(','));\n }\n\n // 4. Handle Filters (Simple vs AST)\n if (options.filters) {\n // If looks like AST (not plain object map)\n // TODO: robust check. safely assuming map for simplified find, and recommending .query() for AST\n if (this.isFilterAST(options.filters)) {\n queryParams.set('filters', JSON.stringify(options.filters));\n } else {\n Object.entries(options.filters).forEach(([k, v]) => {\n if (v !== undefined && v !== null) {\n queryParams.append(k, String(v));\n }\n });\n }\n }\n \n // 5. Handle Aggregations & GroupBy (Pass through as JSON if present)\n if (options.aggregations) {\n queryParams.set('aggregations', JSON.stringify(options.aggregations));\n }\n if (options.groupBy) {\n queryParams.set('groupBy', options.groupBy.join(','));\n }\n\n const res = await this.fetch(`${this.baseUrl}${route}/${object}?${queryParams.toString()}`);\n return res.json();\n },\n\n get: async <T = any>(object: string, id: string): Promise<T> => {\n const route = this.getRoute('data');\n const res = await this.fetch(`${this.baseUrl}${route}/${object}/${id}`);\n return res.json();\n },\n\n create: async <T = any>(object: string, data: Partial<T>): Promise<T> => {\n const route = this.getRoute('data');\n const res = await this.fetch(`${this.baseUrl}${route}/${object}`, {\n method: 'POST',\n body: JSON.stringify(data)\n });\n return res.json();\n },\n\n createMany: async <T = any>(object: string, data: Partial<T>[]): Promise<T[]> => {\n const route = this.getRoute('data');\n const res = await this.fetch(`${this.baseUrl}${route}/${object}/createMany`, {\n method: 'POST',\n body: JSON.stringify(data)\n });\n return res.json();\n },\n\n update: async <T = any>(object: string, id: string, data: Partial<T>): Promise<T> => {\n const route = this.getRoute('data');\n const res = await this.fetch(`${this.baseUrl}${route}/${object}/${id}`, {\n method: 'PATCH',\n body: JSON.stringify(data)\n });\n return res.json();\n },\n\n /**\n * Batch update multiple records\n * Uses the new BatchUpdateRequest schema with full control over options\n */\n batch: async (object: string, request: BatchUpdateRequest): Promise<BatchUpdateResponse> => {\n const route = this.getRoute('data');\n const res = await this.fetch(`${this.baseUrl}${route}/${object}/batch`, {\n method: 'POST',\n body: JSON.stringify(request)\n });\n return res.json();\n },\n\n /**\n * Update multiple records (simplified batch update)\n * Convenience method for batch updates without full BatchUpdateRequest\n */\n updateMany: async <T = any>(\n object: string, \n records: Array<{ id: string; data: Partial<T> }>,\n options?: BatchOptions\n ): Promise<BatchUpdateResponse> => {\n const route = this.getRoute('data');\n const request: UpdateManyRequest = {\n records,\n options\n };\n const res = await this.fetch(`${this.baseUrl}${route}/${object}/updateMany`, {\n method: 'POST',\n body: JSON.stringify(request)\n });\n return res.json();\n },\n\n delete: async (object: string, id: string): Promise<{ success: boolean }> => {\n const route = this.getRoute('data');\n const res = await this.fetch(`${this.baseUrl}${route}/${object}/${id}`, {\n method: 'DELETE'\n });\n return res.json();\n },\n\n /**\n * Delete multiple records by IDs\n */\n deleteMany: async(object: string, ids: string[], options?: BatchOptions): Promise<BatchUpdateResponse> => {\n const route = this.getRoute('data');\n const request: DeleteManyRequest = {\n ids,\n options\n };\n const res = await this.fetch(`${this.baseUrl}${route}/${object}/deleteMany`, {\n method: 'POST',\n body: JSON.stringify(request)\n });\n return res.json();\n }\n };\n\n\n\n /**\n * Private Helpers\n */\n\n private isFilterAST(filter: any): boolean {\n // Basic check: if array, it's [field, op, val] or [logic, node, node]\n // If object but not basic KV map... harder to tell without schema\n // For now, assume if it passes Array.isArray it's an AST root\n return Array.isArray(filter);\n }\n\n private async fetch(url: string, options: RequestInit = {}): Promise<Response> {\n this.logger.debug('HTTP request', { \n method: options.method || 'GET',\n url,\n hasBody: !!options.body\n });\n \n const headers: Record<string, string> = {\n 'Content-Type': 'application/json',\n ...(options.headers as Record<string, string> || {}),\n };\n\n if (this.token) {\n headers['Authorization'] = `Bearer ${this.token}`;\n }\n\n const res = await this.fetchImpl(url, { ...options, headers });\n \n this.logger.debug('HTTP response', { \n method: options.method || 'GET',\n url,\n status: res.status,\n ok: res.ok\n });\n \n if (!res.ok) {\n let errorBody: any;\n try {\n errorBody = await res.json();\n } catch {\n errorBody = { message: res.statusText };\n }\n \n this.logger.error('HTTP request failed', undefined, { \n method: options.method || 'GET',\n url,\n status: res.status,\n error: errorBody\n });\n \n // Create a standardized error if the response includes error details\n const errorMessage = errorBody?.message || errorBody?.error?.message || res.statusText;\n const errorCode = errorBody?.code || errorBody?.error?.code;\n const error = new Error(`[ObjectStack] ${errorCode ? `${errorCode}: ` : ''}${errorMessage}`) as any;\n \n // Attach error details for programmatic access\n error.code = errorCode;\n error.category = errorBody?.category;\n error.httpStatus = res.status;\n error.retryable = errorBody?.retryable;\n error.details = errorBody?.details || errorBody;\n \n throw error;\n }\n \n return res;\n }\n\n /**\n * Get the conventional route path for a given API endpoint type\n * ObjectStack uses standard conventions: /api/v1/data, /api/v1/metadata, /api/v1/ui\n */\n private getRoute(type: 'data' | 'metadata' | 'ui' | 'auth' | 'analytics' | 'hub' | 'storage' | 'automation'): string {\n // 1. Use discovered routes if available\n // Note: Spec uses 'endpoints', mapped dynamically\n if (this.discoveryInfo?.endpoints && (this.discoveryInfo.endpoints as any)[type]) {\n return (this.discoveryInfo.endpoints as any)[type];\n }\n\n // 2. Fallback to conventions\n // Note: HttpDispatcher expects /metadata, not /meta\n const routeMap: Record<string, string> = {\n data: '/api/v1/data',\n metadata: '/api/v1/metadata',\n ui: '/api/v1/ui',\n auth: '/api/v1/auth',\n analytics: '/api/v1/analytics',\n hub: '/api/v1/hub',\n storage: '/api/v1/storage',\n automation: '/api/v1/automation'\n };\n \n return routeMap[type] || `/api/v1/${type}`;\n }\n}\n\n// Re-export type-safe query builder\nexport { QueryBuilder, FilterBuilder, createQuery, createFilter } from './query-builder';\n\n// Re-export commonly used types from @objectstack/spec/api for convenience\nexport type {\n BatchUpdateRequest,\n BatchUpdateResponse,\n UpdateManyRequest,\n DeleteManyRequest,\n BatchOptions,\n BatchRecord,\n BatchOperationResult,\n MetadataCacheRequest,\n MetadataCacheResponse,\n StandardErrorCode,\n ErrorCategory,\n GetDiscoveryResponse,\n GetMetaTypesResponse,\n GetMetaItemsResponse\n} from '@objectstack/spec/api';\n","/**\n * Type-Safe Query Builder\n * \n * Provides a fluent API for building ObjectStack queries with:\n * - Compile-time type checking\n * - Intelligent code completion\n * - Runtime validation\n * - Type-safe filters and selections\n */\n\nimport { QueryAST, FilterCondition, SortNode } from '@objectstack/spec/data';\n\n/**\n * Type-safe filter builder\n */\nexport class FilterBuilder<T = any> {\n private conditions: FilterCondition[] = [];\n\n /**\n * Equality filter: field = value\n */\n equals<K extends keyof T>(field: K, value: T[K]): this {\n this.conditions.push([field as string, '=', value]);\n return this;\n }\n\n /**\n * Not equals filter: field != value\n */\n notEquals<K extends keyof T>(field: K, value: T[K]): this {\n this.conditions.push([field as string, '!=', value]);\n return this;\n }\n\n /**\n * Greater than filter: field > value\n */\n greaterThan<K extends keyof T>(field: K, value: T[K]): this {\n this.conditions.push([field as string, '>', value]);\n return this;\n }\n\n /**\n * Greater than or equal filter: field >= value\n */\n greaterThanOrEqual<K extends keyof T>(field: K, value: T[K]): this {\n this.conditions.push([field as string, '>=', value]);\n return this;\n }\n\n /**\n * Less than filter: field < value\n */\n lessThan<K extends keyof T>(field: K, value: T[K]): this {\n this.conditions.push([field as string, '<', value]);\n return this;\n }\n\n /**\n * Less than or equal filter: field <= value\n */\n lessThanOrEqual<K extends keyof T>(field: K, value: T[K]): this {\n this.conditions.push([field as string, '<=', value]);\n return this;\n }\n\n /**\n * IN filter: field IN (value1, value2, ...)\n */\n in<K extends keyof T>(field: K, values: T[K][]): this {\n this.conditions.push([field as string, 'in', values]);\n return this;\n }\n\n /**\n * NOT IN filter: field NOT IN (value1, value2, ...)\n */\n notIn<K extends keyof T>(field: K, values: T[K][]): this {\n this.conditions.push([field as string, 'not_in', values]);\n return this;\n }\n\n /**\n * LIKE filter: field LIKE pattern\n */\n like<K extends keyof T>(field: K, pattern: string): this {\n this.conditions.push([field as string, 'like', pattern]);\n return this;\n }\n\n /**\n * IS NULL filter: field IS NULL\n */\n isNull<K extends keyof T>(field: K): this {\n this.conditions.push([field as string, 'is_null', null]);\n return this;\n }\n\n /**\n * IS NOT NULL filter: field IS NOT NULL\n */\n isNotNull<K extends keyof T>(field: K): this {\n this.conditions.push([field as string, 'is_not_null', null]);\n return this;\n }\n\n /**\n * Build the filter condition\n */\n build(): FilterCondition {\n if (this.conditions.length === 0) {\n throw new Error('Filter builder has no conditions');\n }\n if (this.conditions.length === 1) {\n return this.conditions[0];\n }\n // Combine multiple conditions with AND\n return ['and', ...this.conditions];\n }\n\n /**\n * Get raw conditions array\n */\n getConditions(): FilterCondition[] {\n return this.conditions;\n }\n}\n\n/**\n * Type-safe query builder\n */\nexport class QueryBuilder<T = any> {\n private query: Partial<QueryAST> = {};\n private _object: string;\n\n constructor(object: string) {\n this._object = object;\n this.query.object = object;\n }\n\n /**\n * Select specific fields\n */\n select<K extends keyof T>(...fields: K[]): this {\n this.query.fields = fields as string[];\n return this;\n }\n\n /**\n * Add filters using a builder function\n */\n where(builderFn: (builder: FilterBuilder<T>) => void): this {\n const builder = new FilterBuilder<T>();\n builderFn(builder);\n const conditions = builder.getConditions();\n \n if (conditions.length === 1) {\n this.query.where = conditions[0];\n } else if (conditions.length > 1) {\n this.query.where = ['and', ...conditions] as FilterCondition;\n }\n \n return this;\n }\n\n /**\n * Add raw filter condition\n */\n filter(condition: FilterCondition): this {\n this.query.where = condition;\n return this;\n }\n\n /**\n * Sort by fields\n */\n orderBy<K extends keyof T>(field: K, order: 'asc' | 'desc' = 'asc'): this {\n if (!this.query.orderBy) {\n this.query.orderBy = [];\n }\n (this.query.orderBy as SortNode[]).push({\n field: field as string,\n order\n });\n return this;\n }\n\n /**\n * Limit the number of results\n */\n limit(count: number): this {\n this.query.limit = count;\n return this;\n }\n\n /**\n * Skip records (for pagination)\n */\n skip(count: number): this {\n this.query.offset = count;\n return this;\n }\n\n /**\n * Paginate results\n */\n paginate(page: number, pageSize: number): this {\n this.query.limit = pageSize;\n this.query.offset = (page - 1) * pageSize;\n return this;\n }\n\n /**\n * Group by fields\n */\n groupBy<K extends keyof T>(...fields: K[]): this {\n this.query.groupBy = fields as string[];\n return this;\n }\n\n /**\n * Build the final query AST\n */\n build(): QueryAST {\n return {\n object: this._object,\n ...this.query\n } as QueryAST;\n }\n\n /**\n * Get the current query state\n */\n getQuery(): Partial<QueryAST> {\n return { ...this.query };\n }\n}\n\n/**\n * Create a type-safe query builder for an object\n */\nexport function createQuery<T = any>(object: string): QueryBuilder<T> {\n return new QueryBuilder<T>(object);\n}\n\n/**\n * Create a type-safe filter builder\n */\nexport function createFilter<T = any>(): FilterBuilder<T> {\n return new FilterBuilder<T>();\n}\n"],"mappings":";;;;;;;;;;;;;;;;;;;;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAqBA,kBAAqC;;;ACN9B,IAAM,gBAAN,MAA6B;AAAA,EAA7B;AACL,SAAQ,aAAgC,CAAC;AAAA;AAAA;AAAA;AAAA;AAAA,EAKzC,OAA0B,OAAU,OAAmB;AACrD,SAAK,WAAW,KAAK,CAAC,OAAiB,KAAK,KAAK,CAAC;AAClD,WAAO;AAAA,EACT;AAAA;AAAA;AAAA;AAAA,EAKA,UAA6B,OAAU,OAAmB;AACxD,SAAK,WAAW,KAAK,CAAC,OAAiB,MAAM,KAAK,CAAC;AACnD,WAAO;AAAA,EACT;AAAA;AAAA;AAAA;AAAA,EAKA,YAA+B,OAAU,OAAmB;AAC1D,SAAK,WAAW,KAAK,CAAC,OAAiB,KAAK,KAAK,CAAC;AAClD,WAAO;AAAA,EACT;AAAA;AAAA;AAAA;AAAA,EAKA,mBAAsC,OAAU,OAAmB;AACjE,SAAK,WAAW,KAAK,CAAC,OAAiB,MAAM,KAAK,CAAC;AACnD,WAAO;AAAA,EACT;AAAA;AAAA;AAAA;AAAA,EAKA,SAA4B,OAAU,OAAmB;AACvD,SAAK,WAAW,KAAK,CAAC,OAAiB,KAAK,KAAK,CAAC;AAClD,WAAO;AAAA,EACT;AAAA;AAAA;AAAA;AAAA,EAKA,gBAAmC,OAAU,OAAmB;AAC9D,SAAK,WAAW,KAAK,CAAC,OAAiB,MAAM,KAAK,CAAC;AACnD,WAAO;AAAA,EACT;AAAA;AAAA;AAAA;AAAA,EAKA,GAAsB,OAAU,QAAsB;AACpD,SAAK,WAAW,KAAK,CAAC,OAAiB,MAAM,MAAM,CAAC;AACpD,WAAO;AAAA,EACT;AAAA;AAAA;AAAA;AAAA,EAKA,MAAyB,OAAU,QAAsB;AACvD,SAAK,WAAW,KAAK,CAAC,OAAiB,UAAU,MAAM,CAAC;AACxD,WAAO;AAAA,EACT;AAAA;AAAA;AAAA;AAAA,EAKA,KAAwB,OAAU,SAAuB;AACvD,SAAK,WAAW,KAAK,CAAC,OAAiB,QAAQ,OAAO,CAAC;AACvD,WAAO;AAAA,EACT;AAAA;AAAA;AAAA;AAAA,EAKA,OAA0B,OAAgB;AACxC,SAAK,WAAW,KAAK,CAAC,OAAiB,WAAW,IAAI,CAAC;AACvD,WAAO;AAAA,EACT;AAAA;AAAA;AAAA;AAAA,EAKA,UAA6B,OAAgB;AAC3C,SAAK,WAAW,KAAK,CAAC,OAAiB,eAAe,IAAI,CAAC;AAC3D,WAAO;AAAA,EACT;AAAA;AAAA;AAAA;AAAA,EAKA,QAAyB;AACvB,QAAI,KAAK,WAAW,WAAW,GAAG;AAChC,YAAM,IAAI,MAAM,kCAAkC;AAAA,IACpD;AACA,QAAI,KAAK,WAAW,WAAW,GAAG;AAChC,aAAO,KAAK,WAAW,CAAC;AAAA,IAC1B;AAEA,WAAO,CAAC,OAAO,GAAG,KAAK,UAAU;AAAA,EACnC;AAAA;AAAA;AAAA;AAAA,EAKA,gBAAmC;AACjC,WAAO,KAAK;AAAA,EACd;AACF;AAKO,IAAM,eAAN,MAA4B;AAAA,EAIjC,YAAY,QAAgB;AAH5B,SAAQ,QAA2B,CAAC;AAIlC,SAAK,UAAU;AACf,SAAK,MAAM,SAAS;AAAA,EACtB;AAAA;AAAA;AAAA;AAAA,EAKA,UAA6B,QAAmB;AAC9C,SAAK,MAAM,SAAS;AACpB,WAAO;AAAA,EACT;AAAA;AAAA;AAAA;AAAA,EAKA,MAAM,WAAsD;AAC1D,UAAM,UAAU,IAAI,cAAiB;AACrC,cAAU,OAAO;AACjB,UAAM,aAAa,QAAQ,cAAc;AAEzC,QAAI,WAAW,WAAW,GAAG;AAC3B,WAAK,MAAM,QAAQ,WAAW,CAAC;AAAA,IACjC,WAAW,WAAW,SAAS,GAAG;AAChC,WAAK,MAAM,QAAQ,CAAC,OAAO,GAAG,UAAU;AAAA,IAC1C;AAEA,WAAO;AAAA,EACT;AAAA;AAAA;AAAA;AAAA,EAKA,OAAO,WAAkC;AACvC,SAAK,MAAM,QAAQ;AACnB,WAAO;AAAA,EACT;AAAA;AAAA;AAAA;AAAA,EAKA,QAA2B,OAAU,QAAwB,OAAa;AACxE,QAAI,CAAC,KAAK,MAAM,SAAS;AACvB,WAAK,MAAM,UAAU,CAAC;AAAA,IACxB;AACA,IAAC,KAAK,MAAM,QAAuB,KAAK;AAAA,MACtC;AAAA,MACA;AAAA,IACF,CAAC;AACD,WAAO;AAAA,EACT;AAAA;AAAA;AAAA;AAAA,EAKA,MAAM,OAAqB;AACzB,SAAK,MAAM,QAAQ;AACnB,WAAO;AAAA,EACT;AAAA;AAAA;AAAA;AAAA,EAKA,KAAK,OAAqB;AACxB,SAAK,MAAM,SAAS;AACpB,WAAO;AAAA,EACT;AAAA;AAAA;AAAA;AAAA,EAKA,SAAS,MAAc,UAAwB;AAC7C,SAAK,MAAM,QAAQ;AACnB,SAAK,MAAM,UAAU,OAAO,KAAK;AACjC,WAAO;AAAA,EACT;AAAA;AAAA;AAAA;AAAA,EAKA,WAA8B,QAAmB;AAC/C,SAAK,MAAM,UAAU;AACrB,WAAO;AAAA,EACT;AAAA;AAAA;AAAA;AAAA,EAKA,QAAkB;AAChB,WAAO;AAAA,MACL,QAAQ,KAAK;AAAA,MACb,GAAG,KAAK;AAAA,IACV;AAAA,EACF;AAAA;AAAA;AAAA;AAAA,EAKA,WAA8B;AAC5B,WAAO,EAAE,GAAG,KAAK,MAAM;AAAA,EACzB;AACF;AAKO,SAAS,YAAqB,QAAiC;AACpE,SAAO,IAAI,aAAgB,MAAM;AACnC;AAKO,SAAS,eAA0C;AACxD,SAAO,IAAI,cAAiB;AAC9B;;;ADnLO,IAAM,oBAAN,MAAwB;AAAA,EAO7B,YAAY,QAAsB;AA2ClC;AAAA;AAAA;AAAA,gBAAO;AAAA;AAAA;AAAA;AAAA;AAAA,MAKL,UAAU,YAA2C;AACjD,cAAM,QAAQ,KAAK,SAAS,UAAU;AACtC,cAAM,MAAM,MAAM,KAAK,MAAM,GAAG,KAAK,OAAO,GAAG,KAAK,EAAE;AACtD,eAAO,IAAI,KAAK;AAAA,MACpB;AAAA;AAAA;AAAA;AAAA;AAAA,MAMA,UAAU,OAAO,SAAgD;AAC7D,cAAM,QAAQ,KAAK,SAAS,UAAU;AACtC,cAAM,MAAM,MAAM,KAAK,MAAM,GAAG,KAAK,OAAO,GAAG,KAAK,IAAI,IAAI,EAAE;AAC9D,eAAO,IAAI,KAAK;AAAA,MACpB;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA,MAOA,WAAW,OAAO,SAAiB;AAC/B,cAAM,QAAQ,KAAK,SAAS,UAAU;AACtC,cAAM,MAAM,MAAM,KAAK,MAAM,GAAG,KAAK,OAAO,GAAG,KAAK,WAAW,IAAI,EAAE;AACrE,eAAO,IAAI,KAAK;AAAA,MACpB;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA,MAOA,SAAS,OAAO,MAAc,SAAiB;AAC3C,cAAM,QAAQ,KAAK,SAAS,UAAU;AACtC,cAAM,MAAM,MAAM,KAAK,MAAM,GAAG,KAAK,OAAO,GAAG,KAAK,IAAI,IAAI,IAAI,IAAI,EAAE;AACtE,eAAO,IAAI,KAAK;AAAA,MACpB;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA,MAQA,UAAU,OAAO,MAAc,MAAc,SAAc;AACvD,cAAM,QAAQ,KAAK,SAAS,UAAU;AACtC,cAAM,MAAM,MAAM,KAAK,MAAM,GAAG,KAAK,OAAO,GAAG,KAAK,IAAI,IAAI,IAAI,IAAI,IAAI;AAAA,UACpE,QAAQ;AAAA,UACR,MAAM,KAAK,UAAU,IAAI;AAAA,QAC7B,CAAC;AACD,eAAO,IAAI,KAAK;AAAA,MACpB;AAAA;AAAA;AAAA;AAAA;AAAA,MAMA,WAAW,OAAO,MAAc,iBAAwE;AACpG,cAAM,QAAQ,KAAK,SAAS,UAAU;AACtC,cAAM,UAAkC,CAAC;AAEzC,YAAI,cAAc,aAAa;AAC7B,kBAAQ,eAAe,IAAI,aAAa;AAAA,QAC1C;AACA,YAAI,cAAc,iBAAiB;AACjC,kBAAQ,mBAAmB,IAAI,aAAa;AAAA,QAC9C;AAEA,cAAM,MAAM,MAAM,KAAK,MAAM,GAAG,KAAK,OAAO,GAAG,KAAK,WAAW,IAAI,IAAI;AAAA,UACrE;AAAA,QACF,CAAC;AAGD,YAAI,IAAI,WAAW,KAAK;AACtB,iBAAO;AAAA,YACL,aAAa;AAAA,YACb,MAAM,cAAc,cAAc;AAAA,cAChC,OAAO,aAAa,YAAY,QAAQ,WAAW,EAAE;AAAA,cACrD,MAAM,aAAa,YAAY,WAAW,IAAI;AAAA,YAChD,IAAI;AAAA,UACN;AAAA,QACF;AAEA,cAAM,OAAO,MAAM,IAAI,KAAK;AAC5B,cAAM,OAAO,IAAI,QAAQ,IAAI,MAAM;AACnC,cAAM,eAAe,IAAI,QAAQ,IAAI,eAAe;AAEpD,eAAO;AAAA,UACL;AAAA,UACA,MAAM,OAAO;AAAA,YACX,OAAO,KAAK,QAAQ,WAAW,EAAE;AAAA,YACjC,MAAM,KAAK,WAAW,IAAI;AAAA,UAC5B,IAAI;AAAA,UACJ,cAAc,gBAAgB;AAAA,UAC9B,aAAa;AAAA,QACf;AAAA,MACJ;AAAA,MAEA,SAAS,OAAO,QAAgB,OAAwB,WAAW;AAC/D,cAAM,QAAQ,KAAK,SAAS,IAAI;AAChC,cAAM,MAAM,MAAM,KAAK,MAAM,GAAG,KAAK,OAAO,GAAG,KAAK,SAAS,MAAM,SAAS,IAAI,EAAE;AAClF,eAAO,IAAI,KAAK;AAAA,MACpB;AAAA,IACF;AAKA;AAAA;AAAA;AAAA,qBAAY;AAAA,MACV,OAAO,OAAO,YAAiB;AAC7B,cAAM,QAAQ,KAAK,SAAS,WAAW;AACvC,cAAM,MAAM,MAAM,KAAK,MAAM,GAAG,KAAK,OAAO,GAAG,KAAK,UAAU;AAAA,UAC3D,QAAQ;AAAA,UACR,MAAM,KAAK,UAAU,OAAO;AAAA,QAC/B,CAAC;AACD,eAAO,IAAI,KAAK;AAAA,MAClB;AAAA,MACA,MAAM,OAAO,SAAiB;AAC1B,cAAM,QAAQ,KAAK,SAAS,WAAW;AACvC,cAAM,MAAM,MAAM,KAAK,MAAM,GAAG,KAAK,OAAO,GAAG,KAAK,SAAS,IAAI,EAAE;AACnE,eAAO,IAAI,KAAK;AAAA,MACpB;AAAA,MACA,SAAS,OAAO,YAAiB;AAC7B,cAAM,QAAQ,KAAK,SAAS,WAAW;AACvC,cAAM,MAAM,MAAM,KAAK,MAAM,GAAG,KAAK,OAAO,GAAG,KAAK,YAAY;AAAA,UAC5D,QAAQ;AAAA,UACR,MAAM,KAAK,UAAU,OAAO;AAAA,QAC/B,CAAC;AACD,eAAO,IAAI,KAAK;AAAA,MACrB;AAAA,IACF;AAKA;AAAA;AAAA;AAAA,eAAM;AAAA,MACJ,QAAQ;AAAA,QACJ,MAAM,YAAY;AACd,gBAAM,QAAQ,KAAK,SAAS,KAAK;AACjC,gBAAM,MAAM,MAAM,KAAK,MAAM,GAAG,KAAK,OAAO,GAAG,KAAK,SAAS;AAC7D,iBAAO,IAAI,KAAK;AAAA,QACpB;AAAA,QACA,QAAQ,OAAO,YAAiB;AAC5B,gBAAM,QAAQ,KAAK,SAAS,KAAK;AACjC,gBAAM,MAAM,MAAM,KAAK,MAAM,GAAG,KAAK,OAAO,GAAG,KAAK,WAAW;AAAA,YAC3D,QAAQ;AAAA,YACR,MAAM,KAAK,UAAU,OAAO;AAAA,UAChC,CAAC;AACD,iBAAO,IAAI,KAAK;AAAA,QACpB;AAAA,MACJ;AAAA,MACA,SAAS;AAAA,QACL,SAAS,OAAO,KAAa,YAAqB;AAC9C,gBAAM,QAAQ,KAAK,SAAS,KAAK;AACjC,gBAAM,MAAM,MAAM,KAAK,MAAM,GAAG,KAAK,OAAO,GAAG,KAAK,oBAAoB;AAAA,YACpE,QAAQ;AAAA,YACR,MAAM,KAAK,UAAU,EAAE,KAAK,QAAQ,CAAC;AAAA,UACzC,CAAC;AACD,iBAAO,IAAI,KAAK;AAAA,QACpB;AAAA,MACJ;AAAA,IACF;AAKA;AAAA;AAAA;AAAA,gBAAO;AAAA,MACL,OAAO,OAAO,YAAoD;AAC9D,cAAM,QAAQ,KAAK,SAAS,MAAM;AAClC,cAAM,MAAM,MAAM,KAAK,MAAM,GAAG,KAAK,OAAO,GAAG,KAAK,UAAU;AAAA,UAC1D,QAAQ;AAAA,UACR,MAAM,KAAK,UAAU,OAAO;AAAA,QAChC,CAAC;AACD,cAAM,OAAO,MAAM,IAAI,KAAK;AAE5B,YAAI,KAAK,MAAM,OAAO;AAClB,eAAK,QAAQ,KAAK,KAAK;AAAA,QAC3B;AACA,eAAO;AAAA,MACX;AAAA,MAEA,QAAQ,YAAY;AAChB,cAAM,QAAQ,KAAK,SAAS,MAAM;AAClC,cAAM,KAAK,MAAM,GAAG,KAAK,OAAO,GAAG,KAAK,WAAW,EAAE,QAAQ,OAAO,CAAC;AACrE,aAAK,QAAQ;AAAA,MACjB;AAAA,MAEA,IAAI,YAAsC;AACtC,cAAM,QAAQ,KAAK,SAAS,MAAM;AAClC,cAAM,MAAM,MAAM,KAAK,MAAM,GAAG,KAAK,OAAO,GAAG,KAAK,KAAK;AACzD,eAAO,IAAI,KAAK;AAAA,MACpB;AAAA,IACF;AAKA;AAAA;AAAA;AAAA,mBAAU;AAAA,MACR,QAAQ,OAAO,MAAW,QAAgB,WAAwC;AAE9E,cAAM,eAAuC;AAAA,UACzC,UAAU,KAAK;AAAA,UACf,UAAU,KAAK;AAAA,UACf,MAAM,KAAK;AAAA,UACX;AAAA,QACJ;AAEA,cAAM,QAAQ,KAAK,SAAS,SAAS;AACrC,cAAM,eAAe,MAAM,KAAK,MAAM,GAAG,KAAK,OAAO,GAAG,KAAK,qBAAqB;AAAA,UAC9E,QAAQ;AAAA,UACR,MAAM,KAAK,UAAU,YAAY;AAAA,QACrC,CAAC;AACD,cAAM,EAAE,MAAM,UAAU,IAAI,MAAM,aAAa,KAAK;AAIpD,cAAM,YAAY,MAAM,KAAK,UAAU,UAAU,WAAW;AAAA,UACxD,QAAQ,UAAU;AAAA,UAClB,SAAS,UAAU;AAAA,UACnB,MAAM;AAAA,QACV,CAAC;AAED,YAAI,CAAC,UAAU,IAAI;AACf,gBAAM,IAAI,MAAM,0BAA0B,UAAU,UAAU,EAAE;AAAA,QACpE;AAGA,cAAM,cAAqC;AAAA,UACvC,QAAQ,UAAU;AAAA,QACtB;AACA,cAAM,cAAc,MAAM,KAAK,MAAM,GAAG,KAAK,OAAO,GAAG,KAAK,oBAAoB;AAAA,UAC5E,QAAQ;AAAA,UACR,MAAM,KAAK,UAAU,WAAW;AAAA,QACpC,CAAC;AAED,eAAO,YAAY,KAAK;AAAA,MAC5B;AAAA,MAEA,gBAAgB,OAAO,WAAoC;AACvD,cAAM,QAAQ,KAAK,SAAS,SAAS;AACrC,cAAM,MAAM,MAAM,KAAK,MAAM,GAAG,KAAK,OAAO,GAAG,KAAK,UAAU,MAAM,MAAM;AAC1E,cAAM,OAAO,MAAM,IAAI,KAAK;AAC5B,eAAO,KAAK;AAAA,MAChB;AAAA,IACF;AAKA;AAAA;AAAA;AAAA,sBAAa;AAAA,MACT,SAAS,OAAO,aAAqB,YAAiB;AAClD,cAAM,QAAQ,KAAK,SAAS,YAAY;AACxC,cAAM,MAAM,MAAM,KAAK,MAAM,GAAG,KAAK,OAAO,GAAG,KAAK,YAAY,WAAW,IAAI;AAAA,UAC3E,QAAQ;AAAA,UACR,MAAM,KAAK,UAAU,OAAO;AAAA,QAChC,CAAC;AACD,eAAO,IAAI,KAAK;AAAA,MACpB;AAAA,IACJ;AAKA;AAAA;AAAA;AAAA,gBAAO;AAAA;AAAA;AAAA;AAAA;AAAA,MAKL,OAAO,OAAgB,QAAgB,UAA0D;AAC/F,cAAM,QAAQ,KAAK,SAAS,MAAM;AAGlC,cAAM,MAAM,MAAM,KAAK,MAAM,GAAG,KAAK,OAAO,GAAG,KAAK,IAAI,MAAM,UAAU;AAAA,UACtE,QAAQ;AAAA,UACR,MAAM,KAAK,UAAU,KAAK;AAAA,QAC5B,CAAC;AACD,eAAO,IAAI,KAAK;AAAA,MAClB;AAAA,MAEA,MAAM,OAAgB,QAAgB,UAAwB,CAAC,MAAmC;AAC9F,cAAM,QAAQ,KAAK,SAAS,MAAM;AAClC,cAAM,cAAc,IAAI,gBAAgB;AAGxC,YAAI,QAAQ,IAAK,aAAY,IAAI,OAAO,QAAQ,IAAI,SAAS,CAAC;AAC9D,YAAI,QAAQ,KAAM,aAAY,IAAI,QAAQ,QAAQ,KAAK,SAAS,CAAC;AAGjE,YAAI,QAAQ,MAAM;AAEd,cAAI,MAAM,QAAQ,QAAQ,IAAI,KAAK,OAAO,QAAQ,KAAK,CAAC,MAAM,UAAU;AACnE,wBAAY,IAAI,QAAQ,KAAK,UAAU,QAAQ,IAAI,CAAC;AAAA,UACzD,OAAO;AACF,kBAAM,UAAU,MAAM,QAAQ,QAAQ,IAAI,IAAI,QAAQ,KAAK,KAAK,GAAG,IAAI,QAAQ;AAC/E,wBAAY,IAAI,QAAQ,OAAiB;AAAA,UAC9C;AAAA,QACJ;AAGA,YAAI,QAAQ,QAAQ;AAChB,sBAAY,IAAI,UAAU,QAAQ,OAAO,KAAK,GAAG,CAAC;AAAA,QACtD;AAGA,YAAI,QAAQ,SAAS;AAGhB,cAAI,KAAK,YAAY,QAAQ,OAAO,GAAG;AACnC,wBAAY,IAAI,WAAW,KAAK,UAAU,QAAQ,OAAO,CAAC;AAAA,UAC9D,OAAO;AACH,mBAAO,QAAQ,QAAQ,OAAO,EAAE,QAAQ,CAAC,CAAC,GAAG,CAAC,MAAM;AAChD,kBAAI,MAAM,UAAa,MAAM,MAAM;AAChC,4BAAY,OAAO,GAAG,OAAO,CAAC,CAAC;AAAA,cAClC;AAAA,YACJ,CAAC;AAAA,UACL;AAAA,QACL;AAGA,YAAI,QAAQ,cAAc;AACtB,sBAAY,IAAI,gBAAgB,KAAK,UAAU,QAAQ,YAAY,CAAC;AAAA,QACxE;AACA,YAAI,QAAQ,SAAS;AAChB,sBAAY,IAAI,WAAW,QAAQ,QAAQ,KAAK,GAAG,CAAC;AAAA,QACzD;AAEA,cAAM,MAAM,MAAM,KAAK,MAAM,GAAG,KAAK,OAAO,GAAG,KAAK,IAAI,MAAM,IAAI,YAAY,SAAS,CAAC,EAAE;AAC1F,eAAO,IAAI,KAAK;AAAA,MACpB;AAAA,MAEA,KAAK,OAAgB,QAAgB,OAA2B;AAC5D,cAAM,QAAQ,KAAK,SAAS,MAAM;AAClC,cAAM,MAAM,MAAM,KAAK,MAAM,GAAG,KAAK,OAAO,GAAG,KAAK,IAAI,MAAM,IAAI,EAAE,EAAE;AACtE,eAAO,IAAI,KAAK;AAAA,MACpB;AAAA,MAEA,QAAQ,OAAgB,QAAgB,SAAiC;AACrE,cAAM,QAAQ,KAAK,SAAS,MAAM;AAClC,cAAM,MAAM,MAAM,KAAK,MAAM,GAAG,KAAK,OAAO,GAAG,KAAK,IAAI,MAAM,IAAI;AAAA,UAC9D,QAAQ;AAAA,UACR,MAAM,KAAK,UAAU,IAAI;AAAA,QAC7B,CAAC;AACD,eAAO,IAAI,KAAK;AAAA,MACpB;AAAA,MAEA,YAAY,OAAgB,QAAgB,SAAqC;AAC7E,cAAM,QAAQ,KAAK,SAAS,MAAM;AAClC,cAAM,MAAM,MAAM,KAAK,MAAM,GAAG,KAAK,OAAO,GAAG,KAAK,IAAI,MAAM,eAAe;AAAA,UACzE,QAAQ;AAAA,UACR,MAAM,KAAK,UAAU,IAAI;AAAA,QAC7B,CAAC;AACD,eAAO,IAAI,KAAK;AAAA,MACpB;AAAA,MAEA,QAAQ,OAAgB,QAAgB,IAAY,SAAiC;AACjF,cAAM,QAAQ,KAAK,SAAS,MAAM;AAClC,cAAM,MAAM,MAAM,KAAK,MAAM,GAAG,KAAK,OAAO,GAAG,KAAK,IAAI,MAAM,IAAI,EAAE,IAAI;AAAA,UACpE,QAAQ;AAAA,UACR,MAAM,KAAK,UAAU,IAAI;AAAA,QAC7B,CAAC;AACD,eAAO,IAAI,KAAK;AAAA,MACpB;AAAA;AAAA;AAAA;AAAA;AAAA,MAMA,OAAO,OAAO,QAAgB,YAA8D;AACxF,cAAM,QAAQ,KAAK,SAAS,MAAM;AAClC,cAAM,MAAM,MAAM,KAAK,MAAM,GAAG,KAAK,OAAO,GAAG,KAAK,IAAI,MAAM,UAAU;AAAA,UACpE,QAAQ;AAAA,UACR,MAAM,KAAK,UAAU,OAAO;AAAA,QAChC,CAAC;AACD,eAAO,IAAI,KAAK;AAAA,MACpB;AAAA;AAAA;AAAA;AAAA;AAAA,MAMA,YAAY,OACV,QACA,SACA,YACiC;AAC/B,cAAM,QAAQ,KAAK,SAAS,MAAM;AAClC,cAAM,UAA6B;AAAA,UACjC;AAAA,UACA;AAAA,QACF;AACA,cAAM,MAAM,MAAM,KAAK,MAAM,GAAG,KAAK,OAAO,GAAG,KAAK,IAAI,MAAM,eAAe;AAAA,UACzE,QAAQ;AAAA,UACR,MAAM,KAAK,UAAU,OAAO;AAAA,QAChC,CAAC;AACD,eAAO,IAAI,KAAK;AAAA,MACpB;AAAA,MAEA,QAAQ,OAAO,QAAgB,OAA8C;AACzE,cAAM,QAAQ,KAAK,SAAS,MAAM;AAClC,cAAM,MAAM,MAAM,KAAK,MAAM,GAAG,KAAK,OAAO,GAAG,KAAK,IAAI,MAAM,IAAI,EAAE,IAAI;AAAA,UACpE,QAAQ;AAAA,QACZ,CAAC;AACD,eAAO,IAAI,KAAK;AAAA,MACpB;AAAA;AAAA;AAAA;AAAA,MAKA,YAAY,OAAM,QAAgB,KAAe,YAAyD;AACtG,cAAM,QAAQ,KAAK,SAAS,MAAM;AAClC,cAAM,UAA6B;AAAA,UACjC;AAAA,UACA;AAAA,QACF;AACA,cAAM,MAAM,MAAM,KAAK,MAAM,GAAG,KAAK,OAAO,GAAG,KAAK,IAAI,MAAM,eAAe;AAAA,UACxE,QAAQ;AAAA,UACR,MAAM,KAAK,UAAU,OAAO;AAAA,QACjC,CAAC;AACD,eAAO,IAAI,KAAK;AAAA,MACpB;AAAA,IACF;AAndE,SAAK,UAAU,OAAO,QAAQ,QAAQ,OAAO,EAAE;AAC/C,SAAK,QAAQ,OAAO;AACpB,SAAK,YAAY,OAAO,SAAS,WAAW,MAAM,KAAK,UAAU;AAGjE,SAAK,SAAS,OAAO,cAAU,0BAAa;AAAA,MAC1C,OAAO,OAAO,QAAQ,UAAU;AAAA,MAChC,QAAQ;AAAA,IACV,CAAC;AAED,SAAK,OAAO,MAAM,8BAA8B,EAAE,SAAS,KAAK,QAAQ,CAAC;AAAA,EAC3E;AAAA;AAAA;AAAA;AAAA,EAKA,MAAM,UAAU;AACd,SAAK,OAAO,MAAM,oCAAoC,EAAE,SAAS,KAAK,QAAQ,CAAC;AAE/E,QAAI;AAEF,YAAM,MAAM,MAAM,KAAK,MAAM,GAAG,KAAK,OAAO,SAAS;AAErD,YAAM,OAAO,MAAM,IAAI,KAAK;AAC5B,WAAK,gBAAgB;AAErB,WAAK,OAAO,KAAK,mCAAmC;AAAA,QAClD,SAAS,KAAK;AAAA,QACd,SAAS,KAAK;AAAA,QACd,cAAc,KAAK;AAAA,MACrB,CAAC;AAED,aAAO;AAAA,IACT,SAAS,GAAG;AACV,WAAK,OAAO,MAAM,2CAA2C,GAAY,EAAE,SAAS,KAAK,QAAQ,CAAC;AAClG,YAAM;AAAA,IACR;AAAA,EACF;AAAA;AAAA;AAAA;AAAA,EAsbQ,YAAY,QAAsB;AAIxC,WAAO,MAAM,QAAQ,MAAM;AAAA,EAC7B;AAAA,EAEA,MAAc,MAAM,KAAa,UAAuB,CAAC,GAAsB;AAC7E,SAAK,OAAO,MAAM,gBAAgB;AAAA,MAChC,QAAQ,QAAQ,UAAU;AAAA,MAC1B;AAAA,MACA,SAAS,CAAC,CAAC,QAAQ;AAAA,IACrB,CAAC;AAED,UAAM,UAAkC;AAAA,MACtC,gBAAgB;AAAA,MAChB,GAAI,QAAQ,WAAqC,CAAC;AAAA,IACpD;AAEA,QAAI,KAAK,OAAO;AACZ,cAAQ,eAAe,IAAI,UAAU,KAAK,KAAK;AAAA,IACnD;AAEA,UAAM,MAAM,MAAM,KAAK,UAAU,KAAK,EAAE,GAAG,SAAS,QAAQ,CAAC;AAE7D,SAAK,OAAO,MAAM,iBAAiB;AAAA,MACjC,QAAQ,QAAQ,UAAU;AAAA,MAC1B;AAAA,MACA,QAAQ,IAAI;AAAA,MACZ,IAAI,IAAI;AAAA,IACV,CAAC;AAED,QAAI,CAAC,IAAI,IAAI;AACT,UAAI;AACJ,UAAI;AACA,oBAAY,MAAM,IAAI,KAAK;AAAA,MAC/B,QAAQ;AACJ,oBAAY,EAAE,SAAS,IAAI,WAAW;AAAA,MAC1C;AAEA,WAAK,OAAO,MAAM,uBAAuB,QAAW;AAAA,QAClD,QAAQ,QAAQ,UAAU;AAAA,QAC1B;AAAA,QACA,QAAQ,IAAI;AAAA,QACZ,OAAO;AAAA,MACT,CAAC;AAGD,YAAM,eAAe,WAAW,WAAW,WAAW,OAAO,WAAW,IAAI;AAC5E,YAAM,YAAY,WAAW,QAAQ,WAAW,OAAO;AACvD,YAAM,QAAQ,IAAI,MAAM,iBAAiB,YAAY,GAAG,SAAS,OAAO,EAAE,GAAG,YAAY,EAAE;AAG3F,YAAM,OAAO;AACb,YAAM,WAAW,WAAW;AAC5B,YAAM,aAAa,IAAI;AACvB,YAAM,YAAY,WAAW;AAC7B,YAAM,UAAU,WAAW,WAAW;AAEtC,YAAM;AAAA,IACV;AAEA,WAAO;AAAA,EACT;AAAA;AAAA;AAAA;AAAA;AAAA,EAMQ,SAAS,MAAoG;AAGnH,QAAI,KAAK,eAAe,aAAc,KAAK,cAAc,UAAkB,IAAI,GAAG;AAC9E,aAAQ,KAAK,cAAc,UAAkB,IAAI;AAAA,IACrD;AAIA,UAAM,WAAmC;AAAA,MACvC,MAAM;AAAA,MACN,UAAU;AAAA,MACV,IAAI;AAAA,MACJ,MAAM;AAAA,MACN,WAAW;AAAA,MACX,KAAK;AAAA,MACL,SAAS;AAAA,MACT,YAAY;AAAA,IACd;AAEA,WAAO,SAAS,IAAI,KAAK,WAAW,IAAI;AAAA,EAC1C;AACF;","names":[]}
1
+ {"version":3,"sources":["../src/index.ts","../src/query-builder.ts"],"sourcesContent":["import { QueryAST, SortNode, AggregationNode } from '@objectstack/spec/data';\nimport { \n BatchUpdateRequest, \n BatchUpdateResponse, \n UpdateManyRequest,\n DeleteManyRequest,\n BatchOptions,\n MetadataCacheRequest,\n MetadataCacheResponse,\n StandardErrorCode,\n ErrorCategory,\n GetDiscoveryResponse,\n GetMetaTypesResponse,\n GetMetaItemsResponse,\n LoginRequest,\n SessionResponse,\n GetPresignedUrlRequest,\n PresignedUrlResponse,\n CompleteUploadRequest,\n FileUploadResponse\n} from '@objectstack/spec/api';\nimport { Logger, createLogger } from '@objectstack/core';\n\nexport interface ClientConfig {\n baseUrl: string;\n token?: string;\n /**\n * Custom fetch implementation (e.g. node-fetch or for Next.js caching)\n */\n fetch?: (input: RequestInfo | URL, init?: RequestInit) => Promise<Response>;\n /**\n * Logger instance for debugging\n */\n logger?: Logger;\n /**\n * Enable debug logging\n */\n debug?: boolean;\n}\n\n/**\n * Discovery Result\n * Re-export from @objectstack/spec/api for convenience\n */\nexport type DiscoveryResult = GetDiscoveryResponse;\n\nexport interface QueryOptions {\n select?: string[]; // Simplified Selection\n filters?: Record<string, any>; // Map or AST\n sort?: string | string[] | SortNode[]; // 'name' or ['-created_at'] or AST\n top?: number;\n skip?: number;\n // Advanced features\n aggregations?: AggregationNode[];\n groupBy?: string[];\n}\n\nexport interface PaginatedResult<T = any> {\n value: T[];\n count: number;\n}\n\nexport interface StandardError {\n code: StandardErrorCode;\n message: string;\n category: ErrorCategory;\n httpStatus: number;\n retryable: boolean;\n details?: Record<string, any>;\n}\n\nexport class ObjectStackClient {\n private baseUrl: string;\n private token?: string;\n private fetchImpl: (input: RequestInfo | URL, init?: RequestInit) => Promise<Response>;\n private discoveryInfo?: DiscoveryResult;\n private logger: Logger;\n\n constructor(config: ClientConfig) {\n this.baseUrl = config.baseUrl.replace(/\\/$/, ''); // Remove trailing slash\n this.token = config.token;\n this.fetchImpl = config.fetch || globalThis.fetch.bind(globalThis);\n \n // Initialize logger\n this.logger = config.logger || createLogger({ \n level: config.debug ? 'debug' : 'info',\n format: 'pretty'\n });\n \n this.logger.debug('ObjectStack client created', { baseUrl: this.baseUrl });\n }\n\n /**\n * Initialize the client by discovering server capabilities.\n */\n async connect() {\n this.logger.debug('Connecting to ObjectStack server', { baseUrl: this.baseUrl });\n \n try {\n let data: DiscoveryResult | undefined;\n\n // 1. Try Standard Discovery (.well-known)\n try {\n let wellKnownUrl: string;\n try {\n // If baseUrl is absolute, get origin\n const url = new URL(this.baseUrl);\n wellKnownUrl = `${url.origin}/.well-known/objectstack`;\n } catch {\n // If baseUrl is relative, use absolute path from root\n wellKnownUrl = '/.well-known/objectstack';\n }\n\n this.logger.debug('Probing .well-known discovery', { url: wellKnownUrl });\n const res = await this.fetchImpl(wellKnownUrl);\n if (res.ok) {\n data = await res.json();\n this.logger.debug('Discovered via .well-known');\n }\n } catch (e) {\n this.logger.debug('Standard discovery probe failed', { error: (e as Error).message });\n }\n\n // 2. Fallback to Legacy/Direct Path /api/v1\n if (!data) {\n const fallbackUrl = `${this.baseUrl}/api/v1`;\n this.logger.debug('Falling back to legacy discovery', { url: fallbackUrl });\n const res = await this.fetchImpl(fallbackUrl);\n if (!res.ok) {\n throw new Error(`Failed to connect to ${fallbackUrl}: ${res.statusText}`);\n }\n data = await res.json();\n }\n\n if (!data) {\n throw new Error('Connection failed: No discovery data returned');\n }\n\n this.discoveryInfo = data;\n \n this.logger.info('Connected to ObjectStack server', { \n version: data.version,\n apiName: data.apiName,\n capabilities: data.capabilities \n });\n \n return data as DiscoveryResult;\n } catch (e) {\n this.logger.error('Failed to connect to ObjectStack server', e as Error, { baseUrl: this.baseUrl });\n throw e;\n }\n }\n\n /**\n * Metadata Operations\n */\n meta = {\n /**\n * Get all available metadata types\n * Returns types like 'object', 'plugin', 'view', etc.\n */\n getTypes: async (): Promise<GetMetaTypesResponse> => {\n const route = this.getRoute('metadata');\n const res = await this.fetch(`${this.baseUrl}${route}`);\n return res.json();\n },\n\n /**\n * Get all items of a specific metadata type\n * @param type - Metadata type name (e.g., 'object', 'plugin')\n */\n getItems: async (type: string): Promise<GetMetaItemsResponse> => {\n const route = this.getRoute('metadata');\n const res = await this.fetch(`${this.baseUrl}${route}/${type}`);\n return res.json();\n },\n\n /**\n * Get a specific object definition by name\n * @deprecated Use `getItem('object', name)` instead for consistency with spec protocol\n * @param name - Object name (snake_case identifier)\n */\n getObject: async (name: string) => {\n const route = this.getRoute('metadata');\n const res = await this.fetch(`${this.baseUrl}${route}/object/${name}`);\n return res.json();\n },\n\n /**\n * Get a specific metadata item by type and name\n * @param type - Metadata type (e.g., 'object', 'plugin')\n * @param name - Item name (snake_case identifier)\n */\n getItem: async (type: string, name: string) => {\n const route = this.getRoute('metadata');\n const res = await this.fetch(`${this.baseUrl}${route}/${type}/${name}`);\n return res.json();\n },\n\n /**\n * Save a metadata item\n * @param type - Metadata type (e.g., 'object', 'plugin')\n * @param name - Item name\n * @param item - The metadata content to save\n */\n saveItem: async (type: string, name: string, item: any) => {\n const route = this.getRoute('metadata');\n const res = await this.fetch(`${this.baseUrl}${route}/${type}/${name}`, {\n method: 'PUT',\n body: JSON.stringify(item)\n });\n return res.json();\n },\n \n /**\n * Get object metadata with cache support\n * Supports ETag-based conditional requests for efficient caching\n */\n getCached: async (name: string, cacheOptions?: MetadataCacheRequest): Promise<MetadataCacheResponse> => {\n const route = this.getRoute('metadata');\n const headers: Record<string, string> = {};\n \n if (cacheOptions?.ifNoneMatch) {\n headers['If-None-Match'] = cacheOptions.ifNoneMatch;\n }\n if (cacheOptions?.ifModifiedSince) {\n headers['If-Modified-Since'] = cacheOptions.ifModifiedSince;\n }\n \n const res = await this.fetch(`${this.baseUrl}${route}/object/${name}`, {\n headers\n });\n \n // Check for 304 Not Modified\n if (res.status === 304) {\n return {\n notModified: true,\n etag: cacheOptions?.ifNoneMatch ? { \n value: cacheOptions.ifNoneMatch.replace(/^W\\/|\"/g, ''),\n weak: cacheOptions.ifNoneMatch.startsWith('W/')\n } : undefined\n };\n }\n \n const data = await res.json();\n const etag = res.headers.get('ETag');\n const lastModified = res.headers.get('Last-Modified');\n \n return {\n data,\n etag: etag ? { \n value: etag.replace(/^W\\/|\"/g, ''), \n weak: etag.startsWith('W/') \n } : undefined,\n lastModified: lastModified || undefined,\n notModified: false\n };\n },\n \n getView: async (object: string, type: 'list' | 'form' = 'list') => {\n const route = this.getRoute('ui');\n const res = await this.fetch(`${this.baseUrl}${route}/view/${object}?type=${type}`);\n return res.json();\n }\n };\n\n /**\n * Analytics Services\n */\n analytics = {\n query: async (payload: any) => {\n const route = this.getRoute('analytics');\n const res = await this.fetch(`${this.baseUrl}${route}/query`, {\n method: 'POST',\n body: JSON.stringify(payload)\n });\n return res.json();\n },\n meta: async (cube: string) => {\n const route = this.getRoute('analytics');\n const res = await this.fetch(`${this.baseUrl}${route}/meta/${cube}`);\n return res.json();\n },\n explain: async (payload: any) => {\n const route = this.getRoute('analytics');\n const res = await this.fetch(`${this.baseUrl}${route}/explain`, {\n method: 'POST',\n body: JSON.stringify(payload)\n });\n return res.json();\n }\n };\n\n /**\n * Hub Management Services\n */\n hub = {\n spaces: {\n list: async () => {\n const route = this.getRoute('hub');\n const res = await this.fetch(`${this.baseUrl}${route}/spaces`);\n return res.json();\n },\n create: async (payload: any) => {\n const route = this.getRoute('hub');\n const res = await this.fetch(`${this.baseUrl}${route}/spaces`, {\n method: 'POST',\n body: JSON.stringify(payload)\n });\n return res.json();\n }\n },\n plugins: {\n install: async (pkg: string, version?: string) => {\n const route = this.getRoute('hub');\n const res = await this.fetch(`${this.baseUrl}${route}/plugins/install`, {\n method: 'POST',\n body: JSON.stringify({ pkg, version })\n });\n return res.json();\n }\n }\n };\n\n /**\n * Authentication Services\n */\n auth = {\n login: async (request: LoginRequest): Promise<SessionResponse> => {\n const route = this.getRoute('auth');\n const res = await this.fetch(`${this.baseUrl}${route}/login`, {\n method: 'POST',\n body: JSON.stringify(request)\n });\n const data = await res.json();\n // Auto-set token if present in response\n if (data.data?.token) {\n this.token = data.data.token;\n }\n return data;\n },\n \n logout: async () => {\n const route = this.getRoute('auth');\n await this.fetch(`${this.baseUrl}${route}/logout`, { method: 'POST' });\n this.token = undefined;\n },\n\n me: async (): Promise<SessionResponse> => {\n const route = this.getRoute('auth');\n const res = await this.fetch(`${this.baseUrl}${route}/me`);\n return res.json();\n }\n };\n\n /**\n * Storage Services\n */\n storage = {\n upload: async (file: any, scope: string = 'user'): Promise<FileUploadResponse> => {\n // 1. Get Presigned URL\n const presignedReq: GetPresignedUrlRequest = {\n filename: file.name,\n mimeType: file.type,\n size: file.size,\n scope\n };\n \n const route = this.getRoute('storage');\n const presignedRes = await this.fetch(`${this.baseUrl}${route}/upload/presigned`, {\n method: 'POST',\n body: JSON.stringify(presignedReq)\n });\n const { data: presigned } = await presignedRes.json() as { data: PresignedUrlResponse['data'] };\n\n // 2. Upload to Cloud directly (Bypass API Middleware to avoid Auth headers if using S3)\n // Use fetchImpl directly\n const uploadRes = await this.fetchImpl(presigned.uploadUrl, {\n method: presigned.method,\n headers: presigned.headers,\n body: file\n });\n\n if (!uploadRes.ok) {\n throw new Error(`Storage Upload Failed: ${uploadRes.statusText}`);\n }\n\n // 3. Complete Upload\n const completeReq: CompleteUploadRequest = {\n fileId: presigned.fileId\n };\n const completeRes = await this.fetch(`${this.baseUrl}${route}/upload/complete`, {\n method: 'POST',\n body: JSON.stringify(completeReq)\n });\n \n return completeRes.json();\n },\n \n getDownloadUrl: async (fileId: string): Promise<string> => {\n const route = this.getRoute('storage');\n const res = await this.fetch(`${this.baseUrl}${route}/files/${fileId}/url`);\n const data = await res.json();\n return data.url;\n }\n };\n\n /**\n * Automation Services\n */\n automation = {\n trigger: async (triggerName: string, payload: any) => {\n const route = this.getRoute('automation');\n const res = await this.fetch(`${this.baseUrl}${route}/trigger/${triggerName}`, {\n method: 'POST',\n body: JSON.stringify(payload)\n });\n return res.json();\n }\n };\n\n /**\n * Data Operations\n */\n data = {\n /**\n * Advanced Query using ObjectStack Query Protocol\n * Supports both simplified options and full AST\n */\n query: async <T = any>(object: string, query: Partial<QueryAST>): Promise<PaginatedResult<T>> => {\n const route = this.getRoute('data');\n // POST for complex query to avoid URL length limits and allow clean JSON AST\n // Convention: POST /api/v1/data/:object/query\n const res = await this.fetch(`${this.baseUrl}${route}/${object}/query`, {\n method: 'POST',\n body: JSON.stringify(query)\n });\n return res.json();\n },\n\n find: async <T = any>(object: string, options: QueryOptions = {}): Promise<PaginatedResult<T>> => {\n const route = this.getRoute('data');\n const queryParams = new URLSearchParams();\n \n // 1. Handle Pagination\n if (options.top) queryParams.set('top', options.top.toString());\n if (options.skip) queryParams.set('skip', options.skip.toString());\n\n // 2. Handle Sort\n if (options.sort) {\n // Check if it's AST \n if (Array.isArray(options.sort) && typeof options.sort[0] === 'object') {\n queryParams.set('sort', JSON.stringify(options.sort));\n } else {\n const sortVal = Array.isArray(options.sort) ? options.sort.join(',') : options.sort;\n queryParams.set('sort', sortVal as string);\n }\n }\n \n // 3. Handle Select\n if (options.select) {\n queryParams.set('select', options.select.join(','));\n }\n\n // 4. Handle Filters (Simple vs AST)\n if (options.filters) {\n // If looks like AST (not plain object map)\n // TODO: robust check. safely assuming map for simplified find, and recommending .query() for AST\n if (this.isFilterAST(options.filters)) {\n queryParams.set('filters', JSON.stringify(options.filters));\n } else {\n Object.entries(options.filters).forEach(([k, v]) => {\n if (v !== undefined && v !== null) {\n queryParams.append(k, String(v));\n }\n });\n }\n }\n \n // 5. Handle Aggregations & GroupBy (Pass through as JSON if present)\n if (options.aggregations) {\n queryParams.set('aggregations', JSON.stringify(options.aggregations));\n }\n if (options.groupBy) {\n queryParams.set('groupBy', options.groupBy.join(','));\n }\n\n const res = await this.fetch(`${this.baseUrl}${route}/${object}?${queryParams.toString()}`);\n return res.json();\n },\n\n get: async <T = any>(object: string, id: string): Promise<T> => {\n const route = this.getRoute('data');\n const res = await this.fetch(`${this.baseUrl}${route}/${object}/${id}`);\n return res.json();\n },\n\n create: async <T = any>(object: string, data: Partial<T>): Promise<T> => {\n const route = this.getRoute('data');\n const res = await this.fetch(`${this.baseUrl}${route}/${object}`, {\n method: 'POST',\n body: JSON.stringify(data)\n });\n return res.json();\n },\n\n createMany: async <T = any>(object: string, data: Partial<T>[]): Promise<T[]> => {\n const route = this.getRoute('data');\n const res = await this.fetch(`${this.baseUrl}${route}/${object}/createMany`, {\n method: 'POST',\n body: JSON.stringify(data)\n });\n return res.json();\n },\n\n update: async <T = any>(object: string, id: string, data: Partial<T>): Promise<T> => {\n const route = this.getRoute('data');\n const res = await this.fetch(`${this.baseUrl}${route}/${object}/${id}`, {\n method: 'PATCH',\n body: JSON.stringify(data)\n });\n return res.json();\n },\n\n /**\n * Batch update multiple records\n * Uses the new BatchUpdateRequest schema with full control over options\n */\n batch: async (object: string, request: BatchUpdateRequest): Promise<BatchUpdateResponse> => {\n const route = this.getRoute('data');\n const res = await this.fetch(`${this.baseUrl}${route}/${object}/batch`, {\n method: 'POST',\n body: JSON.stringify(request)\n });\n return res.json();\n },\n\n /**\n * Update multiple records (simplified batch update)\n * Convenience method for batch updates without full BatchUpdateRequest\n */\n updateMany: async <T = any>(\n object: string, \n records: Array<{ id: string; data: Partial<T> }>,\n options?: BatchOptions\n ): Promise<BatchUpdateResponse> => {\n const route = this.getRoute('data');\n const request: UpdateManyRequest = {\n records,\n options\n };\n const res = await this.fetch(`${this.baseUrl}${route}/${object}/updateMany`, {\n method: 'POST',\n body: JSON.stringify(request)\n });\n return res.json();\n },\n\n delete: async (object: string, id: string): Promise<{ success: boolean }> => {\n const route = this.getRoute('data');\n const res = await this.fetch(`${this.baseUrl}${route}/${object}/${id}`, {\n method: 'DELETE'\n });\n return res.json();\n },\n\n /**\n * Delete multiple records by IDs\n */\n deleteMany: async(object: string, ids: string[], options?: BatchOptions): Promise<BatchUpdateResponse> => {\n const route = this.getRoute('data');\n const request: DeleteManyRequest = {\n ids,\n options\n };\n const res = await this.fetch(`${this.baseUrl}${route}/${object}/deleteMany`, {\n method: 'POST',\n body: JSON.stringify(request)\n });\n return res.json();\n }\n };\n\n\n\n /**\n * Private Helpers\n */\n\n private isFilterAST(filter: any): boolean {\n // Basic check: if array, it's [field, op, val] or [logic, node, node]\n // If object but not basic KV map... harder to tell without schema\n // For now, assume if it passes Array.isArray it's an AST root\n return Array.isArray(filter);\n }\n\n private async fetch(url: string, options: RequestInit = {}): Promise<Response> {\n this.logger.debug('HTTP request', { \n method: options.method || 'GET',\n url,\n hasBody: !!options.body\n });\n \n const headers: Record<string, string> = {\n 'Content-Type': 'application/json',\n ...(options.headers as Record<string, string> || {}),\n };\n\n if (this.token) {\n headers['Authorization'] = `Bearer ${this.token}`;\n }\n\n const res = await this.fetchImpl(url, { ...options, headers });\n \n this.logger.debug('HTTP response', { \n method: options.method || 'GET',\n url,\n status: res.status,\n ok: res.ok\n });\n \n if (!res.ok) {\n let errorBody: any;\n try {\n errorBody = await res.json();\n } catch {\n errorBody = { message: res.statusText };\n }\n \n this.logger.error('HTTP request failed', undefined, { \n method: options.method || 'GET',\n url,\n status: res.status,\n error: errorBody\n });\n \n // Create a standardized error if the response includes error details\n const errorMessage = errorBody?.message || errorBody?.error?.message || res.statusText;\n const errorCode = errorBody?.code || errorBody?.error?.code;\n const error = new Error(`[ObjectStack] ${errorCode ? `${errorCode}: ` : ''}${errorMessage}`) as any;\n \n // Attach error details for programmatic access\n error.code = errorCode;\n error.category = errorBody?.category;\n error.httpStatus = res.status;\n error.retryable = errorBody?.retryable;\n error.details = errorBody?.details || errorBody;\n \n throw error;\n }\n \n return res;\n }\n\n /**\n * Get the conventional route path for a given API endpoint type\n * ObjectStack uses standard conventions: /api/v1/data, /api/v1/metadata, /api/v1/ui\n */\n private getRoute(type: 'data' | 'metadata' | 'ui' | 'auth' | 'analytics' | 'hub' | 'storage' | 'automation'): string {\n // 1. Use discovered routes if available\n // Note: Spec uses 'endpoints', mapped dynamically\n if (this.discoveryInfo?.endpoints && (this.discoveryInfo.endpoints as any)[type]) {\n return (this.discoveryInfo.endpoints as any)[type];\n }\n\n // 2. Fallback to conventions\n // Note: HttpDispatcher expects /metadata, not /meta\n const routeMap: Record<string, string> = {\n data: '/api/v1/data',\n metadata: '/api/v1/metadata',\n ui: '/api/v1/ui',\n auth: '/api/v1/auth',\n analytics: '/api/v1/analytics',\n hub: '/api/v1/hub',\n storage: '/api/v1/storage',\n automation: '/api/v1/automation'\n };\n \n return routeMap[type] || `/api/v1/${type}`;\n }\n}\n\n// Re-export type-safe query builder\nexport { QueryBuilder, FilterBuilder, createQuery, createFilter } from './query-builder';\n\n// Re-export commonly used types from @objectstack/spec/api for convenience\nexport type {\n BatchUpdateRequest,\n BatchUpdateResponse,\n UpdateManyRequest,\n DeleteManyRequest,\n BatchOptions,\n BatchRecord,\n BatchOperationResult,\n MetadataCacheRequest,\n MetadataCacheResponse,\n StandardErrorCode,\n ErrorCategory,\n GetDiscoveryResponse,\n GetMetaTypesResponse,\n GetMetaItemsResponse\n} from '@objectstack/spec/api';\n","/**\n * Type-Safe Query Builder\n * \n * Provides a fluent API for building ObjectStack queries with:\n * - Compile-time type checking\n * - Intelligent code completion\n * - Runtime validation\n * - Type-safe filters and selections\n */\n\nimport { QueryAST, FilterCondition, SortNode } from '@objectstack/spec/data';\n\n/**\n * Type-safe filter builder\n */\nexport class FilterBuilder<T = any> {\n private conditions: FilterCondition[] = [];\n\n /**\n * Equality filter: field = value\n */\n equals<K extends keyof T>(field: K, value: T[K]): this {\n this.conditions.push([field as string, '=', value]);\n return this;\n }\n\n /**\n * Not equals filter: field != value\n */\n notEquals<K extends keyof T>(field: K, value: T[K]): this {\n this.conditions.push([field as string, '!=', value]);\n return this;\n }\n\n /**\n * Greater than filter: field > value\n */\n greaterThan<K extends keyof T>(field: K, value: T[K]): this {\n this.conditions.push([field as string, '>', value]);\n return this;\n }\n\n /**\n * Greater than or equal filter: field >= value\n */\n greaterThanOrEqual<K extends keyof T>(field: K, value: T[K]): this {\n this.conditions.push([field as string, '>=', value]);\n return this;\n }\n\n /**\n * Less than filter: field < value\n */\n lessThan<K extends keyof T>(field: K, value: T[K]): this {\n this.conditions.push([field as string, '<', value]);\n return this;\n }\n\n /**\n * Less than or equal filter: field <= value\n */\n lessThanOrEqual<K extends keyof T>(field: K, value: T[K]): this {\n this.conditions.push([field as string, '<=', value]);\n return this;\n }\n\n /**\n * IN filter: field IN (value1, value2, ...)\n */\n in<K extends keyof T>(field: K, values: T[K][]): this {\n this.conditions.push([field as string, 'in', values]);\n return this;\n }\n\n /**\n * NOT IN filter: field NOT IN (value1, value2, ...)\n */\n notIn<K extends keyof T>(field: K, values: T[K][]): this {\n this.conditions.push([field as string, 'not_in', values]);\n return this;\n }\n\n /**\n * LIKE filter: field LIKE pattern\n */\n like<K extends keyof T>(field: K, pattern: string): this {\n this.conditions.push([field as string, 'like', pattern]);\n return this;\n }\n\n /**\n * IS NULL filter: field IS NULL\n */\n isNull<K extends keyof T>(field: K): this {\n this.conditions.push([field as string, 'is_null', null]);\n return this;\n }\n\n /**\n * IS NOT NULL filter: field IS NOT NULL\n */\n isNotNull<K extends keyof T>(field: K): this {\n this.conditions.push([field as string, 'is_not_null', null]);\n return this;\n }\n\n /**\n * Build the filter condition\n */\n build(): FilterCondition {\n if (this.conditions.length === 0) {\n throw new Error('Filter builder has no conditions');\n }\n if (this.conditions.length === 1) {\n return this.conditions[0];\n }\n // Combine multiple conditions with AND\n return ['and', ...this.conditions];\n }\n\n /**\n * Get raw conditions array\n */\n getConditions(): FilterCondition[] {\n return this.conditions;\n }\n}\n\n/**\n * Type-safe query builder\n */\nexport class QueryBuilder<T = any> {\n private query: Partial<QueryAST> = {};\n private _object: string;\n\n constructor(object: string) {\n this._object = object;\n this.query.object = object;\n }\n\n /**\n * Select specific fields\n */\n select<K extends keyof T>(...fields: K[]): this {\n this.query.fields = fields as string[];\n return this;\n }\n\n /**\n * Add filters using a builder function\n */\n where(builderFn: (builder: FilterBuilder<T>) => void): this {\n const builder = new FilterBuilder<T>();\n builderFn(builder);\n const conditions = builder.getConditions();\n \n if (conditions.length === 1) {\n this.query.where = conditions[0];\n } else if (conditions.length > 1) {\n this.query.where = ['and', ...conditions] as FilterCondition;\n }\n \n return this;\n }\n\n /**\n * Add raw filter condition\n */\n filter(condition: FilterCondition): this {\n this.query.where = condition;\n return this;\n }\n\n /**\n * Sort by fields\n */\n orderBy<K extends keyof T>(field: K, order: 'asc' | 'desc' = 'asc'): this {\n if (!this.query.orderBy) {\n this.query.orderBy = [];\n }\n (this.query.orderBy as SortNode[]).push({\n field: field as string,\n order\n });\n return this;\n }\n\n /**\n * Limit the number of results\n */\n limit(count: number): this {\n this.query.limit = count;\n return this;\n }\n\n /**\n * Skip records (for pagination)\n */\n skip(count: number): this {\n this.query.offset = count;\n return this;\n }\n\n /**\n * Paginate results\n */\n paginate(page: number, pageSize: number): this {\n this.query.limit = pageSize;\n this.query.offset = (page - 1) * pageSize;\n return this;\n }\n\n /**\n * Group by fields\n */\n groupBy<K extends keyof T>(...fields: K[]): this {\n this.query.groupBy = fields as string[];\n return this;\n }\n\n /**\n * Build the final query AST\n */\n build(): QueryAST {\n return {\n object: this._object,\n ...this.query\n } as QueryAST;\n }\n\n /**\n * Get the current query state\n */\n getQuery(): Partial<QueryAST> {\n return { ...this.query };\n }\n}\n\n/**\n * Create a type-safe query builder for an object\n */\nexport function createQuery<T = any>(object: string): QueryBuilder<T> {\n return new QueryBuilder<T>(object);\n}\n\n/**\n * Create a type-safe filter builder\n */\nexport function createFilter<T = any>(): FilterBuilder<T> {\n return new FilterBuilder<T>();\n}\n"],"mappings":"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","names":[]}
package/dist/index.mjs CHANGED
@@ -588,8 +588,36 @@ var ObjectStackClient = class {
588
588
  async connect() {
589
589
  this.logger.debug("Connecting to ObjectStack server", { baseUrl: this.baseUrl });
590
590
  try {
591
- const res = await this.fetch(`${this.baseUrl}/api/v1`);
592
- const data = await res.json();
591
+ let data;
592
+ try {
593
+ let wellKnownUrl;
594
+ try {
595
+ const url = new URL(this.baseUrl);
596
+ wellKnownUrl = `${url.origin}/.well-known/objectstack`;
597
+ } catch {
598
+ wellKnownUrl = "/.well-known/objectstack";
599
+ }
600
+ this.logger.debug("Probing .well-known discovery", { url: wellKnownUrl });
601
+ const res = await this.fetchImpl(wellKnownUrl);
602
+ if (res.ok) {
603
+ data = await res.json();
604
+ this.logger.debug("Discovered via .well-known");
605
+ }
606
+ } catch (e) {
607
+ this.logger.debug("Standard discovery probe failed", { error: e.message });
608
+ }
609
+ if (!data) {
610
+ const fallbackUrl = `${this.baseUrl}/api/v1`;
611
+ this.logger.debug("Falling back to legacy discovery", { url: fallbackUrl });
612
+ const res = await this.fetchImpl(fallbackUrl);
613
+ if (!res.ok) {
614
+ throw new Error(`Failed to connect to ${fallbackUrl}: ${res.statusText}`);
615
+ }
616
+ data = await res.json();
617
+ }
618
+ if (!data) {
619
+ throw new Error("Connection failed: No discovery data returned");
620
+ }
593
621
  this.discoveryInfo = data;
594
622
  this.logger.info("Connected to ObjectStack server", {
595
623
  version: data.version,
@@ -1 +1 @@
1
- {"version":3,"sources":["../src/index.ts","../src/query-builder.ts"],"sourcesContent":["import { QueryAST, SortNode, AggregationNode } from '@objectstack/spec/data';\nimport { \n BatchUpdateRequest, \n BatchUpdateResponse, \n UpdateManyRequest,\n DeleteManyRequest,\n BatchOptions,\n MetadataCacheRequest,\n MetadataCacheResponse,\n StandardErrorCode,\n ErrorCategory,\n GetDiscoveryResponse,\n GetMetaTypesResponse,\n GetMetaItemsResponse,\n LoginRequest,\n SessionResponse,\n GetPresignedUrlRequest,\n PresignedUrlResponse,\n CompleteUploadRequest,\n FileUploadResponse\n} from '@objectstack/spec/api';\nimport { Logger, createLogger } from '@objectstack/core';\n\nexport interface ClientConfig {\n baseUrl: string;\n token?: string;\n /**\n * Custom fetch implementation (e.g. node-fetch or for Next.js caching)\n */\n fetch?: (input: RequestInfo | URL, init?: RequestInit) => Promise<Response>;\n /**\n * Logger instance for debugging\n */\n logger?: Logger;\n /**\n * Enable debug logging\n */\n debug?: boolean;\n}\n\n/**\n * Discovery Result\n * Re-export from @objectstack/spec/api for convenience\n */\nexport type DiscoveryResult = GetDiscoveryResponse;\n\nexport interface QueryOptions {\n select?: string[]; // Simplified Selection\n filters?: Record<string, any>; // Map or AST\n sort?: string | string[] | SortNode[]; // 'name' or ['-created_at'] or AST\n top?: number;\n skip?: number;\n // Advanced features\n aggregations?: AggregationNode[];\n groupBy?: string[];\n}\n\nexport interface PaginatedResult<T = any> {\n value: T[];\n count: number;\n}\n\nexport interface StandardError {\n code: StandardErrorCode;\n message: string;\n category: ErrorCategory;\n httpStatus: number;\n retryable: boolean;\n details?: Record<string, any>;\n}\n\nexport class ObjectStackClient {\n private baseUrl: string;\n private token?: string;\n private fetchImpl: (input: RequestInfo | URL, init?: RequestInit) => Promise<Response>;\n private discoveryInfo?: DiscoveryResult;\n private logger: Logger;\n\n constructor(config: ClientConfig) {\n this.baseUrl = config.baseUrl.replace(/\\/$/, ''); // Remove trailing slash\n this.token = config.token;\n this.fetchImpl = config.fetch || globalThis.fetch.bind(globalThis);\n \n // Initialize logger\n this.logger = config.logger || createLogger({ \n level: config.debug ? 'debug' : 'info',\n format: 'pretty'\n });\n \n this.logger.debug('ObjectStack client created', { baseUrl: this.baseUrl });\n }\n\n /**\n * Initialize the client by discovering server capabilities.\n */\n async connect() {\n this.logger.debug('Connecting to ObjectStack server', { baseUrl: this.baseUrl });\n \n try {\n // Connect to the discovery endpoint at /api/v1\n const res = await this.fetch(`${this.baseUrl}/api/v1`);\n \n const data = await res.json();\n this.discoveryInfo = data;\n \n this.logger.info('Connected to ObjectStack server', { \n version: data.version,\n apiName: data.apiName,\n capabilities: data.capabilities \n });\n \n return data as DiscoveryResult;\n } catch (e) {\n this.logger.error('Failed to connect to ObjectStack server', e as Error, { baseUrl: this.baseUrl });\n throw e;\n }\n }\n\n /**\n * Metadata Operations\n */\n meta = {\n /**\n * Get all available metadata types\n * Returns types like 'object', 'plugin', 'view', etc.\n */\n getTypes: async (): Promise<GetMetaTypesResponse> => {\n const route = this.getRoute('metadata');\n const res = await this.fetch(`${this.baseUrl}${route}`);\n return res.json();\n },\n\n /**\n * Get all items of a specific metadata type\n * @param type - Metadata type name (e.g., 'object', 'plugin')\n */\n getItems: async (type: string): Promise<GetMetaItemsResponse> => {\n const route = this.getRoute('metadata');\n const res = await this.fetch(`${this.baseUrl}${route}/${type}`);\n return res.json();\n },\n\n /**\n * Get a specific object definition by name\n * @deprecated Use `getItem('object', name)` instead for consistency with spec protocol\n * @param name - Object name (snake_case identifier)\n */\n getObject: async (name: string) => {\n const route = this.getRoute('metadata');\n const res = await this.fetch(`${this.baseUrl}${route}/object/${name}`);\n return res.json();\n },\n\n /**\n * Get a specific metadata item by type and name\n * @param type - Metadata type (e.g., 'object', 'plugin')\n * @param name - Item name (snake_case identifier)\n */\n getItem: async (type: string, name: string) => {\n const route = this.getRoute('metadata');\n const res = await this.fetch(`${this.baseUrl}${route}/${type}/${name}`);\n return res.json();\n },\n\n /**\n * Save a metadata item\n * @param type - Metadata type (e.g., 'object', 'plugin')\n * @param name - Item name\n * @param item - The metadata content to save\n */\n saveItem: async (type: string, name: string, item: any) => {\n const route = this.getRoute('metadata');\n const res = await this.fetch(`${this.baseUrl}${route}/${type}/${name}`, {\n method: 'PUT',\n body: JSON.stringify(item)\n });\n return res.json();\n },\n \n /**\n * Get object metadata with cache support\n * Supports ETag-based conditional requests for efficient caching\n */\n getCached: async (name: string, cacheOptions?: MetadataCacheRequest): Promise<MetadataCacheResponse> => {\n const route = this.getRoute('metadata');\n const headers: Record<string, string> = {};\n \n if (cacheOptions?.ifNoneMatch) {\n headers['If-None-Match'] = cacheOptions.ifNoneMatch;\n }\n if (cacheOptions?.ifModifiedSince) {\n headers['If-Modified-Since'] = cacheOptions.ifModifiedSince;\n }\n \n const res = await this.fetch(`${this.baseUrl}${route}/object/${name}`, {\n headers\n });\n \n // Check for 304 Not Modified\n if (res.status === 304) {\n return {\n notModified: true,\n etag: cacheOptions?.ifNoneMatch ? { \n value: cacheOptions.ifNoneMatch.replace(/^W\\/|\"/g, ''),\n weak: cacheOptions.ifNoneMatch.startsWith('W/')\n } : undefined\n };\n }\n \n const data = await res.json();\n const etag = res.headers.get('ETag');\n const lastModified = res.headers.get('Last-Modified');\n \n return {\n data,\n etag: etag ? { \n value: etag.replace(/^W\\/|\"/g, ''), \n weak: etag.startsWith('W/') \n } : undefined,\n lastModified: lastModified || undefined,\n notModified: false\n };\n },\n \n getView: async (object: string, type: 'list' | 'form' = 'list') => {\n const route = this.getRoute('ui');\n const res = await this.fetch(`${this.baseUrl}${route}/view/${object}?type=${type}`);\n return res.json();\n }\n };\n\n /**\n * Analytics Services\n */\n analytics = {\n query: async (payload: any) => {\n const route = this.getRoute('analytics');\n const res = await this.fetch(`${this.baseUrl}${route}/query`, {\n method: 'POST',\n body: JSON.stringify(payload)\n });\n return res.json();\n },\n meta: async (cube: string) => {\n const route = this.getRoute('analytics');\n const res = await this.fetch(`${this.baseUrl}${route}/meta/${cube}`);\n return res.json();\n },\n explain: async (payload: any) => {\n const route = this.getRoute('analytics');\n const res = await this.fetch(`${this.baseUrl}${route}/explain`, {\n method: 'POST',\n body: JSON.stringify(payload)\n });\n return res.json();\n }\n };\n\n /**\n * Hub Management Services\n */\n hub = {\n spaces: {\n list: async () => {\n const route = this.getRoute('hub');\n const res = await this.fetch(`${this.baseUrl}${route}/spaces`);\n return res.json();\n },\n create: async (payload: any) => {\n const route = this.getRoute('hub');\n const res = await this.fetch(`${this.baseUrl}${route}/spaces`, {\n method: 'POST',\n body: JSON.stringify(payload)\n });\n return res.json();\n }\n },\n plugins: {\n install: async (pkg: string, version?: string) => {\n const route = this.getRoute('hub');\n const res = await this.fetch(`${this.baseUrl}${route}/plugins/install`, {\n method: 'POST',\n body: JSON.stringify({ pkg, version })\n });\n return res.json();\n }\n }\n };\n\n /**\n * Authentication Services\n */\n auth = {\n login: async (request: LoginRequest): Promise<SessionResponse> => {\n const route = this.getRoute('auth');\n const res = await this.fetch(`${this.baseUrl}${route}/login`, {\n method: 'POST',\n body: JSON.stringify(request)\n });\n const data = await res.json();\n // Auto-set token if present in response\n if (data.data?.token) {\n this.token = data.data.token;\n }\n return data;\n },\n \n logout: async () => {\n const route = this.getRoute('auth');\n await this.fetch(`${this.baseUrl}${route}/logout`, { method: 'POST' });\n this.token = undefined;\n },\n\n me: async (): Promise<SessionResponse> => {\n const route = this.getRoute('auth');\n const res = await this.fetch(`${this.baseUrl}${route}/me`);\n return res.json();\n }\n };\n\n /**\n * Storage Services\n */\n storage = {\n upload: async (file: any, scope: string = 'user'): Promise<FileUploadResponse> => {\n // 1. Get Presigned URL\n const presignedReq: GetPresignedUrlRequest = {\n filename: file.name,\n mimeType: file.type,\n size: file.size,\n scope\n };\n \n const route = this.getRoute('storage');\n const presignedRes = await this.fetch(`${this.baseUrl}${route}/upload/presigned`, {\n method: 'POST',\n body: JSON.stringify(presignedReq)\n });\n const { data: presigned } = await presignedRes.json() as { data: PresignedUrlResponse['data'] };\n\n // 2. Upload to Cloud directly (Bypass API Middleware to avoid Auth headers if using S3)\n // Use fetchImpl directly\n const uploadRes = await this.fetchImpl(presigned.uploadUrl, {\n method: presigned.method,\n headers: presigned.headers,\n body: file\n });\n\n if (!uploadRes.ok) {\n throw new Error(`Storage Upload Failed: ${uploadRes.statusText}`);\n }\n\n // 3. Complete Upload\n const completeReq: CompleteUploadRequest = {\n fileId: presigned.fileId\n };\n const completeRes = await this.fetch(`${this.baseUrl}${route}/upload/complete`, {\n method: 'POST',\n body: JSON.stringify(completeReq)\n });\n \n return completeRes.json();\n },\n \n getDownloadUrl: async (fileId: string): Promise<string> => {\n const route = this.getRoute('storage');\n const res = await this.fetch(`${this.baseUrl}${route}/files/${fileId}/url`);\n const data = await res.json();\n return data.url;\n }\n };\n\n /**\n * Automation Services\n */\n automation = {\n trigger: async (triggerName: string, payload: any) => {\n const route = this.getRoute('automation');\n const res = await this.fetch(`${this.baseUrl}${route}/trigger/${triggerName}`, {\n method: 'POST',\n body: JSON.stringify(payload)\n });\n return res.json();\n }\n };\n\n /**\n * Data Operations\n */\n data = {\n /**\n * Advanced Query using ObjectStack Query Protocol\n * Supports both simplified options and full AST\n */\n query: async <T = any>(object: string, query: Partial<QueryAST>): Promise<PaginatedResult<T>> => {\n const route = this.getRoute('data');\n // POST for complex query to avoid URL length limits and allow clean JSON AST\n // Convention: POST /api/v1/data/:object/query\n const res = await this.fetch(`${this.baseUrl}${route}/${object}/query`, {\n method: 'POST',\n body: JSON.stringify(query)\n });\n return res.json();\n },\n\n find: async <T = any>(object: string, options: QueryOptions = {}): Promise<PaginatedResult<T>> => {\n const route = this.getRoute('data');\n const queryParams = new URLSearchParams();\n \n // 1. Handle Pagination\n if (options.top) queryParams.set('top', options.top.toString());\n if (options.skip) queryParams.set('skip', options.skip.toString());\n\n // 2. Handle Sort\n if (options.sort) {\n // Check if it's AST \n if (Array.isArray(options.sort) && typeof options.sort[0] === 'object') {\n queryParams.set('sort', JSON.stringify(options.sort));\n } else {\n const sortVal = Array.isArray(options.sort) ? options.sort.join(',') : options.sort;\n queryParams.set('sort', sortVal as string);\n }\n }\n \n // 3. Handle Select\n if (options.select) {\n queryParams.set('select', options.select.join(','));\n }\n\n // 4. Handle Filters (Simple vs AST)\n if (options.filters) {\n // If looks like AST (not plain object map)\n // TODO: robust check. safely assuming map for simplified find, and recommending .query() for AST\n if (this.isFilterAST(options.filters)) {\n queryParams.set('filters', JSON.stringify(options.filters));\n } else {\n Object.entries(options.filters).forEach(([k, v]) => {\n if (v !== undefined && v !== null) {\n queryParams.append(k, String(v));\n }\n });\n }\n }\n \n // 5. Handle Aggregations & GroupBy (Pass through as JSON if present)\n if (options.aggregations) {\n queryParams.set('aggregations', JSON.stringify(options.aggregations));\n }\n if (options.groupBy) {\n queryParams.set('groupBy', options.groupBy.join(','));\n }\n\n const res = await this.fetch(`${this.baseUrl}${route}/${object}?${queryParams.toString()}`);\n return res.json();\n },\n\n get: async <T = any>(object: string, id: string): Promise<T> => {\n const route = this.getRoute('data');\n const res = await this.fetch(`${this.baseUrl}${route}/${object}/${id}`);\n return res.json();\n },\n\n create: async <T = any>(object: string, data: Partial<T>): Promise<T> => {\n const route = this.getRoute('data');\n const res = await this.fetch(`${this.baseUrl}${route}/${object}`, {\n method: 'POST',\n body: JSON.stringify(data)\n });\n return res.json();\n },\n\n createMany: async <T = any>(object: string, data: Partial<T>[]): Promise<T[]> => {\n const route = this.getRoute('data');\n const res = await this.fetch(`${this.baseUrl}${route}/${object}/createMany`, {\n method: 'POST',\n body: JSON.stringify(data)\n });\n return res.json();\n },\n\n update: async <T = any>(object: string, id: string, data: Partial<T>): Promise<T> => {\n const route = this.getRoute('data');\n const res = await this.fetch(`${this.baseUrl}${route}/${object}/${id}`, {\n method: 'PATCH',\n body: JSON.stringify(data)\n });\n return res.json();\n },\n\n /**\n * Batch update multiple records\n * Uses the new BatchUpdateRequest schema with full control over options\n */\n batch: async (object: string, request: BatchUpdateRequest): Promise<BatchUpdateResponse> => {\n const route = this.getRoute('data');\n const res = await this.fetch(`${this.baseUrl}${route}/${object}/batch`, {\n method: 'POST',\n body: JSON.stringify(request)\n });\n return res.json();\n },\n\n /**\n * Update multiple records (simplified batch update)\n * Convenience method for batch updates without full BatchUpdateRequest\n */\n updateMany: async <T = any>(\n object: string, \n records: Array<{ id: string; data: Partial<T> }>,\n options?: BatchOptions\n ): Promise<BatchUpdateResponse> => {\n const route = this.getRoute('data');\n const request: UpdateManyRequest = {\n records,\n options\n };\n const res = await this.fetch(`${this.baseUrl}${route}/${object}/updateMany`, {\n method: 'POST',\n body: JSON.stringify(request)\n });\n return res.json();\n },\n\n delete: async (object: string, id: string): Promise<{ success: boolean }> => {\n const route = this.getRoute('data');\n const res = await this.fetch(`${this.baseUrl}${route}/${object}/${id}`, {\n method: 'DELETE'\n });\n return res.json();\n },\n\n /**\n * Delete multiple records by IDs\n */\n deleteMany: async(object: string, ids: string[], options?: BatchOptions): Promise<BatchUpdateResponse> => {\n const route = this.getRoute('data');\n const request: DeleteManyRequest = {\n ids,\n options\n };\n const res = await this.fetch(`${this.baseUrl}${route}/${object}/deleteMany`, {\n method: 'POST',\n body: JSON.stringify(request)\n });\n return res.json();\n }\n };\n\n\n\n /**\n * Private Helpers\n */\n\n private isFilterAST(filter: any): boolean {\n // Basic check: if array, it's [field, op, val] or [logic, node, node]\n // If object but not basic KV map... harder to tell without schema\n // For now, assume if it passes Array.isArray it's an AST root\n return Array.isArray(filter);\n }\n\n private async fetch(url: string, options: RequestInit = {}): Promise<Response> {\n this.logger.debug('HTTP request', { \n method: options.method || 'GET',\n url,\n hasBody: !!options.body\n });\n \n const headers: Record<string, string> = {\n 'Content-Type': 'application/json',\n ...(options.headers as Record<string, string> || {}),\n };\n\n if (this.token) {\n headers['Authorization'] = `Bearer ${this.token}`;\n }\n\n const res = await this.fetchImpl(url, { ...options, headers });\n \n this.logger.debug('HTTP response', { \n method: options.method || 'GET',\n url,\n status: res.status,\n ok: res.ok\n });\n \n if (!res.ok) {\n let errorBody: any;\n try {\n errorBody = await res.json();\n } catch {\n errorBody = { message: res.statusText };\n }\n \n this.logger.error('HTTP request failed', undefined, { \n method: options.method || 'GET',\n url,\n status: res.status,\n error: errorBody\n });\n \n // Create a standardized error if the response includes error details\n const errorMessage = errorBody?.message || errorBody?.error?.message || res.statusText;\n const errorCode = errorBody?.code || errorBody?.error?.code;\n const error = new Error(`[ObjectStack] ${errorCode ? `${errorCode}: ` : ''}${errorMessage}`) as any;\n \n // Attach error details for programmatic access\n error.code = errorCode;\n error.category = errorBody?.category;\n error.httpStatus = res.status;\n error.retryable = errorBody?.retryable;\n error.details = errorBody?.details || errorBody;\n \n throw error;\n }\n \n return res;\n }\n\n /**\n * Get the conventional route path for a given API endpoint type\n * ObjectStack uses standard conventions: /api/v1/data, /api/v1/metadata, /api/v1/ui\n */\n private getRoute(type: 'data' | 'metadata' | 'ui' | 'auth' | 'analytics' | 'hub' | 'storage' | 'automation'): string {\n // 1. Use discovered routes if available\n // Note: Spec uses 'endpoints', mapped dynamically\n if (this.discoveryInfo?.endpoints && (this.discoveryInfo.endpoints as any)[type]) {\n return (this.discoveryInfo.endpoints as any)[type];\n }\n\n // 2. Fallback to conventions\n // Note: HttpDispatcher expects /metadata, not /meta\n const routeMap: Record<string, string> = {\n data: '/api/v1/data',\n metadata: '/api/v1/metadata',\n ui: '/api/v1/ui',\n auth: '/api/v1/auth',\n analytics: '/api/v1/analytics',\n hub: '/api/v1/hub',\n storage: '/api/v1/storage',\n automation: '/api/v1/automation'\n };\n \n return routeMap[type] || `/api/v1/${type}`;\n }\n}\n\n// Re-export type-safe query builder\nexport { QueryBuilder, FilterBuilder, createQuery, createFilter } from './query-builder';\n\n// Re-export commonly used types from @objectstack/spec/api for convenience\nexport type {\n BatchUpdateRequest,\n BatchUpdateResponse,\n UpdateManyRequest,\n DeleteManyRequest,\n BatchOptions,\n BatchRecord,\n BatchOperationResult,\n MetadataCacheRequest,\n MetadataCacheResponse,\n StandardErrorCode,\n ErrorCategory,\n GetDiscoveryResponse,\n GetMetaTypesResponse,\n GetMetaItemsResponse\n} from '@objectstack/spec/api';\n","/**\n * Type-Safe Query Builder\n * \n * Provides a fluent API for building ObjectStack queries with:\n * - Compile-time type checking\n * - Intelligent code completion\n * - Runtime validation\n * - Type-safe filters and selections\n */\n\nimport { QueryAST, FilterCondition, SortNode } from '@objectstack/spec/data';\n\n/**\n * Type-safe filter builder\n */\nexport class FilterBuilder<T = any> {\n private conditions: FilterCondition[] = [];\n\n /**\n * Equality filter: field = value\n */\n equals<K extends keyof T>(field: K, value: T[K]): this {\n this.conditions.push([field as string, '=', value]);\n return this;\n }\n\n /**\n * Not equals filter: field != value\n */\n notEquals<K extends keyof T>(field: K, value: T[K]): this {\n this.conditions.push([field as string, '!=', value]);\n return this;\n }\n\n /**\n * Greater than filter: field > value\n */\n greaterThan<K extends keyof T>(field: K, value: T[K]): this {\n this.conditions.push([field as string, '>', value]);\n return this;\n }\n\n /**\n * Greater than or equal filter: field >= value\n */\n greaterThanOrEqual<K extends keyof T>(field: K, value: T[K]): this {\n this.conditions.push([field as string, '>=', value]);\n return this;\n }\n\n /**\n * Less than filter: field < value\n */\n lessThan<K extends keyof T>(field: K, value: T[K]): this {\n this.conditions.push([field as string, '<', value]);\n return this;\n }\n\n /**\n * Less than or equal filter: field <= value\n */\n lessThanOrEqual<K extends keyof T>(field: K, value: T[K]): this {\n this.conditions.push([field as string, '<=', value]);\n return this;\n }\n\n /**\n * IN filter: field IN (value1, value2, ...)\n */\n in<K extends keyof T>(field: K, values: T[K][]): this {\n this.conditions.push([field as string, 'in', values]);\n return this;\n }\n\n /**\n * NOT IN filter: field NOT IN (value1, value2, ...)\n */\n notIn<K extends keyof T>(field: K, values: T[K][]): this {\n this.conditions.push([field as string, 'not_in', values]);\n return this;\n }\n\n /**\n * LIKE filter: field LIKE pattern\n */\n like<K extends keyof T>(field: K, pattern: string): this {\n this.conditions.push([field as string, 'like', pattern]);\n return this;\n }\n\n /**\n * IS NULL filter: field IS NULL\n */\n isNull<K extends keyof T>(field: K): this {\n this.conditions.push([field as string, 'is_null', null]);\n return this;\n }\n\n /**\n * IS NOT NULL filter: field IS NOT NULL\n */\n isNotNull<K extends keyof T>(field: K): this {\n this.conditions.push([field as string, 'is_not_null', null]);\n return this;\n }\n\n /**\n * Build the filter condition\n */\n build(): FilterCondition {\n if (this.conditions.length === 0) {\n throw new Error('Filter builder has no conditions');\n }\n if (this.conditions.length === 1) {\n return this.conditions[0];\n }\n // Combine multiple conditions with AND\n return ['and', ...this.conditions];\n }\n\n /**\n * Get raw conditions array\n */\n getConditions(): FilterCondition[] {\n return this.conditions;\n }\n}\n\n/**\n * Type-safe query builder\n */\nexport class QueryBuilder<T = any> {\n private query: Partial<QueryAST> = {};\n private _object: string;\n\n constructor(object: string) {\n this._object = object;\n this.query.object = object;\n }\n\n /**\n * Select specific fields\n */\n select<K extends keyof T>(...fields: K[]): this {\n this.query.fields = fields as string[];\n return this;\n }\n\n /**\n * Add filters using a builder function\n */\n where(builderFn: (builder: FilterBuilder<T>) => void): this {\n const builder = new FilterBuilder<T>();\n builderFn(builder);\n const conditions = builder.getConditions();\n \n if (conditions.length === 1) {\n this.query.where = conditions[0];\n } else if (conditions.length > 1) {\n this.query.where = ['and', ...conditions] as FilterCondition;\n }\n \n return this;\n }\n\n /**\n * Add raw filter condition\n */\n filter(condition: FilterCondition): this {\n this.query.where = condition;\n return this;\n }\n\n /**\n * Sort by fields\n */\n orderBy<K extends keyof T>(field: K, order: 'asc' | 'desc' = 'asc'): this {\n if (!this.query.orderBy) {\n this.query.orderBy = [];\n }\n (this.query.orderBy as SortNode[]).push({\n field: field as string,\n order\n });\n return this;\n }\n\n /**\n * Limit the number of results\n */\n limit(count: number): this {\n this.query.limit = count;\n return this;\n }\n\n /**\n * Skip records (for pagination)\n */\n skip(count: number): this {\n this.query.offset = count;\n return this;\n }\n\n /**\n * Paginate results\n */\n paginate(page: number, pageSize: number): this {\n this.query.limit = pageSize;\n this.query.offset = (page - 1) * pageSize;\n return this;\n }\n\n /**\n * Group by fields\n */\n groupBy<K extends keyof T>(...fields: K[]): this {\n this.query.groupBy = fields as string[];\n return this;\n }\n\n /**\n * Build the final query AST\n */\n build(): QueryAST {\n return {\n object: this._object,\n ...this.query\n } as QueryAST;\n }\n\n /**\n * Get the current query state\n */\n getQuery(): Partial<QueryAST> {\n return { ...this.query };\n }\n}\n\n/**\n * Create a type-safe query builder for an object\n */\nexport function createQuery<T = any>(object: string): QueryBuilder<T> {\n return new QueryBuilder<T>(object);\n}\n\n/**\n * Create a type-safe filter builder\n */\nexport function createFilter<T = any>(): FilterBuilder<T> {\n return new FilterBuilder<T>();\n}\n"],"mappings":"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","names":[]}
1
+ {"version":3,"sources":["../src/index.ts","../src/query-builder.ts"],"sourcesContent":["import { QueryAST, SortNode, AggregationNode } from '@objectstack/spec/data';\nimport { \n BatchUpdateRequest, \n BatchUpdateResponse, \n UpdateManyRequest,\n DeleteManyRequest,\n BatchOptions,\n MetadataCacheRequest,\n MetadataCacheResponse,\n StandardErrorCode,\n ErrorCategory,\n GetDiscoveryResponse,\n GetMetaTypesResponse,\n GetMetaItemsResponse,\n LoginRequest,\n SessionResponse,\n GetPresignedUrlRequest,\n PresignedUrlResponse,\n CompleteUploadRequest,\n FileUploadResponse\n} from '@objectstack/spec/api';\nimport { Logger, createLogger } from '@objectstack/core';\n\nexport interface ClientConfig {\n baseUrl: string;\n token?: string;\n /**\n * Custom fetch implementation (e.g. node-fetch or for Next.js caching)\n */\n fetch?: (input: RequestInfo | URL, init?: RequestInit) => Promise<Response>;\n /**\n * Logger instance for debugging\n */\n logger?: Logger;\n /**\n * Enable debug logging\n */\n debug?: boolean;\n}\n\n/**\n * Discovery Result\n * Re-export from @objectstack/spec/api for convenience\n */\nexport type DiscoveryResult = GetDiscoveryResponse;\n\nexport interface QueryOptions {\n select?: string[]; // Simplified Selection\n filters?: Record<string, any>; // Map or AST\n sort?: string | string[] | SortNode[]; // 'name' or ['-created_at'] or AST\n top?: number;\n skip?: number;\n // Advanced features\n aggregations?: AggregationNode[];\n groupBy?: string[];\n}\n\nexport interface PaginatedResult<T = any> {\n value: T[];\n count: number;\n}\n\nexport interface StandardError {\n code: StandardErrorCode;\n message: string;\n category: ErrorCategory;\n httpStatus: number;\n retryable: boolean;\n details?: Record<string, any>;\n}\n\nexport class ObjectStackClient {\n private baseUrl: string;\n private token?: string;\n private fetchImpl: (input: RequestInfo | URL, init?: RequestInit) => Promise<Response>;\n private discoveryInfo?: DiscoveryResult;\n private logger: Logger;\n\n constructor(config: ClientConfig) {\n this.baseUrl = config.baseUrl.replace(/\\/$/, ''); // Remove trailing slash\n this.token = config.token;\n this.fetchImpl = config.fetch || globalThis.fetch.bind(globalThis);\n \n // Initialize logger\n this.logger = config.logger || createLogger({ \n level: config.debug ? 'debug' : 'info',\n format: 'pretty'\n });\n \n this.logger.debug('ObjectStack client created', { baseUrl: this.baseUrl });\n }\n\n /**\n * Initialize the client by discovering server capabilities.\n */\n async connect() {\n this.logger.debug('Connecting to ObjectStack server', { baseUrl: this.baseUrl });\n \n try {\n let data: DiscoveryResult | undefined;\n\n // 1. Try Standard Discovery (.well-known)\n try {\n let wellKnownUrl: string;\n try {\n // If baseUrl is absolute, get origin\n const url = new URL(this.baseUrl);\n wellKnownUrl = `${url.origin}/.well-known/objectstack`;\n } catch {\n // If baseUrl is relative, use absolute path from root\n wellKnownUrl = '/.well-known/objectstack';\n }\n\n this.logger.debug('Probing .well-known discovery', { url: wellKnownUrl });\n const res = await this.fetchImpl(wellKnownUrl);\n if (res.ok) {\n data = await res.json();\n this.logger.debug('Discovered via .well-known');\n }\n } catch (e) {\n this.logger.debug('Standard discovery probe failed', { error: (e as Error).message });\n }\n\n // 2. Fallback to Legacy/Direct Path /api/v1\n if (!data) {\n const fallbackUrl = `${this.baseUrl}/api/v1`;\n this.logger.debug('Falling back to legacy discovery', { url: fallbackUrl });\n const res = await this.fetchImpl(fallbackUrl);\n if (!res.ok) {\n throw new Error(`Failed to connect to ${fallbackUrl}: ${res.statusText}`);\n }\n data = await res.json();\n }\n\n if (!data) {\n throw new Error('Connection failed: No discovery data returned');\n }\n\n this.discoveryInfo = data;\n \n this.logger.info('Connected to ObjectStack server', { \n version: data.version,\n apiName: data.apiName,\n capabilities: data.capabilities \n });\n \n return data as DiscoveryResult;\n } catch (e) {\n this.logger.error('Failed to connect to ObjectStack server', e as Error, { baseUrl: this.baseUrl });\n throw e;\n }\n }\n\n /**\n * Metadata Operations\n */\n meta = {\n /**\n * Get all available metadata types\n * Returns types like 'object', 'plugin', 'view', etc.\n */\n getTypes: async (): Promise<GetMetaTypesResponse> => {\n const route = this.getRoute('metadata');\n const res = await this.fetch(`${this.baseUrl}${route}`);\n return res.json();\n },\n\n /**\n * Get all items of a specific metadata type\n * @param type - Metadata type name (e.g., 'object', 'plugin')\n */\n getItems: async (type: string): Promise<GetMetaItemsResponse> => {\n const route = this.getRoute('metadata');\n const res = await this.fetch(`${this.baseUrl}${route}/${type}`);\n return res.json();\n },\n\n /**\n * Get a specific object definition by name\n * @deprecated Use `getItem('object', name)` instead for consistency with spec protocol\n * @param name - Object name (snake_case identifier)\n */\n getObject: async (name: string) => {\n const route = this.getRoute('metadata');\n const res = await this.fetch(`${this.baseUrl}${route}/object/${name}`);\n return res.json();\n },\n\n /**\n * Get a specific metadata item by type and name\n * @param type - Metadata type (e.g., 'object', 'plugin')\n * @param name - Item name (snake_case identifier)\n */\n getItem: async (type: string, name: string) => {\n const route = this.getRoute('metadata');\n const res = await this.fetch(`${this.baseUrl}${route}/${type}/${name}`);\n return res.json();\n },\n\n /**\n * Save a metadata item\n * @param type - Metadata type (e.g., 'object', 'plugin')\n * @param name - Item name\n * @param item - The metadata content to save\n */\n saveItem: async (type: string, name: string, item: any) => {\n const route = this.getRoute('metadata');\n const res = await this.fetch(`${this.baseUrl}${route}/${type}/${name}`, {\n method: 'PUT',\n body: JSON.stringify(item)\n });\n return res.json();\n },\n \n /**\n * Get object metadata with cache support\n * Supports ETag-based conditional requests for efficient caching\n */\n getCached: async (name: string, cacheOptions?: MetadataCacheRequest): Promise<MetadataCacheResponse> => {\n const route = this.getRoute('metadata');\n const headers: Record<string, string> = {};\n \n if (cacheOptions?.ifNoneMatch) {\n headers['If-None-Match'] = cacheOptions.ifNoneMatch;\n }\n if (cacheOptions?.ifModifiedSince) {\n headers['If-Modified-Since'] = cacheOptions.ifModifiedSince;\n }\n \n const res = await this.fetch(`${this.baseUrl}${route}/object/${name}`, {\n headers\n });\n \n // Check for 304 Not Modified\n if (res.status === 304) {\n return {\n notModified: true,\n etag: cacheOptions?.ifNoneMatch ? { \n value: cacheOptions.ifNoneMatch.replace(/^W\\/|\"/g, ''),\n weak: cacheOptions.ifNoneMatch.startsWith('W/')\n } : undefined\n };\n }\n \n const data = await res.json();\n const etag = res.headers.get('ETag');\n const lastModified = res.headers.get('Last-Modified');\n \n return {\n data,\n etag: etag ? { \n value: etag.replace(/^W\\/|\"/g, ''), \n weak: etag.startsWith('W/') \n } : undefined,\n lastModified: lastModified || undefined,\n notModified: false\n };\n },\n \n getView: async (object: string, type: 'list' | 'form' = 'list') => {\n const route = this.getRoute('ui');\n const res = await this.fetch(`${this.baseUrl}${route}/view/${object}?type=${type}`);\n return res.json();\n }\n };\n\n /**\n * Analytics Services\n */\n analytics = {\n query: async (payload: any) => {\n const route = this.getRoute('analytics');\n const res = await this.fetch(`${this.baseUrl}${route}/query`, {\n method: 'POST',\n body: JSON.stringify(payload)\n });\n return res.json();\n },\n meta: async (cube: string) => {\n const route = this.getRoute('analytics');\n const res = await this.fetch(`${this.baseUrl}${route}/meta/${cube}`);\n return res.json();\n },\n explain: async (payload: any) => {\n const route = this.getRoute('analytics');\n const res = await this.fetch(`${this.baseUrl}${route}/explain`, {\n method: 'POST',\n body: JSON.stringify(payload)\n });\n return res.json();\n }\n };\n\n /**\n * Hub Management Services\n */\n hub = {\n spaces: {\n list: async () => {\n const route = this.getRoute('hub');\n const res = await this.fetch(`${this.baseUrl}${route}/spaces`);\n return res.json();\n },\n create: async (payload: any) => {\n const route = this.getRoute('hub');\n const res = await this.fetch(`${this.baseUrl}${route}/spaces`, {\n method: 'POST',\n body: JSON.stringify(payload)\n });\n return res.json();\n }\n },\n plugins: {\n install: async (pkg: string, version?: string) => {\n const route = this.getRoute('hub');\n const res = await this.fetch(`${this.baseUrl}${route}/plugins/install`, {\n method: 'POST',\n body: JSON.stringify({ pkg, version })\n });\n return res.json();\n }\n }\n };\n\n /**\n * Authentication Services\n */\n auth = {\n login: async (request: LoginRequest): Promise<SessionResponse> => {\n const route = this.getRoute('auth');\n const res = await this.fetch(`${this.baseUrl}${route}/login`, {\n method: 'POST',\n body: JSON.stringify(request)\n });\n const data = await res.json();\n // Auto-set token if present in response\n if (data.data?.token) {\n this.token = data.data.token;\n }\n return data;\n },\n \n logout: async () => {\n const route = this.getRoute('auth');\n await this.fetch(`${this.baseUrl}${route}/logout`, { method: 'POST' });\n this.token = undefined;\n },\n\n me: async (): Promise<SessionResponse> => {\n const route = this.getRoute('auth');\n const res = await this.fetch(`${this.baseUrl}${route}/me`);\n return res.json();\n }\n };\n\n /**\n * Storage Services\n */\n storage = {\n upload: async (file: any, scope: string = 'user'): Promise<FileUploadResponse> => {\n // 1. Get Presigned URL\n const presignedReq: GetPresignedUrlRequest = {\n filename: file.name,\n mimeType: file.type,\n size: file.size,\n scope\n };\n \n const route = this.getRoute('storage');\n const presignedRes = await this.fetch(`${this.baseUrl}${route}/upload/presigned`, {\n method: 'POST',\n body: JSON.stringify(presignedReq)\n });\n const { data: presigned } = await presignedRes.json() as { data: PresignedUrlResponse['data'] };\n\n // 2. Upload to Cloud directly (Bypass API Middleware to avoid Auth headers if using S3)\n // Use fetchImpl directly\n const uploadRes = await this.fetchImpl(presigned.uploadUrl, {\n method: presigned.method,\n headers: presigned.headers,\n body: file\n });\n\n if (!uploadRes.ok) {\n throw new Error(`Storage Upload Failed: ${uploadRes.statusText}`);\n }\n\n // 3. Complete Upload\n const completeReq: CompleteUploadRequest = {\n fileId: presigned.fileId\n };\n const completeRes = await this.fetch(`${this.baseUrl}${route}/upload/complete`, {\n method: 'POST',\n body: JSON.stringify(completeReq)\n });\n \n return completeRes.json();\n },\n \n getDownloadUrl: async (fileId: string): Promise<string> => {\n const route = this.getRoute('storage');\n const res = await this.fetch(`${this.baseUrl}${route}/files/${fileId}/url`);\n const data = await res.json();\n return data.url;\n }\n };\n\n /**\n * Automation Services\n */\n automation = {\n trigger: async (triggerName: string, payload: any) => {\n const route = this.getRoute('automation');\n const res = await this.fetch(`${this.baseUrl}${route}/trigger/${triggerName}`, {\n method: 'POST',\n body: JSON.stringify(payload)\n });\n return res.json();\n }\n };\n\n /**\n * Data Operations\n */\n data = {\n /**\n * Advanced Query using ObjectStack Query Protocol\n * Supports both simplified options and full AST\n */\n query: async <T = any>(object: string, query: Partial<QueryAST>): Promise<PaginatedResult<T>> => {\n const route = this.getRoute('data');\n // POST for complex query to avoid URL length limits and allow clean JSON AST\n // Convention: POST /api/v1/data/:object/query\n const res = await this.fetch(`${this.baseUrl}${route}/${object}/query`, {\n method: 'POST',\n body: JSON.stringify(query)\n });\n return res.json();\n },\n\n find: async <T = any>(object: string, options: QueryOptions = {}): Promise<PaginatedResult<T>> => {\n const route = this.getRoute('data');\n const queryParams = new URLSearchParams();\n \n // 1. Handle Pagination\n if (options.top) queryParams.set('top', options.top.toString());\n if (options.skip) queryParams.set('skip', options.skip.toString());\n\n // 2. Handle Sort\n if (options.sort) {\n // Check if it's AST \n if (Array.isArray(options.sort) && typeof options.sort[0] === 'object') {\n queryParams.set('sort', JSON.stringify(options.sort));\n } else {\n const sortVal = Array.isArray(options.sort) ? options.sort.join(',') : options.sort;\n queryParams.set('sort', sortVal as string);\n }\n }\n \n // 3. Handle Select\n if (options.select) {\n queryParams.set('select', options.select.join(','));\n }\n\n // 4. Handle Filters (Simple vs AST)\n if (options.filters) {\n // If looks like AST (not plain object map)\n // TODO: robust check. safely assuming map for simplified find, and recommending .query() for AST\n if (this.isFilterAST(options.filters)) {\n queryParams.set('filters', JSON.stringify(options.filters));\n } else {\n Object.entries(options.filters).forEach(([k, v]) => {\n if (v !== undefined && v !== null) {\n queryParams.append(k, String(v));\n }\n });\n }\n }\n \n // 5. Handle Aggregations & GroupBy (Pass through as JSON if present)\n if (options.aggregations) {\n queryParams.set('aggregations', JSON.stringify(options.aggregations));\n }\n if (options.groupBy) {\n queryParams.set('groupBy', options.groupBy.join(','));\n }\n\n const res = await this.fetch(`${this.baseUrl}${route}/${object}?${queryParams.toString()}`);\n return res.json();\n },\n\n get: async <T = any>(object: string, id: string): Promise<T> => {\n const route = this.getRoute('data');\n const res = await this.fetch(`${this.baseUrl}${route}/${object}/${id}`);\n return res.json();\n },\n\n create: async <T = any>(object: string, data: Partial<T>): Promise<T> => {\n const route = this.getRoute('data');\n const res = await this.fetch(`${this.baseUrl}${route}/${object}`, {\n method: 'POST',\n body: JSON.stringify(data)\n });\n return res.json();\n },\n\n createMany: async <T = any>(object: string, data: Partial<T>[]): Promise<T[]> => {\n const route = this.getRoute('data');\n const res = await this.fetch(`${this.baseUrl}${route}/${object}/createMany`, {\n method: 'POST',\n body: JSON.stringify(data)\n });\n return res.json();\n },\n\n update: async <T = any>(object: string, id: string, data: Partial<T>): Promise<T> => {\n const route = this.getRoute('data');\n const res = await this.fetch(`${this.baseUrl}${route}/${object}/${id}`, {\n method: 'PATCH',\n body: JSON.stringify(data)\n });\n return res.json();\n },\n\n /**\n * Batch update multiple records\n * Uses the new BatchUpdateRequest schema with full control over options\n */\n batch: async (object: string, request: BatchUpdateRequest): Promise<BatchUpdateResponse> => {\n const route = this.getRoute('data');\n const res = await this.fetch(`${this.baseUrl}${route}/${object}/batch`, {\n method: 'POST',\n body: JSON.stringify(request)\n });\n return res.json();\n },\n\n /**\n * Update multiple records (simplified batch update)\n * Convenience method for batch updates without full BatchUpdateRequest\n */\n updateMany: async <T = any>(\n object: string, \n records: Array<{ id: string; data: Partial<T> }>,\n options?: BatchOptions\n ): Promise<BatchUpdateResponse> => {\n const route = this.getRoute('data');\n const request: UpdateManyRequest = {\n records,\n options\n };\n const res = await this.fetch(`${this.baseUrl}${route}/${object}/updateMany`, {\n method: 'POST',\n body: JSON.stringify(request)\n });\n return res.json();\n },\n\n delete: async (object: string, id: string): Promise<{ success: boolean }> => {\n const route = this.getRoute('data');\n const res = await this.fetch(`${this.baseUrl}${route}/${object}/${id}`, {\n method: 'DELETE'\n });\n return res.json();\n },\n\n /**\n * Delete multiple records by IDs\n */\n deleteMany: async(object: string, ids: string[], options?: BatchOptions): Promise<BatchUpdateResponse> => {\n const route = this.getRoute('data');\n const request: DeleteManyRequest = {\n ids,\n options\n };\n const res = await this.fetch(`${this.baseUrl}${route}/${object}/deleteMany`, {\n method: 'POST',\n body: JSON.stringify(request)\n });\n return res.json();\n }\n };\n\n\n\n /**\n * Private Helpers\n */\n\n private isFilterAST(filter: any): boolean {\n // Basic check: if array, it's [field, op, val] or [logic, node, node]\n // If object but not basic KV map... harder to tell without schema\n // For now, assume if it passes Array.isArray it's an AST root\n return Array.isArray(filter);\n }\n\n private async fetch(url: string, options: RequestInit = {}): Promise<Response> {\n this.logger.debug('HTTP request', { \n method: options.method || 'GET',\n url,\n hasBody: !!options.body\n });\n \n const headers: Record<string, string> = {\n 'Content-Type': 'application/json',\n ...(options.headers as Record<string, string> || {}),\n };\n\n if (this.token) {\n headers['Authorization'] = `Bearer ${this.token}`;\n }\n\n const res = await this.fetchImpl(url, { ...options, headers });\n \n this.logger.debug('HTTP response', { \n method: options.method || 'GET',\n url,\n status: res.status,\n ok: res.ok\n });\n \n if (!res.ok) {\n let errorBody: any;\n try {\n errorBody = await res.json();\n } catch {\n errorBody = { message: res.statusText };\n }\n \n this.logger.error('HTTP request failed', undefined, { \n method: options.method || 'GET',\n url,\n status: res.status,\n error: errorBody\n });\n \n // Create a standardized error if the response includes error details\n const errorMessage = errorBody?.message || errorBody?.error?.message || res.statusText;\n const errorCode = errorBody?.code || errorBody?.error?.code;\n const error = new Error(`[ObjectStack] ${errorCode ? `${errorCode}: ` : ''}${errorMessage}`) as any;\n \n // Attach error details for programmatic access\n error.code = errorCode;\n error.category = errorBody?.category;\n error.httpStatus = res.status;\n error.retryable = errorBody?.retryable;\n error.details = errorBody?.details || errorBody;\n \n throw error;\n }\n \n return res;\n }\n\n /**\n * Get the conventional route path for a given API endpoint type\n * ObjectStack uses standard conventions: /api/v1/data, /api/v1/metadata, /api/v1/ui\n */\n private getRoute(type: 'data' | 'metadata' | 'ui' | 'auth' | 'analytics' | 'hub' | 'storage' | 'automation'): string {\n // 1. Use discovered routes if available\n // Note: Spec uses 'endpoints', mapped dynamically\n if (this.discoveryInfo?.endpoints && (this.discoveryInfo.endpoints as any)[type]) {\n return (this.discoveryInfo.endpoints as any)[type];\n }\n\n // 2. Fallback to conventions\n // Note: HttpDispatcher expects /metadata, not /meta\n const routeMap: Record<string, string> = {\n data: '/api/v1/data',\n metadata: '/api/v1/metadata',\n ui: '/api/v1/ui',\n auth: '/api/v1/auth',\n analytics: '/api/v1/analytics',\n hub: '/api/v1/hub',\n storage: '/api/v1/storage',\n automation: '/api/v1/automation'\n };\n \n return routeMap[type] || `/api/v1/${type}`;\n }\n}\n\n// Re-export type-safe query builder\nexport { QueryBuilder, FilterBuilder, createQuery, createFilter } from './query-builder';\n\n// Re-export commonly used types from @objectstack/spec/api for convenience\nexport type {\n BatchUpdateRequest,\n BatchUpdateResponse,\n UpdateManyRequest,\n DeleteManyRequest,\n BatchOptions,\n BatchRecord,\n BatchOperationResult,\n MetadataCacheRequest,\n MetadataCacheResponse,\n StandardErrorCode,\n ErrorCategory,\n GetDiscoveryResponse,\n GetMetaTypesResponse,\n GetMetaItemsResponse\n} from '@objectstack/spec/api';\n","/**\n * Type-Safe Query Builder\n * \n * Provides a fluent API for building ObjectStack queries with:\n * - Compile-time type checking\n * - Intelligent code completion\n * - Runtime validation\n * - Type-safe filters and selections\n */\n\nimport { QueryAST, FilterCondition, SortNode } from '@objectstack/spec/data';\n\n/**\n * Type-safe filter builder\n */\nexport class FilterBuilder<T = any> {\n private conditions: FilterCondition[] = [];\n\n /**\n * Equality filter: field = value\n */\n equals<K extends keyof T>(field: K, value: T[K]): this {\n this.conditions.push([field as string, '=', value]);\n return this;\n }\n\n /**\n * Not equals filter: field != value\n */\n notEquals<K extends keyof T>(field: K, value: T[K]): this {\n this.conditions.push([field as string, '!=', value]);\n return this;\n }\n\n /**\n * Greater than filter: field > value\n */\n greaterThan<K extends keyof T>(field: K, value: T[K]): this {\n this.conditions.push([field as string, '>', value]);\n return this;\n }\n\n /**\n * Greater than or equal filter: field >= value\n */\n greaterThanOrEqual<K extends keyof T>(field: K, value: T[K]): this {\n this.conditions.push([field as string, '>=', value]);\n return this;\n }\n\n /**\n * Less than filter: field < value\n */\n lessThan<K extends keyof T>(field: K, value: T[K]): this {\n this.conditions.push([field as string, '<', value]);\n return this;\n }\n\n /**\n * Less than or equal filter: field <= value\n */\n lessThanOrEqual<K extends keyof T>(field: K, value: T[K]): this {\n this.conditions.push([field as string, '<=', value]);\n return this;\n }\n\n /**\n * IN filter: field IN (value1, value2, ...)\n */\n in<K extends keyof T>(field: K, values: T[K][]): this {\n this.conditions.push([field as string, 'in', values]);\n return this;\n }\n\n /**\n * NOT IN filter: field NOT IN (value1, value2, ...)\n */\n notIn<K extends keyof T>(field: K, values: T[K][]): this {\n this.conditions.push([field as string, 'not_in', values]);\n return this;\n }\n\n /**\n * LIKE filter: field LIKE pattern\n */\n like<K extends keyof T>(field: K, pattern: string): this {\n this.conditions.push([field as string, 'like', pattern]);\n return this;\n }\n\n /**\n * IS NULL filter: field IS NULL\n */\n isNull<K extends keyof T>(field: K): this {\n this.conditions.push([field as string, 'is_null', null]);\n return this;\n }\n\n /**\n * IS NOT NULL filter: field IS NOT NULL\n */\n isNotNull<K extends keyof T>(field: K): this {\n this.conditions.push([field as string, 'is_not_null', null]);\n return this;\n }\n\n /**\n * Build the filter condition\n */\n build(): FilterCondition {\n if (this.conditions.length === 0) {\n throw new Error('Filter builder has no conditions');\n }\n if (this.conditions.length === 1) {\n return this.conditions[0];\n }\n // Combine multiple conditions with AND\n return ['and', ...this.conditions];\n }\n\n /**\n * Get raw conditions array\n */\n getConditions(): FilterCondition[] {\n return this.conditions;\n }\n}\n\n/**\n * Type-safe query builder\n */\nexport class QueryBuilder<T = any> {\n private query: Partial<QueryAST> = {};\n private _object: string;\n\n constructor(object: string) {\n this._object = object;\n this.query.object = object;\n }\n\n /**\n * Select specific fields\n */\n select<K extends keyof T>(...fields: K[]): this {\n this.query.fields = fields as string[];\n return this;\n }\n\n /**\n * Add filters using a builder function\n */\n where(builderFn: (builder: FilterBuilder<T>) => void): this {\n const builder = new FilterBuilder<T>();\n builderFn(builder);\n const conditions = builder.getConditions();\n \n if (conditions.length === 1) {\n this.query.where = conditions[0];\n } else if (conditions.length > 1) {\n this.query.where = ['and', ...conditions] as FilterCondition;\n }\n \n return this;\n }\n\n /**\n * Add raw filter condition\n */\n filter(condition: FilterCondition): this {\n this.query.where = condition;\n return this;\n }\n\n /**\n * Sort by fields\n */\n orderBy<K extends keyof T>(field: K, order: 'asc' | 'desc' = 'asc'): this {\n if (!this.query.orderBy) {\n this.query.orderBy = [];\n }\n (this.query.orderBy as SortNode[]).push({\n field: field as string,\n order\n });\n return this;\n }\n\n /**\n * Limit the number of results\n */\n limit(count: number): this {\n this.query.limit = count;\n return this;\n }\n\n /**\n * Skip records (for pagination)\n */\n skip(count: number): this {\n this.query.offset = count;\n return this;\n }\n\n /**\n * Paginate results\n */\n paginate(page: number, pageSize: number): this {\n this.query.limit = pageSize;\n this.query.offset = (page - 1) * pageSize;\n return this;\n }\n\n /**\n * Group by fields\n */\n groupBy<K extends keyof T>(...fields: K[]): this {\n this.query.groupBy = fields as string[];\n return this;\n }\n\n /**\n * Build the final query AST\n */\n build(): QueryAST {\n return {\n object: this._object,\n ...this.query\n } as QueryAST;\n }\n\n /**\n * Get the current query state\n */\n getQuery(): Partial<QueryAST> {\n return { ...this.query };\n }\n}\n\n/**\n * Create a type-safe query builder for an object\n */\nexport function createQuery<T = any>(object: string): QueryBuilder<T> {\n return new QueryBuilder<T>(object);\n}\n\n/**\n * Create a type-safe filter builder\n */\nexport function createFilter<T = any>(): FilterBuilder<T> {\n return new FilterBuilder<T>();\n}\n"],"mappings":"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","names":[]}
package/package.json CHANGED
@@ -1,17 +1,25 @@
1
1
  {
2
2
  "name": "@objectstack/client",
3
- "version": "1.0.6",
3
+ "version": "1.0.7",
4
4
  "license": "Apache-2.0",
5
5
  "description": "Official Client SDK for ObjectStack Protocol",
6
6
  "main": "dist/index.js",
7
7
  "types": "dist/index.d.ts",
8
8
  "dependencies": {
9
- "@objectstack/spec": "1.0.6",
10
- "@objectstack/core": "1.0.6"
9
+ "@objectstack/core": "1.0.7",
10
+ "@objectstack/spec": "1.0.7"
11
11
  },
12
12
  "devDependencies": {
13
+ "@hono/node-server": "^1.2.0",
14
+ "msw": "^2.0.0",
13
15
  "typescript": "^5.0.0",
14
- "vitest": "^4.0.18"
16
+ "vitest": "^4.0.18",
17
+ "@objectstack/driver-memory": "1.0.7",
18
+ "@objectstack/hono": "1.0.7",
19
+ "@objectstack/objectql": "1.0.7",
20
+ "@objectstack/plugin-hono-server": "1.0.7",
21
+ "@objectstack/plugin-msw": "1.0.7",
22
+ "@objectstack/runtime": "1.0.7"
15
23
  },
16
24
  "scripts": {
17
25
  "build": "tsup --config ../../tsup.config.ts",
@@ -0,0 +1,144 @@
1
+ import { describe, it, expect, beforeAll, afterAll } from 'vitest';
2
+ import { LiteKernel } from '@objectstack/core';
3
+ import { ObjectQL, ObjectQLPlugin, SchemaRegistry } from '@objectstack/objectql';
4
+ import { InMemoryDriver } from '@objectstack/driver-memory';
5
+ import { HonoServerPlugin } from '@objectstack/plugin-hono-server';
6
+ import { ObjectStackClient } from './index';
7
+
8
+ describe('ObjectStackClient (with Hono Server)', () => {
9
+ let baseUrl: string;
10
+ let kernel: LiteKernel;
11
+
12
+ beforeAll(async () => {
13
+ // 1. Setup Kernel
14
+ kernel = new LiteKernel();
15
+ kernel.use(new ObjectQLPlugin());
16
+
17
+ // 2. Setup Hono Plugin
18
+ const honoPlugin = new HonoServerPlugin({
19
+ port: 0,
20
+ registerStandardEndpoints: true
21
+ });
22
+ kernel.use(honoPlugin);
23
+
24
+ // --- BROKER SHIM START ---
25
+ // HttpDispatcher requires a broker to function. We inject a simple shim.
26
+ (kernel as any).broker = {
27
+ call: async (action: string, params: any, opts: any) => {
28
+ const parts = action.split('.');
29
+ const service = parts[0];
30
+ const method = parts[1];
31
+
32
+ if (service === 'data') {
33
+ const ql = kernel.getService<any>('objectql'); // Use 'objectql' service name for clarity
34
+ if (method === 'create') {
35
+ const res = await ql.insert(params.object, params.data);
36
+ // Ensure we return the full object including input data + ID
37
+ return { ...params.data, ...res };
38
+ }
39
+ // Params from HttpDispatcher: { object, id, ...query }
40
+ if (method === 'get') {
41
+ // Ensure we search by 'id' explicitly for InMemoryDriver
42
+ return ql.findOne(params.object, { where: { id: params.id } });
43
+ }
44
+ // Params from HttpDispatcher: { object, filters }
45
+ if (method === 'query') {
46
+ // HttpDispatcher passes filters as simple object (map)
47
+ // ObjectQL expects { filter, select, sort, ... }
48
+ const data = await ql.find(params.object, { filter: params.filters });
49
+ // HttpDispatcher expects { data, count }
50
+ return { data, count: data.length };
51
+ }
52
+ if (method === 'find') return ql.find(params.object, { filter: params.filters });
53
+ }
54
+
55
+ if (service === 'metadata') {
56
+ // ObjectQLPlugin registers itself as 'metadata' but doesn't implement all broker methods directly
57
+ // We use SchemaRegistry for lookups
58
+ if (method === 'getObject') {
59
+ return SchemaRegistry.getObject(params.objectName);
60
+ }
61
+ if (method === 'objects') {
62
+ return SchemaRegistry.getAllObjects();
63
+ }
64
+ }
65
+
66
+ if (service === 'auth' && method === 'login') {
67
+ return { token: 'mock-token', user: { id: 'admin', name: 'Admin' } };
68
+ }
69
+
70
+ console.warn(`[BrokerShim] Action not implemented: ${action}`);
71
+ throw new Error(`Action ${action} not implemented in shim`);
72
+ }
73
+ };
74
+ // --- BROKER SHIM END ---
75
+
76
+ await kernel.bootstrap();
77
+
78
+ // 3. Setup Driver
79
+ const ql = kernel.getService<ObjectQL>('objectql');
80
+ ql.registerDriver(new InMemoryDriver(), true);
81
+
82
+ // 4. Load Metadata (Schema)
83
+ SchemaRegistry.registerObject({
84
+ name: 'customer',
85
+ label: 'Customer',
86
+ fields: {
87
+ name: { type: 'text', label: 'Name' },
88
+ email: { type: 'text', label: 'Email' }
89
+ }
90
+ });
91
+
92
+ // 5. Get Port from Service
93
+ const httpServer = kernel.getService<any>('http.server');
94
+ const port = httpServer.getPort();
95
+ baseUrl = `http://localhost:${port}`;
96
+
97
+ console.log(`Test server running at ${baseUrl}`);
98
+ });
99
+
100
+ afterAll(async () => {
101
+ if (kernel) await kernel.shutdown();
102
+ });
103
+
104
+ it('should connect to hono server', async () => {
105
+ const client = new ObjectStackClient({ baseUrl });
106
+ await client.connect();
107
+
108
+ // Client creates URL like ${baseUrl}/api/v1
109
+ expect(client).toBeDefined();
110
+ });
111
+
112
+ it('should create and retrieve data via hono', async () => {
113
+ const client = new ObjectStackClient({ baseUrl });
114
+ await client.connect();
115
+
116
+ // Create
117
+ const createdResponse = await client.data.create('customer', {
118
+ name: 'Hono User',
119
+ email: 'hono@example.com'
120
+ });
121
+
122
+ expect(createdResponse.success).toBe(true);
123
+ expect(createdResponse.data.name).toBe('Hono User');
124
+ expect(createdResponse.data.id).toBeDefined();
125
+
126
+ // Retrieve
127
+ const retrievedResponse = await client.data.get('customer', createdResponse.data.id);
128
+ expect(retrievedResponse.success).toBe(true);
129
+ expect(retrievedResponse.data.name).toBe('Hono User');
130
+ });
131
+
132
+ it('should find data via hono', async () => {
133
+ const client = new ObjectStackClient({ baseUrl });
134
+ await client.connect();
135
+
136
+ const resultsResponse = await client.data.find('customer', {
137
+ where: { name: 'Hono User' }
138
+ });
139
+
140
+ expect(resultsResponse.success).toBe(true);
141
+ expect(resultsResponse.data.length).toBeGreaterThan(0);
142
+ expect(resultsResponse.data[0].name).toBe('Hono User');
143
+ });
144
+ });
@@ -0,0 +1,198 @@
1
+ import { describe, it, expect, beforeAll, afterAll, afterEach } from 'vitest';
2
+ import { setupServer } from 'msw/node';
3
+ import { http, HttpResponse } from 'msw';
4
+ import { LiteKernel } from '@objectstack/core';
5
+ import { ObjectQLPlugin, SchemaRegistry } from '@objectstack/objectql';
6
+ import { InMemoryDriver } from '@objectstack/driver-memory';
7
+ import { MSWPlugin } from '@objectstack/plugin-msw';
8
+ import { ObjectStackClient } from './index';
9
+
10
+ // 127.0.0.1 usage logic remains
11
+ const BASE_URL = 'http://127.0.0.1';
12
+
13
+ describe('ObjectStackClient (with MSW Plugin)', () => {
14
+ let kernel: LiteKernel;
15
+ let mswPlugin: MSWPlugin;
16
+ let server: any;
17
+
18
+ beforeAll(async () => {
19
+ // 1. Setup Kernel
20
+ kernel = new LiteKernel();
21
+ kernel.use(new ObjectQLPlugin());
22
+
23
+ // 2. Setup MSW Plugin (headless mode)
24
+ mswPlugin = new MSWPlugin({
25
+ enableBrowser: false,
26
+ // baseUrl: '/api/v1' // Default is /api/v1
27
+ });
28
+ kernel.use(mswPlugin);
29
+
30
+ // --- BROKER SHIM START ---
31
+ // HttpDispatcher requires a broker to function. We inject a simple shim.
32
+ (kernel as any).broker = {
33
+ call: async (action: string, params: any, opts: any) => {
34
+ const parts = action.split('.');
35
+ const service = parts[0];
36
+ const method = parts[1];
37
+
38
+ if (service === 'data') {
39
+ const ql = kernel.getService<any>('objectql');
40
+ if (method === 'create') {
41
+ const res = await ql.insert(params.object, params.data);
42
+ return { ...params.data, ...res };
43
+ }
44
+ if (method === 'get') {
45
+ // Ensure we search by 'id' explicitly for InMemoryDriver
46
+ return ql.findOne(params.object, { where: { id: params.id } });
47
+ }
48
+ if (method === 'query') {
49
+ const data = await ql.find(params.object, { filter: params.filters });
50
+ return { data, count: data.length };
51
+ }
52
+ if (method === 'find') return ql.find(params.object, { filter: params.filters });
53
+ }
54
+
55
+ if (service === 'metadata') {
56
+ if (method === 'getObject') return SchemaRegistry.getObject(params.objectName);
57
+ if (method === 'objects') return SchemaRegistry.getAllObjects();
58
+ }
59
+
60
+ console.warn(`[BrokerShim] Action not implemented: ${action}`);
61
+ return null;
62
+ }
63
+ };
64
+ // --- BROKER SHIM END ---
65
+
66
+ await kernel.bootstrap();
67
+
68
+ // 3. Setup Driver & Schema
69
+ const ql = kernel.getService<any>('objectql');
70
+ ql.registerDriver(new InMemoryDriver(), true);
71
+
72
+ SchemaRegistry.registerObject({
73
+ name: 'customer',
74
+ fields: {
75
+ name: { type: 'text' }
76
+ }
77
+ });
78
+
79
+ // Add some data
80
+ await ql.insert('customer', { id: '101', name: 'Alice' });
81
+ await ql.insert('customer', { id: '102', name: 'Bob' });
82
+
83
+ // 4. Get handlers and start MSW Node Server
84
+ const handlers = mswPlugin.getHandlers();
85
+
86
+ // Manual handlers to cover gaps in MSWPlugin generation
87
+ handlers.push(
88
+ http.get('http://127.0.0.1/api/v1', () => {
89
+ return HttpResponse.json({
90
+ name: 'ObjectOS',
91
+ version: '1.0.0',
92
+ routes: {
93
+ data: '/api/v1/data',
94
+ metadata: '/api/v1/metadata',
95
+ auth: '/api/v1/auth'
96
+ },
97
+ capabilities: ['data', 'metadata'],
98
+ features: {}
99
+ });
100
+ }),
101
+
102
+ http.get('http://127.0.0.1/api/v1/metadata/object/:name', async ({ params }) => {
103
+ try {
104
+ const res = await (kernel as any).broker.call('metadata.getObject', { objectName: params.name });
105
+ return HttpResponse.json({ success: true, data: res });
106
+ } catch (e: any) { return HttpResponse.json({ success: false, error: e.message }, { status: 404 }); }
107
+ }),
108
+
109
+ http.get('http://127.0.0.1/api/v1/data/:object', async ({ params }) => {
110
+ try {
111
+ // Simplifying: ignoring query filters for this test
112
+ const res = await (kernel as any).broker.call('data.find', { object: params.object, filters: {} });
113
+ return HttpResponse.json({ success: true, data: res });
114
+ } catch (e: any) { return HttpResponse.json({ success: false, error: e.message }, { status: 500 }); }
115
+ }),
116
+
117
+ http.post('http://127.0.0.1/api/v1/data/:object', async ({ params, request }) => {
118
+ try {
119
+ const body = await request.json();
120
+ const res = await (kernel as any).broker.call('data.create', { object: params.object, data: body });
121
+ return HttpResponse.json({ success: true, data: res }, { status: 201 });
122
+ } catch (e: any) { return HttpResponse.json({ success: false, error: e.message }, { status: 500 }); }
123
+ }),
124
+
125
+ http.get('http://127.0.0.1/api/v1/data/:object/:id', async ({ params }) => {
126
+ try {
127
+ const res = await (kernel as any).broker.call('data.get', { object: params.object, id: params.id });
128
+ return HttpResponse.json({ success: true, data: res });
129
+ } catch (e: any) { return HttpResponse.json({ success: false, error: e.message }, { status: 404 }); }
130
+ })
131
+ );
132
+
133
+ server = setupServer(...handlers);
134
+ server.listen({ onUnhandledRequest: 'error' });
135
+ });
136
+
137
+ // Reset handlers after each test to ensure clean state
138
+ afterEach(() => server.resetHandlers());
139
+
140
+ afterAll(async () => {
141
+ if (server) server.close();
142
+ if (kernel) await kernel.shutdown();
143
+ });
144
+
145
+ it('should connect and discover endpoints', async () => {
146
+ // MSWPlugin configured with baseUrl: '' creates handlers at root.
147
+ // Client connects to /api/v1.
148
+ // To make them match in THIS test file where I used baseUrl: '',
149
+ // I should have configured MSWPlugin with baseUrl: '/api/v1' or left default.
150
+
151
+ // RE-FIXING SETUP in beforeAll (via edit).
152
+ // If I change MSWPlugin config to default ('/api/v1'),
153
+ // then Client(BASE_URL).connect() -> fetches BASE_URL/api/v1 -> matches MSW handler /api/v1.
154
+
155
+ const client = new ObjectStackClient({ baseUrl: BASE_URL });
156
+ await client.connect();
157
+
158
+ expect(client['discoveryInfo']).toBeDefined();
159
+ });
160
+
161
+ it('should fetch object metadata', async () => {
162
+ const client = new ObjectStackClient({ baseUrl: BASE_URL });
163
+ await client.connect();
164
+
165
+ const customerRes = await client.meta.getItem('object', 'customer');
166
+ expect(customerRes.success).toBe(true);
167
+ expect(customerRes.data.name).toBe('customer');
168
+ });
169
+
170
+ it('should find data records', async () => {
171
+ const client = new ObjectStackClient({ baseUrl: BASE_URL });
172
+ await client.connect();
173
+
174
+ const resultsRes = await client.data.find('customer');
175
+ expect(resultsRes.success).toBe(true);
176
+ expect(resultsRes.data.length).toBe(2);
177
+ expect(resultsRes.data[0].name).toBe('Alice');
178
+ });
179
+
180
+ it('should get single record', async () => {
181
+ const client = new ObjectStackClient({ baseUrl: BASE_URL });
182
+ await client.connect();
183
+
184
+ const recordRes = await client.data.get('customer', '101');
185
+ expect(recordRes.success).toBe(true);
186
+ expect(recordRes.data.name).toBe('Alice');
187
+ });
188
+
189
+ it('should create record', async () => {
190
+ const client = new ObjectStackClient({ baseUrl: BASE_URL });
191
+ await client.connect();
192
+
193
+ const newRecordRes = await client.data.create('customer', { name: 'Charlie' });
194
+ expect(newRecordRes.success).toBe(true);
195
+ expect(newRecordRes.data.name).toBe('Charlie');
196
+ expect(newRecordRes.data.id).toBeDefined();
197
+ });
198
+ });
package/src/index.ts CHANGED
@@ -97,10 +97,45 @@ export class ObjectStackClient {
97
97
  this.logger.debug('Connecting to ObjectStack server', { baseUrl: this.baseUrl });
98
98
 
99
99
  try {
100
- // Connect to the discovery endpoint at /api/v1
101
- const res = await this.fetch(`${this.baseUrl}/api/v1`);
102
-
103
- const data = await res.json();
100
+ let data: DiscoveryResult | undefined;
101
+
102
+ // 1. Try Standard Discovery (.well-known)
103
+ try {
104
+ let wellKnownUrl: string;
105
+ try {
106
+ // If baseUrl is absolute, get origin
107
+ const url = new URL(this.baseUrl);
108
+ wellKnownUrl = `${url.origin}/.well-known/objectstack`;
109
+ } catch {
110
+ // If baseUrl is relative, use absolute path from root
111
+ wellKnownUrl = '/.well-known/objectstack';
112
+ }
113
+
114
+ this.logger.debug('Probing .well-known discovery', { url: wellKnownUrl });
115
+ const res = await this.fetchImpl(wellKnownUrl);
116
+ if (res.ok) {
117
+ data = await res.json();
118
+ this.logger.debug('Discovered via .well-known');
119
+ }
120
+ } catch (e) {
121
+ this.logger.debug('Standard discovery probe failed', { error: (e as Error).message });
122
+ }
123
+
124
+ // 2. Fallback to Legacy/Direct Path /api/v1
125
+ if (!data) {
126
+ const fallbackUrl = `${this.baseUrl}/api/v1`;
127
+ this.logger.debug('Falling back to legacy discovery', { url: fallbackUrl });
128
+ const res = await this.fetchImpl(fallbackUrl);
129
+ if (!res.ok) {
130
+ throw new Error(`Failed to connect to ${fallbackUrl}: ${res.statusText}`);
131
+ }
132
+ data = await res.json();
133
+ }
134
+
135
+ if (!data) {
136
+ throw new Error('Connection failed: No discovery data returned');
137
+ }
138
+
104
139
  this.discoveryInfo = data;
105
140
 
106
141
  this.logger.info('Connected to ObjectStack server', {