@decaf-ts/for-couchdb 0.3.1 → 0.3.3

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 (61) hide show
  1. package/LICENSE.md +646 -144
  2. package/README.md +371 -1
  3. package/dist/for-couchdb.cjs +733 -24
  4. package/dist/for-couchdb.esm.cjs +733 -24
  5. package/lib/adapter.cjs +186 -1
  6. package/lib/adapter.d.ts +234 -0
  7. package/lib/constants.cjs +29 -1
  8. package/lib/constants.d.ts +28 -0
  9. package/lib/errors.cjs +18 -1
  10. package/lib/errors.d.ts +17 -0
  11. package/lib/esm/adapter.d.ts +234 -0
  12. package/lib/esm/adapter.js +186 -1
  13. package/lib/esm/constants.d.ts +28 -0
  14. package/lib/esm/constants.js +29 -1
  15. package/lib/esm/errors.d.ts +17 -0
  16. package/lib/esm/errors.js +18 -1
  17. package/lib/esm/index.d.ts +6 -13
  18. package/lib/esm/index.js +7 -14
  19. package/lib/esm/indexes/generator.d.ts +47 -0
  20. package/lib/esm/indexes/generator.js +58 -1
  21. package/lib/esm/interfaces/CouchDBRepository.d.ts +10 -0
  22. package/lib/esm/interfaces/CouchDBRepository.js +1 -1
  23. package/lib/esm/model/CouchDBSequence.d.ts +19 -0
  24. package/lib/esm/model/CouchDBSequence.js +12 -1
  25. package/lib/esm/query/Paginator.d.ts +111 -0
  26. package/lib/esm/query/Paginator.js +117 -8
  27. package/lib/esm/query/Statement.d.ts +134 -0
  28. package/lib/esm/query/Statement.js +143 -1
  29. package/lib/esm/query/constants.d.ts +42 -0
  30. package/lib/esm/query/constants.js +43 -1
  31. package/lib/esm/query/translate.d.ts +31 -0
  32. package/lib/esm/query/translate.js +32 -1
  33. package/lib/esm/sequences/Sequence.d.ts +0 -2
  34. package/lib/esm/sequences/Sequence.js +2 -4
  35. package/lib/esm/types.d.ts +55 -12
  36. package/lib/esm/types.js +1 -1
  37. package/lib/esm/utils.d.ts +105 -0
  38. package/lib/esm/utils.js +106 -1
  39. package/lib/index.cjs +7 -14
  40. package/lib/index.d.ts +6 -13
  41. package/lib/indexes/generator.cjs +58 -1
  42. package/lib/indexes/generator.d.ts +47 -0
  43. package/lib/interfaces/CouchDBRepository.cjs +1 -1
  44. package/lib/interfaces/CouchDBRepository.d.ts +10 -0
  45. package/lib/model/CouchDBSequence.cjs +12 -1
  46. package/lib/model/CouchDBSequence.d.ts +19 -0
  47. package/lib/query/Paginator.cjs +117 -8
  48. package/lib/query/Paginator.d.ts +111 -0
  49. package/lib/query/Statement.cjs +143 -1
  50. package/lib/query/Statement.d.ts +134 -0
  51. package/lib/query/constants.cjs +43 -1
  52. package/lib/query/constants.d.ts +42 -0
  53. package/lib/query/translate.cjs +32 -1
  54. package/lib/query/translate.d.ts +31 -0
  55. package/lib/sequences/Sequence.cjs +2 -4
  56. package/lib/sequences/Sequence.d.ts +0 -2
  57. package/lib/types.cjs +1 -1
  58. package/lib/types.d.ts +55 -12
  59. package/lib/utils.cjs +106 -1
  60. package/lib/utils.d.ts +105 -0
  61. package/package.json +2 -2
package/lib/adapter.cjs CHANGED
@@ -18,6 +18,49 @@ const Sequence_1 = require("./sequences/Sequence.cjs");
18
18
  const errors_1 = require("./errors.cjs");
19
19
  const query_1 = require("./query/index.cjs");
20
20
  const core_2 = require("@decaf-ts/core");
21
+ /**
22
+ * @description Abstract adapter for CouchDB database operations
23
+ * @summary Provides a base implementation for CouchDB database operations, including CRUD operations, sequence management, and error handling
24
+ * @template Y - The scope type
25
+ * @template F - The repository flags type
26
+ * @template C - The context type
27
+ * @param {Y} scope - The scope for the adapter
28
+ * @param {string} flavour - The flavour of the adapter
29
+ * @param {string} [alias] - Optional alias for the adapter
30
+ * @class
31
+ * @example
32
+ * // Example of extending CouchDBAdapter
33
+ * class MyCouchDBAdapter extends CouchDBAdapter<MyScope, MyFlags, MyContext> {
34
+ * constructor(scope: MyScope) {
35
+ * super(scope, 'my-couchdb', 'my-alias');
36
+ * }
37
+ *
38
+ * // Implement abstract methods
39
+ * async index<M extends Model>(...models: Constructor<M>[]): Promise<void> {
40
+ * // Implementation
41
+ * }
42
+ *
43
+ * async raw<R>(rawInput: MangoQuery, docsOnly: boolean): Promise<R> {
44
+ * // Implementation
45
+ * }
46
+ *
47
+ * async create(tableName: string, id: string | number, model: Record<string, any>, ...args: any[]): Promise<Record<string, any>> {
48
+ * // Implementation
49
+ * }
50
+ *
51
+ * async read(tableName: string, id: string | number, ...args: any[]): Promise<Record<string, any>> {
52
+ * // Implementation
53
+ * }
54
+ *
55
+ * async update(tableName: string, id: string | number, model: Record<string, any>, ...args: any[]): Promise<Record<string, any>> {
56
+ * // Implementation
57
+ * }
58
+ *
59
+ * async delete(tableName: string, id: string | number, ...args: any[]): Promise<Record<string, any>> {
60
+ * // Implementation
61
+ * }
62
+ * }
63
+ */
21
64
  class CouchDBAdapter extends core_1.Adapter {
22
65
  constructor(scope, flavour, alias) {
23
66
  super(scope, flavour, alias);
@@ -26,16 +69,40 @@ class CouchDBAdapter extends core_1.Adapter {
26
69
  (0, db_decorators_1.prefixMethod)(this, m, this[name + "Prefix"]);
27
70
  });
28
71
  }
72
+ /**
73
+ * @description Creates a new CouchDB statement for querying
74
+ * @summary Factory method that creates a new CouchDBStatement instance for building queries
75
+ * @template M - The model type
76
+ * @return {CouchDBStatement<M, any>} A new CouchDBStatement instance
77
+ */
29
78
  Statement() {
30
79
  return new query_1.CouchDBStatement(this);
31
80
  }
81
+ /**
82
+ * @description Creates a new CouchDB sequence
83
+ * @summary Factory method that creates a new CouchDBSequence instance for managing sequences
84
+ * @param {SequenceOptions} options - The options for the sequence
85
+ * @return {Promise<Sequence>} A promise that resolves to a new Sequence instance
86
+ */
32
87
  async Sequence(options) {
33
88
  return new Sequence_1.CouchDBSequence(options, this);
34
89
  }
90
+ /**
91
+ * @description Initializes the adapter by creating indexes for all managed models
92
+ * @summary Sets up the necessary database indexes for all models managed by this adapter
93
+ * @return {Promise<void>} A promise that resolves when initialization is complete
94
+ */
35
95
  async initialize() {
36
96
  const managedModels = core_1.Adapter.models(this.flavour);
37
97
  return this.index(...managedModels);
38
98
  }
99
+ /**
100
+ * @description Assigns metadata to a model
101
+ * @summary Adds revision metadata to a model as a non-enumerable property
102
+ * @param {Record<string, any>} model - The model to assign metadata to
103
+ * @param {string} rev - The revision string to assign
104
+ * @return {Record<string, any>} The model with metadata assigned
105
+ */
39
106
  assignMetadata(model, rev) {
40
107
  Object.defineProperty(model, core_1.PersistenceKeys.METADATA, {
41
108
  enumerable: false,
@@ -45,6 +112,13 @@ class CouchDBAdapter extends core_1.Adapter {
45
112
  });
46
113
  return model;
47
114
  }
115
+ /**
116
+ * @description Assigns metadata to multiple models
117
+ * @summary Adds revision metadata to multiple models as non-enumerable properties
118
+ * @param models - The models to assign metadata to
119
+ * @param {string[]} revs - The revision strings to assign
120
+ * @return The models with metadata assigned
121
+ */
48
122
  assignMultipleMetadata(models, revs) {
49
123
  models.forEach((m, i) => {
50
124
  core_1.Repository.setMetadata(m, revs[i]);
@@ -52,6 +126,14 @@ class CouchDBAdapter extends core_1.Adapter {
52
126
  });
53
127
  return models;
54
128
  }
129
+ /**
130
+ * @description Prepares a record for creation
131
+ * @summary Adds necessary CouchDB fields to a record before creation
132
+ * @param {string} tableName - The name of the table
133
+ * @param {string|number} id - The ID of the record
134
+ * @param {Record<string, any>} model - The model to prepare
135
+ * @return A tuple containing the tableName, id, and prepared record
136
+ */
55
137
  createPrefix(tableName, id, model) {
56
138
  const record = {};
57
139
  record[constants_1.CouchDBKeys.TABLE] = tableName;
@@ -59,6 +141,15 @@ class CouchDBAdapter extends core_1.Adapter {
59
141
  Object.assign(record, model);
60
142
  return [tableName, id, record];
61
143
  }
144
+ /**
145
+ * @description Prepares multiple records for creation
146
+ * @summary Adds necessary CouchDB fields to multiple records before creation
147
+ * @param {string} tableName - The name of the table
148
+ * @param {string[]|number[]} ids - The IDs of the records
149
+ * @param models - The models to prepare
150
+ * @return A tuple containing the tableName, ids, and prepared records
151
+ * @throws {InternalError} If ids and models arrays have different lengths
152
+ */
62
153
  createAllPrefix(tableName, ids, models) {
63
154
  if (ids.length !== models.length)
64
155
  throw new db_decorators_1.InternalError("Ids and models must have the same length");
@@ -71,6 +162,15 @@ class CouchDBAdapter extends core_1.Adapter {
71
162
  });
72
163
  return [tableName, ids, records];
73
164
  }
165
+ /**
166
+ * @description Prepares a record for update
167
+ * @summary Adds necessary CouchDB fields to a record before update
168
+ * @param {string} tableName - The name of the table
169
+ * @param {string|number} id - The ID of the record
170
+ * @param model - The model to prepare
171
+ * @return A tuple containing the tableName, id, and prepared record
172
+ * @throws {InternalError} If no revision number is found in the model
173
+ */
74
174
  updatePrefix(tableName, id, model) {
75
175
  const record = {};
76
176
  record[constants_1.CouchDBKeys.TABLE] = tableName;
@@ -82,6 +182,15 @@ class CouchDBAdapter extends core_1.Adapter {
82
182
  record[constants_1.CouchDBKeys.REV] = rev;
83
183
  return [tableName, id, record];
84
184
  }
185
+ /**
186
+ * @description Prepares multiple records for update
187
+ * @summary Adds necessary CouchDB fields to multiple records before update
188
+ * @param {string} tableName - The name of the table
189
+ * @param {string[]|number[]} ids - The IDs of the records
190
+ * @param models - The models to prepare
191
+ * @return A tuple containing the tableName, ids, and prepared records
192
+ * @throws {InternalError} If ids and models arrays have different lengths or if no revision number is found in a model
193
+ */
85
194
  updateAllPrefix(tableName, ids, models) {
86
195
  if (ids.length !== models.length)
87
196
  throw new db_decorators_1.InternalError("Ids and models must have the same length");
@@ -98,15 +207,91 @@ class CouchDBAdapter extends core_1.Adapter {
98
207
  });
99
208
  return [tableName, ids, records];
100
209
  }
210
+ /**
211
+ * @description Generates a CouchDB document ID
212
+ * @summary Combines the table name and ID to create a CouchDB document ID
213
+ * @param {string} tableName - The name of the table
214
+ * @param {string|number} id - The ID of the record
215
+ * @return {string} The generated CouchDB document ID
216
+ */
101
217
  generateId(tableName, id) {
102
218
  return [tableName, id].join(constants_1.CouchDBKeys.SEPARATOR);
103
219
  }
220
+ /**
221
+ * @description Parses an error and converts it to a BaseError
222
+ * @summary Converts various error types to appropriate BaseError subtypes
223
+ * @param {Error|string} err - The error to parse
224
+ * @param {string} [reason] - Optional reason for the error
225
+ * @return {BaseError} The parsed error as a BaseError
226
+ */
104
227
  parseError(err, reason) {
105
228
  return CouchDBAdapter.parseError(err, reason);
106
229
  }
230
+ /**
231
+ * @description Checks if an attribute is reserved
232
+ * @summary Determines if an attribute name is reserved in CouchDB
233
+ * @param {string} attr - The attribute name to check
234
+ * @return {boolean} True if the attribute is reserved, false otherwise
235
+ */
107
236
  isReserved(attr) {
108
237
  return !!attr.match(constants_1.reservedAttributes);
109
238
  }
239
+ /**
240
+ * @description Static method to parse an error and convert it to a BaseError
241
+ * @summary Converts various error types to appropriate BaseError subtypes based on error codes and messages
242
+ * @param {Error|string} err - The error to parse
243
+ * @param {string} [reason] - Optional reason for the error
244
+ * @return {BaseError} The parsed error as a BaseError
245
+ * @mermaid
246
+ * sequenceDiagram
247
+ * participant Caller
248
+ * participant parseError
249
+ * participant ErrorTypes
250
+ *
251
+ * Caller->>parseError: err, reason
252
+ * Note over parseError: Check if err is already a BaseError
253
+ * alt err is BaseError
254
+ * parseError-->>Caller: return err
255
+ * else err is string
256
+ * Note over parseError: Extract code from string
257
+ * alt code matches "already exist|update conflict"
258
+ * parseError->>ErrorTypes: new ConflictError(code)
259
+ * ErrorTypes-->>Caller: ConflictError
260
+ * else code matches "missing|deleted"
261
+ * parseError->>ErrorTypes: new NotFoundError(code)
262
+ * ErrorTypes-->>Caller: NotFoundError
263
+ * end
264
+ * else err has code property
265
+ * Note over parseError: Extract code and reason
266
+ * else err has statusCode property
267
+ * Note over parseError: Extract code and reason
268
+ * else
269
+ * Note over parseError: Use err.message as code
270
+ * end
271
+ *
272
+ * Note over parseError: Switch on code
273
+ * alt code is 401, 412, or 409
274
+ * parseError->>ErrorTypes: new ConflictError(reason)
275
+ * ErrorTypes-->>Caller: ConflictError
276
+ * else code is 404
277
+ * parseError->>ErrorTypes: new NotFoundError(reason)
278
+ * ErrorTypes-->>Caller: NotFoundError
279
+ * else code is 400
280
+ * alt code matches "No index exists"
281
+ * parseError->>ErrorTypes: new IndexError(err)
282
+ * ErrorTypes-->>Caller: IndexError
283
+ * else
284
+ * parseError->>ErrorTypes: new InternalError(err)
285
+ * ErrorTypes-->>Caller: InternalError
286
+ * end
287
+ * else code matches "ECONNREFUSED"
288
+ * parseError->>ErrorTypes: new ConnectionError(err)
289
+ * ErrorTypes-->>Caller: ConnectionError
290
+ * else
291
+ * parseError->>ErrorTypes: new InternalError(err)
292
+ * ErrorTypes-->>Caller: InternalError
293
+ * end
294
+ */
110
295
  static parseError(err, reason) {
111
296
  if (err instanceof db_decorators_1.BaseError)
112
297
  return err;
@@ -196,4 +381,4 @@ __decorate([
196
381
  __metadata("design:paramtypes", [String, Array, Array]),
197
382
  __metadata("design:returntype", void 0)
198
383
  ], CouchDBAdapter.prototype, "updateAllPrefix", null);
199
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"adapter.js","sourceRoot":"","sources":["../src/adapter.ts"],"names":[],"mappings":"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","sourcesContent":["import {\n  Adapter,\n  Sequence,\n  type SequenceOptions,\n  PersistenceKeys,\n  ConnectionError,\n  Repository,\n} from \"@decaf-ts/core\";\nimport { CouchDBKeys, reservedAttributes } from \"./constants\";\nimport {\n  BaseError,\n  ConflictError,\n  Context,\n  InternalError,\n  NotFoundError,\n  prefixMethod,\n  RepositoryFlags,\n} from \"@decaf-ts/db-decorators\";\nimport \"reflect-metadata\";\n\nimport { CouchDBSequence } from \"./sequences/Sequence\";\nimport { Constructor, Model } from \"@decaf-ts/decorator-validation\";\nimport { IndexError } from \"./errors\";\nimport { MangoQuery } from \"./types\";\nimport { CouchDBStatement } from \"./query\";\nimport { final } from \"@decaf-ts/core\";\n\nexport abstract class CouchDBAdapter<\n  Y,\n  F extends RepositoryFlags,\n  C extends Context<F>,\n> extends Adapter<Y, MangoQuery, F, C> {\n  protected constructor(scope: Y, flavour: string, alias?: string) {\n    super(scope, flavour, alias);\n    [this.create, this.createAll, this.update, this.updateAll].forEach((m) => {\n      const name = m.name;\n      prefixMethod(this, m, (this as any)[name + \"Prefix\"]);\n    });\n  }\n\n  @final()\n  Statement<M extends Model>(): CouchDBStatement<M, any> {\n    return new CouchDBStatement(this);\n  }\n\n  @final()\n  async Sequence(options: SequenceOptions): Promise<Sequence> {\n    return new CouchDBSequence(options, this);\n  }\n\n  async initialize(): Promise<void> {\n    const managedModels = Adapter.models(this.flavour);\n    return this.index(...managedModels);\n  }\n\n  protected abstract index<M extends Model>(\n    ...models: Constructor<M>[]\n  ): Promise<void>;\n\n  abstract override raw<R>(rawInput: MangoQuery, docsOnly: boolean): Promise<R>;\n\n  @final()\n  protected assignMetadata(\n    model: Record<string, any>,\n    rev: string\n  ): Record<string, any> {\n    Object.defineProperty(model, PersistenceKeys.METADATA, {\n      enumerable: false,\n      configurable: false,\n      writable: false,\n      value: rev,\n    });\n    return model;\n  }\n\n  @final()\n  protected assignMultipleMetadata(\n    models: Record<string, any>[],\n    revs: string[]\n  ): Record<string, any>[] {\n    models.forEach((m, i) => {\n      Repository.setMetadata(m as any, revs[i]);\n      return m;\n    });\n    return models;\n  }\n\n  @final()\n  protected createPrefix(\n    tableName: string,\n    id: string | number,\n    model: Record<string, any>\n  ) {\n    const record: Record<string, any> = {};\n    record[CouchDBKeys.TABLE] = tableName;\n    record[CouchDBKeys.ID] = this.generateId(tableName, id);\n    Object.assign(record, model);\n    return [tableName, id, record];\n  }\n\n  abstract override create(\n    tableName: string,\n    id: string | number,\n    model: Record<string, any>,\n    ...args: any[]\n  ): Promise<Record<string, any>>;\n\n  @final()\n  protected createAllPrefix(\n    tableName: string,\n    ids: string[] | number[],\n    models: Record<string, any>[]\n  ) {\n    if (ids.length !== models.length)\n      throw new InternalError(\"Ids and models must have the same length\");\n\n    const records = ids.map((id, count) => {\n      const record: Record<string, any> = {};\n      record[CouchDBKeys.TABLE] = tableName;\n      record[CouchDBKeys.ID] = this.generateId(tableName, id);\n      Object.assign(record, models[count]);\n      return record;\n    });\n    return [tableName, ids, records];\n  }\n\n  abstract override read(\n    tableName: string,\n    id: string | number,\n    ...args: any[]\n  ): Promise<Record<string, any>>;\n\n  @final()\n  updatePrefix(\n    tableName: string,\n    id: string | number,\n    model: Record<string, any>\n  ) {\n    const record: Record<string, any> = {};\n    record[CouchDBKeys.TABLE] = tableName;\n    record[CouchDBKeys.ID] = this.generateId(tableName, id);\n    const rev = model[PersistenceKeys.METADATA];\n    if (!rev)\n      throw new InternalError(\n        `No revision number found for record with id ${id}`\n      );\n    Object.assign(record, model);\n    record[CouchDBKeys.REV] = rev;\n    return [tableName, id, record];\n  }\n\n  abstract override update(\n    tableName: string,\n    id: string | number,\n    model: Record<string, any>,\n    ...args: any[]\n  ): Promise<Record<string, any>>;\n\n  @final()\n  protected updateAllPrefix(\n    tableName: string,\n    ids: string[] | number[],\n    models: Record<string, any>[]\n  ) {\n    if (ids.length !== models.length)\n      throw new InternalError(\"Ids and models must have the same length\");\n\n    const records = ids.map((id, count) => {\n      const record: Record<string, any> = {};\n      record[CouchDBKeys.TABLE] = tableName;\n      record[CouchDBKeys.ID] = this.generateId(tableName, id);\n      const rev = models[count][PersistenceKeys.METADATA];\n      if (!rev)\n        throw new InternalError(\n          `No revision number found for record with id ${id}`\n        );\n      Object.assign(record, models[count]);\n      record[CouchDBKeys.REV] = rev;\n      return record;\n    });\n    return [tableName, ids, records];\n  }\n\n  abstract override delete(\n    tableName: string,\n    id: string | number,\n    ...args: any[]\n  ): Promise<Record<string, any>>;\n\n  protected generateId(tableName: string, id: string | number) {\n    return [tableName, id].join(CouchDBKeys.SEPARATOR);\n  }\n\n  parseError(err: Error | string, reason?: string): BaseError {\n    return CouchDBAdapter.parseError(err, reason);\n  }\n\n  protected override isReserved(attr: string): boolean {\n    return !!attr.match(reservedAttributes);\n  }\n\n  protected static parseError(err: Error | string, reason?: string): BaseError {\n    if (err instanceof BaseError) return err as any;\n    let code: string = \"\";\n    if (typeof err === \"string\") {\n      code = err;\n      if (code.match(/already exist|update conflict/g))\n        return new ConflictError(code);\n      if (code.match(/missing|deleted/g)) return new NotFoundError(code);\n    } else if ((err as any).code) {\n      code = (err as any).code;\n      reason = reason || err.message;\n    } else if ((err as any).statusCode) {\n      code = (err as any).statusCode;\n      reason = reason || err.message;\n    } else {\n      code = err.message;\n    }\n\n    switch (code.toString()) {\n      case \"401\":\n      case \"412\":\n      case \"409\":\n        return new ConflictError(reason as string);\n      case \"404\":\n        return new NotFoundError(reason as string);\n      case \"400\":\n        if (code.toString().match(/No\\sindex\\sexists/g))\n          return new IndexError(err);\n        return new InternalError(err);\n      default:\n        if (code.toString().match(/ECONNREFUSED/g))\n          return new ConnectionError(err);\n        return new InternalError(err);\n    }\n  }\n}\n"]}
384
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"adapter.js","sourceRoot":"","sources":["../src/adapter.ts"],"names":[],"mappings":"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","sourcesContent":["import {\n  Adapter,\n  Sequence,\n  type SequenceOptions,\n  PersistenceKeys,\n  ConnectionError,\n  Repository,\n} from \"@decaf-ts/core\";\nimport { CouchDBKeys, reservedAttributes } from \"./constants\";\nimport {\n  BaseError,\n  ConflictError,\n  Context,\n  InternalError,\n  NotFoundError,\n  prefixMethod,\n  RepositoryFlags,\n} from \"@decaf-ts/db-decorators\";\nimport \"reflect-metadata\";\n\nimport { CouchDBSequence } from \"./sequences/Sequence\";\nimport { Constructor, Model } from \"@decaf-ts/decorator-validation\";\nimport { IndexError } from \"./errors\";\nimport { MangoQuery } from \"./types\";\nimport { CouchDBStatement } from \"./query\";\nimport { final } from \"@decaf-ts/core\";\n\n/**\n * @description Abstract adapter for CouchDB database operations\n * @summary Provides a base implementation for CouchDB database operations, including CRUD operations, sequence management, and error handling\n * @template Y - The scope type\n * @template F - The repository flags type\n * @template C - The context type\n * @param {Y} scope - The scope for the adapter\n * @param {string} flavour - The flavour of the adapter\n * @param {string} [alias] - Optional alias for the adapter\n * @class\n * @example\n * // Example of extending CouchDBAdapter\n * class MyCouchDBAdapter extends CouchDBAdapter<MyScope, MyFlags, MyContext> {\n *   constructor(scope: MyScope) {\n *     super(scope, 'my-couchdb', 'my-alias');\n *   }\n *\n *   // Implement abstract methods\n *   async index<M extends Model>(...models: Constructor<M>[]): Promise<void> {\n *     // Implementation\n *   }\n *\n *   async raw<R>(rawInput: MangoQuery, docsOnly: boolean): Promise<R> {\n *     // Implementation\n *   }\n *\n *   async create(tableName: string, id: string | number, model: Record<string, any>, ...args: any[]): Promise<Record<string, any>> {\n *     // Implementation\n *   }\n *\n *   async read(tableName: string, id: string | number, ...args: any[]): Promise<Record<string, any>> {\n *     // Implementation\n *   }\n *\n *   async update(tableName: string, id: string | number, model: Record<string, any>, ...args: any[]): Promise<Record<string, any>> {\n *     // Implementation\n *   }\n *\n *   async delete(tableName: string, id: string | number, ...args: any[]): Promise<Record<string, any>> {\n *     // Implementation\n *   }\n * }\n */\nexport abstract class CouchDBAdapter<\n  Y,\n  F extends RepositoryFlags,\n  C extends Context<F>,\n> extends Adapter<Y, MangoQuery, F, C> {\n  protected constructor(scope: Y, flavour: string, alias?: string) {\n    super(scope, flavour, alias);\n    [this.create, this.createAll, this.update, this.updateAll].forEach((m) => {\n      const name = m.name;\n      prefixMethod(this, m, (this as any)[name + \"Prefix\"]);\n    });\n  }\n\n  /**\n   * @description Creates a new CouchDB statement for querying\n   * @summary Factory method that creates a new CouchDBStatement instance for building queries\n   * @template M - The model type\n   * @return {CouchDBStatement<M, any>} A new CouchDBStatement instance\n   */\n  @final()\n  Statement<M extends Model>(): CouchDBStatement<M, any> {\n    return new CouchDBStatement(this);\n  }\n\n  /**\n   * @description Creates a new CouchDB sequence\n   * @summary Factory method that creates a new CouchDBSequence instance for managing sequences\n   * @param {SequenceOptions} options - The options for the sequence\n   * @return {Promise<Sequence>} A promise that resolves to a new Sequence instance\n   */\n  @final()\n  async Sequence(options: SequenceOptions): Promise<Sequence> {\n    return new CouchDBSequence(options, this);\n  }\n\n  /**\n   * @description Initializes the adapter by creating indexes for all managed models\n   * @summary Sets up the necessary database indexes for all models managed by this adapter\n   * @return {Promise<void>} A promise that resolves when initialization is complete\n   */\n  async initialize(): Promise<void> {\n    const managedModels = Adapter.models(this.flavour);\n    return this.index(...managedModels);\n  }\n\n  /**\n   * @description Creates indexes for the given models\n   * @summary Abstract method that must be implemented to create database indexes for the specified models\n   * @template M - The model type\n   * @param {...Constructor<M>} models - The model constructors to create indexes for\n   * @return {Promise<void>} A promise that resolves when all indexes are created\n   */\n  protected abstract index<M extends Model>(\n    ...models: Constructor<M>[]\n  ): Promise<void>;\n\n  /**\n   * @description Executes a raw Mango query against the database\n   * @summary Abstract method that must be implemented to execute raw Mango queries\n   * @template R - The result type\n   * @param {MangoQuery} rawInput - The raw Mango query to execute\n   * @param {boolean} docsOnly - Whether to return only the documents or the full response\n   * @return {Promise<R>} A promise that resolves to the query result\n   */\n  abstract override raw<R>(rawInput: MangoQuery, docsOnly: boolean): Promise<R>;\n\n  /**\n   * @description Assigns metadata to a model\n   * @summary Adds revision metadata to a model as a non-enumerable property\n   * @param {Record<string, any>} model - The model to assign metadata to\n   * @param {string} rev - The revision string to assign\n   * @return {Record<string, any>} The model with metadata assigned\n   */\n  @final()\n  protected assignMetadata(\n    model: Record<string, any>,\n    rev: string\n  ): Record<string, any> {\n    Object.defineProperty(model, PersistenceKeys.METADATA, {\n      enumerable: false,\n      configurable: false,\n      writable: false,\n      value: rev,\n    });\n    return model;\n  }\n\n  /**\n   * @description Assigns metadata to multiple models\n   * @summary Adds revision metadata to multiple models as non-enumerable properties\n   * @param models - The models to assign metadata to\n   * @param {string[]} revs - The revision strings to assign\n   * @return The models with metadata assigned\n   */\n  @final()\n  protected assignMultipleMetadata(\n    models: Record<string, any>[],\n    revs: string[]\n  ): Record<string, any>[] {\n    models.forEach((m, i) => {\n      Repository.setMetadata(m as any, revs[i]);\n      return m;\n    });\n    return models;\n  }\n\n  /**\n   * @description Prepares a record for creation\n   * @summary Adds necessary CouchDB fields to a record before creation\n   * @param {string} tableName - The name of the table\n   * @param {string|number} id - The ID of the record\n   * @param {Record<string, any>} model - The model to prepare\n   * @return A tuple containing the tableName, id, and prepared record\n   */\n  @final()\n  protected createPrefix(\n    tableName: string,\n    id: string | number,\n    model: Record<string, any>\n  ) {\n    const record: Record<string, any> = {};\n    record[CouchDBKeys.TABLE] = tableName;\n    record[CouchDBKeys.ID] = this.generateId(tableName, id);\n    Object.assign(record, model);\n    return [tableName, id, record];\n  }\n\n  /**\n   * @description Creates a new record in the database\n   * @summary Abstract method that must be implemented to create a new record\n   * @param {string} tableName - The name of the table\n   * @param {string|number} id - The ID of the record\n   * @param {Record<string, any>} model - The model to create\n   * @param {...any[]} args - Additional arguments\n   * @return {Promise<Record<string, any>>} A promise that resolves to the created record\n   */\n  abstract override create(\n    tableName: string,\n    id: string | number,\n    model: Record<string, any>,\n    ...args: any[]\n  ): Promise<Record<string, any>>;\n\n  /**\n   * @description Prepares multiple records for creation\n   * @summary Adds necessary CouchDB fields to multiple records before creation\n   * @param {string} tableName - The name of the table\n   * @param {string[]|number[]} ids - The IDs of the records\n   * @param models - The models to prepare\n   * @return A tuple containing the tableName, ids, and prepared records\n   * @throws {InternalError} If ids and models arrays have different lengths\n   */\n  @final()\n  protected createAllPrefix(\n    tableName: string,\n    ids: string[] | number[],\n    models: Record<string, any>[]\n  ) {\n    if (ids.length !== models.length)\n      throw new InternalError(\"Ids and models must have the same length\");\n\n    const records = ids.map((id, count) => {\n      const record: Record<string, any> = {};\n      record[CouchDBKeys.TABLE] = tableName;\n      record[CouchDBKeys.ID] = this.generateId(tableName, id);\n      Object.assign(record, models[count]);\n      return record;\n    });\n    return [tableName, ids, records];\n  }\n\n  /**\n   * @description Reads a record from the database\n   * @summary Abstract method that must be implemented to read a record\n   * @param {string} tableName - The name of the table\n   * @param {string|number} id - The ID of the record\n   * @param {...any[]} args - Additional arguments\n   * @return {Promise<Record<string, any>>} A promise that resolves to the read record\n   */\n  abstract override read(\n    tableName: string,\n    id: string | number,\n    ...args: any[]\n  ): Promise<Record<string, any>>;\n\n  /**\n   * @description Prepares a record for update\n   * @summary Adds necessary CouchDB fields to a record before update\n   * @param {string} tableName - The name of the table\n   * @param {string|number} id - The ID of the record\n   * @param model - The model to prepare\n   * @return A tuple containing the tableName, id, and prepared record\n   * @throws {InternalError} If no revision number is found in the model\n   */\n  @final()\n  updatePrefix(\n    tableName: string,\n    id: string | number,\n    model: Record<string, any>\n  ) {\n    const record: Record<string, any> = {};\n    record[CouchDBKeys.TABLE] = tableName;\n    record[CouchDBKeys.ID] = this.generateId(tableName, id);\n    const rev = model[PersistenceKeys.METADATA];\n    if (!rev)\n      throw new InternalError(\n        `No revision number found for record with id ${id}`\n      );\n    Object.assign(record, model);\n    record[CouchDBKeys.REV] = rev;\n    return [tableName, id, record];\n  }\n\n  /**\n   * @description Updates a record in the database\n   * @summary Abstract method that must be implemented to update a record\n   * @param {string} tableName - The name of the table\n   * @param {string|number} id - The ID of the record\n   * @param {Record<string, any>} model - The model to update\n   * @param {any[]} args - Additional arguments\n   * @return A promise that resolves to the updated record\n   */\n  abstract override update(\n    tableName: string,\n    id: string | number,\n    model: Record<string, any>,\n    ...args: any[]\n  ): Promise<Record<string, any>>;\n\n  /**\n   * @description Prepares multiple records for update\n   * @summary Adds necessary CouchDB fields to multiple records before update\n   * @param {string} tableName - The name of the table\n   * @param {string[]|number[]} ids - The IDs of the records\n   * @param models - The models to prepare\n   * @return A tuple containing the tableName, ids, and prepared records\n   * @throws {InternalError} If ids and models arrays have different lengths or if no revision number is found in a model\n   */\n  @final()\n  protected updateAllPrefix(\n    tableName: string,\n    ids: string[] | number[],\n    models: Record<string, any>[]\n  ) {\n    if (ids.length !== models.length)\n      throw new InternalError(\"Ids and models must have the same length\");\n\n    const records = ids.map((id, count) => {\n      const record: Record<string, any> = {};\n      record[CouchDBKeys.TABLE] = tableName;\n      record[CouchDBKeys.ID] = this.generateId(tableName, id);\n      const rev = models[count][PersistenceKeys.METADATA];\n      if (!rev)\n        throw new InternalError(\n          `No revision number found for record with id ${id}`\n        );\n      Object.assign(record, models[count]);\n      record[CouchDBKeys.REV] = rev;\n      return record;\n    });\n    return [tableName, ids, records];\n  }\n\n  /**\n   * @description Deletes a record from the database\n   * @summary Abstract method that must be implemented to delete a record\n   * @param {string} tableName - The name of the table\n   * @param {string|number} id - The ID of the record\n   * @param {any[]} args - Additional arguments\n   * @return A promise that resolves to the deleted record\n   */\n  abstract override delete(\n    tableName: string,\n    id: string | number,\n    ...args: any[]\n  ): Promise<Record<string, any>>;\n\n  /**\n   * @description Generates a CouchDB document ID\n   * @summary Combines the table name and ID to create a CouchDB document ID\n   * @param {string} tableName - The name of the table\n   * @param {string|number} id - The ID of the record\n   * @return {string} The generated CouchDB document ID\n   */\n  protected generateId(tableName: string, id: string | number) {\n    return [tableName, id].join(CouchDBKeys.SEPARATOR);\n  }\n\n  /**\n   * @description Parses an error and converts it to a BaseError\n   * @summary Converts various error types to appropriate BaseError subtypes\n   * @param {Error|string} err - The error to parse\n   * @param {string} [reason] - Optional reason for the error\n   * @return {BaseError} The parsed error as a BaseError\n   */\n  parseError(err: Error | string, reason?: string): BaseError {\n    return CouchDBAdapter.parseError(err, reason);\n  }\n\n  /**\n   * @description Checks if an attribute is reserved\n   * @summary Determines if an attribute name is reserved in CouchDB\n   * @param {string} attr - The attribute name to check\n   * @return {boolean} True if the attribute is reserved, false otherwise\n   */\n  protected override isReserved(attr: string): boolean {\n    return !!attr.match(reservedAttributes);\n  }\n\n  /**\n   * @description Static method to parse an error and convert it to a BaseError\n   * @summary Converts various error types to appropriate BaseError subtypes based on error codes and messages\n   * @param {Error|string} err - The error to parse\n   * @param {string} [reason] - Optional reason for the error\n   * @return {BaseError} The parsed error as a BaseError\n   * @mermaid\n   * sequenceDiagram\n   *   participant Caller\n   *   participant parseError\n   *   participant ErrorTypes\n   *\n   *   Caller->>parseError: err, reason\n   *   Note over parseError: Check if err is already a BaseError\n   *   alt err is BaseError\n   *     parseError-->>Caller: return err\n   *   else err is string\n   *     Note over parseError: Extract code from string\n   *     alt code matches \"already exist|update conflict\"\n   *       parseError->>ErrorTypes: new ConflictError(code)\n   *       ErrorTypes-->>Caller: ConflictError\n   *     else code matches \"missing|deleted\"\n   *       parseError->>ErrorTypes: new NotFoundError(code)\n   *       ErrorTypes-->>Caller: NotFoundError\n   *     end\n   *   else err has code property\n   *     Note over parseError: Extract code and reason\n   *   else err has statusCode property\n   *     Note over parseError: Extract code and reason\n   *   else\n   *     Note over parseError: Use err.message as code\n   *   end\n   *\n   *   Note over parseError: Switch on code\n   *   alt code is 401, 412, or 409\n   *     parseError->>ErrorTypes: new ConflictError(reason)\n   *     ErrorTypes-->>Caller: ConflictError\n   *   else code is 404\n   *     parseError->>ErrorTypes: new NotFoundError(reason)\n   *     ErrorTypes-->>Caller: NotFoundError\n   *   else code is 400\n   *     alt code matches \"No index exists\"\n   *       parseError->>ErrorTypes: new IndexError(err)\n   *       ErrorTypes-->>Caller: IndexError\n   *     else\n   *       parseError->>ErrorTypes: new InternalError(err)\n   *       ErrorTypes-->>Caller: InternalError\n   *     end\n   *   else code matches \"ECONNREFUSED\"\n   *     parseError->>ErrorTypes: new ConnectionError(err)\n   *     ErrorTypes-->>Caller: ConnectionError\n   *   else\n   *     parseError->>ErrorTypes: new InternalError(err)\n   *     ErrorTypes-->>Caller: InternalError\n   *   end\n   */\n  protected static parseError(err: Error | string, reason?: string): BaseError {\n    if (err instanceof BaseError) return err as any;\n    let code: string = \"\";\n    if (typeof err === \"string\") {\n      code = err;\n      if (code.match(/already exist|update conflict/g))\n        return new ConflictError(code);\n      if (code.match(/missing|deleted/g)) return new NotFoundError(code);\n    } else if ((err as any).code) {\n      code = (err as any).code;\n      reason = reason || err.message;\n    } else if ((err as any).statusCode) {\n      code = (err as any).statusCode;\n      reason = reason || err.message;\n    } else {\n      code = err.message;\n    }\n\n    switch (code.toString()) {\n      case \"401\":\n      case \"412\":\n      case \"409\":\n        return new ConflictError(reason as string);\n      case \"404\":\n        return new NotFoundError(reason as string);\n      case \"400\":\n        if (code.toString().match(/No\\sindex\\sexists/g))\n          return new IndexError(err);\n        return new InternalError(err);\n      default:\n        if (code.toString().match(/ECONNREFUSED/g))\n          return new ConnectionError(err);\n        return new InternalError(err);\n    }\n  }\n}\n"]}
package/lib/adapter.d.ts CHANGED
@@ -4,25 +4,259 @@ import "reflect-metadata";
4
4
  import { Constructor, Model } from "@decaf-ts/decorator-validation";
5
5
  import { MangoQuery } from "./types";
6
6
  import { CouchDBStatement } from "./query";
7
+ /**
8
+ * @description Abstract adapter for CouchDB database operations
9
+ * @summary Provides a base implementation for CouchDB database operations, including CRUD operations, sequence management, and error handling
10
+ * @template Y - The scope type
11
+ * @template F - The repository flags type
12
+ * @template C - The context type
13
+ * @param {Y} scope - The scope for the adapter
14
+ * @param {string} flavour - The flavour of the adapter
15
+ * @param {string} [alias] - Optional alias for the adapter
16
+ * @class
17
+ * @example
18
+ * // Example of extending CouchDBAdapter
19
+ * class MyCouchDBAdapter extends CouchDBAdapter<MyScope, MyFlags, MyContext> {
20
+ * constructor(scope: MyScope) {
21
+ * super(scope, 'my-couchdb', 'my-alias');
22
+ * }
23
+ *
24
+ * // Implement abstract methods
25
+ * async index<M extends Model>(...models: Constructor<M>[]): Promise<void> {
26
+ * // Implementation
27
+ * }
28
+ *
29
+ * async raw<R>(rawInput: MangoQuery, docsOnly: boolean): Promise<R> {
30
+ * // Implementation
31
+ * }
32
+ *
33
+ * async create(tableName: string, id: string | number, model: Record<string, any>, ...args: any[]): Promise<Record<string, any>> {
34
+ * // Implementation
35
+ * }
36
+ *
37
+ * async read(tableName: string, id: string | number, ...args: any[]): Promise<Record<string, any>> {
38
+ * // Implementation
39
+ * }
40
+ *
41
+ * async update(tableName: string, id: string | number, model: Record<string, any>, ...args: any[]): Promise<Record<string, any>> {
42
+ * // Implementation
43
+ * }
44
+ *
45
+ * async delete(tableName: string, id: string | number, ...args: any[]): Promise<Record<string, any>> {
46
+ * // Implementation
47
+ * }
48
+ * }
49
+ */
7
50
  export declare abstract class CouchDBAdapter<Y, F extends RepositoryFlags, C extends Context<F>> extends Adapter<Y, MangoQuery, F, C> {
8
51
  protected constructor(scope: Y, flavour: string, alias?: string);
52
+ /**
53
+ * @description Creates a new CouchDB statement for querying
54
+ * @summary Factory method that creates a new CouchDBStatement instance for building queries
55
+ * @template M - The model type
56
+ * @return {CouchDBStatement<M, any>} A new CouchDBStatement instance
57
+ */
9
58
  Statement<M extends Model>(): CouchDBStatement<M, any>;
59
+ /**
60
+ * @description Creates a new CouchDB sequence
61
+ * @summary Factory method that creates a new CouchDBSequence instance for managing sequences
62
+ * @param {SequenceOptions} options - The options for the sequence
63
+ * @return {Promise<Sequence>} A promise that resolves to a new Sequence instance
64
+ */
10
65
  Sequence(options: SequenceOptions): Promise<Sequence>;
66
+ /**
67
+ * @description Initializes the adapter by creating indexes for all managed models
68
+ * @summary Sets up the necessary database indexes for all models managed by this adapter
69
+ * @return {Promise<void>} A promise that resolves when initialization is complete
70
+ */
11
71
  initialize(): Promise<void>;
72
+ /**
73
+ * @description Creates indexes for the given models
74
+ * @summary Abstract method that must be implemented to create database indexes for the specified models
75
+ * @template M - The model type
76
+ * @param {...Constructor<M>} models - The model constructors to create indexes for
77
+ * @return {Promise<void>} A promise that resolves when all indexes are created
78
+ */
12
79
  protected abstract index<M extends Model>(...models: Constructor<M>[]): Promise<void>;
80
+ /**
81
+ * @description Executes a raw Mango query against the database
82
+ * @summary Abstract method that must be implemented to execute raw Mango queries
83
+ * @template R - The result type
84
+ * @param {MangoQuery} rawInput - The raw Mango query to execute
85
+ * @param {boolean} docsOnly - Whether to return only the documents or the full response
86
+ * @return {Promise<R>} A promise that resolves to the query result
87
+ */
13
88
  abstract raw<R>(rawInput: MangoQuery, docsOnly: boolean): Promise<R>;
89
+ /**
90
+ * @description Assigns metadata to a model
91
+ * @summary Adds revision metadata to a model as a non-enumerable property
92
+ * @param {Record<string, any>} model - The model to assign metadata to
93
+ * @param {string} rev - The revision string to assign
94
+ * @return {Record<string, any>} The model with metadata assigned
95
+ */
14
96
  protected assignMetadata(model: Record<string, any>, rev: string): Record<string, any>;
97
+ /**
98
+ * @description Assigns metadata to multiple models
99
+ * @summary Adds revision metadata to multiple models as non-enumerable properties
100
+ * @param models - The models to assign metadata to
101
+ * @param {string[]} revs - The revision strings to assign
102
+ * @return The models with metadata assigned
103
+ */
15
104
  protected assignMultipleMetadata(models: Record<string, any>[], revs: string[]): Record<string, any>[];
105
+ /**
106
+ * @description Prepares a record for creation
107
+ * @summary Adds necessary CouchDB fields to a record before creation
108
+ * @param {string} tableName - The name of the table
109
+ * @param {string|number} id - The ID of the record
110
+ * @param {Record<string, any>} model - The model to prepare
111
+ * @return A tuple containing the tableName, id, and prepared record
112
+ */
16
113
  protected createPrefix(tableName: string, id: string | number, model: Record<string, any>): (string | number | Record<string, any>)[];
114
+ /**
115
+ * @description Creates a new record in the database
116
+ * @summary Abstract method that must be implemented to create a new record
117
+ * @param {string} tableName - The name of the table
118
+ * @param {string|number} id - The ID of the record
119
+ * @param {Record<string, any>} model - The model to create
120
+ * @param {...any[]} args - Additional arguments
121
+ * @return {Promise<Record<string, any>>} A promise that resolves to the created record
122
+ */
17
123
  abstract create(tableName: string, id: string | number, model: Record<string, any>, ...args: any[]): Promise<Record<string, any>>;
124
+ /**
125
+ * @description Prepares multiple records for creation
126
+ * @summary Adds necessary CouchDB fields to multiple records before creation
127
+ * @param {string} tableName - The name of the table
128
+ * @param {string[]|number[]} ids - The IDs of the records
129
+ * @param models - The models to prepare
130
+ * @return A tuple containing the tableName, ids, and prepared records
131
+ * @throws {InternalError} If ids and models arrays have different lengths
132
+ */
18
133
  protected createAllPrefix(tableName: string, ids: string[] | number[], models: Record<string, any>[]): (string | string[] | number[] | Record<string, any>[])[];
134
+ /**
135
+ * @description Reads a record from the database
136
+ * @summary Abstract method that must be implemented to read a record
137
+ * @param {string} tableName - The name of the table
138
+ * @param {string|number} id - The ID of the record
139
+ * @param {...any[]} args - Additional arguments
140
+ * @return {Promise<Record<string, any>>} A promise that resolves to the read record
141
+ */
19
142
  abstract read(tableName: string, id: string | number, ...args: any[]): Promise<Record<string, any>>;
143
+ /**
144
+ * @description Prepares a record for update
145
+ * @summary Adds necessary CouchDB fields to a record before update
146
+ * @param {string} tableName - The name of the table
147
+ * @param {string|number} id - The ID of the record
148
+ * @param model - The model to prepare
149
+ * @return A tuple containing the tableName, id, and prepared record
150
+ * @throws {InternalError} If no revision number is found in the model
151
+ */
20
152
  updatePrefix(tableName: string, id: string | number, model: Record<string, any>): (string | number | Record<string, any>)[];
153
+ /**
154
+ * @description Updates a record in the database
155
+ * @summary Abstract method that must be implemented to update a record
156
+ * @param {string} tableName - The name of the table
157
+ * @param {string|number} id - The ID of the record
158
+ * @param {Record<string, any>} model - The model to update
159
+ * @param {any[]} args - Additional arguments
160
+ * @return A promise that resolves to the updated record
161
+ */
21
162
  abstract update(tableName: string, id: string | number, model: Record<string, any>, ...args: any[]): Promise<Record<string, any>>;
163
+ /**
164
+ * @description Prepares multiple records for update
165
+ * @summary Adds necessary CouchDB fields to multiple records before update
166
+ * @param {string} tableName - The name of the table
167
+ * @param {string[]|number[]} ids - The IDs of the records
168
+ * @param models - The models to prepare
169
+ * @return A tuple containing the tableName, ids, and prepared records
170
+ * @throws {InternalError} If ids and models arrays have different lengths or if no revision number is found in a model
171
+ */
22
172
  protected updateAllPrefix(tableName: string, ids: string[] | number[], models: Record<string, any>[]): (string | string[] | number[] | Record<string, any>[])[];
173
+ /**
174
+ * @description Deletes a record from the database
175
+ * @summary Abstract method that must be implemented to delete a record
176
+ * @param {string} tableName - The name of the table
177
+ * @param {string|number} id - The ID of the record
178
+ * @param {any[]} args - Additional arguments
179
+ * @return A promise that resolves to the deleted record
180
+ */
23
181
  abstract delete(tableName: string, id: string | number, ...args: any[]): Promise<Record<string, any>>;
182
+ /**
183
+ * @description Generates a CouchDB document ID
184
+ * @summary Combines the table name and ID to create a CouchDB document ID
185
+ * @param {string} tableName - The name of the table
186
+ * @param {string|number} id - The ID of the record
187
+ * @return {string} The generated CouchDB document ID
188
+ */
24
189
  protected generateId(tableName: string, id: string | number): string;
190
+ /**
191
+ * @description Parses an error and converts it to a BaseError
192
+ * @summary Converts various error types to appropriate BaseError subtypes
193
+ * @param {Error|string} err - The error to parse
194
+ * @param {string} [reason] - Optional reason for the error
195
+ * @return {BaseError} The parsed error as a BaseError
196
+ */
25
197
  parseError(err: Error | string, reason?: string): BaseError;
198
+ /**
199
+ * @description Checks if an attribute is reserved
200
+ * @summary Determines if an attribute name is reserved in CouchDB
201
+ * @param {string} attr - The attribute name to check
202
+ * @return {boolean} True if the attribute is reserved, false otherwise
203
+ */
26
204
  protected isReserved(attr: string): boolean;
205
+ /**
206
+ * @description Static method to parse an error and convert it to a BaseError
207
+ * @summary Converts various error types to appropriate BaseError subtypes based on error codes and messages
208
+ * @param {Error|string} err - The error to parse
209
+ * @param {string} [reason] - Optional reason for the error
210
+ * @return {BaseError} The parsed error as a BaseError
211
+ * @mermaid
212
+ * sequenceDiagram
213
+ * participant Caller
214
+ * participant parseError
215
+ * participant ErrorTypes
216
+ *
217
+ * Caller->>parseError: err, reason
218
+ * Note over parseError: Check if err is already a BaseError
219
+ * alt err is BaseError
220
+ * parseError-->>Caller: return err
221
+ * else err is string
222
+ * Note over parseError: Extract code from string
223
+ * alt code matches "already exist|update conflict"
224
+ * parseError->>ErrorTypes: new ConflictError(code)
225
+ * ErrorTypes-->>Caller: ConflictError
226
+ * else code matches "missing|deleted"
227
+ * parseError->>ErrorTypes: new NotFoundError(code)
228
+ * ErrorTypes-->>Caller: NotFoundError
229
+ * end
230
+ * else err has code property
231
+ * Note over parseError: Extract code and reason
232
+ * else err has statusCode property
233
+ * Note over parseError: Extract code and reason
234
+ * else
235
+ * Note over parseError: Use err.message as code
236
+ * end
237
+ *
238
+ * Note over parseError: Switch on code
239
+ * alt code is 401, 412, or 409
240
+ * parseError->>ErrorTypes: new ConflictError(reason)
241
+ * ErrorTypes-->>Caller: ConflictError
242
+ * else code is 404
243
+ * parseError->>ErrorTypes: new NotFoundError(reason)
244
+ * ErrorTypes-->>Caller: NotFoundError
245
+ * else code is 400
246
+ * alt code matches "No index exists"
247
+ * parseError->>ErrorTypes: new IndexError(err)
248
+ * ErrorTypes-->>Caller: IndexError
249
+ * else
250
+ * parseError->>ErrorTypes: new InternalError(err)
251
+ * ErrorTypes-->>Caller: InternalError
252
+ * end
253
+ * else code matches "ECONNREFUSED"
254
+ * parseError->>ErrorTypes: new ConnectionError(err)
255
+ * ErrorTypes-->>Caller: ConnectionError
256
+ * else
257
+ * parseError->>ErrorTypes: new InternalError(err)
258
+ * ErrorTypes-->>Caller: InternalError
259
+ * end
260
+ */
27
261
  protected static parseError(err: Error | string, reason?: string): BaseError;
28
262
  }