@decaf-ts/for-http 0.2.1 → 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.
- package/LICENSE.md +21 -157
- package/dist/for-http.cjs +310 -12
- package/dist/for-http.esm.cjs +310 -12
- package/lib/RestRepository.cjs +30 -1
- package/lib/RestRepository.d.ts +29 -0
- package/lib/RestService.cjs +154 -11
- package/lib/RestService.d.ts +153 -10
- package/lib/adapter.cjs +114 -1
- package/lib/adapter.d.ts +159 -0
- package/lib/axios/axios.cjs +95 -1
- package/lib/axios/axios.d.ts +94 -0
- package/lib/axios/constants.cjs +7 -1
- package/lib/axios/constants.d.ts +6 -0
- package/lib/axios/index.cjs +10 -1
- package/lib/axios/types.cjs +1 -1
- package/lib/axios/types.d.ts +6 -0
- package/lib/esm/RestRepository.d.ts +29 -0
- package/lib/esm/RestRepository.js +30 -1
- package/lib/esm/RestService.d.ts +153 -10
- package/lib/esm/RestService.js +154 -11
- package/lib/esm/adapter.d.ts +159 -0
- package/lib/esm/adapter.js +115 -2
- package/lib/esm/axios/axios.d.ts +94 -0
- package/lib/esm/axios/axios.js +97 -3
- package/lib/esm/axios/constants.d.ts +6 -0
- package/lib/esm/axios/constants.js +7 -1
- package/lib/esm/axios/index.js +14 -5
- package/lib/esm/axios/types.d.ts +6 -0
- package/lib/esm/axios/types.js +1 -1
- package/lib/esm/index.d.ts +14 -1
- package/lib/esm/index.js +19 -6
- package/lib/esm/types.d.ts +15 -0
- package/lib/esm/types.js +1 -1
- package/lib/index.cjs +15 -2
- package/lib/index.d.ts +14 -1
- package/lib/types.cjs +1 -1
- package/lib/types.d.ts +15 -0
- package/package.json +2 -2
package/lib/RestRepository.cjs
CHANGED
|
@@ -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,
|
|
40
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiUmVzdFJlcG9zaXRvcnkuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi9zcmMvUmVzdFJlcG9zaXRvcnkudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6Ijs7O0FBQUEseUNBQTRDO0FBTTVDOzs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7O0dBNEJHO0FBQ0gsTUFBYSxjQU1YLFNBQVEsaUJBQW1CO0lBQzNCLFlBQVksT0FBVSxFQUFFLEtBQXNCO1FBQzVDLEtBQUssQ0FBQyxPQUFPLEVBQUUsS0FBSyxDQUFDLENBQUM7SUFDeEIsQ0FBQztDQUNGO0FBVkQsd0NBVUMiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgeyBSZXBvc2l0b3J5IH0gZnJvbSBcIkBkZWNhZi10cy9jb3JlXCI7XG5pbXBvcnQgeyBDb25zdHJ1Y3RvciwgTW9kZWwgfSBmcm9tIFwiQGRlY2FmLXRzL2RlY29yYXRvci12YWxpZGF0aW9uXCI7XG5pbXBvcnQgeyBIdHRwQWRhcHRlciB9IGZyb20gXCIuL2FkYXB0ZXJcIjtcbmltcG9ydCB7IENvbnRleHQgfSBmcm9tIFwiQGRlY2FmLXRzL2RiLWRlY29yYXRvcnNcIjtcbmltcG9ydCB7IEh0dHBGbGFncyB9IGZyb20gXCIuL3R5cGVzXCI7XG5cbi8qKlxuICogQGRlc2NyaXB0aW9uIFJlcG9zaXRvcnkgZm9yIFJFU1QgQVBJIGludGVyYWN0aW9uc1xuICogQHN1bW1hcnkgQSBzcGVjaWFsaXplZCByZXBvc2l0b3J5IGltcGxlbWVudGF0aW9uIGZvciBpbnRlcmFjdGluZyB3aXRoIFJFU1QgQVBJcy5cbiAqIFRoaXMgY2xhc3MgZXh0ZW5kcyB0aGUgY29yZSBSZXBvc2l0b3J5IGNsYXNzIGFuZCB3b3JrcyB3aXRoIEhUVFAgYWRhcHRlcnMgdG9cbiAqIHByb3ZpZGUgQ1JVRCBvcGVyYXRpb25zIGZvciBtb2RlbHMgdmlhIFJFU1QgZW5kcG9pbnRzLlxuICogVGhpcyBJcyBOT1QgdGhlIGRlZmF1bHQgcmVwb3NpdG9yeSBmb3IgdGhlIEhUVFAgYWRhcHRlci4gVGhhdCB3b3VsZCBiZSB7QGxpbmsgUmVzdFNlcnZpY2V9LlxuICogVXNlIHRoaXMgb25seSBpbiB0aGUgc3BlY2lmaWMgY2FzZSBvZiBuZWVkaW5nIHRvIHJ1biB0aGUgQ1VSRCBtb2RlbCBsb2dpYyAoZGVjb3JhdGlvbikgYmVmb3JlIHN1Ym1pdHRpbmcgdG8gdGhlIGJhY2tlbmRcbiAqIEB0ZW1wbGF0ZSBNIC0gVGhlIG1vZGVsIHR5cGUsIGV4dGVuZGluZyBNb2RlbFxuICogQHRlbXBsYXRlIFEgLSBUaGUgcXVlcnkgdHlwZSB1c2VkIGJ5IHRoZSBhZGFwdGVyXG4gKiBAdGVtcGxhdGUgQSAtIFRoZSBIVFRQIGFkYXB0ZXIgdHlwZSwgZXh0ZW5kaW5nIEh0dHBBZGFwdGVyXG4gKiBAdGVtcGxhdGUgRiAtIFRoZSBIVFRQIGZsYWdzIHR5cGUsIGV4dGVuZGluZyBIdHRwRmxhZ3NcbiAqIEB0ZW1wbGF0ZSBDIC0gVGhlIGNvbnRleHQgdHlwZSwgZXh0ZW5kaW5nIENvbnRleHQ8Rj5cbiAqIEBwYXJhbSB7QX0gYWRhcHRlciAtIFRoZSBIVFRQIGFkYXB0ZXIgaW5zdGFuY2VcbiAqIEBwYXJhbSB7Q29uc3RydWN0b3I8TT59IFtjbGF6el0gLSBPcHRpb25hbCBjb25zdHJ1Y3RvciBmb3IgdGhlIG1vZGVsIGNsYXNzXG4gKiBAZXhhbXBsZVxuICogYGBgdHlwZXNjcmlwdFxuICogLy8gQ3JlYXRlIGEgcmVwb3NpdG9yeSBmb3IgVXNlciBtb2RlbCB3aXRoIEF4aW9zIGFkYXB0ZXJcbiAqIGNvbnN0IGF4aW9zQWRhcHRlciA9IG5ldyBBeGlvc0FkYXB0ZXIoe1xuICogICBwcm90b2NvbDogJ2h0dHBzJyxcbiAqICAgaG9zdDogJ2FwaS5leGFtcGxlLmNvbSdcbiAqIH0pO1xuICogY29uc3QgdXNlclJlcG9zaXRvcnkgPSBuZXcgUmVzdFJlcG9zaXRvcnkoYXhpb3NBZGFwdGVyLCBVc2VyKTtcbiAqXG4gKiAvLyBVc2UgdGhlIHJlcG9zaXRvcnkgZm9yIENSVUQgb3BlcmF0aW9uc1xuICogY29uc3QgdXNlciA9IGF3YWl0IHVzZXJSZXBvc2l0b3J5LmZpbmRCeUlkKCcxMjMnKTtcbiAqIGBgYFxuICogQGNsYXNzIFJlc3RSZXBvc2l0b3J5XG4gKiBAc2VlIHtAbGluayBSZXN0U2VydmljZX1cbiAqL1xuZXhwb3J0IGNsYXNzIFJlc3RSZXBvc2l0b3J5PFxuICBNIGV4dGVuZHMgTW9kZWwsXG4gIFEsXG4gIEEgZXh0ZW5kcyBIdHRwQWRhcHRlcjxhbnksIFEsIEYsIEM+LFxuICBGIGV4dGVuZHMgSHR0cEZsYWdzID0gSHR0cEZsYWdzLFxuICBDIGV4dGVuZHMgQ29udGV4dDxGPiA9IENvbnRleHQ8Rj4sXG4+IGV4dGVuZHMgUmVwb3NpdG9yeTxNLCBRLCBBPiB7XG4gIGNvbnN0cnVjdG9yKGFkYXB0ZXI6IEEsIGNsYXp6PzogQ29uc3RydWN0b3I8TT4pIHtcbiAgICBzdXBlcihhZGFwdGVyLCBjbGF6eik7XG4gIH1cbn1cbiJdfQ==
|
package/lib/RestRepository.d.ts
CHANGED
|
@@ -3,6 +3,35 @@ import { Constructor, Model } from "@decaf-ts/decorator-validation";
|
|
|
3
3
|
import { HttpAdapter } from "./adapter";
|
|
4
4
|
import { Context } from "@decaf-ts/db-decorators";
|
|
5
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
|
+
*/
|
|
6
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> {
|
|
7
36
|
constructor(adapter: A, clazz?: Constructor<M>);
|
|
8
37
|
}
|
package/lib/RestService.cjs
CHANGED
|
@@ -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
|
-
* @
|
|
77
|
-
* @
|
|
78
|
-
*
|
|
79
|
-
* @
|
|
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
|
-
* @
|
|
89
|
-
* @
|
|
90
|
-
*
|
|
91
|
-
* @
|
|
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
|
-
* @
|
|
101
|
-
* @
|
|
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} 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\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  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"]}
|
package/lib/RestService.d.ts
CHANGED
|
@@ -3,42 +3,185 @@ import { Constructor, Model } from "@decaf-ts/decorator-validation";
|
|
|
3
3
|
import { Observable, Observer } from "@decaf-ts/core";
|
|
4
4
|
import { HttpAdapter } from "./adapter";
|
|
5
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
|
+
*/
|
|
6
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 {
|
|
7
42
|
private readonly _class;
|
|
8
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
|
+
*/
|
|
9
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
|
+
*/
|
|
10
58
|
get pk(): keyof M;
|
|
11
59
|
protected observers: Observer[];
|
|
12
60
|
private readonly _adapter;
|
|
13
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
|
+
*/
|
|
14
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
|
+
*/
|
|
15
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
|
+
*/
|
|
16
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
|
+
*/
|
|
17
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
|
+
*/
|
|
18
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
|
+
*/
|
|
19
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
|
+
*/
|
|
20
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
|
+
*/
|
|
21
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
|
+
*/
|
|
22
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
|
+
*/
|
|
23
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
|
+
*/
|
|
24
160
|
updateAll(models: M[], ...args: any[]): Promise<M[]>;
|
|
25
161
|
/**
|
|
26
|
-
* @
|
|
27
|
-
* @
|
|
28
|
-
*
|
|
29
|
-
* @
|
|
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
|
|
30
168
|
*/
|
|
31
169
|
observe(observer: Observer): void;
|
|
32
170
|
/**
|
|
33
|
-
* @
|
|
34
|
-
* @
|
|
35
|
-
*
|
|
36
|
-
* @
|
|
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
|
|
37
177
|
*/
|
|
38
178
|
unObserve(observer: Observer): void;
|
|
39
179
|
/**
|
|
40
|
-
* @
|
|
41
|
-
* @
|
|
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
|
|
42
185
|
*/
|
|
43
186
|
updateObservers(...args: any[]): Promise<void>;
|
|
44
187
|
}
|