@decaf-ts/for-http 0.2.0 → 0.2.2

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.
@@ -2,10 +2,39 @@
2
2
  Object.defineProperty(exports, "__esModule", { value: true });
3
3
  exports.RestRepository = void 0;
4
4
  const core_1 = require("@decaf-ts/core");
5
+ /**
6
+ * @description Repository for REST API interactions
7
+ * @summary A specialized repository implementation for interacting with REST APIs.
8
+ * This class extends the core Repository class and works with HTTP adapters to
9
+ * provide CRUD operations for models via REST endpoints.
10
+ * This Is NOT the default repository for the HTTP adapter. That would be {@link RestService}.
11
+ * Use this only in the specific case of needing to run the CURD model logic (decoration) before submitting to the backend
12
+ * @template M - The model type, extending Model
13
+ * @template Q - The query type used by the adapter
14
+ * @template A - The HTTP adapter type, extending HttpAdapter
15
+ * @template F - The HTTP flags type, extending HttpFlags
16
+ * @template C - The context type, extending Context<F>
17
+ * @param {A} adapter - The HTTP adapter instance
18
+ * @param {Constructor<M>} [clazz] - Optional constructor for the model class
19
+ * @example
20
+ * ```typescript
21
+ * // Create a repository for User model with Axios adapter
22
+ * const axiosAdapter = new AxiosAdapter({
23
+ * protocol: 'https',
24
+ * host: 'api.example.com'
25
+ * });
26
+ * const userRepository = new RestRepository(axiosAdapter, User);
27
+ *
28
+ * // Use the repository for CRUD operations
29
+ * const user = await userRepository.findById('123');
30
+ * ```
31
+ * @class RestRepository
32
+ * @see {@link RestService}
33
+ */
5
34
  class RestRepository extends core_1.Repository {
6
35
  constructor(adapter, clazz) {
7
36
  super(adapter, clazz);
8
37
  }
9
38
  }
10
39
  exports.RestRepository = RestRepository;
11
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiUmVzdFJlcG9zaXRvcnkuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi9zcmMvUmVzdFJlcG9zaXRvcnkudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6Ijs7O0FBQUEseUNBQTRDO0FBSzVDLE1BQWEsY0FNWCxTQUFRLGlCQUFtQjtJQUMzQixZQUFZLE9BQVUsRUFBRSxLQUFzQjtRQUM1QyxLQUFLLENBQUMsT0FBTyxFQUFFLEtBQUssQ0FBQyxDQUFDO0lBQ3hCLENBQUM7Q0FDRjtBQVZELHdDQVVDIiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHsgUmVwb3NpdG9yeSB9IGZyb20gXCJAZGVjYWYtdHMvY29yZVwiO1xuaW1wb3J0IHsgQ29uc3RydWN0b3IsIE1vZGVsIH0gZnJvbSBcIkBkZWNhZi10cy9kZWNvcmF0b3ItdmFsaWRhdGlvblwiO1xuaW1wb3J0IHsgSHR0cEFkYXB0ZXIgfSBmcm9tIFwiLi9hZGFwdGVyXCI7XG5pbXBvcnQgeyBDb250ZXh0LCBSZXBvc2l0b3J5RmxhZ3MgfSBmcm9tIFwiQGRlY2FmLXRzL2RiLWRlY29yYXRvcnNcIjtcblxuZXhwb3J0IGNsYXNzIFJlc3RSZXBvc2l0b3J5PFxuICBNIGV4dGVuZHMgTW9kZWwsXG4gIFEsXG4gIEEgZXh0ZW5kcyBIdHRwQWRhcHRlcjxhbnksIFEsIEYsIEM+LFxuICBGIGV4dGVuZHMgUmVwb3NpdG9yeUZsYWdzLFxuICBDIGV4dGVuZHMgQ29udGV4dDxGPiA9IENvbnRleHQ8Rj4sXG4+IGV4dGVuZHMgUmVwb3NpdG9yeTxNLCBRLCBBPiB7XG4gIGNvbnN0cnVjdG9yKGFkYXB0ZXI6IEEsIGNsYXp6PzogQ29uc3RydWN0b3I8TT4pIHtcbiAgICBzdXBlcihhZGFwdGVyLCBjbGF6eik7XG4gIH1cbn1cbiJdfQ==
40
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiUmVzdFJlcG9zaXRvcnkuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi9zcmMvUmVzdFJlcG9zaXRvcnkudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6Ijs7O0FBQUEseUNBQTRDO0FBTTVDOzs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7O0dBNEJHO0FBQ0gsTUFBYSxjQU1YLFNBQVEsaUJBQW1CO0lBQzNCLFlBQVksT0FBVSxFQUFFLEtBQXNCO1FBQzVDLEtBQUssQ0FBQyxPQUFPLEVBQUUsS0FBSyxDQUFDLENBQUM7SUFDeEIsQ0FBQztDQUNGO0FBVkQsd0NBVUMiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgeyBSZXBvc2l0b3J5IH0gZnJvbSBcIkBkZWNhZi10cy9jb3JlXCI7XG5pbXBvcnQgeyBDb25zdHJ1Y3RvciwgTW9kZWwgfSBmcm9tIFwiQGRlY2FmLXRzL2RlY29yYXRvci12YWxpZGF0aW9uXCI7XG5pbXBvcnQgeyBIdHRwQWRhcHRlciB9IGZyb20gXCIuL2FkYXB0ZXJcIjtcbmltcG9ydCB7IENvbnRleHQgfSBmcm9tIFwiQGRlY2FmLXRzL2RiLWRlY29yYXRvcnNcIjtcbmltcG9ydCB7IEh0dHBGbGFncyB9IGZyb20gXCIuL3R5cGVzXCI7XG5cbi8qKlxuICogQGRlc2NyaXB0aW9uIFJlcG9zaXRvcnkgZm9yIFJFU1QgQVBJIGludGVyYWN0aW9uc1xuICogQHN1bW1hcnkgQSBzcGVjaWFsaXplZCByZXBvc2l0b3J5IGltcGxlbWVudGF0aW9uIGZvciBpbnRlcmFjdGluZyB3aXRoIFJFU1QgQVBJcy5cbiAqIFRoaXMgY2xhc3MgZXh0ZW5kcyB0aGUgY29yZSBSZXBvc2l0b3J5IGNsYXNzIGFuZCB3b3JrcyB3aXRoIEhUVFAgYWRhcHRlcnMgdG9cbiAqIHByb3ZpZGUgQ1JVRCBvcGVyYXRpb25zIGZvciBtb2RlbHMgdmlhIFJFU1QgZW5kcG9pbnRzLlxuICogVGhpcyBJcyBOT1QgdGhlIGRlZmF1bHQgcmVwb3NpdG9yeSBmb3IgdGhlIEhUVFAgYWRhcHRlci4gVGhhdCB3b3VsZCBiZSB7QGxpbmsgUmVzdFNlcnZpY2V9LlxuICogVXNlIHRoaXMgb25seSBpbiB0aGUgc3BlY2lmaWMgY2FzZSBvZiBuZWVkaW5nIHRvIHJ1biB0aGUgQ1VSRCBtb2RlbCBsb2dpYyAoZGVjb3JhdGlvbikgYmVmb3JlIHN1Ym1pdHRpbmcgdG8gdGhlIGJhY2tlbmRcbiAqIEB0ZW1wbGF0ZSBNIC0gVGhlIG1vZGVsIHR5cGUsIGV4dGVuZGluZyBNb2RlbFxuICogQHRlbXBsYXRlIFEgLSBUaGUgcXVlcnkgdHlwZSB1c2VkIGJ5IHRoZSBhZGFwdGVyXG4gKiBAdGVtcGxhdGUgQSAtIFRoZSBIVFRQIGFkYXB0ZXIgdHlwZSwgZXh0ZW5kaW5nIEh0dHBBZGFwdGVyXG4gKiBAdGVtcGxhdGUgRiAtIFRoZSBIVFRQIGZsYWdzIHR5cGUsIGV4dGVuZGluZyBIdHRwRmxhZ3NcbiAqIEB0ZW1wbGF0ZSBDIC0gVGhlIGNvbnRleHQgdHlwZSwgZXh0ZW5kaW5nIENvbnRleHQ8Rj5cbiAqIEBwYXJhbSB7QX0gYWRhcHRlciAtIFRoZSBIVFRQIGFkYXB0ZXIgaW5zdGFuY2VcbiAqIEBwYXJhbSB7Q29uc3RydWN0b3I8TT59IFtjbGF6el0gLSBPcHRpb25hbCBjb25zdHJ1Y3RvciBmb3IgdGhlIG1vZGVsIGNsYXNzXG4gKiBAZXhhbXBsZVxuICogYGBgdHlwZXNjcmlwdFxuICogLy8gQ3JlYXRlIGEgcmVwb3NpdG9yeSBmb3IgVXNlciBtb2RlbCB3aXRoIEF4aW9zIGFkYXB0ZXJcbiAqIGNvbnN0IGF4aW9zQWRhcHRlciA9IG5ldyBBeGlvc0FkYXB0ZXIoe1xuICogICBwcm90b2NvbDogJ2h0dHBzJyxcbiAqICAgaG9zdDogJ2FwaS5leGFtcGxlLmNvbSdcbiAqIH0pO1xuICogY29uc3QgdXNlclJlcG9zaXRvcnkgPSBuZXcgUmVzdFJlcG9zaXRvcnkoYXhpb3NBZGFwdGVyLCBVc2VyKTtcbiAqXG4gKiAvLyBVc2UgdGhlIHJlcG9zaXRvcnkgZm9yIENSVUQgb3BlcmF0aW9uc1xuICogY29uc3QgdXNlciA9IGF3YWl0IHVzZXJSZXBvc2l0b3J5LmZpbmRCeUlkKCcxMjMnKTtcbiAqIGBgYFxuICogQGNsYXNzIFJlc3RSZXBvc2l0b3J5XG4gKiBAc2VlIHtAbGluayBSZXN0U2VydmljZX1cbiAqL1xuZXhwb3J0IGNsYXNzIFJlc3RSZXBvc2l0b3J5PFxuICBNIGV4dGVuZHMgTW9kZWwsXG4gIFEsXG4gIEEgZXh0ZW5kcyBIdHRwQWRhcHRlcjxhbnksIFEsIEYsIEM+LFxuICBGIGV4dGVuZHMgSHR0cEZsYWdzID0gSHR0cEZsYWdzLFxuICBDIGV4dGVuZHMgQ29udGV4dDxGPiA9IENvbnRleHQ8Rj4sXG4+IGV4dGVuZHMgUmVwb3NpdG9yeTxNLCBRLCBBPiB7XG4gIGNvbnN0cnVjdG9yKGFkYXB0ZXI6IEEsIGNsYXp6PzogQ29uc3RydWN0b3I8TT4pIHtcbiAgICBzdXBlcihhZGFwdGVyLCBjbGF6eik7XG4gIH1cbn1cbiJdfQ==
@@ -1,7 +1,37 @@
1
1
  import { Repository } from "@decaf-ts/core";
2
2
  import { Constructor, Model } from "@decaf-ts/decorator-validation";
3
3
  import { HttpAdapter } from "./adapter";
4
- import { Context, RepositoryFlags } from "@decaf-ts/db-decorators";
5
- export declare class RestRepository<M extends Model, Q, A extends HttpAdapter<any, Q, F, C>, F extends RepositoryFlags, C extends Context<F> = Context<F>> extends Repository<M, Q, A> {
4
+ import { Context } from "@decaf-ts/db-decorators";
5
+ import { HttpFlags } from "./types";
6
+ /**
7
+ * @description Repository for REST API interactions
8
+ * @summary A specialized repository implementation for interacting with REST APIs.
9
+ * This class extends the core Repository class and works with HTTP adapters to
10
+ * provide CRUD operations for models via REST endpoints.
11
+ * This Is NOT the default repository for the HTTP adapter. That would be {@link RestService}.
12
+ * Use this only in the specific case of needing to run the CURD model logic (decoration) before submitting to the backend
13
+ * @template M - The model type, extending Model
14
+ * @template Q - The query type used by the adapter
15
+ * @template A - The HTTP adapter type, extending HttpAdapter
16
+ * @template F - The HTTP flags type, extending HttpFlags
17
+ * @template C - The context type, extending Context<F>
18
+ * @param {A} adapter - The HTTP adapter instance
19
+ * @param {Constructor<M>} [clazz] - Optional constructor for the model class
20
+ * @example
21
+ * ```typescript
22
+ * // Create a repository for User model with Axios adapter
23
+ * const axiosAdapter = new AxiosAdapter({
24
+ * protocol: 'https',
25
+ * host: 'api.example.com'
26
+ * });
27
+ * const userRepository = new RestRepository(axiosAdapter, User);
28
+ *
29
+ * // Use the repository for CRUD operations
30
+ * const user = await userRepository.findById('123');
31
+ * ```
32
+ * @class RestRepository
33
+ * @see {@link RestService}
34
+ */
35
+ export declare class RestRepository<M extends Model, Q, A extends HttpAdapter<any, Q, F, C>, F extends HttpFlags = HttpFlags, C extends Context<F> = Context<F>> extends Repository<M, Q, A> {
6
36
  constructor(adapter: A, clazz?: Constructor<M>);
7
37
  }
@@ -3,53 +3,164 @@ Object.defineProperty(exports, "__esModule", { value: true });
3
3
  exports.RestService = void 0;
4
4
  const db_decorators_1 = require("@decaf-ts/db-decorators");
5
5
  const core_1 = require("@decaf-ts/core");
6
+ /**
7
+ * @description Service class for REST API operations
8
+ * @summary Provides a comprehensive implementation for interacting with REST APIs.
9
+ * This class implements CRUD operations for single and bulk operations, as well as
10
+ * the Observable pattern to notify observers of changes. It works with HTTP adapters
11
+ * to perform the actual API requests and handles model conversion.
12
+ * @template M - The model type, extending Model
13
+ * @template Q - The query type used by the adapter
14
+ * @template A - The HTTP adapter type, extending HttpAdapter
15
+ * @template F - The HTTP flags type, extending HttpFlags
16
+ * @template C - The context type, extending Context<F>
17
+ * @param {A} adapter - The HTTP adapter instance
18
+ * @param {Constructor<M>} [clazz] - Optional constructor for the model class
19
+ * @example
20
+ * ```typescript
21
+ * // Create a service for User model with Axios adapter
22
+ * const axiosAdapter = new AxiosAdapter({
23
+ * protocol: 'https',
24
+ * host: 'api.example.com'
25
+ * });
26
+ * const userService = new RestService(axiosAdapter, User);
27
+ *
28
+ * // Create a new user
29
+ * const user = new User({ name: 'John Doe', email: 'john@example.com' });
30
+ * const createdUser = await userService.create(user);
31
+ *
32
+ * // Update a user
33
+ * createdUser.name = 'Jane Doe';
34
+ * const updatedUser = await userService.update(createdUser);
35
+ *
36
+ * // Delete a user
37
+ * await userService.delete(updatedUser.id);
38
+ * ```
39
+ * @class
40
+ */
6
41
  class RestService {
42
+ /**
43
+ * @description Gets the model class constructor
44
+ * @summary Retrieves the model class constructor associated with this service.
45
+ * Throws an error if no class definition is found.
46
+ * @return {Constructor<M>} The model class constructor
47
+ * @throws {InternalError} If no class definition is found
48
+ */
7
49
  get class() {
8
50
  if (!this._class)
9
51
  throw new db_decorators_1.InternalError("No class definition found for this repository");
10
52
  return this._class;
11
53
  }
54
+ /**
55
+ * @description Gets the primary key property name
56
+ * @summary Retrieves the name of the primary key property for the model.
57
+ * If not already determined, it finds the primary key using the model class.
58
+ * @return The primary key property name
59
+ */
12
60
  get pk() {
13
61
  if (!this._pk)
14
62
  this._pk = (0, db_decorators_1.findPrimaryKey)(new this.class()).id;
15
63
  return this._pk;
16
64
  }
65
+ /**
66
+ * @description Gets the HTTP adapter
67
+ * @summary Retrieves the HTTP adapter associated with this service.
68
+ * Throws an error if no adapter is found.
69
+ * @return {A} The HTTP adapter instance
70
+ * @throws {InternalError} If no adapter is found
71
+ */
17
72
  get adapter() {
18
73
  if (!this._adapter)
19
74
  throw new db_decorators_1.InternalError("No adapter found for this repository. did you use the @uses decorator or pass it in the constructor?");
20
75
  return this._adapter;
21
76
  }
77
+ /**
78
+ * @description Gets the table name for the model
79
+ * @summary Retrieves the table name associated with the model class.
80
+ * If not already determined, it gets the table name from the Repository utility.
81
+ * @return {string} The table name
82
+ */
22
83
  get tableName() {
23
84
  if (!this._tableName)
24
85
  this._tableName = core_1.Repository.table(this.class);
25
86
  return this._tableName;
26
87
  }
88
+ /**
89
+ * @description Initializes a new RestService instance
90
+ * @summary Creates a new service instance with the specified adapter and optional model class.
91
+ * The constructor stores the adapter and model class for later use in CRUD operations.
92
+ * @param {A} adapter - The HTTP adapter instance to use for API requests
93
+ * @param {Constructor<M>} [clazz] - Optional constructor for the model class
94
+ */
27
95
  constructor(adapter, clazz) {
28
96
  this.observers = [];
29
97
  this._adapter = adapter;
30
98
  if (clazz)
31
99
  this._class = clazz;
32
100
  }
101
+ /**
102
+ * @description Creates a new resource
103
+ * @summary Creates a new resource in the REST API using the provided model.
104
+ * The method prepares the model for the adapter, sends the create request,
105
+ * and then converts the response back to a model instance.
106
+ * @param {M} model - The model instance to create
107
+ * @param {...any[]} args - Additional arguments to pass to the adapter
108
+ * @return {Promise<M>} A promise that resolves with the created model instance
109
+ */
33
110
  async create(model, ...args) {
34
111
  // eslint-disable-next-line prefer-const
35
112
  let { record, id } = this.adapter.prepare(model, this.pk);
36
113
  record = await this.adapter.create(this.tableName, id, record, ...args);
37
114
  return this.adapter.revert(record, this.class, this.pk, id);
38
115
  }
116
+ /**
117
+ * @description Retrieves a resource by ID
118
+ * @summary Fetches a resource from the REST API using the provided ID.
119
+ * The method sends the read request and converts the response to a model instance.
120
+ * @param {string|number} id - The identifier of the resource to retrieve
121
+ * @param {...any[]} args - Additional arguments to pass to the adapter
122
+ * @return {Promise<M>} A promise that resolves with the retrieved model instance
123
+ */
39
124
  async read(id, ...args) {
40
125
  const m = await this.adapter.read(this.tableName, id, ...args);
41
126
  return this.adapter.revert(m, this.class, this.pk, id);
42
127
  }
128
+ /**
129
+ * @description Updates an existing resource
130
+ * @summary Updates an existing resource in the REST API using the provided model.
131
+ * The method prepares the model for the adapter, sends the update request,
132
+ * and then converts the response back to a model instance.
133
+ * @param {M} model - The model instance with updated data
134
+ * @param {...any[]} args - Additional arguments to pass to the adapter
135
+ * @return {Promise<M>} A promise that resolves with the updated model instance
136
+ */
43
137
  async update(model, ...args) {
44
138
  // eslint-disable-next-line prefer-const
45
139
  let { record, id } = this.adapter.prepare(model, this.pk);
46
140
  record = await this.adapter.update(this.tableName, id, record, ...args);
47
141
  return this.adapter.revert(record, this.class, this.pk, id);
48
142
  }
143
+ /**
144
+ * @description Deletes a resource by ID
145
+ * @summary Removes a resource from the REST API using the provided ID.
146
+ * The method sends the delete request and converts the response to a model instance.
147
+ * @param {string|number} id - The identifier of the resource to delete
148
+ * @param {...any[]} args - Additional arguments to pass to the adapter
149
+ * @return {Promise<M>} A promise that resolves with the deleted model instance
150
+ */
49
151
  async delete(id, ...args) {
50
152
  const m = await this.adapter.delete(this.tableName, id, ...args);
51
153
  return this.adapter.revert(m, this.class, this.pk, id);
52
154
  }
155
+ /**
156
+ * @description Creates multiple resources
157
+ * @summary Creates multiple resources in the REST API using the provided models.
158
+ * The method prepares each model for the adapter, sends a bulk create request,
159
+ * and then converts the responses back to model instances.
160
+ * @param {M[]} models - The model instances to create
161
+ * @param {...any[]} args - Additional arguments to pass to the adapter
162
+ * @return {Promise<M[]>} A promise that resolves with an array of created model instances
163
+ */
53
164
  async createAll(models, ...args) {
54
165
  if (!models.length)
55
166
  return models;
@@ -59,24 +170,51 @@ class RestService {
59
170
  records = await this.adapter.createAll(this.tableName, ids, records, ...args);
60
171
  return records.map((r, i) => this.adapter.revert(r, this.class, this.pk, ids[i]));
61
172
  }
173
+ /**
174
+ * @description Deletes multiple resources by IDs
175
+ * @summary Removes multiple resources from the REST API using the provided IDs.
176
+ * The method sends a bulk delete request and converts the responses to model instances.
177
+ * @param {string[]|number[]} keys - The identifiers of the resources to delete
178
+ * @param {...any[]} args - Additional arguments to pass to the adapter
179
+ * @return {Promise<M[]>} A promise that resolves with an array of deleted model instances
180
+ */
62
181
  async deleteAll(keys, ...args) {
63
182
  const results = await this.adapter.deleteAll(this.tableName, keys, ...args);
64
183
  return results.map((r, i) => this.adapter.revert(r, this.class, this.pk, keys[i]));
65
184
  }
185
+ /**
186
+ * @description Retrieves multiple resources by IDs
187
+ * @summary Fetches multiple resources from the REST API using the provided IDs.
188
+ * The method sends a bulk read request and converts the responses to model instances.
189
+ * @param {string[]|number[]} keys - The identifiers of the resources to retrieve
190
+ * @param {...any[]} args - Additional arguments to pass to the adapter
191
+ * @return {Promise<M[]>} A promise that resolves with an array of retrieved model instances
192
+ */
66
193
  async readAll(keys, ...args) {
67
194
  const records = await this.adapter.readAll(this.tableName, keys, ...args);
68
195
  return records.map((r, i) => this.adapter.revert(r, this.class, this.pk, keys[i]));
69
196
  }
197
+ /**
198
+ * @description Updates multiple resources
199
+ * @summary Updates multiple resources in the REST API using the provided models.
200
+ * The method prepares each model for the adapter, sends a bulk update request,
201
+ * and then converts the responses back to model instances.
202
+ * @param {M[]} models - The model instances with updated data
203
+ * @param {...any[]} args - Additional arguments to pass to the adapter
204
+ * @return {Promise<M[]>} A promise that resolves with an array of updated model instances
205
+ */
70
206
  async updateAll(models, ...args) {
71
207
  const records = models.map((m) => this.adapter.prepare(m, this.pk));
72
208
  const updated = await this.adapter.updateAll(this.tableName, records.map((r) => r.id), records.map((r) => r.record), ...args);
73
209
  return updated.map((u, i) => this.adapter.revert(u, this.class, this.pk, records[i].id));
74
210
  }
75
211
  /**
76
- * @summary Registers an {@link Observer}
77
- * @param {Observer} observer
78
- *
79
- * @see {Observable#observe}
212
+ * @description Registers an observer
213
+ * @summary Adds an observer to the list of observers that will be notified of changes.
214
+ * Throws an error if the observer is already registered.
215
+ * @param {Observer} observer - The observer to register
216
+ * @return {void}
217
+ * @throws {InternalError} If the observer is already registered
80
218
  */
81
219
  observe(observer) {
82
220
  const index = this.observers.indexOf(observer);
@@ -85,10 +223,12 @@ class RestService {
85
223
  this.observers.push(observer);
86
224
  }
87
225
  /**
88
- * @summary Unregisters an {@link Observer}
89
- * @param {Observer} observer
90
- *
91
- * @see {Observable#unObserve}
226
+ * @description Unregisters an observer
227
+ * @summary Removes an observer from the list of observers.
228
+ * Throws an error if the observer is not found.
229
+ * @param {Observer} observer - The observer to unregister
230
+ * @return {void}
231
+ * @throws {InternalError} If the observer is not found
92
232
  */
93
233
  unObserve(observer) {
94
234
  const index = this.observers.indexOf(observer);
@@ -97,8 +237,11 @@ class RestService {
97
237
  this.observers.splice(index, 1);
98
238
  }
99
239
  /**
100
- * @summary calls all registered {@link Observer}s to update themselves
101
- * @param {any[]} [args] optional arguments to be passed to the {@link Observer#refresh} method
240
+ * @description Notifies all registered observers
241
+ * @summary Calls the refresh method on all registered observers to update themselves.
242
+ * Any errors during observer refresh are logged as warnings but don't stop the process.
243
+ * @param {...any[]} [args] - Optional arguments to pass to the observer refresh method
244
+ * @return {Promise<void>} A promise that resolves when all observers have been updated
102
245
  */
103
246
  async updateObservers(...args) {
104
247
  const results = await Promise.allSettled(this.observers.map((o) => o.refresh(...args)));
@@ -109,4 +252,4 @@ class RestService {
109
252
  }
110
253
  }
111
254
  exports.RestService = RestService;
112
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"RestService.js","sourceRoot":"","sources":["../src/RestService.ts"],"names":[],"mappings":"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","sourcesContent":["import {\n  BulkCrudOperator,\n  Context,\n  CrudOperator,\n  findPrimaryKey,\n  InternalError,\n  RepositoryFlags,\n} from \"@decaf-ts/db-decorators\";\nimport { Constructor, Model } from \"@decaf-ts/decorator-validation\";\nimport { Observable, Observer, Repository } from \"@decaf-ts/core\";\nimport { HttpAdapter } from \"./adapter\";\n\nexport class RestService<\n    M extends Model,\n    Q,\n    A extends HttpAdapter<any, Q, F, C>,\n    F extends RepositoryFlags,\n    C extends Context<F> = Context<F>,\n  >\n  implements CrudOperator<M>, BulkCrudOperator<M>, Observable\n{\n  private readonly _class!: Constructor<M>;\n  private _pk!: keyof M;\n\n  get class() {\n    if (!this._class)\n      throw new InternalError(\"No class definition found for this repository\");\n    return this._class;\n  }\n\n  get pk() {\n    if (!this._pk) this._pk = findPrimaryKey(new this.class()).id;\n    return this._pk;\n  }\n\n  protected observers: Observer[] = [];\n\n  private readonly _adapter!: A;\n  private _tableName!: string;\n\n  protected get adapter(): A {\n    if (!this._adapter)\n      throw new InternalError(\n        \"No adapter found for this repository. did you use the @uses decorator or pass it in the constructor?\"\n      );\n    return this._adapter;\n  }\n\n  protected get tableName() {\n    if (!this._tableName) this._tableName = Repository.table(this.class);\n    return this._tableName;\n  }\n\n  constructor(adapter: A, clazz?: Constructor<M>) {\n    this._adapter = adapter;\n    if (clazz) this._class = clazz;\n  }\n\n  async create(model: M, ...args: any[]): Promise<M> {\n    // eslint-disable-next-line prefer-const\n    let { record, id } = this.adapter.prepare(model, this.pk);\n    record = await this.adapter.create(this.tableName, id, record, ...args);\n    return this.adapter.revert(record, this.class, this.pk, id);\n  }\n\n  async read(id: string | number, ...args: any[]): Promise<M> {\n    const m = await this.adapter.read(this.tableName, id, ...args);\n    return this.adapter.revert(m, this.class, this.pk, id);\n  }\n\n  async update(model: M, ...args: any[]): Promise<M> {\n    // eslint-disable-next-line prefer-const\n    let { record, id } = this.adapter.prepare(model, this.pk);\n    record = await this.adapter.update(this.tableName, id, record, ...args);\n    return this.adapter.revert(record, this.class, this.pk, id);\n  }\n\n  async delete(id: string | number, ...args: any[]): Promise<M> {\n    const m = await this.adapter.delete(this.tableName, id, ...args);\n    return this.adapter.revert(m, this.class, this.pk, id);\n  }\n\n  async createAll(models: M[], ...args: any[]): Promise<M[]> {\n    if (!models.length) return models;\n    const prepared = models.map((m) => this.adapter.prepare(m, this.pk));\n    const ids = prepared.map((p) => p.id);\n    let records = prepared.map((p) => p.record);\n    records = await this.adapter.createAll(\n      this.tableName,\n      ids as (string | number)[],\n      records,\n      ...args\n    );\n    return records.map((r, i) =>\n      this.adapter.revert(r, this.class, this.pk, ids[i] as string | number)\n    );\n  }\n\n  async deleteAll(keys: string[] | number[], ...args: any[]): Promise<M[]> {\n    const results = await this.adapter.deleteAll(this.tableName, keys, ...args);\n    return results.map((r, i) =>\n      this.adapter.revert(r, this.class, this.pk, keys[i])\n    );\n  }\n\n  async readAll(keys: string[] | number[], ...args: any[]): Promise<M[]> {\n    const records = await this.adapter.readAll(this.tableName, keys, ...args);\n    return records.map((r, i) =>\n      this.adapter.revert(r, this.class, this.pk, keys[i])\n    );\n  }\n\n  async updateAll(models: M[], ...args: any[]): Promise<M[]> {\n    const records = models.map((m) => this.adapter.prepare(m, this.pk));\n    const updated = await this.adapter.updateAll(\n      this.tableName,\n      records.map((r) => r.id),\n      records.map((r) => r.record),\n      ...args\n    );\n    return updated.map((u, i) =>\n      this.adapter.revert(u, this.class, this.pk, records[i].id)\n    );\n  }\n\n  /**\n   * @summary Registers an {@link Observer}\n   * @param {Observer} observer\n   *\n   * @see {Observable#observe}\n   */\n  observe(observer: Observer): void {\n    const index = this.observers.indexOf(observer);\n    if (index !== -1) throw new InternalError(\"Observer already registered\");\n    this.observers.push(observer);\n  }\n\n  /**\n   * @summary Unregisters an {@link Observer}\n   * @param {Observer} observer\n   *\n   * @see {Observable#unObserve}\n   */\n  unObserve(observer: Observer): void {\n    const index = this.observers.indexOf(observer);\n    if (index === -1) throw new InternalError(\"Failed to find Observer\");\n    this.observers.splice(index, 1);\n  }\n\n  /**\n   * @summary calls all registered {@link Observer}s to update themselves\n   * @param {any[]} [args] optional arguments to be passed to the {@link Observer#refresh} method\n   */\n  async updateObservers(...args: any[]): Promise<void> {\n    const results = await Promise.allSettled(\n      this.observers.map((o) => o.refresh(...args))\n    );\n    results.forEach((result, i) => {\n      if (result.status === \"rejected\")\n        console.warn(\n          `Failed to update observable ${this.observers[i]}: ${result.reason}`\n        );\n    });\n  }\n}\n"]}
255
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"RestService.js","sourceRoot":"","sources":["../src/RestService.ts"],"names":[],"mappings":"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","sourcesContent":["import {\n  BulkCrudOperator,\n  Context,\n  CrudOperator,\n  findPrimaryKey,\n  InternalError,\n} from \"@decaf-ts/db-decorators\";\nimport { Constructor, Model } from \"@decaf-ts/decorator-validation\";\nimport { Observable, Observer, Repository } from \"@decaf-ts/core\";\nimport { HttpAdapter } from \"./adapter\";\nimport { HttpFlags } from \"./types\";\n\n/**\n * @description Service class for REST API operations\n * @summary Provides a comprehensive implementation for interacting with REST APIs.\n * This class implements CRUD operations for single and bulk operations, as well as\n * the Observable pattern to notify observers of changes. It works with HTTP adapters\n * to perform the actual API requests and handles model conversion.\n * @template M - The model type, extending Model\n * @template Q - The query type used by the adapter\n * @template A - The HTTP adapter type, extending HttpAdapter\n * @template F - The HTTP flags type, extending HttpFlags\n * @template C - The context type, extending Context<F>\n * @param {A} adapter - The HTTP adapter instance\n * @param {Constructor<M>} [clazz] - Optional constructor for the model class\n * @example\n * ```typescript\n * // Create a service for User model with Axios adapter\n * const axiosAdapter = new AxiosAdapter({\n *   protocol: 'https',\n *   host: 'api.example.com'\n * });\n * const userService = new RestService(axiosAdapter, User);\n *\n * // Create a new user\n * const user = new User({ name: 'John Doe', email: 'john@example.com' });\n * const createdUser = await userService.create(user);\n *\n * // Update a user\n * createdUser.name = 'Jane Doe';\n * const updatedUser = await userService.update(createdUser);\n *\n * // Delete a user\n * await userService.delete(updatedUser.id);\n * ```\n * @class\n */\nexport class RestService<\n    M extends Model,\n    Q,\n    A extends HttpAdapter<any, Q, F, C>,\n    F extends HttpFlags = HttpFlags,\n    C extends Context<F> = Context<F>,\n  >\n  implements CrudOperator<M>, BulkCrudOperator<M>, Observable\n{\n  private readonly _class!: Constructor<M>;\n  private _pk!: keyof M;\n\n  /**\n   * @description Gets the model class constructor\n   * @summary Retrieves the model class constructor associated with this service.\n   * Throws an error if no class definition is found.\n   * @return {Constructor<M>} The model class constructor\n   * @throws {InternalError} If no class definition is found\n   */\n  get class() {\n    if (!this._class)\n      throw new InternalError(\"No class definition found for this repository\");\n    return this._class;\n  }\n\n  /**\n   * @description Gets the primary key property name\n   * @summary Retrieves the name of the primary key property for the model.\n   * If not already determined, it finds the primary key using the model class.\n   * @return The primary key property name\n   */\n  get pk() {\n    if (!this._pk) this._pk = findPrimaryKey(new this.class()).id;\n    return this._pk;\n  }\n\n  protected observers: Observer[] = [];\n\n  private readonly _adapter!: A;\n  private _tableName!: string;\n\n  /**\n   * @description Gets the HTTP adapter\n   * @summary Retrieves the HTTP adapter associated with this service.\n   * Throws an error if no adapter is found.\n   * @return {A} The HTTP adapter instance\n   * @throws {InternalError} If no adapter is found\n   */\n  protected get adapter(): A {\n    if (!this._adapter)\n      throw new InternalError(\n        \"No adapter found for this repository. did you use the @uses decorator or pass it in the constructor?\"\n      );\n    return this._adapter;\n  }\n\n  /**\n   * @description Gets the table name for the model\n   * @summary Retrieves the table name associated with the model class.\n   * If not already determined, it gets the table name from the Repository utility.\n   * @return {string} The table name\n   */\n  protected get tableName() {\n    if (!this._tableName) this._tableName = Repository.table(this.class);\n    return this._tableName;\n  }\n\n  /**\n   * @description Initializes a new RestService instance\n   * @summary Creates a new service instance with the specified adapter and optional model class.\n   * The constructor stores the adapter and model class for later use in CRUD operations.\n   * @param {A} adapter - The HTTP adapter instance to use for API requests\n   * @param {Constructor<M>} [clazz] - Optional constructor for the model class\n   */\n  constructor(adapter: A, clazz?: Constructor<M>) {\n    this._adapter = adapter;\n    if (clazz) this._class = clazz;\n  }\n\n  /**\n   * @description Creates a new resource\n   * @summary Creates a new resource in the REST API using the provided model.\n   * The method prepares the model for the adapter, sends the create request,\n   * and then converts the response back to a model instance.\n   * @param {M} model - The model instance to create\n   * @param {...any[]} args - Additional arguments to pass to the adapter\n   * @return {Promise<M>} A promise that resolves with the created model instance\n   */\n  async create(model: M, ...args: any[]): Promise<M> {\n    // eslint-disable-next-line prefer-const\n    let { record, id } = this.adapter.prepare(model, this.pk);\n    record = await this.adapter.create(this.tableName, id, record, ...args);\n    return this.adapter.revert(record, this.class, this.pk, id);\n  }\n\n  /**\n   * @description Retrieves a resource by ID\n   * @summary Fetches a resource from the REST API using the provided ID.\n   * The method sends the read request and converts the response to a model instance.\n   * @param {string|number} id - The identifier of the resource to retrieve\n   * @param {...any[]} args - Additional arguments to pass to the adapter\n   * @return {Promise<M>} A promise that resolves with the retrieved model instance\n   */\n  async read(id: string | number, ...args: any[]): Promise<M> {\n    const m = await this.adapter.read(this.tableName, id, ...args);\n    return this.adapter.revert(m, this.class, this.pk, id);\n  }\n\n  /**\n   * @description Updates an existing resource\n   * @summary Updates an existing resource in the REST API using the provided model.\n   * The method prepares the model for the adapter, sends the update request,\n   * and then converts the response back to a model instance.\n   * @param {M} model - The model instance with updated data\n   * @param {...any[]} args - Additional arguments to pass to the adapter\n   * @return {Promise<M>} A promise that resolves with the updated model instance\n   */\n  async update(model: M, ...args: any[]): Promise<M> {\n    // eslint-disable-next-line prefer-const\n    let { record, id } = this.adapter.prepare(model, this.pk);\n    record = await this.adapter.update(this.tableName, id, record, ...args);\n    return this.adapter.revert(record, this.class, this.pk, id);\n  }\n\n  /**\n   * @description Deletes a resource by ID\n   * @summary Removes a resource from the REST API using the provided ID.\n   * The method sends the delete request and converts the response to a model instance.\n   * @param {string|number} id - The identifier of the resource to delete\n   * @param {...any[]} args - Additional arguments to pass to the adapter\n   * @return {Promise<M>} A promise that resolves with the deleted model instance\n   */\n  async delete(id: string | number, ...args: any[]): Promise<M> {\n    const m = await this.adapter.delete(this.tableName, id, ...args);\n    return this.adapter.revert(m, this.class, this.pk, id);\n  }\n\n  /**\n   * @description Creates multiple resources\n   * @summary Creates multiple resources in the REST API using the provided models.\n   * The method prepares each model for the adapter, sends a bulk create request,\n   * and then converts the responses back to model instances.\n   * @param {M[]} models - The model instances to create\n   * @param {...any[]} args - Additional arguments to pass to the adapter\n   * @return {Promise<M[]>} A promise that resolves with an array of created model instances\n   */\n  async createAll(models: M[], ...args: any[]): Promise<M[]> {\n    if (!models.length) return models;\n    const prepared = models.map((m) => this.adapter.prepare(m, this.pk));\n    const ids = prepared.map((p) => p.id);\n    let records = prepared.map((p) => p.record);\n    records = await this.adapter.createAll(\n      this.tableName,\n      ids as (string | number)[],\n      records,\n      ...args\n    );\n    return records.map((r, i) =>\n      this.adapter.revert(r, this.class, this.pk, ids[i] as string | number)\n    );\n  }\n\n  /**\n   * @description Deletes multiple resources by IDs\n   * @summary Removes multiple resources from the REST API using the provided IDs.\n   * The method sends a bulk delete request and converts the responses to model instances.\n   * @param {string[]|number[]} keys - The identifiers of the resources to delete\n   * @param {...any[]} args - Additional arguments to pass to the adapter\n   * @return {Promise<M[]>} A promise that resolves with an array of deleted model instances\n   */\n  async deleteAll(keys: string[] | number[], ...args: any[]): Promise<M[]> {\n    const results = await this.adapter.deleteAll(this.tableName, keys, ...args);\n    return results.map((r, i) =>\n      this.adapter.revert(r, this.class, this.pk, keys[i])\n    );\n  }\n\n  /**\n   * @description Retrieves multiple resources by IDs\n   * @summary Fetches multiple resources from the REST API using the provided IDs.\n   * The method sends a bulk read request and converts the responses to model instances.\n   * @param {string[]|number[]} keys - The identifiers of the resources to retrieve\n   * @param {...any[]} args - Additional arguments to pass to the adapter\n   * @return {Promise<M[]>} A promise that resolves with an array of retrieved model instances\n   */\n  async readAll(keys: string[] | number[], ...args: any[]): Promise<M[]> {\n    const records = await this.adapter.readAll(this.tableName, keys, ...args);\n    return records.map((r, i) =>\n      this.adapter.revert(r, this.class, this.pk, keys[i])\n    );\n  }\n\n  /**\n   * @description Updates multiple resources\n   * @summary Updates multiple resources in the REST API using the provided models.\n   * The method prepares each model for the adapter, sends a bulk update request,\n   * and then converts the responses back to model instances.\n   * @param {M[]} models - The model instances with updated data\n   * @param {...any[]} args - Additional arguments to pass to the adapter\n   * @return {Promise<M[]>} A promise that resolves with an array of updated model instances\n   */\n  async updateAll(models: M[], ...args: any[]): Promise<M[]> {\n    const records = models.map((m) => this.adapter.prepare(m, this.pk));\n    const updated = await this.adapter.updateAll(\n      this.tableName,\n      records.map((r) => r.id),\n      records.map((r) => r.record),\n      ...args\n    );\n    return updated.map((u, i) =>\n      this.adapter.revert(u, this.class, this.pk, records[i].id)\n    );\n  }\n\n  /**\n   * @description Registers an observer\n   * @summary Adds an observer to the list of observers that will be notified of changes.\n   * Throws an error if the observer is already registered.\n   * @param {Observer} observer - The observer to register\n   * @return {void}\n   * @throws {InternalError} If the observer is already registered\n   */\n  observe(observer: Observer): void {\n    const index = this.observers.indexOf(observer);\n    if (index !== -1) throw new InternalError(\"Observer already registered\");\n    this.observers.push(observer);\n  }\n\n  /**\n   * @description Unregisters an observer\n   * @summary Removes an observer from the list of observers.\n   * Throws an error if the observer is not found.\n   * @param {Observer} observer - The observer to unregister\n   * @return {void}\n   * @throws {InternalError} If the observer is not found\n   */\n  unObserve(observer: Observer): void {\n    const index = this.observers.indexOf(observer);\n    if (index === -1) throw new InternalError(\"Failed to find Observer\");\n    this.observers.splice(index, 1);\n  }\n\n  /**\n   * @description Notifies all registered observers\n   * @summary Calls the refresh method on all registered observers to update themselves.\n   * Any errors during observer refresh are logged as warnings but don't stop the process.\n   * @param {...any[]} [args] - Optional arguments to pass to the observer refresh method\n   * @return {Promise<void>} A promise that resolves when all observers have been updated\n   */\n  async updateObservers(...args: any[]): Promise<void> {\n    const results = await Promise.allSettled(\n      this.observers.map((o) => o.refresh(...args))\n    );\n    results.forEach((result, i) => {\n      if (result.status === \"rejected\")\n        console.warn(\n          `Failed to update observable ${this.observers[i]}: ${result.reason}`\n        );\n    });\n  }\n}\n"]}
@@ -1,43 +1,187 @@
1
- import { BulkCrudOperator, Context, CrudOperator, RepositoryFlags } from "@decaf-ts/db-decorators";
1
+ import { BulkCrudOperator, Context, CrudOperator } from "@decaf-ts/db-decorators";
2
2
  import { Constructor, Model } from "@decaf-ts/decorator-validation";
3
3
  import { Observable, Observer } from "@decaf-ts/core";
4
4
  import { HttpAdapter } from "./adapter";
5
- export declare class RestService<M extends Model, Q, A extends HttpAdapter<any, Q, F, C>, F extends RepositoryFlags, C extends Context<F> = Context<F>> implements CrudOperator<M>, BulkCrudOperator<M>, Observable {
5
+ import { HttpFlags } from "./types";
6
+ /**
7
+ * @description Service class for REST API operations
8
+ * @summary Provides a comprehensive implementation for interacting with REST APIs.
9
+ * This class implements CRUD operations for single and bulk operations, as well as
10
+ * the Observable pattern to notify observers of changes. It works with HTTP adapters
11
+ * to perform the actual API requests and handles model conversion.
12
+ * @template M - The model type, extending Model
13
+ * @template Q - The query type used by the adapter
14
+ * @template A - The HTTP adapter type, extending HttpAdapter
15
+ * @template F - The HTTP flags type, extending HttpFlags
16
+ * @template C - The context type, extending Context<F>
17
+ * @param {A} adapter - The HTTP adapter instance
18
+ * @param {Constructor<M>} [clazz] - Optional constructor for the model class
19
+ * @example
20
+ * ```typescript
21
+ * // Create a service for User model with Axios adapter
22
+ * const axiosAdapter = new AxiosAdapter({
23
+ * protocol: 'https',
24
+ * host: 'api.example.com'
25
+ * });
26
+ * const userService = new RestService(axiosAdapter, User);
27
+ *
28
+ * // Create a new user
29
+ * const user = new User({ name: 'John Doe', email: 'john@example.com' });
30
+ * const createdUser = await userService.create(user);
31
+ *
32
+ * // Update a user
33
+ * createdUser.name = 'Jane Doe';
34
+ * const updatedUser = await userService.update(createdUser);
35
+ *
36
+ * // Delete a user
37
+ * await userService.delete(updatedUser.id);
38
+ * ```
39
+ * @class
40
+ */
41
+ export declare class RestService<M extends Model, Q, A extends HttpAdapter<any, Q, F, C>, F extends HttpFlags = HttpFlags, C extends Context<F> = Context<F>> implements CrudOperator<M>, BulkCrudOperator<M>, Observable {
6
42
  private readonly _class;
7
43
  private _pk;
44
+ /**
45
+ * @description Gets the model class constructor
46
+ * @summary Retrieves the model class constructor associated with this service.
47
+ * Throws an error if no class definition is found.
48
+ * @return {Constructor<M>} The model class constructor
49
+ * @throws {InternalError} If no class definition is found
50
+ */
8
51
  get class(): Constructor<M>;
52
+ /**
53
+ * @description Gets the primary key property name
54
+ * @summary Retrieves the name of the primary key property for the model.
55
+ * If not already determined, it finds the primary key using the model class.
56
+ * @return The primary key property name
57
+ */
9
58
  get pk(): keyof M;
10
59
  protected observers: Observer[];
11
60
  private readonly _adapter;
12
61
  private _tableName;
62
+ /**
63
+ * @description Gets the HTTP adapter
64
+ * @summary Retrieves the HTTP adapter associated with this service.
65
+ * Throws an error if no adapter is found.
66
+ * @return {A} The HTTP adapter instance
67
+ * @throws {InternalError} If no adapter is found
68
+ */
13
69
  protected get adapter(): A;
70
+ /**
71
+ * @description Gets the table name for the model
72
+ * @summary Retrieves the table name associated with the model class.
73
+ * If not already determined, it gets the table name from the Repository utility.
74
+ * @return {string} The table name
75
+ */
14
76
  protected get tableName(): string;
77
+ /**
78
+ * @description Initializes a new RestService instance
79
+ * @summary Creates a new service instance with the specified adapter and optional model class.
80
+ * The constructor stores the adapter and model class for later use in CRUD operations.
81
+ * @param {A} adapter - The HTTP adapter instance to use for API requests
82
+ * @param {Constructor<M>} [clazz] - Optional constructor for the model class
83
+ */
15
84
  constructor(adapter: A, clazz?: Constructor<M>);
85
+ /**
86
+ * @description Creates a new resource
87
+ * @summary Creates a new resource in the REST API using the provided model.
88
+ * The method prepares the model for the adapter, sends the create request,
89
+ * and then converts the response back to a model instance.
90
+ * @param {M} model - The model instance to create
91
+ * @param {...any[]} args - Additional arguments to pass to the adapter
92
+ * @return {Promise<M>} A promise that resolves with the created model instance
93
+ */
16
94
  create(model: M, ...args: any[]): Promise<M>;
95
+ /**
96
+ * @description Retrieves a resource by ID
97
+ * @summary Fetches a resource from the REST API using the provided ID.
98
+ * The method sends the read request and converts the response to a model instance.
99
+ * @param {string|number} id - The identifier of the resource to retrieve
100
+ * @param {...any[]} args - Additional arguments to pass to the adapter
101
+ * @return {Promise<M>} A promise that resolves with the retrieved model instance
102
+ */
17
103
  read(id: string | number, ...args: any[]): Promise<M>;
104
+ /**
105
+ * @description Updates an existing resource
106
+ * @summary Updates an existing resource in the REST API using the provided model.
107
+ * The method prepares the model for the adapter, sends the update request,
108
+ * and then converts the response back to a model instance.
109
+ * @param {M} model - The model instance with updated data
110
+ * @param {...any[]} args - Additional arguments to pass to the adapter
111
+ * @return {Promise<M>} A promise that resolves with the updated model instance
112
+ */
18
113
  update(model: M, ...args: any[]): Promise<M>;
114
+ /**
115
+ * @description Deletes a resource by ID
116
+ * @summary Removes a resource from the REST API using the provided ID.
117
+ * The method sends the delete request and converts the response to a model instance.
118
+ * @param {string|number} id - The identifier of the resource to delete
119
+ * @param {...any[]} args - Additional arguments to pass to the adapter
120
+ * @return {Promise<M>} A promise that resolves with the deleted model instance
121
+ */
19
122
  delete(id: string | number, ...args: any[]): Promise<M>;
123
+ /**
124
+ * @description Creates multiple resources
125
+ * @summary Creates multiple resources in the REST API using the provided models.
126
+ * The method prepares each model for the adapter, sends a bulk create request,
127
+ * and then converts the responses back to model instances.
128
+ * @param {M[]} models - The model instances to create
129
+ * @param {...any[]} args - Additional arguments to pass to the adapter
130
+ * @return {Promise<M[]>} A promise that resolves with an array of created model instances
131
+ */
20
132
  createAll(models: M[], ...args: any[]): Promise<M[]>;
133
+ /**
134
+ * @description Deletes multiple resources by IDs
135
+ * @summary Removes multiple resources from the REST API using the provided IDs.
136
+ * The method sends a bulk delete request and converts the responses to model instances.
137
+ * @param {string[]|number[]} keys - The identifiers of the resources to delete
138
+ * @param {...any[]} args - Additional arguments to pass to the adapter
139
+ * @return {Promise<M[]>} A promise that resolves with an array of deleted model instances
140
+ */
21
141
  deleteAll(keys: string[] | number[], ...args: any[]): Promise<M[]>;
142
+ /**
143
+ * @description Retrieves multiple resources by IDs
144
+ * @summary Fetches multiple resources from the REST API using the provided IDs.
145
+ * The method sends a bulk read request and converts the responses to model instances.
146
+ * @param {string[]|number[]} keys - The identifiers of the resources to retrieve
147
+ * @param {...any[]} args - Additional arguments to pass to the adapter
148
+ * @return {Promise<M[]>} A promise that resolves with an array of retrieved model instances
149
+ */
22
150
  readAll(keys: string[] | number[], ...args: any[]): Promise<M[]>;
151
+ /**
152
+ * @description Updates multiple resources
153
+ * @summary Updates multiple resources in the REST API using the provided models.
154
+ * The method prepares each model for the adapter, sends a bulk update request,
155
+ * and then converts the responses back to model instances.
156
+ * @param {M[]} models - The model instances with updated data
157
+ * @param {...any[]} args - Additional arguments to pass to the adapter
158
+ * @return {Promise<M[]>} A promise that resolves with an array of updated model instances
159
+ */
23
160
  updateAll(models: M[], ...args: any[]): Promise<M[]>;
24
161
  /**
25
- * @summary Registers an {@link Observer}
26
- * @param {Observer} observer
27
- *
28
- * @see {Observable#observe}
162
+ * @description Registers an observer
163
+ * @summary Adds an observer to the list of observers that will be notified of changes.
164
+ * Throws an error if the observer is already registered.
165
+ * @param {Observer} observer - The observer to register
166
+ * @return {void}
167
+ * @throws {InternalError} If the observer is already registered
29
168
  */
30
169
  observe(observer: Observer): void;
31
170
  /**
32
- * @summary Unregisters an {@link Observer}
33
- * @param {Observer} observer
34
- *
35
- * @see {Observable#unObserve}
171
+ * @description Unregisters an observer
172
+ * @summary Removes an observer from the list of observers.
173
+ * Throws an error if the observer is not found.
174
+ * @param {Observer} observer - The observer to unregister
175
+ * @return {void}
176
+ * @throws {InternalError} If the observer is not found
36
177
  */
37
178
  unObserve(observer: Observer): void;
38
179
  /**
39
- * @summary calls all registered {@link Observer}s to update themselves
40
- * @param {any[]} [args] optional arguments to be passed to the {@link Observer#refresh} method
180
+ * @description Notifies all registered observers
181
+ * @summary Calls the refresh method on all registered observers to update themselves.
182
+ * Any errors during observer refresh are logged as warnings but don't stop the process.
183
+ * @param {...any[]} [args] - Optional arguments to pass to the observer refresh method
184
+ * @return {Promise<void>} A promise that resolves when all observers have been updated
41
185
  */
42
186
  updateObservers(...args: any[]): Promise<void>;
43
187
  }