@baasix/sdk 0.1.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (43) hide show
  1. package/LICENSE +21 -0
  2. package/README.md +1197 -0
  3. package/dist/client-DeXa-R9w.d.ts +680 -0
  4. package/dist/client-VT7NckyI.d.cts +680 -0
  5. package/dist/index.cjs +4567 -0
  6. package/dist/index.cjs.map +1 -0
  7. package/dist/index.d.cts +1788 -0
  8. package/dist/index.d.ts +1788 -0
  9. package/dist/index.js +4543 -0
  10. package/dist/index.js.map +1 -0
  11. package/dist/modules/auth.cjs +650 -0
  12. package/dist/modules/auth.cjs.map +1 -0
  13. package/dist/modules/auth.d.cts +384 -0
  14. package/dist/modules/auth.d.ts +384 -0
  15. package/dist/modules/auth.js +648 -0
  16. package/dist/modules/auth.js.map +1 -0
  17. package/dist/modules/files.cjs +269 -0
  18. package/dist/modules/files.cjs.map +1 -0
  19. package/dist/modules/files.d.cts +187 -0
  20. package/dist/modules/files.d.ts +187 -0
  21. package/dist/modules/files.js +267 -0
  22. package/dist/modules/files.js.map +1 -0
  23. package/dist/modules/items.cjs +640 -0
  24. package/dist/modules/items.cjs.map +1 -0
  25. package/dist/modules/items.d.cts +465 -0
  26. package/dist/modules/items.d.ts +465 -0
  27. package/dist/modules/items.js +637 -0
  28. package/dist/modules/items.js.map +1 -0
  29. package/dist/modules/schemas.cjs +322 -0
  30. package/dist/modules/schemas.cjs.map +1 -0
  31. package/dist/modules/schemas.d.cts +260 -0
  32. package/dist/modules/schemas.d.ts +260 -0
  33. package/dist/modules/schemas.js +320 -0
  34. package/dist/modules/schemas.js.map +1 -0
  35. package/dist/storage/index.cjs +162 -0
  36. package/dist/storage/index.cjs.map +1 -0
  37. package/dist/storage/index.d.cts +96 -0
  38. package/dist/storage/index.d.ts +96 -0
  39. package/dist/storage/index.js +157 -0
  40. package/dist/storage/index.js.map +1 -0
  41. package/dist/types-BdjsGANq.d.cts +40 -0
  42. package/dist/types-BdjsGANq.d.ts +40 -0
  43. package/package.json +108 -0
@@ -0,0 +1,637 @@
1
+ // src/modules/items.ts
2
+ var QueryBuilder = class {
3
+ collection;
4
+ client;
5
+ queryParams = {};
6
+ constructor(collection, client) {
7
+ this.collection = collection;
8
+ this.client = client;
9
+ }
10
+ /**
11
+ * Select specific fields to return
12
+ *
13
+ * @example
14
+ * ```typescript
15
+ * items.select(['id', 'name', 'author.*'])
16
+ * items.select('*', 'category.name')
17
+ * ```
18
+ */
19
+ select(...fields) {
20
+ const flatFields = fields.length === 1 && Array.isArray(fields[0]) ? fields[0] : fields;
21
+ this.queryParams.fields = flatFields;
22
+ return this;
23
+ }
24
+ /**
25
+ * Alias for select()
26
+ */
27
+ fields(...fields) {
28
+ return this.select(...fields);
29
+ }
30
+ /**
31
+ * Add filter conditions
32
+ *
33
+ * @example
34
+ * ```typescript
35
+ * // Simple equality
36
+ * items.filter({ status: { eq: 'active' } })
37
+ *
38
+ * // Multiple conditions
39
+ * items.filter({
40
+ * AND: [
41
+ * { status: { eq: 'active' } },
42
+ * { price: { gte: 100 } }
43
+ * ]
44
+ * })
45
+ *
46
+ * // Relation filtering
47
+ * items.filter({ 'author.name': { like: 'John' } })
48
+ * ```
49
+ */
50
+ filter(filter) {
51
+ this.queryParams.filter = filter;
52
+ return this;
53
+ }
54
+ /**
55
+ * Alias for filter()
56
+ */
57
+ where(filter) {
58
+ return this.filter(filter);
59
+ }
60
+ /**
61
+ * Sort results
62
+ *
63
+ * @example
64
+ * ```typescript
65
+ * // Object notation
66
+ * items.sort({ createdAt: 'desc', name: 'asc' })
67
+ *
68
+ * // Array notation with prefix
69
+ * items.sort(['-createdAt', 'name'])
70
+ *
71
+ * // String shorthand
72
+ * items.sort('createdAt:desc')
73
+ * ```
74
+ */
75
+ sort(sort) {
76
+ this.queryParams.sort = sort;
77
+ return this;
78
+ }
79
+ /**
80
+ * Alias for sort()
81
+ */
82
+ orderBy(sort) {
83
+ return this.sort(sort);
84
+ }
85
+ /**
86
+ * Limit number of results
87
+ *
88
+ * @example
89
+ * ```typescript
90
+ * items.limit(20)
91
+ * items.limit(-1) // All results
92
+ * ```
93
+ */
94
+ limit(limit) {
95
+ this.queryParams.limit = limit;
96
+ return this;
97
+ }
98
+ /**
99
+ * Set page number (1-indexed)
100
+ *
101
+ * @example
102
+ * ```typescript
103
+ * items.page(2).limit(20)
104
+ * ```
105
+ */
106
+ page(page) {
107
+ this.queryParams.page = page;
108
+ return this;
109
+ }
110
+ /**
111
+ * Skip a number of results
112
+ *
113
+ * @example
114
+ * ```typescript
115
+ * items.offset(20)
116
+ * ```
117
+ */
118
+ offset(offset) {
119
+ this.queryParams.offset = offset;
120
+ return this;
121
+ }
122
+ /**
123
+ * Full-text search
124
+ *
125
+ * @example
126
+ * ```typescript
127
+ * items.search('keyword', ['title', 'description'])
128
+ * ```
129
+ */
130
+ search(query, fields) {
131
+ this.queryParams.search = query;
132
+ if (fields) {
133
+ this.queryParams.searchFields = fields;
134
+ }
135
+ return this;
136
+ }
137
+ /**
138
+ * Include soft-deleted items
139
+ *
140
+ * @example
141
+ * ```typescript
142
+ * items.withDeleted()
143
+ * ```
144
+ */
145
+ withDeleted() {
146
+ this.queryParams.paranoid = false;
147
+ return this;
148
+ }
149
+ /**
150
+ * Filter related items in O2M/M2M relations
151
+ *
152
+ * @example
153
+ * ```typescript
154
+ * // Only show approved comments
155
+ * items.relFilter({
156
+ * comments: { approved: { eq: true } }
157
+ * })
158
+ * ```
159
+ */
160
+ relFilter(conditions) {
161
+ this.queryParams.relConditions = conditions;
162
+ return this;
163
+ }
164
+ /**
165
+ * Get the built query parameters
166
+ */
167
+ getQuery() {
168
+ return { ...this.queryParams };
169
+ }
170
+ /**
171
+ * Execute the query and return results
172
+ *
173
+ * @example
174
+ * ```typescript
175
+ * const { data, totalCount } = await items
176
+ * .filter({ status: { eq: 'active' } })
177
+ * .sort({ createdAt: 'desc' })
178
+ * .limit(10)
179
+ * .get();
180
+ * ```
181
+ */
182
+ async get() {
183
+ return this.client.get(`/items/${this.collection}`, {
184
+ params: this.buildParams()
185
+ });
186
+ }
187
+ /**
188
+ * Execute the query and return the first result
189
+ *
190
+ * @example
191
+ * ```typescript
192
+ * const item = await items
193
+ * .filter({ slug: { eq: 'my-post' } })
194
+ * .first();
195
+ * ```
196
+ */
197
+ async first() {
198
+ const result = await this.limit(1).get();
199
+ return result.data[0] || null;
200
+ }
201
+ /**
202
+ * Count matching items
203
+ *
204
+ * @example
205
+ * ```typescript
206
+ * const count = await items.filter({ status: { eq: 'active' } }).count();
207
+ * ```
208
+ */
209
+ async count() {
210
+ const result = await this.client.get(
211
+ `/items/${this.collection}`,
212
+ {
213
+ params: {
214
+ ...this.buildParams(),
215
+ limit: 0
216
+ }
217
+ }
218
+ );
219
+ return result.totalCount || 0;
220
+ }
221
+ /**
222
+ * Build query parameters for the request
223
+ */
224
+ buildParams() {
225
+ const params = {};
226
+ if (this.queryParams.fields) {
227
+ params.fields = this.queryParams.fields;
228
+ }
229
+ if (this.queryParams.filter) {
230
+ params.filter = this.queryParams.filter;
231
+ }
232
+ if (this.queryParams.sort) {
233
+ params.sort = this.queryParams.sort;
234
+ }
235
+ if (this.queryParams.limit !== void 0) {
236
+ params.limit = this.queryParams.limit;
237
+ }
238
+ if (this.queryParams.page !== void 0) {
239
+ params.page = this.queryParams.page;
240
+ }
241
+ if (this.queryParams.offset !== void 0) {
242
+ params.offset = this.queryParams.offset;
243
+ }
244
+ if (this.queryParams.search) {
245
+ params.search = this.queryParams.search;
246
+ }
247
+ if (this.queryParams.searchFields) {
248
+ params.searchFields = this.queryParams.searchFields;
249
+ }
250
+ if (this.queryParams.paranoid !== void 0) {
251
+ params.paranoid = this.queryParams.paranoid;
252
+ }
253
+ if (this.queryParams.relConditions) {
254
+ params.relConditions = this.queryParams.relConditions;
255
+ }
256
+ if (this.queryParams.aggregate) {
257
+ params.aggregate = this.queryParams.aggregate;
258
+ }
259
+ if (this.queryParams.groupBy) {
260
+ params.groupBy = this.queryParams.groupBy;
261
+ }
262
+ return params;
263
+ }
264
+ };
265
+ var ItemsModule = class {
266
+ collection;
267
+ client;
268
+ constructor(collection, config) {
269
+ this.collection = collection;
270
+ this.client = config.client;
271
+ }
272
+ /**
273
+ * Create a query builder for fluent query construction
274
+ *
275
+ * @example
276
+ * ```typescript
277
+ * const results = await baasix.items('posts')
278
+ * .query()
279
+ * .select('*', 'author.*')
280
+ * .filter({ status: { eq: 'published' } })
281
+ * .sort({ createdAt: 'desc' })
282
+ * .limit(10)
283
+ * .get();
284
+ * ```
285
+ */
286
+ query() {
287
+ return new QueryBuilder(this.collection, this.client);
288
+ }
289
+ /**
290
+ * Find items with optional query parameters
291
+ *
292
+ * @example
293
+ * ```typescript
294
+ * // Simple query
295
+ * const { data } = await items.find();
296
+ *
297
+ * // With parameters
298
+ * const { data, totalCount } = await items.find({
299
+ * filter: { status: { eq: 'active' } },
300
+ * sort: { createdAt: 'desc' },
301
+ * limit: 20,
302
+ * page: 1,
303
+ * fields: ['id', 'name', 'price']
304
+ * });
305
+ * ```
306
+ */
307
+ async find(params) {
308
+ return this.client.get(`/items/${this.collection}`, {
309
+ params
310
+ });
311
+ }
312
+ /**
313
+ * Alias for find()
314
+ */
315
+ async findMany(params) {
316
+ return this.find(params);
317
+ }
318
+ /**
319
+ * Find a single item by ID
320
+ *
321
+ * @example
322
+ * ```typescript
323
+ * const product = await items.findOne('product-uuid');
324
+ *
325
+ * // With specific fields
326
+ * const product = await items.findOne('product-uuid', {
327
+ * fields: ['id', 'name', 'category.*']
328
+ * });
329
+ * ```
330
+ */
331
+ async findOne(id, params) {
332
+ const response = await this.client.get(
333
+ `/items/${this.collection}/${id}`,
334
+ { params }
335
+ );
336
+ return response.data;
337
+ }
338
+ /**
339
+ * Alias for findOne()
340
+ */
341
+ async get(id, params) {
342
+ return this.findOne(id, params);
343
+ }
344
+ /**
345
+ * Create a new item
346
+ *
347
+ * @example
348
+ * ```typescript
349
+ * const id = await items.create({
350
+ * name: 'New Product',
351
+ * price: 29.99,
352
+ * status: 'draft'
353
+ * });
354
+ * ```
355
+ */
356
+ async create(data) {
357
+ const response = await this.client.post(
358
+ `/items/${this.collection}`,
359
+ data
360
+ );
361
+ return response.data;
362
+ }
363
+ /**
364
+ * Alias for create()
365
+ */
366
+ async insert(data) {
367
+ return this.create(data);
368
+ }
369
+ /**
370
+ * Create multiple items at once
371
+ *
372
+ * @example
373
+ * ```typescript
374
+ * const ids = await items.createMany([
375
+ * { name: 'Product 1', price: 10 },
376
+ * { name: 'Product 2', price: 20 }
377
+ * ]);
378
+ * ```
379
+ */
380
+ async createMany(data) {
381
+ const response = await this.client.post(
382
+ `/items/${this.collection}/bulk`,
383
+ data
384
+ );
385
+ return response.data;
386
+ }
387
+ /**
388
+ * Alias for createMany()
389
+ */
390
+ async insertMany(data) {
391
+ return this.createMany(data);
392
+ }
393
+ /**
394
+ * Update an existing item
395
+ *
396
+ * @example
397
+ * ```typescript
398
+ * await items.update('product-uuid', {
399
+ * price: 24.99,
400
+ * status: 'published'
401
+ * });
402
+ * ```
403
+ */
404
+ async update(id, data) {
405
+ const response = await this.client.patch(
406
+ `/items/${this.collection}/${id}`,
407
+ data
408
+ );
409
+ return response.data;
410
+ }
411
+ /**
412
+ * Update multiple items at once
413
+ *
414
+ * @example
415
+ * ```typescript
416
+ * // Update by IDs with same data
417
+ * await items.updateMany(['id1', 'id2'], { status: 'archived' });
418
+ * ```
419
+ */
420
+ async updateMany(ids, data) {
421
+ const updates = ids.map((id) => ({ id, data }));
422
+ const response = await this.client.patch(
423
+ `/items/${this.collection}/bulk`,
424
+ updates
425
+ );
426
+ return response.data;
427
+ }
428
+ /**
429
+ * Upsert an item (create if not exists, update if exists)
430
+ *
431
+ * @example
432
+ * ```typescript
433
+ * const id = await items.upsert(
434
+ * { sku: 'WIDGET-001' },
435
+ * { name: 'Widget', price: 29.99, sku: 'WIDGET-001' }
436
+ * );
437
+ * ```
438
+ */
439
+ async upsert(filter, data) {
440
+ const existing = await this.find({ filter, limit: 1 });
441
+ if (existing.data.length > 0) {
442
+ return this.update(existing.data[0].id, data);
443
+ }
444
+ return this.create(data);
445
+ }
446
+ /**
447
+ * Delete an item by ID
448
+ *
449
+ * @example
450
+ * ```typescript
451
+ * await items.delete('product-uuid');
452
+ * ```
453
+ */
454
+ async delete(id) {
455
+ await this.client.delete(`/items/${this.collection}/${id}`);
456
+ }
457
+ /**
458
+ * Delete multiple items
459
+ *
460
+ * @example
461
+ * ```typescript
462
+ * await items.deleteMany(['id1', 'id2', 'id3']);
463
+ * ```
464
+ */
465
+ async deleteMany(ids) {
466
+ await this.client.delete(`/items/${this.collection}/bulk`, {
467
+ body: JSON.stringify(ids)
468
+ });
469
+ }
470
+ /**
471
+ * Soft delete an item (if paranoid mode is enabled)
472
+ *
473
+ * @example
474
+ * ```typescript
475
+ * await items.softDelete('product-uuid');
476
+ * ```
477
+ */
478
+ async softDelete(id) {
479
+ await this.update(id, { deletedAt: (/* @__PURE__ */ new Date()).toISOString() });
480
+ }
481
+ /**
482
+ * Restore a soft-deleted item
483
+ *
484
+ * @example
485
+ * ```typescript
486
+ * await items.restore('product-uuid');
487
+ * ```
488
+ */
489
+ async restore(id) {
490
+ await this.update(id, { deletedAt: null });
491
+ }
492
+ /**
493
+ * Aggregate data with grouping
494
+ *
495
+ * @example
496
+ * ```typescript
497
+ * const results = await items.aggregate({
498
+ * aggregate: {
499
+ * total: { function: 'sum', field: 'amount' },
500
+ * count: { function: 'count', field: 'id' },
501
+ * avgPrice: { function: 'avg', field: 'price' }
502
+ * },
503
+ * groupBy: ['category', 'status'],
504
+ * filter: { createdAt: { gte: '$NOW-DAYS_30' } }
505
+ * });
506
+ * ```
507
+ */
508
+ async aggregate(params) {
509
+ const response = await this.client.get(
510
+ `/items/${this.collection}`,
511
+ { params }
512
+ );
513
+ return response.data;
514
+ }
515
+ // ===================
516
+ // Import Operations
517
+ // ===================
518
+ /**
519
+ * Import items from a CSV file
520
+ *
521
+ * @example
522
+ * ```typescript
523
+ * // Browser
524
+ * const fileInput = document.querySelector('input[type="file"]');
525
+ * const file = fileInput.files[0];
526
+ *
527
+ * const result = await baasix.items('products').importCSV(file);
528
+ *
529
+ * console.log(`Imported ${result.imported} items`);
530
+ * ```
531
+ */
532
+ async importCSV(file) {
533
+ const formData = new FormData();
534
+ if (file instanceof File) {
535
+ formData.append("csvFile", file);
536
+ } else {
537
+ formData.append("csvFile", file);
538
+ }
539
+ const response = await this.client.post(
540
+ `/items/${this.collection}/import-csv`,
541
+ formData
542
+ );
543
+ return response.results;
544
+ }
545
+ /**
546
+ * Import items from a JSON file
547
+ *
548
+ * @example
549
+ * ```typescript
550
+ * const file = fileInput.files[0]; // JSON file
551
+ * const result = await baasix.items('products').importJSON(file);
552
+ *
553
+ * console.log(`Imported ${result.imported} items`);
554
+ * ```
555
+ */
556
+ async importJSON(file) {
557
+ const formData = new FormData();
558
+ if (file instanceof File) {
559
+ formData.append("jsonFile", file);
560
+ } else {
561
+ formData.append("jsonFile", file);
562
+ }
563
+ const response = await this.client.post(
564
+ `/items/${this.collection}/import-json`,
565
+ formData
566
+ );
567
+ return response.results;
568
+ }
569
+ /**
570
+ * Import items from an array of objects
571
+ *
572
+ * @example
573
+ * ```typescript
574
+ * const data = [
575
+ * { name: 'Product 1', price: 29.99 },
576
+ * { name: 'Product 2', price: 39.99 }
577
+ * ];
578
+ *
579
+ * const result = await baasix.items('products').importData(data);
580
+ * ```
581
+ */
582
+ async importData(data) {
583
+ const response = await this.client.post(
584
+ `/items/${this.collection}/bulk`,
585
+ data
586
+ );
587
+ return response;
588
+ }
589
+ // ===================
590
+ // Sort Operations
591
+ // ===================
592
+ /**
593
+ * Sort/reorder items (move item before or after another)
594
+ *
595
+ * @example
596
+ * ```typescript
597
+ * // Move item1 before item2
598
+ * await baasix.items('products').sortItem('item1-uuid', 'item2-uuid');
599
+ *
600
+ * // Move item1 after item2
601
+ * await baasix.items('products').sortItem('item1-uuid', 'item2-uuid', 'after');
602
+ * ```
603
+ */
604
+ async sortItem(itemId, targetItemId, mode = "before") {
605
+ await this.client.post(`/utils/sort/${this.collection}`, {
606
+ item: itemId,
607
+ to: targetItemId,
608
+ mode
609
+ });
610
+ }
611
+ /**
612
+ * Reorder multiple items
613
+ *
614
+ * @example
615
+ * ```typescript
616
+ * // Set explicit order
617
+ * await baasix.items('products').reorder([
618
+ * 'item3-uuid',
619
+ * 'item1-uuid',
620
+ * 'item2-uuid'
621
+ * ]);
622
+ * ```
623
+ */
624
+ async reorder(orderedIds) {
625
+ for (let i = 1; i < orderedIds.length; i++) {
626
+ await this.client.post(`/utils/sort/${this.collection}`, {
627
+ item: orderedIds[i],
628
+ to: orderedIds[i - 1],
629
+ mode: "after"
630
+ });
631
+ }
632
+ }
633
+ };
634
+
635
+ export { ItemsModule, QueryBuilder };
636
+ //# sourceMappingURL=items.js.map
637
+ //# sourceMappingURL=items.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"sources":["../../src/modules/items.ts"],"names":[],"mappings":"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","file":"items.js","sourcesContent":["import type { HttpClient } from \"../client\";\nimport type {\n BaseItem,\n BulkResponse,\n DeleteResponse,\n Filter,\n MutationResponse,\n PaginatedResponse,\n QueryParams,\n SingleResponse,\n Sort,\n} from \"../types\";\n\nexport interface ItemsModuleConfig {\n client: HttpClient;\n}\n\n/**\n * Query builder for constructing type-safe queries\n */\nexport class QueryBuilder<T extends BaseItem = BaseItem> {\n private collection: string;\n private client: HttpClient;\n private queryParams: QueryParams = {};\n\n constructor(collection: string, client: HttpClient) {\n this.collection = collection;\n this.client = client;\n }\n\n /**\n * Select specific fields to return\n *\n * @example\n * ```typescript\n * items.select(['id', 'name', 'author.*'])\n * items.select('*', 'category.name')\n * ```\n */\n select(...fields: string[] | [string[]]): this {\n const flatFields = fields.length === 1 && Array.isArray(fields[0])\n ? fields[0]\n : (fields as string[]);\n this.queryParams.fields = flatFields;\n return this;\n }\n\n /**\n * Alias for select()\n */\n fields(...fields: string[] | [string[]]): this {\n return this.select(...fields);\n }\n\n /**\n * Add filter conditions\n *\n * @example\n * ```typescript\n * // Simple equality\n * items.filter({ status: { eq: 'active' } })\n *\n * // Multiple conditions\n * items.filter({\n * AND: [\n * { status: { eq: 'active' } },\n * { price: { gte: 100 } }\n * ]\n * })\n *\n * // Relation filtering\n * items.filter({ 'author.name': { like: 'John' } })\n * ```\n */\n filter(filter: Filter): this {\n this.queryParams.filter = filter;\n return this;\n }\n\n /**\n * Alias for filter()\n */\n where(filter: Filter): this {\n return this.filter(filter);\n }\n\n /**\n * Sort results\n *\n * @example\n * ```typescript\n * // Object notation\n * items.sort({ createdAt: 'desc', name: 'asc' })\n *\n * // Array notation with prefix\n * items.sort(['-createdAt', 'name'])\n *\n * // String shorthand\n * items.sort('createdAt:desc')\n * ```\n */\n sort(sort: Sort): this {\n this.queryParams.sort = sort;\n return this;\n }\n\n /**\n * Alias for sort()\n */\n orderBy(sort: Sort): this {\n return this.sort(sort);\n }\n\n /**\n * Limit number of results\n *\n * @example\n * ```typescript\n * items.limit(20)\n * items.limit(-1) // All results\n * ```\n */\n limit(limit: number): this {\n this.queryParams.limit = limit;\n return this;\n }\n\n /**\n * Set page number (1-indexed)\n *\n * @example\n * ```typescript\n * items.page(2).limit(20)\n * ```\n */\n page(page: number): this {\n this.queryParams.page = page;\n return this;\n }\n\n /**\n * Skip a number of results\n *\n * @example\n * ```typescript\n * items.offset(20)\n * ```\n */\n offset(offset: number): this {\n this.queryParams.offset = offset;\n return this;\n }\n\n /**\n * Full-text search\n *\n * @example\n * ```typescript\n * items.search('keyword', ['title', 'description'])\n * ```\n */\n search(query: string, fields?: string[]): this {\n this.queryParams.search = query;\n if (fields) {\n this.queryParams.searchFields = fields;\n }\n return this;\n }\n\n /**\n * Include soft-deleted items\n *\n * @example\n * ```typescript\n * items.withDeleted()\n * ```\n */\n withDeleted(): this {\n this.queryParams.paranoid = false;\n return this;\n }\n\n /**\n * Filter related items in O2M/M2M relations\n *\n * @example\n * ```typescript\n * // Only show approved comments\n * items.relFilter({\n * comments: { approved: { eq: true } }\n * })\n * ```\n */\n relFilter(conditions: Record<string, Filter>): this {\n this.queryParams.relConditions = conditions;\n return this;\n }\n\n /**\n * Get the built query parameters\n */\n getQuery(): QueryParams {\n return { ...this.queryParams };\n }\n\n /**\n * Execute the query and return results\n *\n * @example\n * ```typescript\n * const { data, totalCount } = await items\n * .filter({ status: { eq: 'active' } })\n * .sort({ createdAt: 'desc' })\n * .limit(10)\n * .get();\n * ```\n */\n async get(): Promise<PaginatedResponse<T>> {\n return this.client.get<PaginatedResponse<T>>(`/items/${this.collection}`, {\n params: this.buildParams(),\n });\n }\n\n /**\n * Execute the query and return the first result\n *\n * @example\n * ```typescript\n * const item = await items\n * .filter({ slug: { eq: 'my-post' } })\n * .first();\n * ```\n */\n async first(): Promise<T | null> {\n const result = await this.limit(1).get();\n return result.data[0] || null;\n }\n\n /**\n * Count matching items\n *\n * @example\n * ```typescript\n * const count = await items.filter({ status: { eq: 'active' } }).count();\n * ```\n */\n async count(): Promise<number> {\n const result = await this.client.get<PaginatedResponse<T>>(\n `/items/${this.collection}`,\n {\n params: {\n ...this.buildParams(),\n limit: 0,\n },\n }\n );\n return result.totalCount || 0;\n }\n\n /**\n * Build query parameters for the request\n */\n private buildParams(): Record<string, unknown> {\n const params: Record<string, unknown> = {};\n\n if (this.queryParams.fields) {\n params.fields = this.queryParams.fields;\n }\n if (this.queryParams.filter) {\n params.filter = this.queryParams.filter;\n }\n if (this.queryParams.sort) {\n params.sort = this.queryParams.sort;\n }\n if (this.queryParams.limit !== undefined) {\n params.limit = this.queryParams.limit;\n }\n if (this.queryParams.page !== undefined) {\n params.page = this.queryParams.page;\n }\n if (this.queryParams.offset !== undefined) {\n params.offset = this.queryParams.offset;\n }\n if (this.queryParams.search) {\n params.search = this.queryParams.search;\n }\n if (this.queryParams.searchFields) {\n params.searchFields = this.queryParams.searchFields;\n }\n if (this.queryParams.paranoid !== undefined) {\n params.paranoid = this.queryParams.paranoid;\n }\n if (this.queryParams.relConditions) {\n params.relConditions = this.queryParams.relConditions;\n }\n if (this.queryParams.aggregate) {\n params.aggregate = this.queryParams.aggregate;\n }\n if (this.queryParams.groupBy) {\n params.groupBy = this.queryParams.groupBy;\n }\n\n return params;\n }\n}\n\n/**\n * Items module for CRUD operations on collections.\n *\n * @example\n * ```typescript\n * const items = baasix.items('products');\n *\n * // List items\n * const products = await items.find({ filter: { status: { eq: 'active' } } });\n *\n * // Get single item\n * const product = await items.findOne('product-id');\n *\n * // Create item\n * const newProduct = await items.create({ name: 'Widget', price: 29.99 });\n *\n * // Update item\n * await items.update('product-id', { price: 24.99 });\n *\n * // Delete item\n * await items.delete('product-id');\n * ```\n */\nexport class ItemsModule<T extends BaseItem = BaseItem> {\n private collection: string;\n private client: HttpClient;\n\n constructor(collection: string, config: ItemsModuleConfig) {\n this.collection = collection;\n this.client = config.client;\n }\n\n /**\n * Create a query builder for fluent query construction\n *\n * @example\n * ```typescript\n * const results = await baasix.items('posts')\n * .query()\n * .select('*', 'author.*')\n * .filter({ status: { eq: 'published' } })\n * .sort({ createdAt: 'desc' })\n * .limit(10)\n * .get();\n * ```\n */\n query(): QueryBuilder<T> {\n return new QueryBuilder<T>(this.collection, this.client);\n }\n\n /**\n * Find items with optional query parameters\n *\n * @example\n * ```typescript\n * // Simple query\n * const { data } = await items.find();\n *\n * // With parameters\n * const { data, totalCount } = await items.find({\n * filter: { status: { eq: 'active' } },\n * sort: { createdAt: 'desc' },\n * limit: 20,\n * page: 1,\n * fields: ['id', 'name', 'price']\n * });\n * ```\n */\n async find(params?: QueryParams): Promise<PaginatedResponse<T>> {\n return this.client.get<PaginatedResponse<T>>(`/items/${this.collection}`, {\n params: params as Record<string, unknown>,\n });\n }\n\n /**\n * Alias for find()\n */\n async findMany(params?: QueryParams): Promise<PaginatedResponse<T>> {\n return this.find(params);\n }\n\n /**\n * Find a single item by ID\n *\n * @example\n * ```typescript\n * const product = await items.findOne('product-uuid');\n *\n * // With specific fields\n * const product = await items.findOne('product-uuid', {\n * fields: ['id', 'name', 'category.*']\n * });\n * ```\n */\n async findOne(\n id: string,\n params?: Pick<QueryParams, \"fields\">\n ): Promise<T> {\n const response = await this.client.get<SingleResponse<T>>(\n `/items/${this.collection}/${id}`,\n { params: params as Record<string, unknown> }\n );\n return response.data;\n }\n\n /**\n * Alias for findOne()\n */\n async get(id: string, params?: Pick<QueryParams, \"fields\">): Promise<T> {\n return this.findOne(id, params);\n }\n\n /**\n * Create a new item\n *\n * @example\n * ```typescript\n * const id = await items.create({\n * name: 'New Product',\n * price: 29.99,\n * status: 'draft'\n * });\n * ```\n */\n async create(data: Partial<T>): Promise<string> {\n const response = await this.client.post<MutationResponse<string>>(\n `/items/${this.collection}`,\n data\n );\n return response.data;\n }\n\n /**\n * Alias for create()\n */\n async insert(data: Partial<T>): Promise<string> {\n return this.create(data);\n }\n\n /**\n * Create multiple items at once\n *\n * @example\n * ```typescript\n * const ids = await items.createMany([\n * { name: 'Product 1', price: 10 },\n * { name: 'Product 2', price: 20 }\n * ]);\n * ```\n */\n async createMany(data: Partial<T>[]): Promise<string[]> {\n const response = await this.client.post<BulkResponse<string[]>>(\n `/items/${this.collection}/bulk`,\n data\n );\n return response.data;\n }\n\n /**\n * Alias for createMany()\n */\n async insertMany(data: Partial<T>[]): Promise<string[]> {\n return this.createMany(data);\n }\n\n /**\n * Update an existing item\n *\n * @example\n * ```typescript\n * await items.update('product-uuid', {\n * price: 24.99,\n * status: 'published'\n * });\n * ```\n */\n async update(id: string, data: Partial<T>): Promise<string> {\n const response = await this.client.patch<MutationResponse<string>>(\n `/items/${this.collection}/${id}`,\n data\n );\n return response.data;\n }\n\n /**\n * Update multiple items at once\n *\n * @example\n * ```typescript\n * // Update by IDs with same data\n * await items.updateMany(['id1', 'id2'], { status: 'archived' });\n * ```\n */\n async updateMany(ids: string[], data: Partial<T>): Promise<string[]> {\n // Transform to array format expected by API: [{id, data}, {id, data}, ...]\n const updates = ids.map(id => ({ id, data }));\n const response = await this.client.patch<BulkResponse<string[]>>(\n `/items/${this.collection}/bulk`,\n updates\n );\n return response.data;\n }\n\n /**\n * Upsert an item (create if not exists, update if exists)\n *\n * @example\n * ```typescript\n * const id = await items.upsert(\n * { sku: 'WIDGET-001' },\n * { name: 'Widget', price: 29.99, sku: 'WIDGET-001' }\n * );\n * ```\n */\n async upsert(\n filter: Filter,\n data: Partial<T>\n ): Promise<string> {\n // Try to find existing\n const existing = await this.find({ filter, limit: 1 });\n if (existing.data.length > 0) {\n return this.update(existing.data[0].id, data);\n }\n return this.create(data);\n }\n\n /**\n * Delete an item by ID\n *\n * @example\n * ```typescript\n * await items.delete('product-uuid');\n * ```\n */\n async delete(id: string): Promise<void> {\n await this.client.delete<DeleteResponse>(`/items/${this.collection}/${id}`);\n }\n\n /**\n * Delete multiple items\n *\n * @example\n * ```typescript\n * await items.deleteMany(['id1', 'id2', 'id3']);\n * ```\n */\n async deleteMany(ids: string[]): Promise<void> {\n await this.client.delete<DeleteResponse>(`/items/${this.collection}/bulk`, {\n body: JSON.stringify(ids),\n });\n }\n\n /**\n * Soft delete an item (if paranoid mode is enabled)\n *\n * @example\n * ```typescript\n * await items.softDelete('product-uuid');\n * ```\n */\n async softDelete(id: string): Promise<void> {\n await this.update(id, { deletedAt: new Date().toISOString() } as unknown as Partial<T>);\n }\n\n /**\n * Restore a soft-deleted item\n *\n * @example\n * ```typescript\n * await items.restore('product-uuid');\n * ```\n */\n async restore(id: string): Promise<void> {\n await this.update(id, { deletedAt: null } as unknown as Partial<T>);\n }\n\n /**\n * Aggregate data with grouping\n *\n * @example\n * ```typescript\n * const results = await items.aggregate({\n * aggregate: {\n * total: { function: 'sum', field: 'amount' },\n * count: { function: 'count', field: 'id' },\n * avgPrice: { function: 'avg', field: 'price' }\n * },\n * groupBy: ['category', 'status'],\n * filter: { createdAt: { gte: '$NOW-DAYS_30' } }\n * });\n * ```\n */\n async aggregate(\n params: Pick<QueryParams, \"aggregate\" | \"groupBy\" | \"filter\">\n ): Promise<Record<string, unknown>[]> {\n const response = await this.client.get<PaginatedResponse<Record<string, unknown>>>(\n `/items/${this.collection}`,\n { params: params as Record<string, unknown> }\n );\n return response.data;\n }\n\n // ===================\n // Import Operations\n // ===================\n\n /**\n * Import items from a CSV file\n *\n * @example\n * ```typescript\n * // Browser\n * const fileInput = document.querySelector('input[type=\"file\"]');\n * const file = fileInput.files[0];\n * \n * const result = await baasix.items('products').importCSV(file);\n * \n * console.log(`Imported ${result.imported} items`);\n * ```\n */\n async importCSV(\n file: File | { uri: string; name: string; type: string }\n ): Promise<ImportResult> {\n const formData = new FormData();\n \n if (file instanceof File) {\n formData.append(\"csvFile\", file);\n } else {\n // React Native style file\n formData.append(\"csvFile\", file as any);\n }\n\n const response = await this.client.post<{ results: ImportResult }>(\n `/items/${this.collection}/import-csv`,\n formData\n );\n return response.results;\n }\n\n /**\n * Import items from a JSON file\n *\n * @example\n * ```typescript\n * const file = fileInput.files[0]; // JSON file\n * const result = await baasix.items('products').importJSON(file);\n * \n * console.log(`Imported ${result.imported} items`);\n * ```\n */\n async importJSON(\n file: File | { uri: string; name: string; type: string }\n ): Promise<ImportResult> {\n const formData = new FormData();\n \n if (file instanceof File) {\n formData.append(\"jsonFile\", file);\n } else {\n formData.append(\"jsonFile\", file as any);\n }\n\n const response = await this.client.post<{ results: ImportResult }>(\n `/items/${this.collection}/import-json`,\n formData\n );\n return response.results;\n }\n\n /**\n * Import items from an array of objects\n *\n * @example\n * ```typescript\n * const data = [\n * { name: 'Product 1', price: 29.99 },\n * { name: 'Product 2', price: 39.99 }\n * ];\n * \n * const result = await baasix.items('products').importData(data);\n * ```\n */\n async importData(data: Partial<T>[]): Promise<BulkResponse> {\n const response = await this.client.post<BulkResponse>(\n `/items/${this.collection}/bulk`,\n data\n );\n return response;\n }\n\n // ===================\n // Sort Operations\n // ===================\n\n /**\n * Sort/reorder items (move item before or after another)\n *\n * @example\n * ```typescript\n * // Move item1 before item2\n * await baasix.items('products').sortItem('item1-uuid', 'item2-uuid');\n * \n * // Move item1 after item2\n * await baasix.items('products').sortItem('item1-uuid', 'item2-uuid', 'after');\n * ```\n */\n async sortItem(\n itemId: string,\n targetItemId: string,\n mode: \"before\" | \"after\" = \"before\"\n ): Promise<void> {\n await this.client.post(`/utils/sort/${this.collection}`, {\n item: itemId,\n to: targetItemId,\n mode,\n });\n }\n\n /**\n * Reorder multiple items\n *\n * @example\n * ```typescript\n * // Set explicit order\n * await baasix.items('products').reorder([\n * 'item3-uuid',\n * 'item1-uuid',\n * 'item2-uuid'\n * ]);\n * ```\n */\n async reorder(orderedIds: string[]): Promise<void> {\n // Sort items one by one to achieve the desired order\n for (let i = 1; i < orderedIds.length; i++) {\n await this.client.post(`/utils/sort/${this.collection}`, {\n item: orderedIds[i],\n to: orderedIds[i - 1],\n mode: \"after\",\n });\n }\n }\n}\n\nexport interface ImportResult {\n imported: number;\n failed: number;\n errors: Array<{ row: number; data: Record<string, unknown>; error: string }>;\n}\n\n// Re-export types from types.ts\nexport type {\n BaseItem,\n BulkResponse,\n DeleteResponse,\n Filter,\n MutationResponse,\n PaginatedResponse,\n QueryParams,\n SingleResponse,\n Sort,\n};\n"]}