@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.
- package/LICENSE.md +22 -0
- package/README.md +647 -0
- package/dist/for-fabric.cjs +6223 -0
- package/dist/for-fabric.esm.cjs +6180 -0
- package/lib/client/FabricClientAdapter.cjs +760 -0
- package/lib/client/FabricClientAdapter.d.ts +381 -0
- package/lib/client/FabricClientDispatch.cjs +186 -0
- package/lib/client/FabricClientDispatch.d.ts +125 -0
- package/lib/client/FabricClientRepository.cjs +131 -0
- package/lib/client/FabricClientRepository.d.ts +100 -0
- package/lib/client/erc20/erc20ClientRepository.cjs +343 -0
- package/lib/client/erc20/erc20ClientRepository.d.ts +254 -0
- package/lib/client/fabric-fs.cjs +234 -0
- package/lib/client/fabric-fs.d.ts +92 -0
- package/lib/client/index.cjs +30 -0
- package/lib/client/index.d.ts +13 -0
- package/lib/client/logging.cjs +102 -0
- package/lib/client/logging.d.ts +60 -0
- package/lib/client/services/LoggedService.cjs +47 -0
- package/lib/client/services/LoggedService.d.ts +42 -0
- package/lib/client/services/constants.cjs +3 -0
- package/lib/client/services/constants.d.ts +15 -0
- package/lib/client/services/enrollementService.cjs +344 -0
- package/lib/client/services/enrollementService.d.ts +176 -0
- package/lib/client/services/index.cjs +18 -0
- package/lib/client/services/index.d.ts +1 -0
- package/lib/contracts/ContractAdapter.cjs +730 -0
- package/lib/contracts/ContractAdapter.d.ts +296 -0
- package/lib/contracts/ContractContext.cjs +85 -0
- package/lib/contracts/ContractContext.d.ts +64 -0
- package/lib/contracts/ContractPrivateDataAdapter.cjs +281 -0
- package/lib/contracts/ContractPrivateDataAdapter.d.ts +74 -0
- package/lib/contracts/FabricConstruction.cjs +441 -0
- package/lib/contracts/FabricConstruction.d.ts +304 -0
- package/lib/contracts/FabricContractRepository.cjs +306 -0
- package/lib/contracts/FabricContractRepository.d.ts +162 -0
- package/lib/contracts/FabricContractRepositoryObservableHandler.cjs +85 -0
- package/lib/contracts/FabricContractRepositoryObservableHandler.d.ts +62 -0
- package/lib/contracts/FabricContractSequence.cjs +139 -0
- package/lib/contracts/FabricContractSequence.d.ts +61 -0
- package/lib/contracts/FabricContractStatement.cjs +119 -0
- package/lib/contracts/FabricContractStatement.d.ts +34 -0
- package/lib/contracts/PrivateSequence.cjs +36 -0
- package/lib/contracts/PrivateSequence.d.ts +15 -0
- package/lib/contracts/crud/crud-contract.cjs +257 -0
- package/lib/contracts/crud/crud-contract.d.ts +168 -0
- package/lib/contracts/crud/index.cjs +19 -0
- package/lib/contracts/crud/index.d.ts +2 -0
- package/lib/contracts/crud/serialized-crud-contract.cjs +172 -0
- package/lib/contracts/crud/serialized-crud-contract.d.ts +37 -0
- package/lib/contracts/erc20/erc20contract.cjs +569 -0
- package/lib/contracts/erc20/erc20contract.d.ts +151 -0
- package/lib/contracts/erc20/index.cjs +21 -0
- package/lib/contracts/erc20/index.d.ts +2 -0
- package/lib/contracts/erc20/models.cjs +209 -0
- package/lib/contracts/erc20/models.d.ts +114 -0
- package/lib/contracts/index.cjs +32 -0
- package/lib/contracts/index.d.ts +15 -0
- package/lib/contracts/logging.cjs +96 -0
- package/lib/contracts/logging.d.ts +49 -0
- package/lib/contracts/private-data.cjs +121 -0
- package/lib/contracts/private-data.d.ts +16 -0
- package/lib/contracts/types.cjs +3 -0
- package/lib/contracts/types.d.ts +26 -0
- package/lib/esm/client/FabricClientAdapter.d.ts +381 -0
- package/lib/esm/client/FabricClientAdapter.js +723 -0
- package/lib/esm/client/FabricClientDispatch.d.ts +125 -0
- package/lib/esm/client/FabricClientDispatch.js +182 -0
- package/lib/esm/client/FabricClientRepository.d.ts +100 -0
- package/lib/esm/client/FabricClientRepository.js +127 -0
- package/lib/esm/client/erc20/erc20ClientRepository.d.ts +254 -0
- package/lib/esm/client/erc20/erc20ClientRepository.js +339 -0
- package/lib/esm/client/fabric-fs.d.ts +92 -0
- package/lib/esm/client/fabric-fs.js +191 -0
- package/lib/esm/client/index.d.ts +13 -0
- package/lib/esm/client/index.js +14 -0
- package/lib/esm/client/logging.d.ts +60 -0
- package/lib/esm/client/logging.js +98 -0
- package/lib/esm/client/services/LoggedService.d.ts +42 -0
- package/lib/esm/client/services/LoggedService.js +43 -0
- package/lib/esm/client/services/constants.d.ts +15 -0
- package/lib/esm/client/services/constants.js +2 -0
- package/lib/esm/client/services/enrollementService.d.ts +176 -0
- package/lib/esm/client/services/enrollementService.js +337 -0
- package/lib/esm/client/services/index.d.ts +1 -0
- package/lib/esm/client/services/index.js +2 -0
- package/lib/esm/contracts/ContractAdapter.d.ts +296 -0
- package/lib/esm/contracts/ContractAdapter.js +724 -0
- package/lib/esm/contracts/ContractContext.d.ts +64 -0
- package/lib/esm/contracts/ContractContext.js +81 -0
- package/lib/esm/contracts/ContractPrivateDataAdapter.d.ts +74 -0
- package/lib/esm/contracts/ContractPrivateDataAdapter.js +277 -0
- package/lib/esm/contracts/FabricConstruction.d.ts +304 -0
- package/lib/esm/contracts/FabricConstruction.js +433 -0
- package/lib/esm/contracts/FabricContractRepository.d.ts +162 -0
- package/lib/esm/contracts/FabricContractRepository.js +302 -0
- package/lib/esm/contracts/FabricContractRepositoryObservableHandler.d.ts +62 -0
- package/lib/esm/contracts/FabricContractRepositoryObservableHandler.js +81 -0
- package/lib/esm/contracts/FabricContractSequence.d.ts +61 -0
- package/lib/esm/contracts/FabricContractSequence.js +135 -0
- package/lib/esm/contracts/FabricContractStatement.d.ts +34 -0
- package/lib/esm/contracts/FabricContractStatement.js +115 -0
- package/lib/esm/contracts/PrivateSequence.d.ts +15 -0
- package/lib/esm/contracts/PrivateSequence.js +33 -0
- package/lib/esm/contracts/crud/crud-contract.d.ts +168 -0
- package/lib/esm/contracts/crud/crud-contract.js +253 -0
- package/lib/esm/contracts/crud/index.d.ts +2 -0
- package/lib/esm/contracts/crud/index.js +3 -0
- package/lib/esm/contracts/crud/serialized-crud-contract.d.ts +37 -0
- package/lib/esm/contracts/crud/serialized-crud-contract.js +168 -0
- package/lib/esm/contracts/erc20/erc20contract.d.ts +151 -0
- package/lib/esm/contracts/erc20/erc20contract.js +565 -0
- package/lib/esm/contracts/erc20/index.d.ts +2 -0
- package/lib/esm/contracts/erc20/index.js +4 -0
- package/lib/esm/contracts/erc20/models.d.ts +114 -0
- package/lib/esm/contracts/erc20/models.js +206 -0
- package/lib/esm/contracts/index.d.ts +15 -0
- package/lib/esm/contracts/index.js +16 -0
- package/lib/esm/contracts/logging.d.ts +49 -0
- package/lib/esm/contracts/logging.js +92 -0
- package/lib/esm/contracts/private-data.d.ts +16 -0
- package/lib/esm/contracts/private-data.js +113 -0
- package/lib/esm/contracts/types.d.ts +26 -0
- package/lib/esm/contracts/types.js +2 -0
- package/lib/esm/index.d.ts +8 -0
- package/lib/esm/index.js +9 -0
- package/lib/esm/shared/ClientSerializer.d.ts +52 -0
- package/lib/esm/shared/ClientSerializer.js +80 -0
- package/lib/esm/shared/DeterministicSerializer.d.ts +40 -0
- package/lib/esm/shared/DeterministicSerializer.js +50 -0
- package/lib/esm/shared/SimpleDeterministicSerializer.d.ts +7 -0
- package/lib/esm/shared/SimpleDeterministicSerializer.js +42 -0
- package/lib/esm/shared/constants.d.ts +39 -0
- package/lib/esm/shared/constants.js +42 -0
- package/lib/esm/shared/crypto.d.ts +107 -0
- package/lib/esm/shared/crypto.js +331 -0
- package/lib/esm/shared/decorators.d.ts +24 -0
- package/lib/esm/shared/decorators.js +98 -0
- package/lib/esm/shared/erc20/erc20-constants.d.ts +25 -0
- package/lib/esm/shared/erc20/erc20-constants.js +27 -0
- package/lib/esm/shared/errors.d.ts +116 -0
- package/lib/esm/shared/errors.js +132 -0
- package/lib/esm/shared/events.d.ts +39 -0
- package/lib/esm/shared/events.js +47 -0
- package/lib/esm/shared/fabric-types.d.ts +33 -0
- package/lib/esm/shared/fabric-types.js +2 -0
- package/lib/esm/shared/index.d.ts +13 -0
- package/lib/esm/shared/index.js +14 -0
- package/lib/esm/shared/interfaces/Checkable.d.ts +21 -0
- package/lib/esm/shared/interfaces/Checkable.js +2 -0
- package/lib/esm/shared/math.d.ts +34 -0
- package/lib/esm/shared/math.js +61 -0
- package/lib/esm/shared/model/Identity.d.ts +42 -0
- package/lib/esm/shared/model/Identity.js +78 -0
- package/lib/esm/shared/model/IdentityCredentials.d.ts +41 -0
- package/lib/esm/shared/model/IdentityCredentials.js +74 -0
- package/lib/esm/shared/model/index.d.ts +1 -0
- package/lib/esm/shared/model/index.js +2 -0
- package/lib/esm/shared/model/utils.d.ts +60 -0
- package/lib/esm/shared/model/utils.js +108 -0
- package/lib/esm/shared/types.d.ts +79 -0
- package/lib/esm/shared/types.js +2 -0
- package/lib/esm/shared/utils.d.ts +55 -0
- package/lib/esm/shared/utils.js +148 -0
- package/lib/index.cjs +25 -0
- package/lib/index.d.ts +8 -0
- package/lib/shared/ClientSerializer.cjs +84 -0
- package/lib/shared/ClientSerializer.d.ts +52 -0
- package/lib/shared/DeterministicSerializer.cjs +54 -0
- package/lib/shared/DeterministicSerializer.d.ts +40 -0
- package/lib/shared/SimpleDeterministicSerializer.cjs +46 -0
- package/lib/shared/SimpleDeterministicSerializer.d.ts +7 -0
- package/lib/shared/constants.cjs +45 -0
- package/lib/shared/constants.d.ts +39 -0
- package/lib/shared/crypto.cjs +369 -0
- package/lib/shared/crypto.d.ts +107 -0
- package/lib/shared/decorators.cjs +105 -0
- package/lib/shared/decorators.d.ts +24 -0
- package/lib/shared/erc20/erc20-constants.cjs +30 -0
- package/lib/shared/erc20/erc20-constants.d.ts +25 -0
- package/lib/shared/errors.cjs +142 -0
- package/lib/shared/errors.d.ts +116 -0
- package/lib/shared/events.cjs +51 -0
- package/lib/shared/events.d.ts +39 -0
- package/lib/shared/fabric-types.cjs +4 -0
- package/lib/shared/fabric-types.d.ts +33 -0
- package/lib/shared/index.cjs +30 -0
- package/lib/shared/index.d.ts +13 -0
- package/lib/shared/interfaces/Checkable.cjs +3 -0
- package/lib/shared/interfaces/Checkable.d.ts +21 -0
- package/lib/shared/math.cjs +66 -0
- package/lib/shared/math.d.ts +34 -0
- package/lib/shared/model/Identity.cjs +81 -0
- package/lib/shared/model/Identity.d.ts +42 -0
- package/lib/shared/model/IdentityCredentials.cjs +77 -0
- package/lib/shared/model/IdentityCredentials.d.ts +41 -0
- package/lib/shared/model/index.cjs +18 -0
- package/lib/shared/model/index.d.ts +1 -0
- package/lib/shared/model/utils.cjs +114 -0
- package/lib/shared/model/utils.d.ts +60 -0
- package/lib/shared/types.cjs +3 -0
- package/lib/shared/types.d.ts +79 -0
- package/lib/shared/utils.cjs +185 -0
- package/lib/shared/utils.d.ts +55 -0
- 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
|
+
}
|