@decaf-ts/for-http 0.2.5 → 0.2.7
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/LICENSE.md +3 -2
- package/README.md +212 -276
- package/dist/for-http.cjs +1 -1
- package/dist/for-http.esm.cjs +1 -1
- package/lib/esm/index.d.ts +1 -1
- package/lib/esm/index.js +1 -1
- package/lib/index.cjs +1 -1
- package/lib/index.d.ts +1 -1
- package/package.json +12 -4
package/LICENSE.md
CHANGED
|
@@ -1,6 +1,6 @@
|
|
|
1
|
-
|
|
1
|
+
MIT License
|
|
2
2
|
|
|
3
|
-
Copyright (c) 2025 Tiago Venceslau
|
|
3
|
+
Copyright (c) 2025 Tiago Venceslau and Contributors
|
|
4
4
|
|
|
5
5
|
Permission is hereby granted, free of charge, to any person obtaining a copy
|
|
6
6
|
of this software and associated documentation files (the "Software"), to deal
|
|
@@ -19,3 +19,4 @@ AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
|
|
19
19
|
LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
|
20
20
|
OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
|
|
21
21
|
SOFTWARE.
|
|
22
|
+
|
package/README.md
CHANGED
|
@@ -1,8 +1,8 @@
|
|
|
1
|
-

|
|
2
2
|
|
|
3
|
-
|
|
3
|
+
# decaf-ts/for-http
|
|
4
4
|
|
|
5
|
-
A
|
|
5
|
+
A lightweight HTTP adapter layer for decaf-ts that enables CRUD-style repositories and services over REST APIs. It defines a generic HttpAdapter with concrete implementations (e.g., Axios), a RestService for simple model-centric operations, and a RestRepository when you need repository decoration logic before submitting to the backend. Includes minimal types for configuration and request flags.
|
|
6
6
|
|
|
7
7
|
|
|
8
8
|

|
|
@@ -28,344 +28,280 @@ A TypeScript library for seamless REST API interactions. This module provides a
|
|
|
28
28
|
|
|
29
29
|
Documentation available [here](https://decaf-ts.github.io/for-http/)
|
|
30
30
|
|
|
31
|
-
|
|
31
|
+
# decaf-ts/for-http — Detailed Description
|
|
32
|
+
|
|
33
|
+
This package provides a small, focused HTTP integration for decaf-ts. It introduces a generic HttpAdapter abstraction that maps REST semantics (create/read/update/delete and bulk variants) onto decaf-ts repositories and services, plus minimal configuration and flags. A ready-to-use Axios adapter is included.
|
|
34
|
+
|
|
35
|
+
Core goals
|
|
36
|
+
- Keep HTTP concerns decoupled from models and repositories.
|
|
37
|
+
- Provide a consistent CRUD surface over REST endpoints.
|
|
38
|
+
- Allow custom clients (via subclassing HttpAdapter) while shipping an Axios implementation.
|
|
39
|
+
- Offer simple configuration (protocol/host) and request-scoped headers via flags/context.
|
|
40
|
+
|
|
41
|
+
Key building blocks
|
|
42
|
+
- HttpConfig: A minimal connection config with protocol and host.
|
|
43
|
+
- HttpFlags: Extends RepositoryFlags to include optional HTTP headers.
|
|
44
|
+
- HttpAdapter<Y, CON, Q, F, C>:
|
|
45
|
+
- Extends the core Adapter to focus on HTTP.
|
|
46
|
+
- Adds default flags() with headers support.
|
|
47
|
+
- Provides URL building (protected url()) and error parsing (parseError()).
|
|
48
|
+
- Declares abstract request(), create(), read(), update(), delete().
|
|
49
|
+
- Declares optional/unsupported-by-default raw(), Sequence(), Statement(), parseCondition() that concrete adapters may implement if needed.
|
|
50
|
+
- repository() returns RestService as the default repository/service implementation for this adapter type.
|
|
51
|
+
- RestService<M, Q, A, F, C>:
|
|
52
|
+
- Lightweight, model-centric service that delegates to the HttpAdapter for CRUD and bulk operations.
|
|
53
|
+
- Converts between model instances and plain records using adapter.prepare() and adapter.revert().
|
|
54
|
+
- Manages a list of observers (observe/unObserve/updateObservers) that can be refreshed after changes.
|
|
55
|
+
- RestRepository<M, Q, A, F, C>:
|
|
56
|
+
- A Repository that works with an HttpAdapter; use it if you need decaf-ts repository decoration/logic before sending to the backend.
|
|
57
|
+
- Not the default repository for the HTTP adapter (that role is fulfilled by RestService); intended for cases where repository lifecycle logic matters.
|
|
58
|
+
- AxiosHttpAdapter:
|
|
59
|
+
- Concrete implementation of HttpAdapter built on Axios.
|
|
60
|
+
- Implements request and CRUD operations using Axios.request/get/post/put/delete.
|
|
61
|
+
- Uses HttpConfig for base URL construction and inherits header flag behavior.
|
|
62
|
+
|
|
63
|
+
Flow overview
|
|
64
|
+
1. Instantiate a concrete adapter (e.g., AxiosHttpAdapter) with an HttpConfig.
|
|
65
|
+
2. Get a repository/service for a given model (the default is RestService via adapter.getRepository(), or instantiate RestService/RestRepository directly).
|
|
66
|
+
3. Call CRUD methods (create/read/update/delete) or their bulk equivalents (createAll/readAll/updateAll/deleteAll) on the service or repository. The service:
|
|
67
|
+
- Derives the table/collection name from the model’s decaf-ts metadata.
|
|
68
|
+
- Uses adapter.prepare() to serialize the model and extract its ID.
|
|
69
|
+
- Invokes the adapter’s HTTP methods.
|
|
70
|
+
- Uses adapter.revert() to rehydrate responses back into model instances.
|
|
71
|
+
4. Optionally provide HttpFlags (e.g., headers) in the Context to influence requests.
|
|
72
|
+
|
|
73
|
+
URL building and error handling
|
|
74
|
+
- HttpAdapter.url() builds URLs as `${protocol}://${host}/${tableName}` and appends encoded query parameters when provided, ensuring spaces are encoded as %20.
|
|
75
|
+
- HttpAdapter.parseError() currently returns the error unchanged (as BaseError) but is intended to be overridden/extended by concrete adapters to normalize HTTP/client errors.
|
|
76
|
+
|
|
77
|
+
Bulk operations
|
|
78
|
+
- RestService implements createAll, readAll, updateAll, deleteAll. These delegate to similarly named adapter methods (which are expected to exist on the base Adapter implementation from @decaf-ts/core), allowing efficient batched operations where supported by the backend.
|
|
79
|
+
|
|
80
|
+
Unsupported APIs by default
|
|
81
|
+
- Some persistence APIs from the core (raw, Sequence, Statement, parseCondition) are not meaningful out of the box for a generic HTTP adapter, so HttpAdapter throws UnsupportedError for them. Concrete adapters targeting specific backends can choose to implement these.
|
|
82
|
+
|
|
83
|
+
When to use RestService vs RestRepository
|
|
84
|
+
- Use RestService by default for straightforward CRUD over REST endpoints.
|
|
85
|
+
- Use RestRepository if you need repository-level decoration logic (e.g., hooks, rules) to run on your models before hitting the HTTP layer.
|
|
86
|
+
|
|
87
|
+
Extending with another HTTP client
|
|
88
|
+
- Subclass HttpAdapter and implement request(), create(), read(), update(), delete().
|
|
89
|
+
- Override parseError() to translate client-specific errors to your app’s BaseError.
|
|
90
|
+
- Optionally implement raw(), Sequence(), Statement(), parseCondition() if your backend/client supports those features.
|
|
91
|
+
|
|
92
|
+
|
|
93
|
+
# How to Use decaf-ts/for-http
|
|
94
|
+
|
|
95
|
+
Below are concise, non-repeating examples demonstrating how to use each public element of this library. All code samples are valid TypeScript.
|
|
96
|
+
|
|
97
|
+
Note: Examples assume your models are decorated with decaf-ts metadata (e.g., table names and primary keys) through @decaf-ts/decorator-validation and @decaf-ts/db-decorators. For brevity, model decoration details are omitted.
|
|
98
|
+
|
|
99
|
+
## Types: HttpConfig and HttpFlags
|
|
100
|
+
|
|
101
|
+
Description: Define the basic connection configuration and optional per-request headers.
|
|
102
|
+
|
|
103
|
+
```ts
|
|
104
|
+
import { HttpConfig, HttpFlags } from "@decaf-ts/for-http";
|
|
32
105
|
|
|
33
|
-
|
|
34
|
-
|
|
35
|
-
|
|
106
|
+
const config: HttpConfig = {
|
|
107
|
+
protocol: "https",
|
|
108
|
+
host: "api.example.com",
|
|
109
|
+
};
|
|
36
110
|
|
|
37
|
-
|
|
111
|
+
// You can pass headers via flags (typically through a Context)
|
|
112
|
+
const flags: HttpFlags = {
|
|
113
|
+
headers: {
|
|
114
|
+
Authorization: "Bearer <token>",
|
|
115
|
+
},
|
|
116
|
+
};
|
|
117
|
+
```
|
|
38
118
|
|
|
39
|
-
|
|
119
|
+
## Adapter: AxiosHttpAdapter
|
|
40
120
|
|
|
41
|
-
|
|
121
|
+
Description: Ready-to-use HTTP adapter based on Axios. Use it to interact with REST endpoints.
|
|
42
122
|
|
|
43
|
-
|
|
123
|
+
```ts
|
|
124
|
+
import { AxiosHttpAdapter } from "@decaf-ts/for-http/axios";
|
|
125
|
+
import { HttpConfig } from "@decaf-ts/for-http";
|
|
44
126
|
|
|
45
|
-
|
|
127
|
+
const config: HttpConfig = { protocol: "https", host: "api.example.com" };
|
|
128
|
+
const adapter = new AxiosHttpAdapter(config);
|
|
129
|
+
```
|
|
46
130
|
|
|
47
|
-
|
|
131
|
+
## Service: RestService
|
|
48
132
|
|
|
49
|
-
|
|
133
|
+
Description: Lightweight, model-centric service that delegates CRUD and bulk operations to the adapter.
|
|
50
134
|
|
|
51
|
-
|
|
52
|
-
|
|
53
|
-
|
|
54
|
-
|
|
55
|
-
- **Extensibility**: Designed to be extended with different HTTP client implementations
|
|
56
|
-
- **Error Handling**: Includes robust error handling and parsing
|
|
57
|
-
- **Observer Pattern**: Implements the observer pattern for reactive programming
|
|
135
|
+
```ts
|
|
136
|
+
import { RestService } from "@decaf-ts/for-http";
|
|
137
|
+
import { AxiosHttpAdapter } from "@decaf-ts/for-http/axios";
|
|
138
|
+
import { HttpConfig } from "@decaf-ts/for-http";
|
|
58
139
|
|
|
59
|
-
|
|
140
|
+
// Example model (assumes proper decaf-ts decorations elsewhere)
|
|
141
|
+
class User {
|
|
142
|
+
id!: string;
|
|
143
|
+
name!: string;
|
|
144
|
+
}
|
|
60
145
|
|
|
146
|
+
const config: HttpConfig = { protocol: "https", host: "api.example.com" };
|
|
147
|
+
const adapter = new AxiosHttpAdapter(config);
|
|
61
148
|
|
|
62
|
-
|
|
149
|
+
// Create a service bound to the User model
|
|
150
|
+
const users = new RestService<User, any, typeof adapter>(adapter, User);
|
|
63
151
|
|
|
64
|
-
|
|
65
|
-
|
|
152
|
+
// Create
|
|
153
|
+
const created = await users.create({ id: "u1", name: "Alice" } as User);
|
|
66
154
|
|
|
67
|
-
|
|
155
|
+
// Read
|
|
156
|
+
const found = await users.read("u1");
|
|
68
157
|
|
|
69
|
-
|
|
158
|
+
// Update
|
|
159
|
+
const updated = await users.update({ id: "u1", name: "Alice Cooper" } as User);
|
|
70
160
|
|
|
71
|
-
|
|
72
|
-
|
|
73
|
-
import { AxiosHttpAdapter } from '@decaf-ts/for-http';
|
|
161
|
+
// Delete
|
|
162
|
+
const removed = await users.delete("u1");
|
|
74
163
|
|
|
75
|
-
//
|
|
76
|
-
const
|
|
77
|
-
|
|
78
|
-
|
|
79
|
-
|
|
164
|
+
// Bulk create
|
|
165
|
+
const many = await users.createAll([
|
|
166
|
+
{ id: "u2", name: "Bob" } as User,
|
|
167
|
+
{ id: "u3", name: "Carol" } as User,
|
|
168
|
+
]);
|
|
80
169
|
|
|
81
|
-
//
|
|
82
|
-
const
|
|
83
|
-
```
|
|
170
|
+
// Bulk read
|
|
171
|
+
const foundMany = await users.readAll(["u2", "u3"]);
|
|
84
172
|
|
|
85
|
-
|
|
173
|
+
// Bulk update
|
|
174
|
+
const updatedMany = await users.updateAll([
|
|
175
|
+
{ id: "u2", name: "Bobby" } as User,
|
|
176
|
+
{ id: "u3", name: "Caroline" } as User,
|
|
177
|
+
]);
|
|
86
178
|
|
|
87
|
-
|
|
88
|
-
|
|
89
|
-
|
|
90
|
-
class User extends Model {
|
|
91
|
-
id: string;
|
|
92
|
-
name: string;
|
|
93
|
-
email: string;
|
|
94
|
-
|
|
95
|
-
constructor(data?: Partial<User>) {
|
|
96
|
-
super();
|
|
97
|
-
Object.assign(this, data);
|
|
98
|
-
}
|
|
99
|
-
}
|
|
179
|
+
// Bulk delete
|
|
180
|
+
const removedMany = await users.deleteAll(["u2", "u3"]);
|
|
100
181
|
```
|
|
101
182
|
|
|
102
|
-
|
|
183
|
+
## Repository: RestRepository
|
|
103
184
|
|
|
104
|
-
|
|
105
|
-
import { RestRepository } from '@decaf-ts/for-http';
|
|
185
|
+
Description: Use this when you need decaf-ts "repository" decoration logic to run before hitting the HTTP backend.
|
|
106
186
|
|
|
107
|
-
|
|
108
|
-
|
|
187
|
+
```ts
|
|
188
|
+
import { RestRepository } from "@decaf-ts/for-http";
|
|
189
|
+
import { AxiosHttpAdapter } from "@decaf-ts/for-http/axios";
|
|
190
|
+
import { HttpConfig } from "@decaf-ts/for-http";
|
|
109
191
|
|
|
110
|
-
|
|
111
|
-
|
|
112
|
-
|
|
113
|
-
|
|
114
|
-
});
|
|
115
|
-
const createdUser = await userRepository.create(newUser);
|
|
192
|
+
class Product {
|
|
193
|
+
id!: number;
|
|
194
|
+
title!: string;
|
|
195
|
+
}
|
|
116
196
|
|
|
117
|
-
|
|
118
|
-
const
|
|
197
|
+
const cfg: HttpConfig = { protocol: "https", host: "store.example.com" };
|
|
198
|
+
const http = new AxiosHttpAdapter(cfg);
|
|
119
199
|
|
|
120
|
-
//
|
|
121
|
-
|
|
122
|
-
await userRepository.update(user);
|
|
200
|
+
// Create a repository for Product
|
|
201
|
+
const products = new RestRepository<Product, any, typeof http>(http, Product);
|
|
123
202
|
|
|
124
|
-
//
|
|
125
|
-
await
|
|
203
|
+
// Typical repository interactions
|
|
204
|
+
const p = await products.findById(101);
|
|
205
|
+
// ... other repository APIs as provided by @decaf-ts/core Repository
|
|
126
206
|
```
|
|
127
207
|
|
|
128
|
-
|
|
208
|
+
## Passing headers via flags/context
|
|
129
209
|
|
|
130
|
-
|
|
131
|
-
import { RestService } from '@decaf-ts/for-http';
|
|
210
|
+
Description: Supply headers for a specific operation using HttpFlags. These are typically carried inside a Context from @decaf-ts/db-decorators.
|
|
132
211
|
|
|
133
|
-
|
|
134
|
-
|
|
212
|
+
```ts
|
|
213
|
+
import { Context } from "@decaf-ts/db-decorators";
|
|
214
|
+
import { OperationKeys } from "@decaf-ts/db-decorators";
|
|
215
|
+
import { AxiosHttpAdapter, AxiosFlags } from "@decaf-ts/for-http/axios";
|
|
216
|
+
import { RestService } from "@decaf-ts/for-http";
|
|
135
217
|
|
|
136
|
-
|
|
137
|
-
const newUser = new User({
|
|
138
|
-
name: 'John Doe',
|
|
139
|
-
email: 'john@example.com'
|
|
140
|
-
});
|
|
141
|
-
const createdUser = await userService.create(newUser);
|
|
218
|
+
class User { id!: string; name!: string; }
|
|
142
219
|
|
|
143
|
-
|
|
144
|
-
const
|
|
220
|
+
const adapter = new AxiosHttpAdapter({ protocol: "https", host: "api.example.com" });
|
|
221
|
+
const users = new RestService<User, any, typeof adapter>(adapter, User);
|
|
145
222
|
|
|
146
|
-
//
|
|
147
|
-
|
|
148
|
-
await userService.update(user);
|
|
223
|
+
// Generate flags for a READ operation (adds an empty headers obj you can override)
|
|
224
|
+
const flags = adapter.flags<User>(OperationKeys.READ, User, { headers: { Authorization: "Bearer <token>" } });
|
|
149
225
|
|
|
150
|
-
//
|
|
151
|
-
|
|
152
|
-
```
|
|
226
|
+
// Place flags into a context (shape depends on @decaf-ts/db-decorators; we cast here for example purposes)
|
|
227
|
+
const ctx = { flags } as unknown as Context<AxiosFlags>;
|
|
153
228
|
|
|
154
|
-
|
|
229
|
+
// Many decaf-ts operations accept an optional context/flags as the last argument
|
|
230
|
+
const user = await users.read("u1", ctx);
|
|
231
|
+
```
|
|
155
232
|
|
|
156
|
-
|
|
157
|
-
import { RestService } from '@decaf-ts/for-http';
|
|
233
|
+
## Subclassing: Custom HttpAdapter
|
|
158
234
|
|
|
159
|
-
|
|
160
|
-
const userService = new RestService(httpAdapter, User);
|
|
235
|
+
Description: Implement a custom adapter for a different HTTP client. You must implement request and CRUD methods at minimum.
|
|
161
236
|
|
|
162
|
-
|
|
163
|
-
|
|
164
|
-
|
|
165
|
-
new User({ name: 'Jane Doe', email: 'jane@example.com' })
|
|
166
|
-
];
|
|
167
|
-
const createdUsers = await userService.createAll(users);
|
|
237
|
+
```ts
|
|
238
|
+
import { HttpAdapter, HttpConfig, HttpFlags } from "@decaf-ts/for-http";
|
|
239
|
+
import { Context } from "@decaf-ts/db-decorators";
|
|
168
240
|
|
|
169
|
-
//
|
|
170
|
-
|
|
171
|
-
const fetchedUsers = await userService.readAll(userIds);
|
|
241
|
+
// Hypothetical client types
|
|
242
|
+
type MyClient = { request: <T>(config: any) => Promise<T>; get: <T>(url: string) => Promise<T>; post: <T>(url: string, body: any) => Promise<T>; put: <T>(url: string, body: any) => Promise<T>; delete: <T>(url: string) => Promise<T>; };
|
|
172
243
|
|
|
173
|
-
|
|
174
|
-
const usersToUpdate = [
|
|
175
|
-
new User({ id: '123', name: 'John Smith' }),
|
|
176
|
-
new User({ id: '456', name: 'Jane Smith' })
|
|
177
|
-
];
|
|
178
|
-
const updatedUsers = await userService.updateAll(usersToUpdate);
|
|
244
|
+
type MyRequestConfig = { url: string; method: "GET"|"POST"|"PUT"|"DELETE"; data?: any; headers?: Record<string,string>; };
|
|
179
245
|
|
|
180
|
-
|
|
181
|
-
await userService.deleteAll(['123', '456']);
|
|
182
|
-
```
|
|
246
|
+
type MyFlags = HttpFlags;
|
|
183
247
|
|
|
184
|
-
|
|
248
|
+
type MyContext = Context<MyFlags>;
|
|
185
249
|
|
|
186
|
-
|
|
187
|
-
|
|
188
|
-
import { Observer } from '@decaf-ts/core';
|
|
250
|
+
class MyHttpAdapter extends HttpAdapter<HttpConfig, MyClient, MyRequestConfig, MyFlags, MyContext> {
|
|
251
|
+
constructor(config: HttpConfig, alias?: string) { super(config, "my-client", alias); }
|
|
189
252
|
|
|
190
|
-
|
|
191
|
-
|
|
192
|
-
|
|
193
|
-
|
|
194
|
-
|
|
195
|
-
|
|
196
|
-
|
|
253
|
+
protected override getClient(): MyClient {
|
|
254
|
+
// create and return your HTTP client instance
|
|
255
|
+
return {
|
|
256
|
+
request: async <T>(c: any) => ({} as T),
|
|
257
|
+
get: async <T>(url: string) => ({} as T),
|
|
258
|
+
post: async <T>(url: string, body: any) => ({} as T),
|
|
259
|
+
put: async <T>(url: string, body: any) => ({} as T),
|
|
260
|
+
delete: async <T>(url: string) => ({} as T),
|
|
261
|
+
};
|
|
197
262
|
}
|
|
198
|
-
};
|
|
199
263
|
|
|
200
|
-
|
|
201
|
-
|
|
202
|
-
|
|
203
|
-
// When operations are performed, observers will be notified
|
|
204
|
-
await userService.create(new User({ name: 'John Doe' }));
|
|
205
|
-
|
|
206
|
-
// Unregister the observer when done
|
|
207
|
-
userService.unObserve(userObserver);
|
|
208
|
-
```
|
|
209
|
-
|
|
210
|
-
### Custom HTTP Adapter Implementation
|
|
211
|
-
|
|
212
|
-
```typescript
|
|
213
|
-
import { HttpAdapter } from '@decaf-ts/for-http';
|
|
214
|
-
import { HttpConfig, HttpFlags } from '@decaf-ts/for-http';
|
|
215
|
-
import { Context } from '@decaf-ts/db-decorators';
|
|
216
|
-
import SomeHttpClient from 'some-http-client';
|
|
217
|
-
|
|
218
|
-
// Create a custom HTTP adapter for a different HTTP client
|
|
219
|
-
class CustomHttpAdapter extends HttpAdapter<
|
|
220
|
-
SomeHttpClient,
|
|
221
|
-
any,
|
|
222
|
-
HttpFlags,
|
|
223
|
-
Context<HttpFlags>
|
|
224
|
-
> {
|
|
225
|
-
constructor(native: SomeHttpClient, config: HttpConfig, alias?: string) {
|
|
226
|
-
super(native, config, 'custom', alias);
|
|
264
|
+
override async request<V>(details: MyRequestConfig): Promise<V> {
|
|
265
|
+
// bridge to your client’s request API
|
|
266
|
+
return this.client.request<V>(details);
|
|
227
267
|
}
|
|
228
268
|
|
|
229
|
-
async
|
|
230
|
-
|
|
269
|
+
async create(table: string, id: string|number, model: Record<string, any>): Promise<Record<string, any>> {
|
|
270
|
+
const url = this.url(table);
|
|
271
|
+
return this.client.post(url, model);
|
|
231
272
|
}
|
|
232
273
|
|
|
233
|
-
async
|
|
234
|
-
|
|
235
|
-
|
|
236
|
-
return this.native.post(url, model);
|
|
237
|
-
} catch (e: any) {
|
|
238
|
-
throw this.parseError(e);
|
|
239
|
-
}
|
|
274
|
+
async read(table: string, id: string|number|bigint): Promise<Record<string, any>> {
|
|
275
|
+
const url = this.url(table, { id: id as string|number });
|
|
276
|
+
return this.client.get(url);
|
|
240
277
|
}
|
|
241
278
|
|
|
242
|
-
async
|
|
243
|
-
|
|
244
|
-
|
|
245
|
-
return this.native.get(url);
|
|
246
|
-
} catch (e: any) {
|
|
247
|
-
throw this.parseError(e);
|
|
248
|
-
}
|
|
279
|
+
async update(table: string, id: string|number, model: Record<string, any>): Promise<Record<string, any>> {
|
|
280
|
+
const url = this.url(table);
|
|
281
|
+
return this.client.put(url, model);
|
|
249
282
|
}
|
|
250
283
|
|
|
251
|
-
async
|
|
252
|
-
|
|
253
|
-
|
|
254
|
-
return this.native.put(url, model);
|
|
255
|
-
} catch (e: any) {
|
|
256
|
-
throw this.parseError(e);
|
|
257
|
-
}
|
|
284
|
+
async delete(table: string, id: string|number|bigint): Promise<Record<string, any>> {
|
|
285
|
+
const url = this.url(table, { id: id as string|number });
|
|
286
|
+
return this.client.delete(url);
|
|
258
287
|
}
|
|
259
288
|
|
|
260
|
-
|
|
261
|
-
try {
|
|
262
|
-
const url = this.url(tableName, { id: id as string | number });
|
|
263
|
-
return this.native.delete(url);
|
|
264
|
-
} catch (e: any) {
|
|
265
|
-
throw this.parseError(e);
|
|
266
|
-
}
|
|
267
|
-
}
|
|
289
|
+
// Optionally override parseError(err) to normalize client-specific errors
|
|
268
290
|
}
|
|
269
291
|
```
|
|
270
292
|
|
|
271
|
-
|
|
272
|
-
|
|
273
|
-
```typescript
|
|
274
|
-
import { HttpFlags } from '@decaf-ts/for-http';
|
|
275
|
-
|
|
276
|
-
// Create custom HTTP flags with headers
|
|
277
|
-
const flags: HttpFlags = {
|
|
278
|
-
headers: {
|
|
279
|
-
'Authorization': 'Bearer token123',
|
|
280
|
-
'Content-Type': 'application/json'
|
|
281
|
-
}
|
|
282
|
-
};
|
|
283
|
-
|
|
284
|
-
// Use flags with repository operations
|
|
285
|
-
const user = await userRepository.findById('123', { flags });
|
|
293
|
+
## Constants and Types (axios)
|
|
286
294
|
|
|
287
|
-
|
|
288
|
-
const createdUser = await userService.create(newUser, { flags });
|
|
289
|
-
```
|
|
290
|
-
|
|
291
|
-
### Complete Application Example
|
|
292
|
-
|
|
293
|
-
```typescript
|
|
294
|
-
import axios from 'axios';
|
|
295
|
-
import {
|
|
296
|
-
AxiosHttpAdapter,
|
|
297
|
-
RestRepository,
|
|
298
|
-
RestService,
|
|
299
|
-
HttpConfig
|
|
300
|
-
} from '@decaf-ts/for-http';
|
|
301
|
-
import { Model } from '@decaf-ts/decorator-validation';
|
|
302
|
-
|
|
303
|
-
// Define a model
|
|
304
|
-
class Product extends Model {
|
|
305
|
-
id: string;
|
|
306
|
-
name: string;
|
|
307
|
-
price: number;
|
|
308
|
-
|
|
309
|
-
constructor(data?: Partial<Product>) {
|
|
310
|
-
super();
|
|
311
|
-
Object.assign(this, data);
|
|
312
|
-
}
|
|
313
|
-
}
|
|
295
|
+
Description: Utilities specific to the Axios implementation.
|
|
314
296
|
|
|
315
|
-
|
|
316
|
-
|
|
317
|
-
protocol: 'https',
|
|
318
|
-
host: 'api.mystore.com'
|
|
319
|
-
};
|
|
297
|
+
```ts
|
|
298
|
+
import { AxiosFlavour, AxiosFlags } from "@decaf-ts/for-http/axios";
|
|
320
299
|
|
|
321
|
-
//
|
|
322
|
-
|
|
323
|
-
|
|
324
|
-
// Create a repository
|
|
325
|
-
const productRepo = new RestRepository(adapter, Product);
|
|
326
|
-
|
|
327
|
-
// Create a service
|
|
328
|
-
const productService = new RestService(adapter, Product);
|
|
329
|
-
|
|
330
|
-
// Example application
|
|
331
|
-
async function manageProducts() {
|
|
332
|
-
try {
|
|
333
|
-
// Create a new product
|
|
334
|
-
const newProduct = new Product({
|
|
335
|
-
name: 'Smartphone',
|
|
336
|
-
price: 699.99
|
|
337
|
-
});
|
|
338
|
-
|
|
339
|
-
const createdProduct = await productRepo.create(newProduct);
|
|
340
|
-
console.log('Created product:', createdProduct);
|
|
341
|
-
|
|
342
|
-
// Get all products
|
|
343
|
-
const products = await productRepo.findAll();
|
|
344
|
-
console.log('All products:', products);
|
|
345
|
-
|
|
346
|
-
// Update a product
|
|
347
|
-
createdProduct.price = 649.99;
|
|
348
|
-
const updatedProduct = await productService.update(createdProduct);
|
|
349
|
-
console.log('Updated product:', updatedProduct);
|
|
350
|
-
|
|
351
|
-
// Delete a product
|
|
352
|
-
await productService.delete(createdProduct.id);
|
|
353
|
-
console.log('Product deleted');
|
|
354
|
-
|
|
355
|
-
// Bulk operations
|
|
356
|
-
const bulkProducts = [
|
|
357
|
-
new Product({ name: 'Laptop', price: 1299.99 }),
|
|
358
|
-
new Product({ name: 'Tablet', price: 499.99 })
|
|
359
|
-
];
|
|
360
|
-
|
|
361
|
-
const createdProducts = await productService.createAll(bulkProducts);
|
|
362
|
-
console.log('Created multiple products:', createdProducts);
|
|
363
|
-
} catch (error) {
|
|
364
|
-
console.error('Error managing products:', error);
|
|
365
|
-
}
|
|
366
|
-
}
|
|
300
|
+
// AxiosFlavour is the adapter flavour identifier string: "axios"
|
|
301
|
+
console.log(AxiosFlavour);
|
|
367
302
|
|
|
368
|
-
|
|
303
|
+
// AxiosFlags is a type alias of HttpFlags; useful for contexts with Axios
|
|
304
|
+
const f: AxiosFlags = { headers: { "X-Trace": "1" } };
|
|
369
305
|
```
|
|
370
306
|
|
|
371
307
|
|
package/dist/for-http.cjs
CHANGED
package/dist/for-http.esm.cjs
CHANGED
|
@@ -470,7 +470,7 @@ class RestRepository extends Repository {
|
|
|
470
470
|
* @summary Version identifier for the module
|
|
471
471
|
* @const VERSION
|
|
472
472
|
*/
|
|
473
|
-
const VERSION = "0.2.
|
|
473
|
+
const VERSION = "0.2.6";
|
|
474
474
|
|
|
475
475
|
export { HttpAdapter, RestRepository, RestService, VERSION };
|
|
476
476
|
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"file":"for-http.esm.cjs","sources":["../src/RestService.ts","../src/adapter.ts","../src/RestRepository.ts","../src/index.ts"],"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 * @class RestService\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 * @mermaid\n * sequenceDiagram\n *   participant Client\n *   participant Service as RestService\n *   participant Adapter as HttpAdapter\n *   participant API\n *   Client->>Service: create(model)\n *   Service->>Adapter: prepare(model, pk)\n *   Service->>Adapter: create(table, id, record)\n *   Adapter->>API: HTTP POST\n *   API-->>Adapter: 201 Created\n *   Adapter-->>Service: record\n *   Service-->>Client: revert(record)\n */\nexport class RestService<\n    M extends Model,\n    Q,\n    A extends HttpAdapter<any, 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   * @mermaid\n   * sequenceDiagram\n   *   participant Client\n   *   participant Service as RestService\n   *   participant Adapter as HttpAdapter\n   *   Client->>Service: createAll(models)\n   *   Service->>Adapter: prepare(model, pk) x N\n   *   Service->>Adapter: createAll(table, ids[], records[])\n   *   Adapter-->>Service: records[]\n   *   Service-->>Client: revert(records[])\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","import {\n  Adapter,\n  Condition,\n  Repository,\n  Sequence,\n  SequenceOptions,\n  UnsupportedError,\n} from \"@decaf-ts/core\";\nimport { BaseError, Context, OperationKeys } from \"@decaf-ts/db-decorators\";\nimport { HttpConfig, HttpFlags } from \"./types\";\nimport { Constructor, Model } from \"@decaf-ts/decorator-validation\";\nimport { RestService } from \"./RestService\";\nimport { Statement } from \"@decaf-ts/core\";\n\n/**\n * @description Abstract HTTP adapter for REST API interactions\n * @summary Provides a base implementation for HTTP adapters with methods for CRUD operations,\n * URL construction, and error handling. This class extends the core Adapter class and\n * implements the necessary methods for HTTP communication. Concrete implementations\n * must provide specific HTTP client functionality.\n * @template Y - The native HTTP client type\n * @template Q - The query type used by the adapter\n * @template F - The HTTP flags type, extending HttpFlags\n * @template C - The context type, extending Context<F>\n * @param {Y} native - The native HTTP client instance\n * @param {HttpConfig} config - Configuration for the HTTP adapter\n * @param {string} flavour - The adapter flavor identifier\n * @param {string} [alias] - Optional alias for the adapter\n * @class HttpAdapter\n * @example\n * ```typescript\n * // Example implementation with Axios\n * class AxiosAdapter extends HttpAdapter<AxiosInstance, AxiosRequestConfig> {\n *   constructor(config: HttpConfig) {\n *     super(axios.create(), config, 'axios');\n *   }\n *\n *   async request<V>(details: AxiosRequestConfig): Promise<V> {\n *     const response = await this.native.request(details);\n *     return response.data;\n *   }\n *\n *   // Implement other abstract methods...\n * }\n * ```\n */\nexport abstract class HttpAdapter<\n  Y extends HttpConfig,\n  CON,\n  Q,\n  F extends HttpFlags = HttpFlags,\n  C extends Context<F> = Context<F>,\n> extends Adapter<Y, CON, Q, F, C> {\n  protected constructor(config: Y, flavour: string, alias?: string) {\n    super(config, flavour, alias);\n  }\n\n  /**\n   * @description Generates operation flags with HTTP headers\n   * @summary Extends the base flags method to include HTTP-specific headers for operations.\n   * This method adds an empty headers object to the flags returned by the parent class.\n   * @template F - The Repository Flags type\n   * @template M - The model type\n   * @param {OperationKeys.CREATE|OperationKeys.READ|OperationKeys.UPDATE|OperationKeys.DELETE} operation - The operation type\n   * @param {Constructor<M>} model - The model constructor\n   * @param {Partial<F>} overrides - Optional flag overrides\n   * @return {F} The flags object with headers\n   */\n  override flags<M extends Model>(\n    operation:\n      | OperationKeys.CREATE\n      | OperationKeys.READ\n      | OperationKeys.UPDATE\n      | OperationKeys.DELETE,\n    model: Constructor<M>,\n    overrides: Partial<F>\n  ) {\n    return Object.assign(super.flags<M>(operation, model, overrides), {\n      headers: {},\n    });\n  }\n\n  /**\n   * @description Returns the repository constructor for this adapter\n   * @summary Provides the RestService class as the repository implementation for this HTTP adapter.\n   * This method is used to create repository instances that work with this adapter type.\n   * @template M - The model type\n   * @return {Constructor<Repository<M, Q, HttpAdapter<Y, Q, F, C>, F, C>>} The repository constructor\n   */\n  override repository<M extends Model>(): Constructor<\n    Repository<M, Q, HttpAdapter<Y, CON, Q, F, C>, F, C>\n  > {\n    return RestService as unknown as Constructor<\n      Repository<M, Q, HttpAdapter<Y, CON, Q, F, C>, F, C>\n    >;\n  }\n\n  /**\n   * @description Constructs a URL for API requests\n   * @summary Builds a complete URL for API requests using the configured protocol and host,\n   * the specified table name, and optional query parameters. The method handles URL encoding.\n   * @param {string} tableName - The name of the table or endpoint\n   * @param {Record<string, string | number>} [queryParams] - Optional query parameters\n   * @return {string} The encoded URL string\n   */\n  protected url(\n    tableName: string,\n    queryParams?: Record<string, string | number>\n  ) {\n    const url = new URL(\n      `${this.config.protocol}://${this.config.host}/${tableName}`\n    );\n    if (queryParams)\n      Object.entries(queryParams).forEach(([key, value]) =>\n        url.searchParams.append(key, value.toString())\n      );\n\n    // ensure spaces are encoded as %20 (not '+') to match expectations\n    return encodeURI(url.toString()).replace(/\\+/g, \"%20\");\n  }\n\n  /**\n   * @description Parses and converts errors to BaseError type\n   * @summary Processes errors that occur during HTTP operations and converts them to\n   * the appropriate BaseError type. Currently returns the error as-is, but can be\n   * extended to handle specific error messages differently.\n   * @param {Error} err - The error to parse\n   * @return {BaseError} The parsed error as a BaseError\n   */\n  parseError(err: Error): BaseError {\n    const { message } = err;\n    switch (message) {\n      default:\n        return err as BaseError;\n    }\n  }\n\n  /**\n   * @description Sends an HTTP request\n   * @summary Abstract method that must be implemented by subclasses to send HTTP requests\n   * using the native HTTP client. This is the core method for making API calls.\n   * @template V - The response value type\n   * @param {Q} details - The request details specific to the HTTP client\n   * @return {Promise<V>} A promise that resolves with the response data\n   */\n  abstract request<V>(details: Q): Promise<V>;\n\n  /**\n   * @description Creates a new resource\n   * @summary Abstract method that must be implemented by subclasses to create a new resource\n   * via HTTP. This typically corresponds to a POST request.\n   * @param {string} tableName - The name of the table or endpoint\n   * @param {string|number} id - The identifier for the resource\n   * @param {Record<string, any>} model - The data model to create\n   * @param {...any[]} args - Additional arguments\n   * @return {Promise<Record<string, any>>} A promise that resolves with the created resource\n   */\n  abstract override create(\n    tableName: string,\n    id: string | number,\n    model: Record<string, any>,\n    ...args: any[]\n  ): Promise<Record<string, any>>;\n\n  /**\n   * @description Retrieves a resource by ID\n   * @summary Abstract method that must be implemented by subclasses to retrieve a resource\n   * via HTTP. This typically corresponds to a GET request.\n   * @param {string} tableName - The name of the table or endpoint\n   * @param {string|number|bigint} id - The identifier for the resource\n   * @param {...any[]} args - Additional arguments\n   * @return {Promise<Record<string, any>>} A promise that resolves with the retrieved resource\n   */\n  abstract override read(\n    tableName: string,\n    id: string | number | bigint,\n    ...args: any[]\n  ): Promise<Record<string, any>>;\n\n  /**\n   * @description Updates an existing resource\n   * @summary Abstract method that must be implemented by subclasses to update a resource\n   * via HTTP. This typically corresponds to a PUT or PATCH request.\n   * @param {string} tableName - The name of the table or endpoint\n   * @param {string|number} id - The identifier for the resource\n   * @param {Record<string, any>} model - The updated data model\n   * @param {...any[]} args - Additional arguments\n   * @return {Promise<Record<string, any>>} A promise that resolves with the updated resource\n   */\n  abstract override update(\n    tableName: string,\n    id: string | number,\n    model: Record<string, any>,\n    ...args: any[]\n  ): Promise<Record<string, any>>;\n\n  /**\n   * @description Deletes a resource by ID\n   * @summary Abstract method that must be implemented by subclasses to delete a resource\n   * via HTTP. This typically corresponds to a DELETE request.\n   * @param {string} tableName - The name of the table or endpoint\n   * @param {string|number|bigint} id - The identifier for the resource to delete\n   * @param {...any[]} args - Additional arguments\n   * @return {Promise<Record<string, any>>} A promise that resolves with the deletion result\n   */\n  abstract override delete(\n    tableName: string,\n    id: string | number | bigint,\n    ...args: any[]\n  ): Promise<Record<string, any>>;\n\n  /**\n   * @description Executes a raw query\n   * @summary Method for executing raw queries directly with the HTTP client.\n   * This method is not supported by default in HTTP adapters and throws an UnsupportedError.\n   * Subclasses can override this method to provide implementation.\n   * @template R - The result type\n   * @param {Q} rawInput - The raw query input\n   * @param {boolean} process - Whether to process the result\n   * @param {...any[]} args - Additional arguments\n   * @return {Promise<R>} A promise that resolves with the query result\n   * @throws {UnsupportedError} Always throws as this method is not supported by default\n   */\n  // eslint-disable-next-line @typescript-eslint/no-unused-vars\n  raw<R>(rawInput: Q, process: boolean, ...args: any[]): Promise<R> {\n    return Promise.reject(\n      new UnsupportedError(\n        \"Api is not natively available for HttpAdapters. If required, please extends this class\"\n      )\n    );\n  }\n\n  /**\n   * @description Creates a sequence\n   * @summary Method for creating a sequence for generating unique identifiers.\n   * This method is not supported by default in HTTP adapters and throws an UnsupportedError.\n   * Subclasses can override this method to provide implementation.\n   * @param {SequenceOptions} options - Options for creating the sequence\n   * @return {Promise<Sequence>} A promise that resolves with the created sequence\n   * @throws {UnsupportedError} Always throws as this method is not supported by default\n   */\n  // eslint-disable-next-line @typescript-eslint/no-unused-vars\n  Sequence(options: SequenceOptions): Promise<Sequence> {\n    return Promise.reject(\n      new UnsupportedError(\n        \"Api is not natively available for HttpAdapters. If required, please extends this class\"\n      )\n    );\n  }\n\n  /**\n   * @description Creates a statement for querying\n   * @summary Method for creating a statement for building and executing queries.\n   * This method is not supported by default in HTTP adapters and throws an UnsupportedError.\n   * Subclasses can override this method to provide implementation.\n   * @template M - The model type\n   * @template ! - The raw query type\n   * @return {Statement<Q, M, any>} A statement object for building queries\n   * @throws {UnsupportedError} Always throws as this method is not supported by default\n   */\n  override Statement<M extends Model>(): Statement<Q, M, any> {\n    throw new UnsupportedError(\n      \"Api is not natively available for HttpAdapters. If required, please extends this class\"\n    );\n  }\n\n  /**\n   * @description Parses a condition into a query\n   * @summary Method for parsing a condition object into a query format understood by the HTTP client.\n   * This method is not supported by default in HTTP adapters and throws an UnsupportedError.\n   * Subclasses can override this method to provide implementation.\n   * @param {Condition<any>} condition - The condition to parse\n   * @return {Q} The parsed query\n   * @throws {UnsupportedError} Always throws as this method is not supported by default\n   */\n  // eslint-disable-next-line @typescript-eslint/no-unused-vars\n  parseCondition(condition: Condition<any>): Q {\n    throw new UnsupportedError(\n      \"Api is not natively available for HttpAdapters. If required, please extends this class\"\n    );\n  }\n}\n","import { Repository } from \"@decaf-ts/core\";\nimport { Constructor, Model } from \"@decaf-ts/decorator-validation\";\nimport { HttpAdapter } from \"./adapter\";\nimport { Context } from \"@decaf-ts/db-decorators\";\nimport { HttpFlags } from \"./types\";\n\n/**\n * @description Repository for REST API interactions\n * @summary A specialized repository implementation for interacting with REST APIs.\n * This class extends the core Repository class and works with HTTP adapters to\n * provide CRUD operations for models via REST endpoints.\n * This Is NOT the default repository for the HTTP adapter. That would be {@link RestService}.\n * Use this only in the specific case of needing to run the CURD model logic (decoration) before submitting to the backend\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 * @class RestRepository\n * @example\n * ```typescript\n * // Create a repository for User model with Axios adapter\n * const axiosAdapter = new AxiosAdapter({\n *   protocol: 'https',\n *   host: 'api.example.com'\n * });\n * const userRepository = new RestRepository(axiosAdapter, User);\n *\n * // Use the repository for CRUD operations\n * const user = await userRepository.findById('123');\n * ```\n * @see {@link RestService}\n */\nexport class RestRepository<\n  M extends Model,\n  Q,\n  A extends HttpAdapter<any, any, Q, F, C>,\n  F extends HttpFlags = HttpFlags,\n  C extends Context<F> = Context<F>,\n> extends Repository<M, Q, A> {\n  constructor(adapter: A, clazz?: Constructor<M>) {\n    super(adapter, clazz);\n  }\n}\n","/**\n * @description HTTP client module for REST API interactions\n * @summary This module provides classes and utilities for interacting with REST APIs.\n * It exposes repository and service classes for making HTTP requests, along with\n * type definitions and adapters for different HTTP clients. The module includes\n * {@link RestRepository} and {@link RestService} for API interactions.\n * @module for-http\n */\nexport * from \"./adapter\";\nexport * from \"./RestRepository\";\nexport * from \"./RestService\";\nexport * from \"./types\";\n\n/**\n * @description Current version of the for-http module\n * @summary Version identifier for the module\n * @const VERSION\n */\nexport const VERSION = \"##VERSION##\";\n"],"names":[],"mappings":"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"}
|
package/lib/esm/index.d.ts
CHANGED
package/lib/esm/index.js
CHANGED
|
@@ -15,5 +15,5 @@ export * from "./types.js";
|
|
|
15
15
|
* @summary Version identifier for the module
|
|
16
16
|
* @const VERSION
|
|
17
17
|
*/
|
|
18
|
-
export const VERSION = "0.2.
|
|
18
|
+
export const VERSION = "0.2.6";
|
|
19
19
|
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi9zcmMvaW5kZXgudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQUE7Ozs7Ozs7R0FPRztBQUNILDZCQUEwQjtBQUMxQixvQ0FBaUM7QUFDakMsaUNBQThCO0FBQzlCLDJCQUF3QjtBQUV4Qjs7OztHQUlHO0FBQ0gsTUFBTSxDQUFDLE1BQU0sT0FBTyxHQUFHLGFBQWEsQ0FBQyIsInNvdXJjZXNDb250ZW50IjpbIi8qKlxuICogQGRlc2NyaXB0aW9uIEhUVFAgY2xpZW50IG1vZHVsZSBmb3IgUkVTVCBBUEkgaW50ZXJhY3Rpb25zXG4gKiBAc3VtbWFyeSBUaGlzIG1vZHVsZSBwcm92aWRlcyBjbGFzc2VzIGFuZCB1dGlsaXRpZXMgZm9yIGludGVyYWN0aW5nIHdpdGggUkVTVCBBUElzLlxuICogSXQgZXhwb3NlcyByZXBvc2l0b3J5IGFuZCBzZXJ2aWNlIGNsYXNzZXMgZm9yIG1ha2luZyBIVFRQIHJlcXVlc3RzLCBhbG9uZyB3aXRoXG4gKiB0eXBlIGRlZmluaXRpb25zIGFuZCBhZGFwdGVycyBmb3IgZGlmZmVyZW50IEhUVFAgY2xpZW50cy4gVGhlIG1vZHVsZSBpbmNsdWRlc1xuICoge0BsaW5rIFJlc3RSZXBvc2l0b3J5fSBhbmQge0BsaW5rIFJlc3RTZXJ2aWNlfSBmb3IgQVBJIGludGVyYWN0aW9ucy5cbiAqIEBtb2R1bGUgZm9yLWh0dHBcbiAqL1xuZXhwb3J0ICogZnJvbSBcIi4vYWRhcHRlclwiO1xuZXhwb3J0ICogZnJvbSBcIi4vUmVzdFJlcG9zaXRvcnlcIjtcbmV4cG9ydCAqIGZyb20gXCIuL1Jlc3RTZXJ2aWNlXCI7XG5leHBvcnQgKiBmcm9tIFwiLi90eXBlc1wiO1xuXG4vKipcbiAqIEBkZXNjcmlwdGlvbiBDdXJyZW50IHZlcnNpb24gb2YgdGhlIGZvci1odHRwIG1vZHVsZVxuICogQHN1bW1hcnkgVmVyc2lvbiBpZGVudGlmaWVyIGZvciB0aGUgbW9kdWxlXG4gKiBAY29uc3QgVkVSU0lPTlxuICovXG5leHBvcnQgY29uc3QgVkVSU0lPTiA9IFwiIyNWRVJTSU9OIyNcIjtcbiJdfQ==
|
package/lib/index.cjs
CHANGED
|
@@ -32,5 +32,5 @@ __exportStar(require("./types.cjs"), exports);
|
|
|
32
32
|
* @summary Version identifier for the module
|
|
33
33
|
* @const VERSION
|
|
34
34
|
*/
|
|
35
|
-
exports.VERSION = "0.2.
|
|
35
|
+
exports.VERSION = "0.2.6";
|
|
36
36
|
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi9zcmMvaW5kZXgudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6Ijs7Ozs7Ozs7Ozs7Ozs7Ozs7QUFBQTs7Ozs7OztHQU9HO0FBQ0gsZ0RBQTBCO0FBQzFCLHVEQUFpQztBQUNqQyxvREFBOEI7QUFDOUIsOENBQXdCO0FBRXhCOzs7O0dBSUc7QUFDVSxRQUFBLE9BQU8sR0FBRyxhQUFhLENBQUMiLCJzb3VyY2VzQ29udGVudCI6WyIvKipcbiAqIEBkZXNjcmlwdGlvbiBIVFRQIGNsaWVudCBtb2R1bGUgZm9yIFJFU1QgQVBJIGludGVyYWN0aW9uc1xuICogQHN1bW1hcnkgVGhpcyBtb2R1bGUgcHJvdmlkZXMgY2xhc3NlcyBhbmQgdXRpbGl0aWVzIGZvciBpbnRlcmFjdGluZyB3aXRoIFJFU1QgQVBJcy5cbiAqIEl0IGV4cG9zZXMgcmVwb3NpdG9yeSBhbmQgc2VydmljZSBjbGFzc2VzIGZvciBtYWtpbmcgSFRUUCByZXF1ZXN0cywgYWxvbmcgd2l0aFxuICogdHlwZSBkZWZpbml0aW9ucyBhbmQgYWRhcHRlcnMgZm9yIGRpZmZlcmVudCBIVFRQIGNsaWVudHMuIFRoZSBtb2R1bGUgaW5jbHVkZXNcbiAqIHtAbGluayBSZXN0UmVwb3NpdG9yeX0gYW5kIHtAbGluayBSZXN0U2VydmljZX0gZm9yIEFQSSBpbnRlcmFjdGlvbnMuXG4gKiBAbW9kdWxlIGZvci1odHRwXG4gKi9cbmV4cG9ydCAqIGZyb20gXCIuL2FkYXB0ZXJcIjtcbmV4cG9ydCAqIGZyb20gXCIuL1Jlc3RSZXBvc2l0b3J5XCI7XG5leHBvcnQgKiBmcm9tIFwiLi9SZXN0U2VydmljZVwiO1xuZXhwb3J0ICogZnJvbSBcIi4vdHlwZXNcIjtcblxuLyoqXG4gKiBAZGVzY3JpcHRpb24gQ3VycmVudCB2ZXJzaW9uIG9mIHRoZSBmb3ItaHR0cCBtb2R1bGVcbiAqIEBzdW1tYXJ5IFZlcnNpb24gaWRlbnRpZmllciBmb3IgdGhlIG1vZHVsZVxuICogQGNvbnN0IFZFUlNJT05cbiAqL1xuZXhwb3J0IGNvbnN0IFZFUlNJT04gPSBcIiMjVkVSU0lPTiMjXCI7XG4iXX0=
|
package/lib/index.d.ts
CHANGED
package/package.json
CHANGED
|
@@ -1,6 +1,6 @@
|
|
|
1
1
|
{
|
|
2
2
|
"name": "@decaf-ts/for-http",
|
|
3
|
-
"version": "0.2.
|
|
3
|
+
"version": "0.2.7",
|
|
4
4
|
"description": "http wrappers for decaf-ts",
|
|
5
5
|
"type": "module",
|
|
6
6
|
"exports": {
|
|
@@ -32,14 +32,21 @@
|
|
|
32
32
|
"coverage": "rimraf ./workdocs/reports/data/*.json && npm run test:all -- --coverage --config=./workdocs/reports/jest.coverage.config.ts",
|
|
33
33
|
"lint": "eslint .",
|
|
34
34
|
"lint-fix": "eslint --fix .",
|
|
35
|
-
"prepare-pr": "npm run lint-fix && npm run build:prod && npm run coverage && npm run docs",
|
|
35
|
+
"prepare-pr": "npm run repo:pr && npm run lint-fix && npm run build:prod && npm run coverage && npm run docs",
|
|
36
36
|
"prepare-release": "npm run lint-fix && npm run build:prod && npm run coverage && npm run docs",
|
|
37
37
|
"release": "./bin/tag-release.sh",
|
|
38
38
|
"clean-publish": "npx clean-publish",
|
|
39
39
|
"drawings": "for FILE in workdocs/drawings/*.drawio; do echo \"converting $FILE to image...\" && docker run --rm -v $(pwd):/data rlespinasse/drawio-export --format png $FILE; done && cp -rf workdocs/drawings/export/* workdocs/resources/",
|
|
40
40
|
"uml": "cd workdocs/uml && for FILE in ./*.puml; do docker run --rm -v $(pwd):/work -w /work miy4/plantuml -DPLANTUML_LIMIT_SIZE=8192 -tpng $FILE; done && cd ../.. && cp -fr workdocs/uml/*.png workdocs/resources/",
|
|
41
41
|
"docs": "npx rimraf ./docs && mkdir docs && npx build-scripts --docs",
|
|
42
|
-
"publish-docs": "docker run -it --rm --user $(id -u):$(id -g) -v \"$(pwd)/workdocs/confluence:/content\" -e ATLASSIAN_API_TOKEN=$(cat .confluence-token) ghcr.io/markdown-confluence/publish:latest"
|
|
42
|
+
"publish-docs": "docker run -it --rm --user $(id -u):$(id -g) -v \"$(pwd)/workdocs/confluence:/content\" -e ATLASSIAN_API_TOKEN=$(cat .confluence-token) ghcr.io/markdown-confluence/publish:latest",
|
|
43
|
+
"repo:init": "codex exec \"$(cat ./.codex/prompts/repo-setup.md)\nbase_path is `./`, initialize the repository\"",
|
|
44
|
+
"repo:setup": "codex exec \"$(cat ./.codex/prompts/repo-setup.md)\nbase_path is ./\"",
|
|
45
|
+
"repo:doc": "codex exec \"$(cat ./.codex/prompts/doc.md) $(cat ./.codex/prompts/bulk-docs.md)\nbase_path is ./\"",
|
|
46
|
+
"repo:tests": "codex exec \"$(cat ./.codex/prompts/bulk-tests.md)\nbase_path is ./ and coverage is 95%\" -s workspace-write",
|
|
47
|
+
"repo:readme": "codex exec \"$(cat ./.codex/prompts/update-readme.md)\nbase_path is ./\"",
|
|
48
|
+
"repo:pr": "npm run repo:doc && npm run repo:tests && npm run repo:readme",
|
|
49
|
+
"sync-codex": "./bin/sync-codex.sh"
|
|
43
50
|
},
|
|
44
51
|
"repository": {
|
|
45
52
|
"type": "git",
|
|
@@ -71,7 +78,7 @@
|
|
|
71
78
|
"typescript",
|
|
72
79
|
"ts"
|
|
73
80
|
],
|
|
74
|
-
"author": "Tiago Venceslau",
|
|
81
|
+
"author": "Tiago Venceslau and Contributors",
|
|
75
82
|
"license": "MIT",
|
|
76
83
|
"bugs": {
|
|
77
84
|
"url": "https://github.com/decaf-ts/for-http/issues"
|
|
@@ -91,6 +98,7 @@
|
|
|
91
98
|
"eslint": "^9.25.1",
|
|
92
99
|
"eslint-config-prettier": "^10.1.2",
|
|
93
100
|
"eslint-plugin-prettier": "^5.2.6",
|
|
101
|
+
"fastify": "^5.6.1",
|
|
94
102
|
"globals": "^16.0.0",
|
|
95
103
|
"jest": "^29.7.0",
|
|
96
104
|
"jest-html-reporters": "^3.1.7",
|