@decaf-ts/db-decorators 0.6.1 → 0.6.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/README.md +571 -10
- package/dist/db-decorators.cjs +1352 -322
- package/dist/db-decorators.esm.cjs +1352 -323
- package/lib/esm/identity/decorators.d.ts +7 -0
- package/lib/esm/identity/decorators.js +11 -4
- package/lib/esm/identity/index.js +3 -3
- package/lib/esm/identity/utils.d.ts +36 -23
- package/lib/esm/identity/utils.js +38 -25
- package/lib/esm/index.d.ts +12 -27
- package/lib/esm/index.js +13 -28
- package/lib/esm/interfaces/BulkCrudOperator.d.ts +39 -0
- package/lib/esm/interfaces/BulkCrudOperator.js +1 -1
- package/lib/esm/interfaces/Contextual.d.ts +17 -0
- package/lib/esm/interfaces/Contextual.js +1 -1
- package/lib/esm/interfaces/CrudOperator.d.ts +26 -23
- package/lib/esm/interfaces/CrudOperator.js +1 -1
- package/lib/esm/interfaces/IRepository.d.ts +10 -2
- package/lib/esm/interfaces/IRepository.js +1 -1
- package/lib/esm/interfaces/index.js +5 -5
- package/lib/esm/model/constants.d.ts +11 -13
- package/lib/esm/model/constants.js +12 -14
- package/lib/esm/model/decorators.d.ts +112 -23
- package/lib/esm/model/decorators.js +119 -29
- package/lib/esm/model/index.d.ts +1 -0
- package/lib/esm/model/index.js +7 -6
- package/lib/esm/model/model.d.ts +2 -141
- package/lib/esm/model/model.js +2 -13
- package/lib/esm/model/overrides.d.ts +1 -0
- package/lib/esm/model/overrides.js +23 -0
- package/lib/esm/model/utils.d.ts +39 -0
- package/lib/esm/model/utils.js +42 -3
- package/lib/esm/model/validation.d.ts +26 -8
- package/lib/esm/model/validation.js +29 -11
- package/lib/esm/operations/Operations.d.ts +65 -3
- package/lib/esm/operations/Operations.js +68 -6
- package/lib/esm/operations/OperationsRegistry.d.ts +44 -16
- package/lib/esm/operations/OperationsRegistry.js +46 -18
- package/lib/esm/operations/constants.d.ts +27 -8
- package/lib/esm/operations/constants.js +16 -9
- package/lib/esm/operations/decorators.d.ts +140 -134
- package/lib/esm/operations/decorators.js +152 -137
- package/lib/esm/operations/index.js +6 -6
- package/lib/esm/operations/types.d.ts +10 -0
- package/lib/esm/operations/types.js +1 -1
- package/lib/esm/repository/BaseRepository.d.ts +322 -0
- package/lib/esm/repository/BaseRepository.js +297 -7
- package/lib/esm/repository/Context.d.ts +153 -2
- package/lib/esm/repository/Context.js +154 -6
- package/lib/esm/repository/Repository.d.ts +89 -0
- package/lib/esm/repository/Repository.js +96 -7
- package/lib/esm/repository/constants.d.ts +7 -0
- package/lib/esm/repository/constants.js +8 -1
- package/lib/esm/repository/errors.d.ts +61 -34
- package/lib/esm/repository/errors.js +62 -35
- package/lib/esm/repository/index.js +9 -9
- package/lib/esm/repository/types.d.ts +25 -0
- package/lib/esm/repository/types.js +1 -1
- package/lib/esm/repository/utils.d.ts +11 -0
- package/lib/esm/repository/utils.js +4 -4
- package/lib/esm/repository/wrappers.d.ts +2 -2
- package/lib/esm/repository/wrappers.js +5 -5
- package/lib/esm/validation/constants.d.ts +20 -5
- package/lib/esm/validation/constants.js +22 -7
- package/lib/esm/validation/decorators.d.ts +101 -19
- package/lib/esm/validation/decorators.js +109 -27
- package/lib/esm/validation/index.js +5 -5
- package/lib/esm/validation/validation.js +10 -2
- package/lib/esm/validation/validators/ReadOnlyValidator.d.ts +32 -8
- package/lib/esm/validation/validators/ReadOnlyValidator.js +34 -10
- package/lib/esm/validation/validators/TimestampValidator.d.ts +37 -3
- package/lib/esm/validation/validators/TimestampValidator.js +39 -5
- package/lib/esm/validation/validators/UpdateValidator.d.ts +28 -11
- package/lib/esm/validation/validators/UpdateValidator.js +23 -8
- package/lib/esm/validation/validators/index.js +4 -4
- package/lib/identity/decorators.cjs +8 -1
- package/lib/identity/decorators.d.ts +7 -0
- package/lib/identity/utils.cjs +35 -22
- package/lib/identity/utils.d.ts +36 -23
- package/lib/index.cjs +14 -28
- package/lib/index.d.ts +12 -27
- package/lib/interfaces/BulkCrudOperator.cjs +1 -1
- package/lib/interfaces/BulkCrudOperator.d.ts +39 -0
- package/lib/interfaces/Contextual.cjs +1 -1
- package/lib/interfaces/Contextual.d.ts +17 -0
- package/lib/interfaces/CrudOperator.cjs +1 -1
- package/lib/interfaces/CrudOperator.d.ts +26 -23
- package/lib/interfaces/IRepository.cjs +1 -1
- package/lib/interfaces/IRepository.d.ts +10 -2
- package/lib/model/constants.cjs +12 -14
- package/lib/model/constants.d.ts +11 -13
- package/lib/model/decorators.cjs +114 -24
- package/lib/model/decorators.d.ts +112 -23
- package/lib/model/index.cjs +2 -1
- package/lib/model/index.d.ts +1 -0
- package/lib/model/model.cjs +1 -13
- package/lib/model/model.d.ts +2 -141
- package/lib/model/overrides.cjs +25 -0
- package/lib/model/overrides.d.ts +1 -0
- package/lib/model/utils.cjs +40 -1
- package/lib/model/utils.d.ts +39 -0
- package/lib/model/validation.cjs +27 -9
- package/lib/model/validation.d.ts +26 -8
- package/lib/operations/Operations.cjs +66 -4
- package/lib/operations/Operations.d.ts +65 -3
- package/lib/operations/OperationsRegistry.cjs +45 -17
- package/lib/operations/OperationsRegistry.d.ts +44 -16
- package/lib/operations/constants.cjs +16 -9
- package/lib/operations/constants.d.ts +27 -8
- package/lib/operations/decorators.cjs +150 -135
- package/lib/operations/decorators.d.ts +140 -134
- package/lib/operations/types.cjs +1 -1
- package/lib/operations/types.d.ts +10 -0
- package/lib/repository/BaseRepository.cjs +291 -1
- package/lib/repository/BaseRepository.d.ts +322 -0
- package/lib/repository/Context.cjs +153 -5
- package/lib/repository/Context.d.ts +153 -2
- package/lib/repository/Repository.cjs +90 -1
- package/lib/repository/Repository.d.ts +89 -0
- package/lib/repository/constants.cjs +8 -1
- package/lib/repository/constants.d.ts +7 -0
- package/lib/repository/errors.cjs +62 -35
- package/lib/repository/errors.d.ts +61 -34
- package/lib/repository/types.cjs +1 -1
- package/lib/repository/types.d.ts +25 -0
- package/lib/repository/utils.cjs +1 -1
- package/lib/repository/utils.d.ts +11 -0
- package/lib/repository/wrappers.cjs +3 -3
- package/lib/repository/wrappers.d.ts +2 -2
- package/lib/validation/constants.cjs +21 -6
- package/lib/validation/constants.d.ts +20 -5
- package/lib/validation/decorators.cjs +102 -20
- package/lib/validation/decorators.d.ts +101 -19
- package/lib/validation/validation.cjs +9 -1
- package/lib/validation/validators/ReadOnlyValidator.cjs +33 -9
- package/lib/validation/validators/ReadOnlyValidator.d.ts +32 -8
- package/lib/validation/validators/TimestampValidator.cjs +38 -4
- package/lib/validation/validators/TimestampValidator.d.ts +37 -3
- package/lib/validation/validators/UpdateValidator.cjs +23 -8
- package/lib/validation/validators/UpdateValidator.d.ts +28 -11
- package/package.json +2 -2
|
@@ -3,17 +3,168 @@ import { Contextual } from "../interfaces/Contextual";
|
|
|
3
3
|
import { OperationKeys } from "../operations/constants";
|
|
4
4
|
import { Constructor, Model } from "@decaf-ts/decorator-validation";
|
|
5
5
|
import { RepositoryFlags } from "./types";
|
|
6
|
+
/**
|
|
7
|
+
* @description Factory type for creating context instances.
|
|
8
|
+
* @summary Defines a function type that creates context instances with specific repository flags.
|
|
9
|
+
* @template F - The repository flags type extending RepositoryFlags
|
|
10
|
+
* @typedef {Function} ContextFactory
|
|
11
|
+
* @memberOf module:db-decorators
|
|
12
|
+
*/
|
|
6
13
|
export type ContextFactory<F extends RepositoryFlags> = <C extends Context<F>>(arg: Omit<F, "timestamp">) => C;
|
|
14
|
+
/**
|
|
15
|
+
* @description Default factory for creating context instances.
|
|
16
|
+
* @summary A factory function that creates new Context instances with the provided repository flags.
|
|
17
|
+
* It automatically adds a timestamp to the context and returns a properly typed context instance.
|
|
18
|
+
* @const DefaultContextFactory
|
|
19
|
+
* @memberOf module:db-decorators
|
|
20
|
+
*/
|
|
7
21
|
export declare const DefaultContextFactory: ContextFactory<any>;
|
|
8
|
-
|
|
22
|
+
/**
|
|
23
|
+
* @description A context management class for handling repository operations.
|
|
24
|
+
* @summary The Context class provides a mechanism for managing repository operations with flags,
|
|
25
|
+
* parent-child relationships, and state accumulation. It allows for hierarchical context chains
|
|
26
|
+
* and maintains operation-specific configurations while supporting type safety through generics.
|
|
27
|
+
*
|
|
28
|
+
* @template F - Type extending RepositoryFlags that defines the context configuration
|
|
29
|
+
*
|
|
30
|
+
* @param {ObjectAccumulator<F>} cache - The internal cache storing accumulated values
|
|
31
|
+
*
|
|
32
|
+
* @class
|
|
33
|
+
*
|
|
34
|
+
* @example
|
|
35
|
+
* ```typescript
|
|
36
|
+
* // Creating a new context with repository flags
|
|
37
|
+
* const context = new Context<RepositoryFlags>();
|
|
38
|
+
*
|
|
39
|
+
* // Accumulating values
|
|
40
|
+
* const enrichedContext = context.accumulate({
|
|
41
|
+
* writeOperation: true,
|
|
42
|
+
* affectedTables: ['users'],
|
|
43
|
+
* operation: OperationKeys.CREATE
|
|
44
|
+
* });
|
|
45
|
+
*
|
|
46
|
+
* // Accessing values
|
|
47
|
+
* const isWrite = enrichedContext.get('writeOperation'); // true
|
|
48
|
+
* const tables = enrichedContext.get('affectedTables'); // ['users']
|
|
49
|
+
* ```
|
|
50
|
+
*
|
|
51
|
+
* @mermaid
|
|
52
|
+
* sequenceDiagram
|
|
53
|
+
* participant C as Client
|
|
54
|
+
* participant Ctx as Context
|
|
55
|
+
* participant Cache as ObjectAccumulator
|
|
56
|
+
*
|
|
57
|
+
* C->>Ctx: new Context()
|
|
58
|
+
* Ctx->>Cache: create cache
|
|
59
|
+
*
|
|
60
|
+
* C->>Ctx: accumulate(value)
|
|
61
|
+
* Ctx->>Cache: accumulate(value)
|
|
62
|
+
* Cache-->>Ctx: updated cache
|
|
63
|
+
* Ctx-->>C: updated context
|
|
64
|
+
*
|
|
65
|
+
* C->>Ctx: get(key)
|
|
66
|
+
* Ctx->>Cache: get(key)
|
|
67
|
+
* alt Key exists in cache
|
|
68
|
+
* Cache-->>Ctx: value
|
|
69
|
+
* else Key not found
|
|
70
|
+
* Ctx->>Ctx: check parent context
|
|
71
|
+
* alt Parent exists
|
|
72
|
+
* Ctx->>Parent: get(key)
|
|
73
|
+
* Parent-->>Ctx: value
|
|
74
|
+
* else No parent
|
|
75
|
+
* Ctx-->>C: throw error
|
|
76
|
+
* end
|
|
77
|
+
* end
|
|
78
|
+
* Ctx-->>C: requested value
|
|
79
|
+
*/
|
|
80
|
+
export declare class Context<F extends RepositoryFlags> {
|
|
81
|
+
constructor();
|
|
9
82
|
static factory: ContextFactory<any>;
|
|
10
83
|
private readonly cache;
|
|
11
|
-
|
|
84
|
+
/**
|
|
85
|
+
* @description Accumulates new values into the context.
|
|
86
|
+
* @summary Merges the provided value object with the existing context state,
|
|
87
|
+
* creating a new immutable cache state.
|
|
88
|
+
*
|
|
89
|
+
* @template F - current accumulator type
|
|
90
|
+
* @template V - Type extending object for the values to accumulate
|
|
91
|
+
* @param {V} value - The object containing values to accumulate
|
|
92
|
+
* @returns A new context instance with accumulated values
|
|
93
|
+
*/
|
|
12
94
|
accumulate<V extends object>(value: V): Context<F & V>;
|
|
13
95
|
get timestamp(): Date;
|
|
96
|
+
/**
|
|
97
|
+
* @description Retrieves a value from the context by key.
|
|
98
|
+
* @summary Attempts to get a value from the current context's cache.
|
|
99
|
+
* If not found, traverses up the parent context chain.
|
|
100
|
+
*
|
|
101
|
+
* @template K - Type extending keyof F for the key to retrieve
|
|
102
|
+
* @template F - Accumulator type
|
|
103
|
+
* @param {K} key - The key to retrieve from the context
|
|
104
|
+
* @returns The value associated with the key
|
|
105
|
+
* @throws {Error} If the key is not found in the context chain
|
|
106
|
+
*/
|
|
14
107
|
get<K extends keyof F>(key: K): F[K];
|
|
108
|
+
/**
|
|
109
|
+
* @description Creates a child context
|
|
110
|
+
* @summary Generates a new context instance with current context as parent
|
|
111
|
+
*
|
|
112
|
+
* @template M - Type extending Model
|
|
113
|
+
* @param {OperationKeys} operation - The operation type
|
|
114
|
+
* @param {Constructor<M>} [model] - Optional model constructor
|
|
115
|
+
* @returns {C} New child context instance
|
|
116
|
+
*/
|
|
15
117
|
child<M extends Model, C extends Context<F>>(operation: OperationKeys, model?: Constructor<M>): C;
|
|
118
|
+
/**
|
|
119
|
+
* @description Creates a child context from another context
|
|
120
|
+
* @summary Generates a new context instance with parent reference
|
|
121
|
+
*
|
|
122
|
+
* @template F - Type extending Repository Flags
|
|
123
|
+
* @template C - Type extending Context<F>
|
|
124
|
+
* @param {C} context - The parent context
|
|
125
|
+
* @param {Partial<F>} [overrides] - Optional flag overrides
|
|
126
|
+
* @returns {C} New child context instance
|
|
127
|
+
*/
|
|
16
128
|
static childFrom<F extends RepositoryFlags, C extends Context<F>>(context: C, overrides?: Partial<F>): C;
|
|
129
|
+
/**
|
|
130
|
+
* @description Creates a new context from operation parameters
|
|
131
|
+
* @summary Generates a context instance for specific operation
|
|
132
|
+
*
|
|
133
|
+
* @template F - Type extending Repository Flags
|
|
134
|
+
* @template M - Type extending Model
|
|
135
|
+
* @param {OperationKeys.DELETE} operation - The operation type
|
|
136
|
+
* @param {Partial<F>} overrides - Flag overrides
|
|
137
|
+
* @param {Constructor<M>} model - The model constructor
|
|
138
|
+
* @param {any} args - Operation arguments
|
|
139
|
+
* @returns {Promise<C>} Promise resolving to new context
|
|
140
|
+
*/
|
|
17
141
|
static from<M extends Model, F extends RepositoryFlags, C extends Context<F>>(operation: OperationKeys.CREATE | OperationKeys.READ | OperationKeys.UPDATE | OperationKeys.DELETE, overrides: Partial<F>, model: Constructor<M>, ...args: any[]): Promise<C>;
|
|
142
|
+
/**
|
|
143
|
+
* @description Prepares arguments for context operations
|
|
144
|
+
* @summary Creates a context args object with the specified operation parameters
|
|
145
|
+
*
|
|
146
|
+
* @template F - Type extending {@link RepositoryFlags}
|
|
147
|
+
* @template M - Type extending {@link Model}
|
|
148
|
+
* @param {OperationKeys.DELETE} operation - The operation type
|
|
149
|
+
* @param {Constructor<M>} model - The model constructor
|
|
150
|
+
* @param {any[]} args - Operation arguments
|
|
151
|
+
* @param {Contextual<F>} [contextual] - Optional contextual object
|
|
152
|
+
* @param {Partial<F>} [overrides] - Optional flag overrides
|
|
153
|
+
* @returns {Promise<ContextArgs>} Promise resolving to context arguments
|
|
154
|
+
*
|
|
155
|
+
* @mermaid
|
|
156
|
+
* sequenceDiagram
|
|
157
|
+
* participant C as Context
|
|
158
|
+
* participant M as Model
|
|
159
|
+
* participant A as Args
|
|
160
|
+
*
|
|
161
|
+
* C->>C: Receive operation request
|
|
162
|
+
* C->>M: Validate model constructor
|
|
163
|
+
* C->>C: Create child context
|
|
164
|
+
* C->>A: Process operation args
|
|
165
|
+
* A->>C: Return context args
|
|
166
|
+
* C->>C: Apply overrides
|
|
167
|
+
* C->>C: Return final context
|
|
168
|
+
*/
|
|
18
169
|
static args<M extends Model, C extends Context<F>, F extends RepositoryFlags>(operation: OperationKeys.CREATE | OperationKeys.READ | OperationKeys.UPDATE | OperationKeys.DELETE, model: Constructor<M>, args: any[], contextual?: Contextual<F>, overrides?: Partial<F>): Promise<ContextArgs<F, C>>;
|
|
19
170
|
}
|
|
@@ -7,10 +7,58 @@ const errors_1 = require("./errors.cjs");
|
|
|
7
7
|
const BaseRepository_1 = require("./BaseRepository.cjs");
|
|
8
8
|
const constants_2 = require("./../model/constants.cjs");
|
|
9
9
|
const Context_1 = require("./Context.cjs");
|
|
10
|
+
/**
|
|
11
|
+
* @description Concrete repository implementation with validation support.
|
|
12
|
+
* @summary The Repository class extends BaseRepository to provide additional validation
|
|
13
|
+
* functionality. It overrides prefix methods to perform model validation before database
|
|
14
|
+
* operations and throws ValidationError when validation fails.
|
|
15
|
+
* @template M - The model type extending Model
|
|
16
|
+
* @template F - The repository flags type, defaults to RepositoryFlags
|
|
17
|
+
* @template C - The context type, defaults to Context<F>
|
|
18
|
+
* @class Repository
|
|
19
|
+
* @example
|
|
20
|
+
* class UserModel extends Model {
|
|
21
|
+
* @id()
|
|
22
|
+
* id: string;
|
|
23
|
+
*
|
|
24
|
+
* @required()
|
|
25
|
+
* @minLength(3)
|
|
26
|
+
* name: string;
|
|
27
|
+
* }
|
|
28
|
+
*
|
|
29
|
+
* class UserRepository extends Repository<UserModel> {
|
|
30
|
+
* constructor() {
|
|
31
|
+
* super(UserModel);
|
|
32
|
+
* }
|
|
33
|
+
*
|
|
34
|
+
* async create(model: UserModel): Promise<UserModel> {
|
|
35
|
+
* // Implementation with automatic validation
|
|
36
|
+
* return model;
|
|
37
|
+
* }
|
|
38
|
+
* }
|
|
39
|
+
*
|
|
40
|
+
* // Using the repository
|
|
41
|
+
* const repo = new UserRepository();
|
|
42
|
+
* try {
|
|
43
|
+
* const user = await repo.create({ name: 'Jo' }); // Will throw ValidationError
|
|
44
|
+
* } catch (error) {
|
|
45
|
+
* console.error(error); // ValidationError: name must be at least 3 characters
|
|
46
|
+
* }
|
|
47
|
+
*/
|
|
10
48
|
class Repository extends BaseRepository_1.BaseRepository {
|
|
11
49
|
constructor(clazz) {
|
|
12
50
|
super(clazz);
|
|
13
51
|
}
|
|
52
|
+
/**
|
|
53
|
+
* @description Prepares a model for creation with validation.
|
|
54
|
+
* @summary Overrides the base createPrefix method to add validation checks.
|
|
55
|
+
* Creates a context, instantiates a new model, enforces decorators, and validates
|
|
56
|
+
* the model before allowing creation to proceed.
|
|
57
|
+
* @param {M} model - The model instance to prepare for creation
|
|
58
|
+
* @param {any[]} args - Additional arguments for the create operation
|
|
59
|
+
* @return A promise that resolves to an array containing the validated model and context arguments
|
|
60
|
+
* @throws {ValidationError} If the model fails validation
|
|
61
|
+
*/
|
|
14
62
|
async createPrefix(model, ...args) {
|
|
15
63
|
const contextArgs = await Context_1.Context.args(constants_1.OperationKeys.CREATE, this.class, args);
|
|
16
64
|
model = new this.class(model);
|
|
@@ -20,6 +68,16 @@ class Repository extends BaseRepository_1.BaseRepository {
|
|
|
20
68
|
throw new errors_1.ValidationError(errors.toString());
|
|
21
69
|
return [model, ...contextArgs.args];
|
|
22
70
|
}
|
|
71
|
+
/**
|
|
72
|
+
* @description Prepares multiple models for creation with validation.
|
|
73
|
+
* @summary Overrides the base createAllPrefix method to add validation checks for multiple models.
|
|
74
|
+
* Creates a context, instantiates new models, enforces decorators, and validates
|
|
75
|
+
* each model before allowing creation to proceed. Collects validation errors from all models.
|
|
76
|
+
* @param {M[]} models - The array of model instances to prepare for creation
|
|
77
|
+
* @param {any[]} args - Additional arguments for the create operation
|
|
78
|
+
* @return {Promise<any[]>} A promise that resolves to an array containing the validated models and context arguments
|
|
79
|
+
* @throws {ValidationError} If any model fails validation, with details about which models failed
|
|
80
|
+
*/
|
|
23
81
|
async createAllPrefix(models, ...args) {
|
|
24
82
|
const contextArgs = await Context_1.Context.args(constants_1.OperationKeys.CREATE, this.class, args);
|
|
25
83
|
await Promise.all(models.map(async (m) => {
|
|
@@ -41,6 +99,18 @@ class Repository extends BaseRepository_1.BaseRepository {
|
|
|
41
99
|
throw new errors_1.ValidationError(errors);
|
|
42
100
|
return [models, ...contextArgs.args];
|
|
43
101
|
}
|
|
102
|
+
/**
|
|
103
|
+
* @description Prepares a model for update with validation.
|
|
104
|
+
* @summary Overrides the base updatePrefix method to add validation checks.
|
|
105
|
+
* Creates a context, validates the primary key, retrieves the existing model,
|
|
106
|
+
* merges the old and new models, enforces decorators, and validates the model
|
|
107
|
+
* before allowing the update to proceed.
|
|
108
|
+
* @param {M} model - The model instance to prepare for update
|
|
109
|
+
* @param {any[]} args - Additional arguments for the update operation
|
|
110
|
+
* @return A promise that resolves to an array containing the validated model and context arguments
|
|
111
|
+
* @throws {InternalError} If the model doesn't have a primary key value
|
|
112
|
+
* @throws {ValidationError} If the model fails validation
|
|
113
|
+
*/
|
|
44
114
|
async updatePrefix(model, ...args) {
|
|
45
115
|
const contextArgs = await Context_1.Context.args(constants_1.OperationKeys.UPDATE, this.class, args);
|
|
46
116
|
const pk = model[this.pk];
|
|
@@ -54,6 +124,18 @@ class Repository extends BaseRepository_1.BaseRepository {
|
|
|
54
124
|
throw new errors_1.ValidationError(errors.toString());
|
|
55
125
|
return [model, ...contextArgs.args];
|
|
56
126
|
}
|
|
127
|
+
/**
|
|
128
|
+
* @description Prepares multiple models for update with validation.
|
|
129
|
+
* @summary Overrides the base updateAllPrefix method to add validation checks for multiple models.
|
|
130
|
+
* Creates a context, validates primary keys, retrieves existing models, merges old and new models,
|
|
131
|
+
* enforces decorators, and validates each model before allowing updates to proceed.
|
|
132
|
+
* Collects validation errors from all models.
|
|
133
|
+
* @param {M[]} models - The array of model instances to prepare for update
|
|
134
|
+
* @param {any[]} args - Additional arguments for the update operation
|
|
135
|
+
* @return A promise that resolves to an array containing the validated models and context arguments
|
|
136
|
+
* @throws {InternalError} If any model doesn't have a primary key value
|
|
137
|
+
* @throws {ValidationError} If any model fails validation, with details about which models failed
|
|
138
|
+
*/
|
|
57
139
|
async updateAllPrefix(models, ...args) {
|
|
58
140
|
const contextArgs = await Context_1.Context.args(constants_1.OperationKeys.UPDATE, this.class, args);
|
|
59
141
|
const ids = models.map((m) => {
|
|
@@ -79,9 +161,16 @@ class Repository extends BaseRepository_1.BaseRepository {
|
|
|
79
161
|
throw new errors_1.ValidationError(errors);
|
|
80
162
|
return [models, ...contextArgs.args];
|
|
81
163
|
}
|
|
164
|
+
/**
|
|
165
|
+
* @description Creates a reflection key for database operations.
|
|
166
|
+
* @summary Generates a key for storing metadata in the reflection system by prefixing
|
|
167
|
+
* the provided key with the database reflection prefix.
|
|
168
|
+
* @param {string} key - The base key to prefix
|
|
169
|
+
* @return {string} The prefixed reflection key
|
|
170
|
+
*/
|
|
82
171
|
static key(key) {
|
|
83
172
|
return constants_2.DBKeys.REFLECT + key;
|
|
84
173
|
}
|
|
85
174
|
}
|
|
86
175
|
exports.Repository = Repository;
|
|
87
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"Repository.js","sourceRoot":"","sources":["../../src/repository/Repository.ts"],"names":[],"mappings":"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","sourcesContent":["import { enforceDBDecorators } from \"./utils\";\nimport { OperationKeys } from \"../operations/constants\";\nimport { InternalError, ValidationError } from \"./errors\";\nimport { BaseRepository } from \"./BaseRepository\";\nimport { Constructor, Model } from \"@decaf-ts/decorator-validation\";\nimport { DBKeys } from \"../model/constants\";\nimport { Context } from \"./Context\";\nimport { RepositoryFlags } from \"./types\";\n\nexport abstract class Repository<\n  M extends Model,\n  F extends RepositoryFlags = RepositoryFlags,\n  C extends Context<F> = Context<F>,\n> extends BaseRepository<M, F, C> {\n  protected constructor(clazz?: Constructor<M>) {\n    super(clazz);\n  }\n\n  protected override async createPrefix(\n    model: M,\n    ...args: any[]\n  ): Promise<[M, ...any[]]> {\n    const contextArgs = await Context.args(\n      OperationKeys.CREATE,\n      this.class,\n      args\n    );\n    model = new this.class(model);\n    await enforceDBDecorators(\n      this,\n      contextArgs.context,\n      model,\n      OperationKeys.CREATE,\n      OperationKeys.ON\n    );\n\n    const errors = model.hasErrors();\n    if (errors) throw new ValidationError(errors.toString());\n\n    return [model, ...contextArgs.args];\n  }\n\n  protected override async createAllPrefix(\n    models: M[],\n    ...args: any[]\n  ): Promise<any[]> {\n    const contextArgs = await Context.args(\n      OperationKeys.CREATE,\n      this.class,\n      args\n    );\n    await Promise.all(\n      models.map(async (m) => {\n        m = new this.class(m);\n        await enforceDBDecorators(\n          this,\n          contextArgs.context,\n          m,\n          OperationKeys.CREATE,\n          OperationKeys.ON\n        );\n        return m;\n      })\n    );\n    const errors = models\n      .map((m) => m.hasErrors())\n      .reduce((accum: string | undefined, e, i) => {\n        if (e)\n          accum =\n            typeof accum === \"string\"\n              ? accum + `\\n - ${i}: ${e.toString()}`\n              : ` - ${i}: ${e.toString()}`;\n        return accum;\n      }, undefined);\n    if (errors) throw new ValidationError(errors);\n    return [models, ...contextArgs.args];\n  }\n\n  protected override async updatePrefix(\n    model: M,\n    ...args: any[]\n  ): Promise<[M, ...args: any[]]> {\n    const contextArgs = await Context.args(\n      OperationKeys.UPDATE,\n      this.class,\n      args\n    );\n    const pk = (model as any)[this.pk];\n    if (!pk)\n      throw new InternalError(\n        `No value for the Id is defined under the property ${this.pk as string}`\n      );\n\n    const oldModel: M = await this.read(pk);\n\n    model = this.merge(oldModel, model);\n\n    await enforceDBDecorators(\n      this,\n      contextArgs.context,\n      model,\n      OperationKeys.UPDATE,\n      OperationKeys.ON,\n      oldModel\n    );\n\n    const errors = model.hasErrors(oldModel as any);\n    if (errors) throw new ValidationError(errors.toString());\n    return [model, ...contextArgs.args];\n  }\n\n  protected override async updateAllPrefix(models: M[], ...args: any[]) {\n    const contextArgs = await Context.args(\n      OperationKeys.UPDATE,\n      this.class,\n      args\n    );\n    const ids = models.map((m) => {\n      const id = m[this.pk];\n      if (typeof id === \"undefined\")\n        throw new InternalError(\n          `No value for the Id is defined under the property ${this.pk as string}`\n        );\n      return id as string;\n    });\n    const oldModels: M[] = await this.readAll(ids, ...contextArgs.args);\n    models = models.map((m, i) => this.merge(oldModels[i], m));\n    await Promise.all(\n      models.map((m, i) =>\n        enforceDBDecorators(\n          this,\n          contextArgs.context,\n          m,\n          OperationKeys.UPDATE,\n          OperationKeys.ON,\n          oldModels[i]\n        )\n      )\n    );\n\n    const errors = models\n      .map((m, i) => m.hasErrors(oldModels[i] as any))\n      .reduce((accum: string | undefined, e, i) => {\n        if (e)\n          accum =\n            typeof accum === \"string\"\n              ? accum + `\\n - ${i}: ${e.toString()}`\n              : ` - ${i}: ${e.toString()}`;\n        return accum;\n      }, undefined);\n    if (errors) throw new ValidationError(errors);\n    return [models, ...contextArgs.args];\n  }\n\n  static key(key: string) {\n    return DBKeys.REFLECT + key;\n  }\n}\n"]}
|
|
176
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"Repository.js","sourceRoot":"","sources":["../../src/repository/Repository.ts"],"names":[],"mappings":"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","sourcesContent":["import { enforceDBDecorators } from \"./utils\";\nimport { OperationKeys } from \"../operations/constants\";\nimport { InternalError, ValidationError } from \"./errors\";\nimport { BaseRepository } from \"./BaseRepository\";\nimport { Constructor, Model } from \"@decaf-ts/decorator-validation\";\nimport { DBKeys } from \"../model/constants\";\nimport { Context } from \"./Context\";\nimport { RepositoryFlags } from \"./types\";\n\n/**\n * @description Concrete repository implementation with validation support.\n * @summary The Repository class extends BaseRepository to provide additional validation\n * functionality. It overrides prefix methods to perform model validation before database\n * operations and throws ValidationError when validation fails.\n * @template M - The model type extending Model\n * @template F - The repository flags type, defaults to RepositoryFlags\n * @template C - The context type, defaults to Context<F>\n * @class Repository\n * @example\n * class UserModel extends Model {\n *   @id()\n *   id: string;\n *\n *   @required()\n *   @minLength(3)\n *   name: string;\n * }\n *\n * class UserRepository extends Repository<UserModel> {\n *   constructor() {\n *     super(UserModel);\n *   }\n *\n *   async create(model: UserModel): Promise<UserModel> {\n *     // Implementation with automatic validation\n *     return model;\n *   }\n * }\n *\n * // Using the repository\n * const repo = new UserRepository();\n * try {\n *   const user = await repo.create({ name: 'Jo' }); // Will throw ValidationError\n * } catch (error) {\n *   console.error(error); // ValidationError: name must be at least 3 characters\n * }\n */\nexport abstract class Repository<\n  M extends Model,\n  F extends RepositoryFlags = RepositoryFlags,\n  C extends Context<F> = Context<F>,\n> extends BaseRepository<M, F, C> {\n  protected constructor(clazz?: Constructor<M>) {\n    super(clazz);\n  }\n\n  /**\n   * @description Prepares a model for creation with validation.\n   * @summary Overrides the base createPrefix method to add validation checks.\n   * Creates a context, instantiates a new model, enforces decorators, and validates\n   * the model before allowing creation to proceed.\n   * @param {M} model - The model instance to prepare for creation\n   * @param {any[]} args - Additional arguments for the create operation\n   * @return A promise that resolves to an array containing the validated model and context arguments\n   * @throws {ValidationError} If the model fails validation\n   */\n  protected override async createPrefix(\n    model: M,\n    ...args: any[]\n  ): Promise<[M, ...any[]]> {\n    const contextArgs = await Context.args(\n      OperationKeys.CREATE,\n      this.class,\n      args\n    );\n    model = new this.class(model);\n    await enforceDBDecorators(\n      this,\n      contextArgs.context,\n      model,\n      OperationKeys.CREATE,\n      OperationKeys.ON\n    );\n\n    const errors = model.hasErrors();\n    if (errors) throw new ValidationError(errors.toString());\n\n    return [model, ...contextArgs.args];\n  }\n\n  /**\n   * @description Prepares multiple models for creation with validation.\n   * @summary Overrides the base createAllPrefix method to add validation checks for multiple models.\n   * Creates a context, instantiates new models, enforces decorators, and validates\n   * each model before allowing creation to proceed. Collects validation errors from all models.\n   * @param {M[]} models - The array of model instances to prepare for creation\n   * @param {any[]} args - Additional arguments for the create operation\n   * @return {Promise<any[]>} A promise that resolves to an array containing the validated models and context arguments\n   * @throws {ValidationError} If any model fails validation, with details about which models failed\n   */\n  protected override async createAllPrefix(\n    models: M[],\n    ...args: any[]\n  ): Promise<any[]> {\n    const contextArgs = await Context.args(\n      OperationKeys.CREATE,\n      this.class,\n      args\n    );\n    await Promise.all(\n      models.map(async (m) => {\n        m = new this.class(m);\n        await enforceDBDecorators(\n          this,\n          contextArgs.context,\n          m,\n          OperationKeys.CREATE,\n          OperationKeys.ON\n        );\n        return m;\n      })\n    );\n    const errors = models\n      .map((m) => m.hasErrors())\n      .reduce((accum: string | undefined, e, i) => {\n        if (e)\n          accum =\n            typeof accum === \"string\"\n              ? accum + `\\n - ${i}: ${e.toString()}`\n              : ` - ${i}: ${e.toString()}`;\n        return accum;\n      }, undefined);\n    if (errors) throw new ValidationError(errors);\n    return [models, ...contextArgs.args];\n  }\n\n  /**\n   * @description Prepares a model for update with validation.\n   * @summary Overrides the base updatePrefix method to add validation checks.\n   * Creates a context, validates the primary key, retrieves the existing model,\n   * merges the old and new models, enforces decorators, and validates the model\n   * before allowing the update to proceed.\n   * @param {M} model - The model instance to prepare for update\n   * @param {any[]} args - Additional arguments for the update operation\n   * @return A promise that resolves to an array containing the validated model and context arguments\n   * @throws {InternalError} If the model doesn't have a primary key value\n   * @throws {ValidationError} If the model fails validation\n   */\n  protected override async updatePrefix(\n    model: M,\n    ...args: any[]\n  ): Promise<[M, ...args: any[]]> {\n    const contextArgs = await Context.args(\n      OperationKeys.UPDATE,\n      this.class,\n      args\n    );\n    const pk = (model as any)[this.pk];\n    if (!pk)\n      throw new InternalError(\n        `No value for the Id is defined under the property ${this.pk as string}`\n      );\n\n    const oldModel: M = await this.read(pk);\n\n    model = this.merge(oldModel, model);\n\n    await enforceDBDecorators(\n      this,\n      contextArgs.context,\n      model,\n      OperationKeys.UPDATE,\n      OperationKeys.ON,\n      oldModel\n    );\n\n    const errors = model.hasErrors(oldModel as any);\n    if (errors) throw new ValidationError(errors.toString());\n    return [model, ...contextArgs.args];\n  }\n\n  /**\n   * @description Prepares multiple models for update with validation.\n   * @summary Overrides the base updateAllPrefix method to add validation checks for multiple models.\n   * Creates a context, validates primary keys, retrieves existing models, merges old and new models,\n   * enforces decorators, and validates each model before allowing updates to proceed.\n   * Collects validation errors from all models.\n   * @param {M[]} models - The array of model instances to prepare for update\n   * @param {any[]} args - Additional arguments for the update operation\n   * @return A promise that resolves to an array containing the validated models and context arguments\n   * @throws {InternalError} If any model doesn't have a primary key value\n   * @throws {ValidationError} If any model fails validation, with details about which models failed\n   */\n  protected override async updateAllPrefix(models: M[], ...args: any[]) {\n    const contextArgs = await Context.args(\n      OperationKeys.UPDATE,\n      this.class,\n      args\n    );\n    const ids = models.map((m) => {\n      const id = m[this.pk];\n      if (typeof id === \"undefined\")\n        throw new InternalError(\n          `No value for the Id is defined under the property ${this.pk as string}`\n        );\n      return id as string;\n    });\n    const oldModels: M[] = await this.readAll(ids, ...contextArgs.args);\n    models = models.map((m, i) => this.merge(oldModels[i], m));\n    await Promise.all(\n      models.map((m, i) =>\n        enforceDBDecorators(\n          this,\n          contextArgs.context,\n          m,\n          OperationKeys.UPDATE,\n          OperationKeys.ON,\n          oldModels[i]\n        )\n      )\n    );\n\n    const errors = models\n      .map((m, i) => m.hasErrors(oldModels[i] as any))\n      .reduce((accum: string | undefined, e, i) => {\n        if (e)\n          accum =\n            typeof accum === \"string\"\n              ? accum + `\\n - ${i}: ${e.toString()}`\n              : ` - ${i}: ${e.toString()}`;\n        return accum;\n      }, undefined);\n    if (errors) throw new ValidationError(errors);\n    return [models, ...contextArgs.args];\n  }\n\n  /**\n   * @description Creates a reflection key for database operations.\n   * @summary Generates a key for storing metadata in the reflection system by prefixing\n   * the provided key with the database reflection prefix.\n   * @param {string} key - The base key to prefix\n   * @return {string} The prefixed reflection key\n   */\n  static key(key: string) {\n    return DBKeys.REFLECT + key;\n  }\n}\n"]}
|
|
@@ -2,11 +2,100 @@ import { BaseRepository } from "./BaseRepository";
|
|
|
2
2
|
import { Constructor, Model } from "@decaf-ts/decorator-validation";
|
|
3
3
|
import { Context } from "./Context";
|
|
4
4
|
import { RepositoryFlags } from "./types";
|
|
5
|
+
/**
|
|
6
|
+
* @description Concrete repository implementation with validation support.
|
|
7
|
+
* @summary The Repository class extends BaseRepository to provide additional validation
|
|
8
|
+
* functionality. It overrides prefix methods to perform model validation before database
|
|
9
|
+
* operations and throws ValidationError when validation fails.
|
|
10
|
+
* @template M - The model type extending Model
|
|
11
|
+
* @template F - The repository flags type, defaults to RepositoryFlags
|
|
12
|
+
* @template C - The context type, defaults to Context<F>
|
|
13
|
+
* @class Repository
|
|
14
|
+
* @example
|
|
15
|
+
* class UserModel extends Model {
|
|
16
|
+
* @id()
|
|
17
|
+
* id: string;
|
|
18
|
+
*
|
|
19
|
+
* @required()
|
|
20
|
+
* @minLength(3)
|
|
21
|
+
* name: string;
|
|
22
|
+
* }
|
|
23
|
+
*
|
|
24
|
+
* class UserRepository extends Repository<UserModel> {
|
|
25
|
+
* constructor() {
|
|
26
|
+
* super(UserModel);
|
|
27
|
+
* }
|
|
28
|
+
*
|
|
29
|
+
* async create(model: UserModel): Promise<UserModel> {
|
|
30
|
+
* // Implementation with automatic validation
|
|
31
|
+
* return model;
|
|
32
|
+
* }
|
|
33
|
+
* }
|
|
34
|
+
*
|
|
35
|
+
* // Using the repository
|
|
36
|
+
* const repo = new UserRepository();
|
|
37
|
+
* try {
|
|
38
|
+
* const user = await repo.create({ name: 'Jo' }); // Will throw ValidationError
|
|
39
|
+
* } catch (error) {
|
|
40
|
+
* console.error(error); // ValidationError: name must be at least 3 characters
|
|
41
|
+
* }
|
|
42
|
+
*/
|
|
5
43
|
export declare abstract class Repository<M extends Model, F extends RepositoryFlags = RepositoryFlags, C extends Context<F> = Context<F>> extends BaseRepository<M, F, C> {
|
|
6
44
|
protected constructor(clazz?: Constructor<M>);
|
|
45
|
+
/**
|
|
46
|
+
* @description Prepares a model for creation with validation.
|
|
47
|
+
* @summary Overrides the base createPrefix method to add validation checks.
|
|
48
|
+
* Creates a context, instantiates a new model, enforces decorators, and validates
|
|
49
|
+
* the model before allowing creation to proceed.
|
|
50
|
+
* @param {M} model - The model instance to prepare for creation
|
|
51
|
+
* @param {any[]} args - Additional arguments for the create operation
|
|
52
|
+
* @return A promise that resolves to an array containing the validated model and context arguments
|
|
53
|
+
* @throws {ValidationError} If the model fails validation
|
|
54
|
+
*/
|
|
7
55
|
protected createPrefix(model: M, ...args: any[]): Promise<[M, ...any[]]>;
|
|
56
|
+
/**
|
|
57
|
+
* @description Prepares multiple models for creation with validation.
|
|
58
|
+
* @summary Overrides the base createAllPrefix method to add validation checks for multiple models.
|
|
59
|
+
* Creates a context, instantiates new models, enforces decorators, and validates
|
|
60
|
+
* each model before allowing creation to proceed. Collects validation errors from all models.
|
|
61
|
+
* @param {M[]} models - The array of model instances to prepare for creation
|
|
62
|
+
* @param {any[]} args - Additional arguments for the create operation
|
|
63
|
+
* @return {Promise<any[]>} A promise that resolves to an array containing the validated models and context arguments
|
|
64
|
+
* @throws {ValidationError} If any model fails validation, with details about which models failed
|
|
65
|
+
*/
|
|
8
66
|
protected createAllPrefix(models: M[], ...args: any[]): Promise<any[]>;
|
|
67
|
+
/**
|
|
68
|
+
* @description Prepares a model for update with validation.
|
|
69
|
+
* @summary Overrides the base updatePrefix method to add validation checks.
|
|
70
|
+
* Creates a context, validates the primary key, retrieves the existing model,
|
|
71
|
+
* merges the old and new models, enforces decorators, and validates the model
|
|
72
|
+
* before allowing the update to proceed.
|
|
73
|
+
* @param {M} model - The model instance to prepare for update
|
|
74
|
+
* @param {any[]} args - Additional arguments for the update operation
|
|
75
|
+
* @return A promise that resolves to an array containing the validated model and context arguments
|
|
76
|
+
* @throws {InternalError} If the model doesn't have a primary key value
|
|
77
|
+
* @throws {ValidationError} If the model fails validation
|
|
78
|
+
*/
|
|
9
79
|
protected updatePrefix(model: M, ...args: any[]): Promise<[M, ...args: any[]]>;
|
|
80
|
+
/**
|
|
81
|
+
* @description Prepares multiple models for update with validation.
|
|
82
|
+
* @summary Overrides the base updateAllPrefix method to add validation checks for multiple models.
|
|
83
|
+
* Creates a context, validates primary keys, retrieves existing models, merges old and new models,
|
|
84
|
+
* enforces decorators, and validates each model before allowing updates to proceed.
|
|
85
|
+
* Collects validation errors from all models.
|
|
86
|
+
* @param {M[]} models - The array of model instances to prepare for update
|
|
87
|
+
* @param {any[]} args - Additional arguments for the update operation
|
|
88
|
+
* @return A promise that resolves to an array containing the validated models and context arguments
|
|
89
|
+
* @throws {InternalError} If any model doesn't have a primary key value
|
|
90
|
+
* @throws {ValidationError} If any model fails validation, with details about which models failed
|
|
91
|
+
*/
|
|
10
92
|
protected updateAllPrefix(models: M[], ...args: any[]): Promise<any[]>;
|
|
93
|
+
/**
|
|
94
|
+
* @description Creates a reflection key for database operations.
|
|
95
|
+
* @summary Generates a key for storing metadata in the reflection system by prefixing
|
|
96
|
+
* the provided key with the database reflection prefix.
|
|
97
|
+
* @param {string} key - The base key to prefix
|
|
98
|
+
* @return {string} The prefixed reflection key
|
|
99
|
+
*/
|
|
11
100
|
static key(key: string): string;
|
|
12
101
|
}
|
|
@@ -1,6 +1,13 @@
|
|
|
1
1
|
"use strict";
|
|
2
2
|
Object.defineProperty(exports, "__esModule", { value: true });
|
|
3
3
|
exports.DefaultRepositoryFlags = void 0;
|
|
4
|
+
/**
|
|
5
|
+
* @description Default configuration flags for repository operations.
|
|
6
|
+
* @summary Provides default values for repository operation flags, excluding the timestamp property.
|
|
7
|
+
* These flags control behavior such as context handling, validation, error handling, and more.
|
|
8
|
+
* @const DefaultRepositoryFlags
|
|
9
|
+
* @memberOf module:db-decorators
|
|
10
|
+
*/
|
|
4
11
|
exports.DefaultRepositoryFlags = {
|
|
5
12
|
parentContext: undefined,
|
|
6
13
|
childContexts: [],
|
|
@@ -12,4 +19,4 @@ exports.DefaultRepositoryFlags = {
|
|
|
12
19
|
breakOnHandlerError: true,
|
|
13
20
|
rebuildWithTransient: true,
|
|
14
21
|
};
|
|
15
|
-
//# sourceMappingURL=data:application/json;base64,
|
|
22
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiY29uc3RhbnRzLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vc3JjL3JlcG9zaXRvcnkvY29uc3RhbnRzLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiI7OztBQUVBOzs7Ozs7R0FNRztBQUNVLFFBQUEsc0JBQXNCLEdBQXVDO0lBQ3hFLGFBQWEsRUFBRSxTQUFTO0lBQ3hCLGFBQWEsRUFBRSxFQUFFO0lBQ2pCLDJCQUEyQixFQUFFLEVBQUU7SUFDL0IsUUFBUSxFQUFFLEVBQUU7SUFDWixjQUFjLEVBQUUsS0FBSztJQUNyQixjQUFjLEVBQUUsRUFBRTtJQUNsQixTQUFTLEVBQUUsU0FBUztJQUNwQixtQkFBbUIsRUFBRSxJQUFJO0lBQ3pCLG9CQUFvQixFQUFFLElBQUk7Q0FDM0IsQ0FBQyIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB7IFJlcG9zaXRvcnlGbGFncyB9IGZyb20gXCIuL3R5cGVzXCI7XG5cbi8qKlxuICogQGRlc2NyaXB0aW9uIERlZmF1bHQgY29uZmlndXJhdGlvbiBmbGFncyBmb3IgcmVwb3NpdG9yeSBvcGVyYXRpb25zLlxuICogQHN1bW1hcnkgUHJvdmlkZXMgZGVmYXVsdCB2YWx1ZXMgZm9yIHJlcG9zaXRvcnkgb3BlcmF0aW9uIGZsYWdzLCBleGNsdWRpbmcgdGhlIHRpbWVzdGFtcCBwcm9wZXJ0eS5cbiAqIFRoZXNlIGZsYWdzIGNvbnRyb2wgYmVoYXZpb3Igc3VjaCBhcyBjb250ZXh0IGhhbmRsaW5nLCB2YWxpZGF0aW9uLCBlcnJvciBoYW5kbGluZywgYW5kIG1vcmUuXG4gKiBAY29uc3QgRGVmYXVsdFJlcG9zaXRvcnlGbGFnc1xuICogQG1lbWJlck9mIG1vZHVsZTpkYi1kZWNvcmF0b3JzXG4gKi9cbmV4cG9ydCBjb25zdCBEZWZhdWx0UmVwb3NpdG9yeUZsYWdzOiBPbWl0PFJlcG9zaXRvcnlGbGFncywgXCJ0aW1lc3RhbXBcIj4gPSB7XG4gIHBhcmVudENvbnRleHQ6IHVuZGVmaW5lZCxcbiAgY2hpbGRDb250ZXh0czogW10sXG4gIGlnbm9yZWRWYWxpZGF0aW9uUHJvcGVydGllczogW10sXG4gIGNhbGxBcmdzOiBbXSxcbiAgd3JpdGVPcGVyYXRpb246IGZhbHNlLFxuICBhZmZlY3RlZFRhYmxlczogW10sXG4gIG9wZXJhdGlvbjogdW5kZWZpbmVkLFxuICBicmVha09uSGFuZGxlckVycm9yOiB0cnVlLFxuICByZWJ1aWxkV2l0aFRyYW5zaWVudDogdHJ1ZSxcbn07XG4iXX0=
|
|
@@ -1,2 +1,9 @@
|
|
|
1
1
|
import { RepositoryFlags } from "./types";
|
|
2
|
+
/**
|
|
3
|
+
* @description Default configuration flags for repository operations.
|
|
4
|
+
* @summary Provides default values for repository operation flags, excluding the timestamp property.
|
|
5
|
+
* These flags control behavior such as context handling, validation, error handling, and more.
|
|
6
|
+
* @const DefaultRepositoryFlags
|
|
7
|
+
* @memberOf module:db-decorators
|
|
8
|
+
*/
|
|
2
9
|
export declare const DefaultRepositoryFlags: Omit<RepositoryFlags, "timestamp">;
|