@decaf-ts/for-fabric 0.0.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.
Files changed (205) hide show
  1. package/LICENSE.md +22 -0
  2. package/README.md +647 -0
  3. package/dist/for-fabric.cjs +6223 -0
  4. package/dist/for-fabric.esm.cjs +6180 -0
  5. package/lib/client/FabricClientAdapter.cjs +760 -0
  6. package/lib/client/FabricClientAdapter.d.ts +381 -0
  7. package/lib/client/FabricClientDispatch.cjs +186 -0
  8. package/lib/client/FabricClientDispatch.d.ts +125 -0
  9. package/lib/client/FabricClientRepository.cjs +131 -0
  10. package/lib/client/FabricClientRepository.d.ts +100 -0
  11. package/lib/client/erc20/erc20ClientRepository.cjs +343 -0
  12. package/lib/client/erc20/erc20ClientRepository.d.ts +254 -0
  13. package/lib/client/fabric-fs.cjs +234 -0
  14. package/lib/client/fabric-fs.d.ts +92 -0
  15. package/lib/client/index.cjs +30 -0
  16. package/lib/client/index.d.ts +13 -0
  17. package/lib/client/logging.cjs +102 -0
  18. package/lib/client/logging.d.ts +60 -0
  19. package/lib/client/services/LoggedService.cjs +47 -0
  20. package/lib/client/services/LoggedService.d.ts +42 -0
  21. package/lib/client/services/constants.cjs +3 -0
  22. package/lib/client/services/constants.d.ts +15 -0
  23. package/lib/client/services/enrollementService.cjs +344 -0
  24. package/lib/client/services/enrollementService.d.ts +176 -0
  25. package/lib/client/services/index.cjs +18 -0
  26. package/lib/client/services/index.d.ts +1 -0
  27. package/lib/contracts/ContractAdapter.cjs +730 -0
  28. package/lib/contracts/ContractAdapter.d.ts +296 -0
  29. package/lib/contracts/ContractContext.cjs +85 -0
  30. package/lib/contracts/ContractContext.d.ts +64 -0
  31. package/lib/contracts/ContractPrivateDataAdapter.cjs +281 -0
  32. package/lib/contracts/ContractPrivateDataAdapter.d.ts +74 -0
  33. package/lib/contracts/FabricConstruction.cjs +441 -0
  34. package/lib/contracts/FabricConstruction.d.ts +304 -0
  35. package/lib/contracts/FabricContractRepository.cjs +306 -0
  36. package/lib/contracts/FabricContractRepository.d.ts +162 -0
  37. package/lib/contracts/FabricContractRepositoryObservableHandler.cjs +85 -0
  38. package/lib/contracts/FabricContractRepositoryObservableHandler.d.ts +62 -0
  39. package/lib/contracts/FabricContractSequence.cjs +139 -0
  40. package/lib/contracts/FabricContractSequence.d.ts +61 -0
  41. package/lib/contracts/FabricContractStatement.cjs +119 -0
  42. package/lib/contracts/FabricContractStatement.d.ts +34 -0
  43. package/lib/contracts/PrivateSequence.cjs +36 -0
  44. package/lib/contracts/PrivateSequence.d.ts +15 -0
  45. package/lib/contracts/crud/crud-contract.cjs +257 -0
  46. package/lib/contracts/crud/crud-contract.d.ts +168 -0
  47. package/lib/contracts/crud/index.cjs +19 -0
  48. package/lib/contracts/crud/index.d.ts +2 -0
  49. package/lib/contracts/crud/serialized-crud-contract.cjs +172 -0
  50. package/lib/contracts/crud/serialized-crud-contract.d.ts +37 -0
  51. package/lib/contracts/erc20/erc20contract.cjs +569 -0
  52. package/lib/contracts/erc20/erc20contract.d.ts +151 -0
  53. package/lib/contracts/erc20/index.cjs +21 -0
  54. package/lib/contracts/erc20/index.d.ts +2 -0
  55. package/lib/contracts/erc20/models.cjs +209 -0
  56. package/lib/contracts/erc20/models.d.ts +114 -0
  57. package/lib/contracts/index.cjs +32 -0
  58. package/lib/contracts/index.d.ts +15 -0
  59. package/lib/contracts/logging.cjs +96 -0
  60. package/lib/contracts/logging.d.ts +49 -0
  61. package/lib/contracts/private-data.cjs +121 -0
  62. package/lib/contracts/private-data.d.ts +16 -0
  63. package/lib/contracts/types.cjs +3 -0
  64. package/lib/contracts/types.d.ts +26 -0
  65. package/lib/esm/client/FabricClientAdapter.d.ts +381 -0
  66. package/lib/esm/client/FabricClientAdapter.js +723 -0
  67. package/lib/esm/client/FabricClientDispatch.d.ts +125 -0
  68. package/lib/esm/client/FabricClientDispatch.js +182 -0
  69. package/lib/esm/client/FabricClientRepository.d.ts +100 -0
  70. package/lib/esm/client/FabricClientRepository.js +127 -0
  71. package/lib/esm/client/erc20/erc20ClientRepository.d.ts +254 -0
  72. package/lib/esm/client/erc20/erc20ClientRepository.js +339 -0
  73. package/lib/esm/client/fabric-fs.d.ts +92 -0
  74. package/lib/esm/client/fabric-fs.js +191 -0
  75. package/lib/esm/client/index.d.ts +13 -0
  76. package/lib/esm/client/index.js +14 -0
  77. package/lib/esm/client/logging.d.ts +60 -0
  78. package/lib/esm/client/logging.js +98 -0
  79. package/lib/esm/client/services/LoggedService.d.ts +42 -0
  80. package/lib/esm/client/services/LoggedService.js +43 -0
  81. package/lib/esm/client/services/constants.d.ts +15 -0
  82. package/lib/esm/client/services/constants.js +2 -0
  83. package/lib/esm/client/services/enrollementService.d.ts +176 -0
  84. package/lib/esm/client/services/enrollementService.js +337 -0
  85. package/lib/esm/client/services/index.d.ts +1 -0
  86. package/lib/esm/client/services/index.js +2 -0
  87. package/lib/esm/contracts/ContractAdapter.d.ts +296 -0
  88. package/lib/esm/contracts/ContractAdapter.js +724 -0
  89. package/lib/esm/contracts/ContractContext.d.ts +64 -0
  90. package/lib/esm/contracts/ContractContext.js +81 -0
  91. package/lib/esm/contracts/ContractPrivateDataAdapter.d.ts +74 -0
  92. package/lib/esm/contracts/ContractPrivateDataAdapter.js +277 -0
  93. package/lib/esm/contracts/FabricConstruction.d.ts +304 -0
  94. package/lib/esm/contracts/FabricConstruction.js +433 -0
  95. package/lib/esm/contracts/FabricContractRepository.d.ts +162 -0
  96. package/lib/esm/contracts/FabricContractRepository.js +302 -0
  97. package/lib/esm/contracts/FabricContractRepositoryObservableHandler.d.ts +62 -0
  98. package/lib/esm/contracts/FabricContractRepositoryObservableHandler.js +81 -0
  99. package/lib/esm/contracts/FabricContractSequence.d.ts +61 -0
  100. package/lib/esm/contracts/FabricContractSequence.js +135 -0
  101. package/lib/esm/contracts/FabricContractStatement.d.ts +34 -0
  102. package/lib/esm/contracts/FabricContractStatement.js +115 -0
  103. package/lib/esm/contracts/PrivateSequence.d.ts +15 -0
  104. package/lib/esm/contracts/PrivateSequence.js +33 -0
  105. package/lib/esm/contracts/crud/crud-contract.d.ts +168 -0
  106. package/lib/esm/contracts/crud/crud-contract.js +253 -0
  107. package/lib/esm/contracts/crud/index.d.ts +2 -0
  108. package/lib/esm/contracts/crud/index.js +3 -0
  109. package/lib/esm/contracts/crud/serialized-crud-contract.d.ts +37 -0
  110. package/lib/esm/contracts/crud/serialized-crud-contract.js +168 -0
  111. package/lib/esm/contracts/erc20/erc20contract.d.ts +151 -0
  112. package/lib/esm/contracts/erc20/erc20contract.js +565 -0
  113. package/lib/esm/contracts/erc20/index.d.ts +2 -0
  114. package/lib/esm/contracts/erc20/index.js +4 -0
  115. package/lib/esm/contracts/erc20/models.d.ts +114 -0
  116. package/lib/esm/contracts/erc20/models.js +206 -0
  117. package/lib/esm/contracts/index.d.ts +15 -0
  118. package/lib/esm/contracts/index.js +16 -0
  119. package/lib/esm/contracts/logging.d.ts +49 -0
  120. package/lib/esm/contracts/logging.js +92 -0
  121. package/lib/esm/contracts/private-data.d.ts +16 -0
  122. package/lib/esm/contracts/private-data.js +113 -0
  123. package/lib/esm/contracts/types.d.ts +26 -0
  124. package/lib/esm/contracts/types.js +2 -0
  125. package/lib/esm/index.d.ts +8 -0
  126. package/lib/esm/index.js +9 -0
  127. package/lib/esm/shared/ClientSerializer.d.ts +52 -0
  128. package/lib/esm/shared/ClientSerializer.js +80 -0
  129. package/lib/esm/shared/DeterministicSerializer.d.ts +40 -0
  130. package/lib/esm/shared/DeterministicSerializer.js +50 -0
  131. package/lib/esm/shared/SimpleDeterministicSerializer.d.ts +7 -0
  132. package/lib/esm/shared/SimpleDeterministicSerializer.js +42 -0
  133. package/lib/esm/shared/constants.d.ts +39 -0
  134. package/lib/esm/shared/constants.js +42 -0
  135. package/lib/esm/shared/crypto.d.ts +107 -0
  136. package/lib/esm/shared/crypto.js +331 -0
  137. package/lib/esm/shared/decorators.d.ts +24 -0
  138. package/lib/esm/shared/decorators.js +98 -0
  139. package/lib/esm/shared/erc20/erc20-constants.d.ts +25 -0
  140. package/lib/esm/shared/erc20/erc20-constants.js +27 -0
  141. package/lib/esm/shared/errors.d.ts +116 -0
  142. package/lib/esm/shared/errors.js +132 -0
  143. package/lib/esm/shared/events.d.ts +39 -0
  144. package/lib/esm/shared/events.js +47 -0
  145. package/lib/esm/shared/fabric-types.d.ts +33 -0
  146. package/lib/esm/shared/fabric-types.js +2 -0
  147. package/lib/esm/shared/index.d.ts +13 -0
  148. package/lib/esm/shared/index.js +14 -0
  149. package/lib/esm/shared/interfaces/Checkable.d.ts +21 -0
  150. package/lib/esm/shared/interfaces/Checkable.js +2 -0
  151. package/lib/esm/shared/math.d.ts +34 -0
  152. package/lib/esm/shared/math.js +61 -0
  153. package/lib/esm/shared/model/Identity.d.ts +42 -0
  154. package/lib/esm/shared/model/Identity.js +78 -0
  155. package/lib/esm/shared/model/IdentityCredentials.d.ts +41 -0
  156. package/lib/esm/shared/model/IdentityCredentials.js +74 -0
  157. package/lib/esm/shared/model/index.d.ts +1 -0
  158. package/lib/esm/shared/model/index.js +2 -0
  159. package/lib/esm/shared/model/utils.d.ts +60 -0
  160. package/lib/esm/shared/model/utils.js +108 -0
  161. package/lib/esm/shared/types.d.ts +79 -0
  162. package/lib/esm/shared/types.js +2 -0
  163. package/lib/esm/shared/utils.d.ts +55 -0
  164. package/lib/esm/shared/utils.js +148 -0
  165. package/lib/index.cjs +25 -0
  166. package/lib/index.d.ts +8 -0
  167. package/lib/shared/ClientSerializer.cjs +84 -0
  168. package/lib/shared/ClientSerializer.d.ts +52 -0
  169. package/lib/shared/DeterministicSerializer.cjs +54 -0
  170. package/lib/shared/DeterministicSerializer.d.ts +40 -0
  171. package/lib/shared/SimpleDeterministicSerializer.cjs +46 -0
  172. package/lib/shared/SimpleDeterministicSerializer.d.ts +7 -0
  173. package/lib/shared/constants.cjs +45 -0
  174. package/lib/shared/constants.d.ts +39 -0
  175. package/lib/shared/crypto.cjs +369 -0
  176. package/lib/shared/crypto.d.ts +107 -0
  177. package/lib/shared/decorators.cjs +105 -0
  178. package/lib/shared/decorators.d.ts +24 -0
  179. package/lib/shared/erc20/erc20-constants.cjs +30 -0
  180. package/lib/shared/erc20/erc20-constants.d.ts +25 -0
  181. package/lib/shared/errors.cjs +142 -0
  182. package/lib/shared/errors.d.ts +116 -0
  183. package/lib/shared/events.cjs +51 -0
  184. package/lib/shared/events.d.ts +39 -0
  185. package/lib/shared/fabric-types.cjs +4 -0
  186. package/lib/shared/fabric-types.d.ts +33 -0
  187. package/lib/shared/index.cjs +30 -0
  188. package/lib/shared/index.d.ts +13 -0
  189. package/lib/shared/interfaces/Checkable.cjs +3 -0
  190. package/lib/shared/interfaces/Checkable.d.ts +21 -0
  191. package/lib/shared/math.cjs +66 -0
  192. package/lib/shared/math.d.ts +34 -0
  193. package/lib/shared/model/Identity.cjs +81 -0
  194. package/lib/shared/model/Identity.d.ts +42 -0
  195. package/lib/shared/model/IdentityCredentials.cjs +77 -0
  196. package/lib/shared/model/IdentityCredentials.d.ts +41 -0
  197. package/lib/shared/model/index.cjs +18 -0
  198. package/lib/shared/model/index.d.ts +1 -0
  199. package/lib/shared/model/utils.cjs +114 -0
  200. package/lib/shared/model/utils.d.ts +60 -0
  201. package/lib/shared/types.cjs +3 -0
  202. package/lib/shared/types.d.ts +79 -0
  203. package/lib/shared/utils.cjs +185 -0
  204. package/lib/shared/utils.d.ts +55 -0
  205. package/package.json +166 -0
@@ -0,0 +1,162 @@
1
+ import { EventIds, ObserverHandler, Repository, WhereOption } from "@decaf-ts/core";
2
+ import { FabricContractAdapter } from "./ContractAdapter";
3
+ import { FabricContractFlags } from "./types";
4
+ import { FabricContractContext } from "./ContractContext";
5
+ import { Constructor, Model } from "@decaf-ts/decorator-validation";
6
+ import { BulkCrudOperationKeys, OperationKeys } from "@decaf-ts/db-decorators";
7
+ import { MangoQuery } from "@decaf-ts/for-couchdb";
8
+ import { Context } from "fabric-contract-api";
9
+ import { ContractLogger } from "./logging";
10
+ /**
11
+ * @description Repository for Hyperledger Fabric chaincode models
12
+ * @summary Provides CRUD operations for models within Fabric chaincode contracts
13
+ * @template M - Type extending Model
14
+ * @template MangoQuery - Query type for CouchDB-like queries
15
+ * @template FabricContractAdapter - Adapter type for Fabric contract operations
16
+ * @template FabricContractFlags - Flags specific to Fabric contract operations
17
+ * @template FabricContractContext - Context type for Fabric contract operations
18
+ *
19
+ * @param {FabricContractAdapter} [adapter] - The adapter for interacting with the state database
20
+ * @param {Constructor<M>} [clazz] - The model constructor
21
+ * @param {Array<OperationKeys | BulkCrudOperationKeys | string>} [trackedEvents] - Events to track for observer notifications
22
+ *
23
+ * @class FabricContractRepository
24
+ * @example
25
+ * ```typescript
26
+ * // In a Fabric chaincode contract class
27
+ * import { FabricContractRepository, FabricContractAdapter } from '@decaf-ts/for-fabric';
28
+ *
29
+ * @table('assets')
30
+ * class Asset extends Model {
31
+ * @id()
32
+ * id: string;
33
+ *
34
+ * @property()
35
+ * data: string;
36
+ * }
37
+ *
38
+ * export class MyContract extends Contract {
39
+ * private adapter = new FabricContractAdapter();
40
+ * private repository: FabricContractRepository<Asset>;
41
+ *
42
+ * constructor() {
43
+ * super('MyContract');
44
+ * this.repository = new FabricContractRepository<Asset>(this.adapter, Asset);
45
+ * }
46
+ *
47
+ * @Transaction()
48
+ * async createAsset(ctx: Context, id: string, data: string): Promise<void> {
49
+ * const asset = new Asset();
50
+ * asset.id = id;
51
+ * asset.data = data;
52
+ *
53
+ * await this.repository.create(asset, { stub: ctx.stub });
54
+ * }
55
+ * }
56
+ * ```
57
+ * @mermaid
58
+ * sequenceDiagram
59
+ * participant Contract
60
+ * participant Repository
61
+ * participant Adapter
62
+ * participant StateDB
63
+ *
64
+ * Contract->>Repository: create(model, ctx)
65
+ * Repository->>Adapter: prepare(model, pk)
66
+ * Repository->>Adapter: create(tableName, id, record, transient, ctx)
67
+ * Adapter->>StateDB: putState(id, serializedData)
68
+ * StateDB-->>Adapter: Success
69
+ * Adapter-->>Repository: record
70
+ * Repository->>Adapter: revert(record, class, pk, id, transient)
71
+ * Adapter-->>Repository: model
72
+ * Repository-->>Contract: model
73
+ */
74
+ export declare class FabricContractRepository<M extends Model> extends Repository<M, MangoQuery, FabricContractAdapter, FabricContractFlags, FabricContractContext> {
75
+ protected trackedEvents?: (OperationKeys | BulkCrudOperationKeys | string)[] | undefined;
76
+ constructor(adapter?: FabricContractAdapter, clazz?: Constructor<M>, trackedEvents?: (OperationKeys | BulkCrudOperationKeys | string)[] | undefined);
77
+ /**
78
+ * @description Creates a logger for a specific chaincode context
79
+ * @summary Returns a ContractLogger instance configured for the current context
80
+ * @param {Ctx} ctx - The Fabric chaincode context
81
+ * @return {ContractLogger} The logger instance
82
+ */
83
+ logFor(ctx: Context): ContractLogger;
84
+ /**
85
+ * @description Creates a single model in the state database
86
+ * @summary Prepares, creates, and reverts a model using the adapter
87
+ * @param {M} model - The model to create
88
+ * @param {...any[]} args - Additional arguments, including the chaincode context
89
+ * @return {Promise<M>} Promise resolving to the created model
90
+ */
91
+ create(model: M, ...args: any[]): Promise<M>;
92
+ read(id: string | number | bigint, ...args: any[]): Promise<M>;
93
+ /**
94
+ * @description Deletes a model from the database by ID.
95
+ * @summary Removes a model instance from the database using its primary key.
96
+ * @param {string|number|bigint} id - The primary key of the model to delete.
97
+ * @param {...any[]} args - Additional arguments.
98
+ * @return {Promise<M>} The deleted model instance.
99
+ */
100
+ delete(id: string, ...args: any[]): Promise<M>;
101
+ /**
102
+ * @description Updates a single model in the state database
103
+ * @summary Prepares, updates, and reverts a model using the adapter
104
+ * @param {M} model - The model to update
105
+ * @param {...any[]} args - Additional arguments, including the chaincode context
106
+ * @return {Promise<M>} Promise resolving to the updated model
107
+ */
108
+ update(model: M, ...args: any[]): Promise<M>;
109
+ /**
110
+ * @description Gets the observer handler for this repository
111
+ * @summary Returns a FabricContractRepositoryObservableHandler instance
112
+ * @return {ObserverHandler} The observer handler
113
+ */
114
+ ObserverHandler(): ObserverHandler;
115
+ /**
116
+ * @description Creates multiple models in the state database
117
+ * @summary Prepares, creates, and reverts multiple models using the adapter
118
+ * @param {M[]} models - The models to create
119
+ * @param {...any[]} args - Additional arguments, including the chaincode context
120
+ * @return {Promise<M[]>} Promise resolving to the created models
121
+ */
122
+ createAll(models: M[], ...args: any[]): Promise<M[]>;
123
+ /**
124
+ * @description Updates multiple models in the state database
125
+ * @summary Prepares, updates, and reverts multiple models using the adapter
126
+ * @param {M[]} models - The models to update
127
+ * @param {...any[]} args - Additional arguments, including the chaincode context
128
+ * @return {Promise<M[]>} Promise resolving to the updated models
129
+ */
130
+ updateAll(models: M[], ...args: any[]): Promise<M[]>;
131
+ /**
132
+ * @description Executes a raw query against the state database
133
+ * @summary Delegates to the adapter's raw method
134
+ * @param {MangoQuery} rawInput - The Mango Query to execute
135
+ * @param {boolean} docsOnly - Whether to return only documents
136
+ * @param {...any[]} args - Additional arguments, including the chaincode context
137
+ * @return {Promise<any>} Promise resolving to the query results
138
+ */
139
+ raw(rawInput: MangoQuery, docsOnly: boolean, ...args: any[]): Promise<unknown>;
140
+ /**
141
+ * @description Updates observers based on tracked events
142
+ * @summary Filters events based on trackedEvents and delegates to the parent method
143
+ * @param {string} table - The table/collection name
144
+ * @param {OperationKeys | BulkCrudOperationKeys | string} event - The event type
145
+ * @param {EventIds} id - The event identifier
146
+ * @param {FabricContractContext} ctx - The Fabric contract context
147
+ * @param {...any[]} args - Additional arguments
148
+ * @return {Promise<void>} Promise that resolves when observers are updated
149
+ */
150
+ updateObservers(table: string, event: OperationKeys | BulkCrudOperationKeys | string, id: EventIds, ctx: FabricContractContext, ...args: any[]): Promise<void>;
151
+ selectWithContext<S extends readonly (keyof M)[]>(selector?: readonly [...S], ctx?: FabricContractContext | Context): Promise<WhereOption<M, M[]> | WhereOption<M, Pick<M, S[number]>[]>>;
152
+ /**
153
+ * @description Prepares multiple models for creation.
154
+ * @summary Validates multiple models and prepares them for creation in the database.
155
+ * @param {M[]} models - The models to create.
156
+ * @param {...any[]} args - Additional arguments.
157
+ * @return The prepared models and context arguments.
158
+ * @throws {ValidationError} If any model fails validation.
159
+ */
160
+ protected createAllPrefix(models: M[], ...args: any[]): Promise<any[]>;
161
+ protected createPrefix(model: M, ...args: any[]): Promise<[M, ...any[]]>;
162
+ }
@@ -0,0 +1,302 @@
1
+ import { Repository, } from "@decaf-ts/core";
2
+ import { FabricContractContext } from "./ContractContext.js";
3
+ import { ConflictError, Context as Ctx, Context as CTX, enforceDBDecorators, InternalError, OperationKeys, ValidationError, } from "@decaf-ts/db-decorators";
4
+ import { FabricContractRepositoryObservableHandler } from "./FabricContractRepositoryObservableHandler.js";
5
+ import { FabricContractSequence } from "./FabricContractSequence.js";
6
+ import { Logging } from "@decaf-ts/logging";
7
+ /**
8
+ * @description Repository for Hyperledger Fabric chaincode models
9
+ * @summary Provides CRUD operations for models within Fabric chaincode contracts
10
+ * @template M - Type extending Model
11
+ * @template MangoQuery - Query type for CouchDB-like queries
12
+ * @template FabricContractAdapter - Adapter type for Fabric contract operations
13
+ * @template FabricContractFlags - Flags specific to Fabric contract operations
14
+ * @template FabricContractContext - Context type for Fabric contract operations
15
+ *
16
+ * @param {FabricContractAdapter} [adapter] - The adapter for interacting with the state database
17
+ * @param {Constructor<M>} [clazz] - The model constructor
18
+ * @param {Array<OperationKeys | BulkCrudOperationKeys | string>} [trackedEvents] - Events to track for observer notifications
19
+ *
20
+ * @class FabricContractRepository
21
+ * @example
22
+ * ```typescript
23
+ * // In a Fabric chaincode contract class
24
+ * import { FabricContractRepository, FabricContractAdapter } from '@decaf-ts/for-fabric';
25
+ *
26
+ * @table('assets')
27
+ * class Asset extends Model {
28
+ * @id()
29
+ * id: string;
30
+ *
31
+ * @property()
32
+ * data: string;
33
+ * }
34
+ *
35
+ * export class MyContract extends Contract {
36
+ * private adapter = new FabricContractAdapter();
37
+ * private repository: FabricContractRepository<Asset>;
38
+ *
39
+ * constructor() {
40
+ * super('MyContract');
41
+ * this.repository = new FabricContractRepository<Asset>(this.adapter, Asset);
42
+ * }
43
+ *
44
+ * @Transaction()
45
+ * async createAsset(ctx: Context, id: string, data: string): Promise<void> {
46
+ * const asset = new Asset();
47
+ * asset.id = id;
48
+ * asset.data = data;
49
+ *
50
+ * await this.repository.create(asset, { stub: ctx.stub });
51
+ * }
52
+ * }
53
+ * ```
54
+ * @mermaid
55
+ * sequenceDiagram
56
+ * participant Contract
57
+ * participant Repository
58
+ * participant Adapter
59
+ * participant StateDB
60
+ *
61
+ * Contract->>Repository: create(model, ctx)
62
+ * Repository->>Adapter: prepare(model, pk)
63
+ * Repository->>Adapter: create(tableName, id, record, transient, ctx)
64
+ * Adapter->>StateDB: putState(id, serializedData)
65
+ * StateDB-->>Adapter: Success
66
+ * Adapter-->>Repository: record
67
+ * Repository->>Adapter: revert(record, class, pk, id, transient)
68
+ * Adapter-->>Repository: model
69
+ * Repository-->>Contract: model
70
+ */
71
+ export class FabricContractRepository extends Repository {
72
+ constructor(adapter, clazz, trackedEvents) {
73
+ super(adapter, clazz);
74
+ this.trackedEvents = trackedEvents;
75
+ }
76
+ /**
77
+ * @description Creates a logger for a specific chaincode context
78
+ * @summary Returns a ContractLogger instance configured for the current context
79
+ * @param {Ctx} ctx - The Fabric chaincode context
80
+ * @return {ContractLogger} The logger instance
81
+ */
82
+ logFor(ctx) {
83
+ return Logging.for(FabricContractRepository, {}, ctx);
84
+ }
85
+ /**
86
+ * @description Creates a single model in the state database
87
+ * @summary Prepares, creates, and reverts a model using the adapter
88
+ * @param {M} model - The model to create
89
+ * @param {...any[]} args - Additional arguments, including the chaincode context
90
+ * @return {Promise<M>} Promise resolving to the created model
91
+ */
92
+ async create(model, ...args) {
93
+ const ctx = args[args.length - 1];
94
+ const log = this.logFor(ctx).for(this.create);
95
+ log.info(`Preparing model: ${JSON.stringify(model)}`);
96
+ // eslint-disable-next-line prefer-const
97
+ let { record, id, transient } = this.adapter.prepare(model, this.pk, this.tableName, ...args);
98
+ log.info(`Creating model: ${JSON.stringify(model)}`);
99
+ record = await this.adapter.create(this.tableName, id, record, ...args);
100
+ let c = undefined;
101
+ if (args.length)
102
+ c = args[args.length - 1];
103
+ log.info(`Reverting model: ${JSON.stringify(model)}`);
104
+ return this.adapter.revert(record, this.class, this.pk, id, c && c.get("rebuildWithTransient") ? transient : undefined);
105
+ }
106
+ async read(id, ...args) {
107
+ const m = await this.adapter.read(this.tableName, id, new this.class(), ...args);
108
+ return this.adapter.revert(m, this.class, this.pk, id);
109
+ }
110
+ /**
111
+ * @description Deletes a model from the database by ID.
112
+ * @summary Removes a model instance from the database using its primary key.
113
+ * @param {string|number|bigint} id - The primary key of the model to delete.
114
+ * @param {...any[]} args - Additional arguments.
115
+ * @return {Promise<M>} The deleted model instance.
116
+ */
117
+ async delete(id, ...args) {
118
+ const m = await this.adapter.delete(this.tableName, id, new this.class(), ...args);
119
+ return this.adapter.revert(m, this.class, this.pk, id);
120
+ }
121
+ /**
122
+ * @description Updates a single model in the state database
123
+ * @summary Prepares, updates, and reverts a model using the adapter
124
+ * @param {M} model - The model to update
125
+ * @param {...any[]} args - Additional arguments, including the chaincode context
126
+ * @return {Promise<M>} Promise resolving to the updated model
127
+ */
128
+ async update(model, ...args) {
129
+ const ctx = args[args.length - 1];
130
+ const log = this.logFor(ctx).for(this.update);
131
+ log.info(`Preparing model: ${JSON.stringify(model)}`);
132
+ // eslint-disable-next-line prefer-const
133
+ let { record, id, transient } = this.adapter.prepare(model, this.pk, this.tableName, ...args);
134
+ log.info(`Updating model: ${JSON.stringify(model)}`);
135
+ record = await this.adapter.update(this.tableName, id, record, ...args);
136
+ let c = undefined;
137
+ if (args.length)
138
+ c = args[args.length - 1];
139
+ log.info(`Reverting model: ${JSON.stringify(model)}`);
140
+ return this.adapter.revert(record, this.class, this.pk, id, c && c.get("rebuildWithTransient") ? transient : undefined);
141
+ }
142
+ /**
143
+ * @description Gets the observer handler for this repository
144
+ * @summary Returns a FabricContractRepositoryObservableHandler instance
145
+ * @return {ObserverHandler} The observer handler
146
+ */
147
+ ObserverHandler() {
148
+ return new FabricContractRepositoryObservableHandler();
149
+ }
150
+ /**
151
+ * @description Creates multiple models in the state database
152
+ * @summary Prepares, creates, and reverts multiple models using the adapter
153
+ * @param {M[]} models - The models to create
154
+ * @param {...any[]} args - Additional arguments, including the chaincode context
155
+ * @return {Promise<M[]>} Promise resolving to the created models
156
+ */
157
+ async createAll(models, ...args) {
158
+ if (!models.length)
159
+ return models;
160
+ const prepared = models.map((m) => this.adapter.prepare(m, this.pk, this.tableName, ...args));
161
+ const ids = prepared.map((p) => p.id);
162
+ let records = prepared.map((p) => p.record);
163
+ const transients = prepared.map((p) => p.transient).filter((e) => !!e);
164
+ let c = undefined;
165
+ if (args.length)
166
+ c = args[args.length - 1];
167
+ records = await this.adapter.createAll(this.tableName, ids, records, ...args);
168
+ return records.map((r, i) => this.adapter.revert(r, this.class, this.pk, ids[i], c && c.get("rebuildWithTransient") ? transients : undefined));
169
+ }
170
+ /**
171
+ * @description Updates multiple models in the state database
172
+ * @summary Prepares, updates, and reverts multiple models using the adapter
173
+ * @param {M[]} models - The models to update
174
+ * @param {...any[]} args - Additional arguments, including the chaincode context
175
+ * @return {Promise<M[]>} Promise resolving to the updated models
176
+ */
177
+ async updateAll(models, ...args) {
178
+ if (!models.length)
179
+ return models;
180
+ const records = models.map((m) => this.adapter.prepare(m, this.pk, this.tableName, ...args));
181
+ const transients = records.map((p) => p.transient).filter((e) => !!e);
182
+ let c = undefined;
183
+ if (args.length)
184
+ c = args[args.length - 1];
185
+ const updated = await this.adapter.updateAll(this.tableName, records.map((r) => r.id), records.map((r) => r.record), ...args);
186
+ return updated.map((u, i) => this.adapter.revert(u, this.class, this.pk, records[i].id, c && c.get("rebuildWithTransient") ? transients : undefined));
187
+ }
188
+ /**
189
+ * @description Executes a raw query against the state database
190
+ * @summary Delegates to the adapter's raw method
191
+ * @param {MangoQuery} rawInput - The Mango Query to execute
192
+ * @param {boolean} docsOnly - Whether to return only documents
193
+ * @param {...any[]} args - Additional arguments, including the chaincode context
194
+ * @return {Promise<any>} Promise resolving to the query results
195
+ */
196
+ async raw(rawInput, docsOnly, ...args) {
197
+ const ctx = args.pop();
198
+ const transformedArgs = await CTX.args("QUERY", this.class, [ctx], this["adapter"], {});
199
+ return this.adapter.raw(rawInput, docsOnly, new this.class(), ...transformedArgs.args);
200
+ }
201
+ /**
202
+ * @description Updates observers based on tracked events
203
+ * @summary Filters events based on trackedEvents and delegates to the parent method
204
+ * @param {string} table - The table/collection name
205
+ * @param {OperationKeys | BulkCrudOperationKeys | string} event - The event type
206
+ * @param {EventIds} id - The event identifier
207
+ * @param {FabricContractContext} ctx - The Fabric contract context
208
+ * @param {...any[]} args - Additional arguments
209
+ * @return {Promise<void>} Promise that resolves when observers are updated
210
+ */
211
+ async updateObservers(table, event, id, ctx, ...args) {
212
+ if (!this.trackedEvents || this.trackedEvents.indexOf(event) !== -1)
213
+ return await super.updateObservers(table, event, id, ctx, ...args);
214
+ }
215
+ async selectWithContext(selector, ctx) {
216
+ let contextArgs;
217
+ if (ctx instanceof FabricContractContext) {
218
+ contextArgs.context = ctx;
219
+ }
220
+ else {
221
+ contextArgs = await Ctx.args(OperationKeys.CREATE, this.class, [ctx], this.adapter, this._overrides || {});
222
+ }
223
+ if (!selector) {
224
+ return this.adapter
225
+ .Statement(contextArgs.context)
226
+ .select()
227
+ .from(this.class);
228
+ }
229
+ return this.adapter
230
+ .Statement(contextArgs.context)
231
+ .select(selector)
232
+ .from(this.class);
233
+ }
234
+ /**
235
+ * @description Prepares multiple models for creation.
236
+ * @summary Validates multiple models and prepares them for creation in the database.
237
+ * @param {M[]} models - The models to create.
238
+ * @param {...any[]} args - Additional arguments.
239
+ * @return The prepared models and context arguments.
240
+ * @throws {ValidationError} If any model fails validation.
241
+ */
242
+ async createAllPrefix(models, ...args) {
243
+ const ctx = args[args.length - 1];
244
+ const contextArgs = await Ctx.args(OperationKeys.CREATE, this.class, args, this.adapter, this._overrides || {});
245
+ if (!models.length)
246
+ return [models, ...contextArgs.args];
247
+ const opts = Repository.getSequenceOptions(models[0]);
248
+ let ids = [];
249
+ if (opts.type) {
250
+ if (!opts.name)
251
+ opts.name = FabricContractSequence.pk(models[0]);
252
+ ids = await (await this.adapter.Sequence(opts)).range(models.length, ctx);
253
+ }
254
+ else {
255
+ ids = models.map((m, i) => {
256
+ if (typeof m[this.pk] === "undefined")
257
+ throw new InternalError(`Primary key is not defined for model in position ${i}`);
258
+ return m[this.pk];
259
+ });
260
+ }
261
+ models = await Promise.all(models.map(async (m, i) => {
262
+ m = new this.class(m);
263
+ if (opts.type)
264
+ m[this.pk] = ids[i];
265
+ await enforceDBDecorators(this, contextArgs.context, m, OperationKeys.CREATE, OperationKeys.ON);
266
+ return m;
267
+ }));
268
+ const ignoredProps = contextArgs.context.get("ignoredValidationProperties") || [];
269
+ const errors = await Promise.all(models.map((m) => Promise.resolve(m.hasErrors(...ignoredProps))));
270
+ const errorMessages = errors.reduce((accum, e, i) => {
271
+ if (e)
272
+ accum =
273
+ typeof accum === "string"
274
+ ? accum + `\n - ${i}: ${e.toString()}`
275
+ : ` - ${i}: ${e.toString()}`;
276
+ return accum;
277
+ }, undefined);
278
+ if (errorMessages)
279
+ throw new ValidationError(errorMessages);
280
+ return [models, ...contextArgs.args];
281
+ }
282
+ async createPrefix(model, ...args) {
283
+ const result = await super.createPrefix(model, ...args);
284
+ const id = result[0][this.pk];
285
+ try {
286
+ const res = await this.read(String(id), ...args);
287
+ if (res) {
288
+ throw new ConflictError(`Conflict detected while creating model with id: ${id} already exists`);
289
+ }
290
+ }
291
+ catch (e) {
292
+ if (e.code === 404) {
293
+ this.logFor(args[args.length - 1]).info(`Record entry with pk ${id} does not exist, creating it now...`);
294
+ }
295
+ else {
296
+ throw e;
297
+ }
298
+ }
299
+ return result;
300
+ }
301
+ }
302
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"FabricContractRepository.js","sourceRoot":"","sources":["../../../src/contracts/FabricContractRepository.ts"],"names":[],"mappings":"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","sourcesContent":["import {\n  EventIds,\n  ObserverHandler,\n  Repository,\n  WhereOption,\n} from \"@decaf-ts/core\";\nimport { FabricContractAdapter } from \"./ContractAdapter\";\nimport { FabricContractFlags } from \"./types\";\nimport { FabricContractContext } from \"./ContractContext\";\nimport { Constructor, Model } from \"@decaf-ts/decorator-validation\";\nimport {\n  BulkCrudOperationKeys,\n  ConflictError,\n  Context as Ctx,\n  Context as CTX,\n  enforceDBDecorators,\n  InternalError,\n  OperationKeys,\n  ValidationError,\n} from \"@decaf-ts/db-decorators\";\nimport { MangoQuery } from \"@decaf-ts/for-couchdb\";\nimport { FabricContractRepositoryObservableHandler } from \"./FabricContractRepositoryObservableHandler\";\nimport { Context } from \"fabric-contract-api\";\nimport { FabricContractSequence } from \"./FabricContractSequence\";\nimport { ContractLogger } from \"./logging\";\nimport { Logging } from \"@decaf-ts/logging\";\n\n/**\n * @description Repository for Hyperledger Fabric chaincode models\n * @summary Provides CRUD operations for models within Fabric chaincode contracts\n * @template M - Type extending Model\n * @template MangoQuery - Query type for CouchDB-like queries\n * @template FabricContractAdapter - Adapter type for Fabric contract operations\n * @template FabricContractFlags - Flags specific to Fabric contract operations\n * @template FabricContractContext - Context type for Fabric contract operations\n *\n * @param {FabricContractAdapter} [adapter] - The adapter for interacting with the state database\n * @param {Constructor<M>} [clazz] - The model constructor\n * @param {Array<OperationKeys | BulkCrudOperationKeys | string>} [trackedEvents] - Events to track for observer notifications\n *\n * @class FabricContractRepository\n * @example\n * ```typescript\n * // In a Fabric chaincode contract class\n * import { FabricContractRepository, FabricContractAdapter } from '@decaf-ts/for-fabric';\n *\n * @table('assets')\n * class Asset extends Model {\n *   @id()\n *   id: string;\n *\n *   @property()\n *   data: string;\n * }\n *\n * export class MyContract extends Contract {\n *   private adapter = new FabricContractAdapter();\n *   private repository: FabricContractRepository<Asset>;\n *\n *   constructor() {\n *     super('MyContract');\n *     this.repository = new FabricContractRepository<Asset>(this.adapter, Asset);\n *   }\n *\n *   @Transaction()\n *   async createAsset(ctx: Context, id: string, data: string): Promise<void> {\n *     const asset = new Asset();\n *     asset.id = id;\n *     asset.data = data;\n *\n *     await this.repository.create(asset, { stub: ctx.stub });\n *   }\n * }\n * ```\n * @mermaid\n * sequenceDiagram\n *   participant Contract\n *   participant Repository\n *   participant Adapter\n *   participant StateDB\n *\n *   Contract->>Repository: create(model, ctx)\n *   Repository->>Adapter: prepare(model, pk)\n *   Repository->>Adapter: create(tableName, id, record, transient, ctx)\n *   Adapter->>StateDB: putState(id, serializedData)\n *   StateDB-->>Adapter: Success\n *   Adapter-->>Repository: record\n *   Repository->>Adapter: revert(record, class, pk, id, transient)\n *   Adapter-->>Repository: model\n *   Repository-->>Contract: model\n */\nexport class FabricContractRepository<M extends Model> extends Repository<\n  M,\n  MangoQuery,\n  FabricContractAdapter,\n  FabricContractFlags,\n  FabricContractContext\n> {\n  constructor(\n    adapter?: FabricContractAdapter,\n    clazz?: Constructor<M>,\n    protected trackedEvents?: (OperationKeys | BulkCrudOperationKeys | string)[]\n  ) {\n    super(adapter, clazz);\n  }\n\n  /**\n   * @description Creates a logger for a specific chaincode context\n   * @summary Returns a ContractLogger instance configured for the current context\n   * @param {Ctx} ctx - The Fabric chaincode context\n   * @return {ContractLogger} The logger instance\n   */\n  public logFor(ctx: Context): ContractLogger {\n    return Logging.for(FabricContractRepository, {}, ctx) as ContractLogger;\n  }\n\n  /**\n   * @description Creates a single model in the state database\n   * @summary Prepares, creates, and reverts a model using the adapter\n   * @param {M} model - The model to create\n   * @param {...any[]} args - Additional arguments, including the chaincode context\n   * @return {Promise<M>} Promise resolving to the created model\n   */\n  override async create(model: M, ...args: any[]): Promise<M> {\n    const ctx = args[args.length - 1] as Context;\n    const log = this.logFor(ctx).for(this.create);\n    log.info(`Preparing model: ${JSON.stringify(model)}`);\n    // eslint-disable-next-line prefer-const\n    let { record, id, transient } = this.adapter.prepare(\n      model,\n      this.pk,\n      this.tableName,\n      ...args\n    );\n    log.info(`Creating model: ${JSON.stringify(model)}`);\n    record = await this.adapter.create(this.tableName, id, record, ...args);\n    let c: FabricContractContext | undefined = undefined;\n    if (args.length) c = args[args.length - 1] as FabricContractContext;\n    log.info(`Reverting model: ${JSON.stringify(model)}`);\n    return this.adapter.revert<M>(\n      record,\n      this.class,\n      this.pk,\n      id,\n      c && c.get(\"rebuildWithTransient\") ? transient : undefined\n    );\n  }\n\n  override async read(\n    id: string | number | bigint,\n    ...args: any[]\n  ): Promise<M> {\n    const m = await this.adapter.read(\n      this.tableName,\n      id as string,\n      new this.class(),\n      ...args\n    );\n    return this.adapter.revert<M>(m, this.class, this.pk, id as string);\n  }\n\n  /**\n   * @description Deletes a model from the database by ID.\n   * @summary Removes a model instance from the database using its primary key.\n   * @param {string|number|bigint} id - The primary key of the model to delete.\n   * @param {...any[]} args - Additional arguments.\n   * @return {Promise<M>} The deleted model instance.\n   */\n  override async delete(id: string, ...args: any[]) {\n    const m = await this.adapter.delete(\n      this.tableName,\n      id,\n      new this.class(),\n      ...args\n    );\n    return this.adapter.revert(m, this.class, this.pk, id);\n  }\n\n  /**\n   * @description Updates a single model in the state database\n   * @summary Prepares, updates, and reverts a model using the adapter\n   * @param {M} model - The model to update\n   * @param {...any[]} args - Additional arguments, including the chaincode context\n   * @return {Promise<M>} Promise resolving to the updated model\n   */\n  override async update(model: M, ...args: any[]): Promise<M> {\n    const ctx = args[args.length - 1] as Context;\n    const log = this.logFor(ctx).for(this.update);\n    log.info(`Preparing model: ${JSON.stringify(model)}`);\n    // eslint-disable-next-line prefer-const\n    let { record, id, transient } = this.adapter.prepare(\n      model,\n      this.pk,\n      this.tableName,\n      ...args\n    );\n    log.info(`Updating model: ${JSON.stringify(model)}`);\n    record = await this.adapter.update(this.tableName, id, record, ...args);\n    let c: FabricContractContext | undefined = undefined;\n    if (args.length) c = args[args.length - 1] as FabricContractContext;\n    log.info(`Reverting model: ${JSON.stringify(model)}`);\n    return this.adapter.revert<M>(\n      record,\n      this.class,\n      this.pk,\n      id,\n      c && c.get(\"rebuildWithTransient\") ? transient : undefined\n    );\n  }\n\n  /**\n   * @description Gets the observer handler for this repository\n   * @summary Returns a FabricContractRepositoryObservableHandler instance\n   * @return {ObserverHandler} The observer handler\n   */\n  override ObserverHandler(): ObserverHandler {\n    return new FabricContractRepositoryObservableHandler();\n  }\n\n  /**\n   * @description Creates multiple models in the state database\n   * @summary Prepares, creates, and reverts multiple models using the adapter\n   * @param {M[]} models - The models to create\n   * @param {...any[]} args - Additional arguments, including the chaincode context\n   * @return {Promise<M[]>} Promise resolving to the created models\n   */\n  override async createAll(models: M[], ...args: any[]): Promise<M[]> {\n    if (!models.length) return models;\n    const prepared = models.map((m) =>\n      this.adapter.prepare(m, this.pk, this.tableName, ...args)\n    );\n    const ids = prepared.map((p) => p.id);\n    let records = prepared.map((p) => p.record);\n    const transients = prepared.map((p) => p.transient).filter((e) => !!e);\n    let c: FabricContractContext | undefined = undefined;\n    if (args.length) c = args[args.length - 1] as FabricContractContext;\n    records = await this.adapter.createAll(\n      this.tableName,\n      ids,\n      records,\n      ...args\n    );\n    return records.map((r, i) =>\n      this.adapter.revert(\n        r,\n        this.class,\n        this.pk,\n        ids[i] as string | number,\n        c && c.get(\"rebuildWithTransient\") ? transients : undefined\n      )\n    );\n  }\n\n  /**\n   * @description Updates multiple models in the state database\n   * @summary Prepares, updates, and reverts multiple models using the adapter\n   * @param {M[]} models - The models to update\n   * @param {...any[]} args - Additional arguments, including the chaincode context\n   * @return {Promise<M[]>} Promise resolving to the updated models\n   */\n  override async updateAll(models: M[], ...args: any[]): Promise<M[]> {\n    if (!models.length) return models;\n    const records = models.map((m) =>\n      this.adapter.prepare(m, this.pk, this.tableName, ...args)\n    );\n    const transients = records.map((p) => p.transient).filter((e) => !!e);\n    let c: FabricContractContext | undefined = undefined;\n    if (args.length) c = args[args.length - 1] as FabricContractContext;\n\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(\n        u,\n        this.class,\n        this.pk,\n        records[i].id,\n        c && c.get(\"rebuildWithTransient\") ? transients : undefined\n      )\n    );\n  }\n\n  /**\n   * @description Executes a raw query against the state database\n   * @summary Delegates to the adapter's raw method\n   * @param {MangoQuery} rawInput - The Mango Query to execute\n   * @param {boolean} docsOnly - Whether to return only documents\n   * @param {...any[]} args - Additional arguments, including the chaincode context\n   * @return {Promise<any>} Promise resolving to the query results\n   */\n  async raw(rawInput: MangoQuery, docsOnly: boolean, ...args: any[]) {\n    const ctx = args.pop();\n    const transformedArgs = await CTX.args(\n      \"QUERY\" as OperationKeys.CREATE,\n      this.class,\n      [ctx],\n      this[\"adapter\"] as any,\n      {}\n    );\n\n    return this.adapter.raw(\n      rawInput,\n      docsOnly,\n      new this.class(),\n      ...transformedArgs.args\n    );\n  }\n\n  /**\n   * @description Updates observers based on tracked events\n   * @summary Filters events based on trackedEvents and delegates to the parent method\n   * @param {string} table - The table/collection name\n   * @param {OperationKeys | BulkCrudOperationKeys | string} event - The event type\n   * @param {EventIds} id - The event identifier\n   * @param {FabricContractContext} ctx - The Fabric contract context\n   * @param {...any[]} args - Additional arguments\n   * @return {Promise<void>} Promise that resolves when observers are updated\n   */\n  override async updateObservers(\n    table: string,\n    event: OperationKeys | BulkCrudOperationKeys | string,\n    id: EventIds,\n    ctx: FabricContractContext,\n    ...args: any[]\n  ): Promise<void> {\n    if (!this.trackedEvents || this.trackedEvents.indexOf(event) !== -1)\n      return await super.updateObservers(table, event, id, ctx, ...args);\n  }\n\n  async selectWithContext<S extends readonly (keyof M)[]>(\n    selector?: readonly [...S],\n    ctx?: FabricContractContext | Context\n  ): Promise<WhereOption<M, M[]> | WhereOption<M, Pick<M, S[number]>[]>> {\n    let contextArgs: any;\n    if (ctx instanceof FabricContractContext) {\n      contextArgs.context = ctx;\n    } else {\n      contextArgs = await Ctx.args(\n        OperationKeys.CREATE,\n        this.class,\n        [ctx],\n        this.adapter,\n        this._overrides || {}\n      );\n    }\n    if (!selector) {\n      return this.adapter\n        .Statement<M>(contextArgs.context as FabricContractContext)\n        .select()\n        .from(this.class);\n    }\n    return this.adapter\n      .Statement<M>(contextArgs.context as FabricContractContext)\n      .select(selector)\n      .from(this.class);\n  }\n\n  /**\n   * @description Prepares multiple models for creation.\n   * @summary Validates multiple models and prepares them for creation in the database.\n   * @param {M[]} models - The models to create.\n   * @param {...any[]} args - Additional arguments.\n   * @return The prepared models and context arguments.\n   * @throws {ValidationError} If any model fails validation.\n   */\n  protected override async createAllPrefix(models: M[], ...args: any[]) {\n    const ctx = args[args.length - 1] as Context;\n\n    const contextArgs = await Ctx.args(\n      OperationKeys.CREATE,\n      this.class,\n      args,\n      this.adapter,\n      this._overrides || {}\n    );\n    if (!models.length) return [models, ...contextArgs.args];\n    const opts = Repository.getSequenceOptions(models[0]);\n    let ids: (string | number | bigint | undefined)[] = [];\n    if (opts.type) {\n      if (!opts.name) opts.name = FabricContractSequence.pk(models[0]);\n      ids = await (\n        (await this.adapter.Sequence(opts)) as FabricContractSequence\n      ).range(models.length, ctx as unknown as FabricContractContext);\n    } else {\n      ids = models.map((m, i) => {\n        if (typeof m[this.pk] === \"undefined\")\n          throw new InternalError(\n            `Primary key is not defined for model in position ${i}`\n          );\n        return m[this.pk] as string;\n      });\n    }\n\n    models = await Promise.all(\n      models.map(async (m, i) => {\n        m = new this.class(m);\n        if (opts.type) m[this.pk] = ids[i] as M[keyof M];\n        await enforceDBDecorators(\n          this,\n          contextArgs.context,\n          m,\n          OperationKeys.CREATE,\n          OperationKeys.ON\n        );\n        return m;\n      })\n    );\n\n    const ignoredProps =\n      contextArgs.context.get(\"ignoredValidationProperties\") || [];\n\n    const errors = await Promise.all(\n      models.map((m) => Promise.resolve(m.hasErrors(...ignoredProps)))\n    );\n\n    const errorMessages = errors.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\n    if (errorMessages) throw new ValidationError(errorMessages);\n    return [models, ...contextArgs.args];\n  }\n\n  protected override async createPrefix(\n    model: M,\n    ...args: any[]\n  ): Promise<[M, ...any[]]> {\n    const result = await super.createPrefix(model, ...args);\n    const id = result[0][this.pk];\n    try {\n      const res = await this.read(String(id), ...args);\n      if (res) {\n        throw new ConflictError(\n          `Conflict detected while creating model with id: ${id} already exists`\n        );\n      }\n    } catch (e) {\n      if ((e as any).code === 404) {\n        this.logFor(args[args.length - 1]).info(\n          `Record entry with pk ${id} does not exist, creating it now...`\n        );\n      } else {\n        throw e;\n      }\n    }\n    return result;\n  }\n}\n"]}
@@ -0,0 +1,62 @@
1
+ import { BulkCrudOperationKeys, OperationKeys } from "@decaf-ts/db-decorators";
2
+ import { EventIds, ObserverHandler } from "@decaf-ts/core";
3
+ import { Logger } from "@decaf-ts/logging";
4
+ import { Context } from "fabric-contract-api";
5
+ /**
6
+ * @description Observer handler for Fabric chaincode events
7
+ * @summary Emits events on the Fabric ledger when repository operations occur
8
+ * @class FabricContractRepositoryObservableHandler
9
+ * @extends {ObserverHandler}
10
+ * @example
11
+ * ```typescript
12
+ * // In a Fabric chaincode contract
13
+ * import { FabricContractRepositoryObservableHandler } from '@decaf-ts/for-fabric';
14
+ *
15
+ * // Create a handler with default supported events
16
+ * const handler = new FabricContractRepositoryObservableHandler();
17
+ *
18
+ * // Emit an event
19
+ * await handler.updateObservers(
20
+ * logger,
21
+ * 'assets',
22
+ * OperationKeys.CREATE,
23
+ * 'asset1',
24
+ * context
25
+ * );
26
+ * ```
27
+ * @mermaid
28
+ * sequenceDiagram
29
+ * participant Repository
30
+ * participant ObservableHandler
31
+ * participant Stub
32
+ * participant Ledger
33
+ *
34
+ * Repository->>ObservableHandler: updateObservers(log, table, event, id, ctx)
35
+ * ObservableHandler->>ObservableHandler: Check if event is supported
36
+ * ObservableHandler->>ObservableHandler: generateFabricEventName(table, event, owner)
37
+ * ObservableHandler->>Stub: setEvent(eventName, payload)
38
+ * Stub->>Ledger: Record event
39
+ */
40
+ export declare class FabricContractRepositoryObservableHandler extends ObserverHandler {
41
+ private supportedEvents;
42
+ /**
43
+ * @description Creates a new FabricContractRepositoryObservableHandler instance
44
+ * @summary Initializes the handler with a list of supported events
45
+ * @param {Array<OperationKeys | BulkCrudOperationKeys | string>} [supportedEvents] - Events that will trigger Fabric events
46
+ */
47
+ constructor(supportedEvents?: (OperationKeys | BulkCrudOperationKeys | string)[]);
48
+ /**
49
+ * @description Updates observers by emitting Fabric events
50
+ * @summary Emits events on the Fabric ledger for supported event types
51
+ * @param {Logger} log - Logger instance for debugging
52
+ * @param {string} table - The table/collection name
53
+ * @param {OperationKeys | BulkCrudOperationKeys | string} event - The event type
54
+ * @param {EventIds} id - The event identifier
55
+ * @param {FabricContractContext} ctx - The Fabric contract context
56
+ * @param {string} [owner] - Optional owner identifier for the event
57
+ * @param {object | string | undefined} [owner] - Optional payload for the event
58
+ *
59
+ * @return {Promise<void>} Promise that resolves when the event is emitted
60
+ */
61
+ updateObservers(log: Logger, table: string, event: OperationKeys | BulkCrudOperationKeys | string, id: EventIds, ctx: Context, owner?: string, payload?: object | string | undefined): Promise<void>;
62
+ }