@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,724 @@
|
|
|
1
|
+
import { CouchDBAdapter, CouchDBKeys } from "@decaf-ts/for-couchdb";
|
|
2
|
+
import { Decoration, list, Model, prop, propMetadata, required, type, } from "@decaf-ts/decorator-validation";
|
|
3
|
+
import { FabricContractContext } from "./ContractContext.js";
|
|
4
|
+
import { afterAny, DBKeys, InternalError, modelToTransient, NotFoundError, onCreate, onCreateUpdate, onDelete, onUpdate, readonly, SerializationError, } from "@decaf-ts/db-decorators";
|
|
5
|
+
import { Object as FabricObject, Property as FabricProperty, } from "fabric-contract-api";
|
|
6
|
+
import { Logging } from "@decaf-ts/logging";
|
|
7
|
+
import { OrderDirection, PersistenceKeys, Repository, sequenceNameForModel, UnsupportedError, index, NumericSequence, Adapter, oneToManyOnUpdate, } from "@decaf-ts/core";
|
|
8
|
+
import { FabricContractRepository } from "./FabricContractRepository.js";
|
|
9
|
+
import { FabricStatement } from "./FabricContractStatement.js";
|
|
10
|
+
import { FabricContractSequence } from "./FabricContractSequence.js";
|
|
11
|
+
import { MissingContextError } from "./../shared/errors.js";
|
|
12
|
+
import { FabricFlavour } from "./../shared/constants.js";
|
|
13
|
+
import { SimpleDeterministicSerializer } from "./../shared/SimpleDeterministicSerializer.js";
|
|
14
|
+
import { oneToManyOnCreate, oneToManyOnDelete, oneToOneOnCreate, oneToOneOnDelete, oneToOneOnUpdate, populate as pop, } from "./FabricConstruction.js";
|
|
15
|
+
import { apply } from "@decaf-ts/reflection";
|
|
16
|
+
/**
|
|
17
|
+
* @description Sets the creator or updater field in a model based on the user in the context
|
|
18
|
+
* @summary Callback function used in decorators to automatically set the created_by or updated_by fields
|
|
19
|
+
* with the username from the context when a document is created or updated
|
|
20
|
+
* @template M - Type extending Model
|
|
21
|
+
* @template R - Type extending NanoRepository<M>
|
|
22
|
+
* @template V - Type extending RelationsMetadata
|
|
23
|
+
* @param {R} this - The repository instance
|
|
24
|
+
* @param {FabricContractContext} context - The operation context containing user information
|
|
25
|
+
* @param {V} data - The relation metadata
|
|
26
|
+
* @param {string} key - The property key to set with the username
|
|
27
|
+
* @param {M} model - The model instance being created or updated
|
|
28
|
+
* @return {Promise<void>} A promise that resolves when the operation is complete
|
|
29
|
+
* @function createdByOnFabricCreateUpdate
|
|
30
|
+
* @memberOf module:fabric.contracts
|
|
31
|
+
* @mermaid
|
|
32
|
+
* sequenceDiagram
|
|
33
|
+
* participant F as createdByOnNanoCreateUpdate
|
|
34
|
+
* participant C as Context
|
|
35
|
+
* participant M as Model
|
|
36
|
+
* F->>C: get("user")
|
|
37
|
+
* C-->>F: user object
|
|
38
|
+
* F->>M: set key to user.name
|
|
39
|
+
* Note over F: If no user in context
|
|
40
|
+
* F-->>F: throw UnsupportedError
|
|
41
|
+
*/
|
|
42
|
+
export async function createdByOnFabricCreateUpdate(context, data, key, model) {
|
|
43
|
+
try {
|
|
44
|
+
const user = context.get("clientIdentity");
|
|
45
|
+
model[key] = user.getID();
|
|
46
|
+
// eslint-disable-next-line @typescript-eslint/no-unused-vars
|
|
47
|
+
}
|
|
48
|
+
catch (e) {
|
|
49
|
+
throw new UnsupportedError("No User found in context. Please provide a user in the context");
|
|
50
|
+
}
|
|
51
|
+
}
|
|
52
|
+
/**
|
|
53
|
+
* @description Primary key auto-assignment callback for Fabric models
|
|
54
|
+
* @summary Generates and assigns a primary key value to the specified model property using a Fabric-backed sequence when the model is created. If the sequence name is not provided in options, it is derived from the model via sequenceNameForModel. The assigned key is defined as non-writable and enumerable.
|
|
55
|
+
* @template M - Type extending Model for the target instance
|
|
56
|
+
* @template R - Type extending FabricContractRepository for repository context
|
|
57
|
+
* @template V - Type extending SequenceOptions to configure sequence behavior
|
|
58
|
+
* @template F - Type extending FabricContractFlags for contextual flags
|
|
59
|
+
* @param {R} this - The repository instance invoking the callback
|
|
60
|
+
* @param {FabricContractContext} context - Fabric contract context containing invocation metadata
|
|
61
|
+
* @param {V} data - Sequence options used to configure or locate the sequence
|
|
62
|
+
* @param {string} key - The primary key property name to assign on the model
|
|
63
|
+
* @param {M} model - The model instance to receive the generated primary key
|
|
64
|
+
* @return {Promise<void>} Resolves when the key is assigned or when no action is required
|
|
65
|
+
* @function pkFabricOnCreate
|
|
66
|
+
* @memberOf module:for-fabric.contracts
|
|
67
|
+
* @mermaid
|
|
68
|
+
* sequenceDiagram
|
|
69
|
+
* participant R as Repository
|
|
70
|
+
* participant C as Context<F>
|
|
71
|
+
* participant S as FabricContractDBSequence
|
|
72
|
+
* participant M as Model
|
|
73
|
+
* R->>R: derive sequence name if missing
|
|
74
|
+
* R->>S: adapter.Sequence(options)
|
|
75
|
+
* S-->>R: sequence instance
|
|
76
|
+
* R->>S: next(context)
|
|
77
|
+
* S-->>R: next value
|
|
78
|
+
* R->>M: define non-writable primary key
|
|
79
|
+
*/
|
|
80
|
+
export async function pkFabricOnCreate(context, data, key, model) {
|
|
81
|
+
if (!data.type || model[key]) {
|
|
82
|
+
return;
|
|
83
|
+
}
|
|
84
|
+
const setPrimaryKeyValue = function (target, propertyKey, value) {
|
|
85
|
+
Object.defineProperty(target, propertyKey, {
|
|
86
|
+
enumerable: true,
|
|
87
|
+
writable: false,
|
|
88
|
+
configurable: true,
|
|
89
|
+
value: value,
|
|
90
|
+
});
|
|
91
|
+
};
|
|
92
|
+
if (!data.name)
|
|
93
|
+
data.name = sequenceNameForModel(model, "pk");
|
|
94
|
+
let sequence;
|
|
95
|
+
try {
|
|
96
|
+
sequence = (await this.adapter.Sequence(data));
|
|
97
|
+
}
|
|
98
|
+
catch (e) {
|
|
99
|
+
throw new InternalError(`Failed to instantiate Sequence ${data.name}: ${e}`);
|
|
100
|
+
}
|
|
101
|
+
const next = await sequence.next(context);
|
|
102
|
+
setPrimaryKeyValue(model, key, next);
|
|
103
|
+
}
|
|
104
|
+
/**
|
|
105
|
+
* @description Adapter for Hyperledger Fabric chaincode state database operations
|
|
106
|
+
* @summary Provides a CouchDB-like interface for interacting with the Fabric state database from within a chaincode contract
|
|
107
|
+
* @template void - No configuration needed for contract adapter
|
|
108
|
+
* @template FabricContractFlags - Flags specific to Fabric contract operations
|
|
109
|
+
* @template FabricContractContext - Context type for Fabric contract operations
|
|
110
|
+
* @class FabricContractAdapter
|
|
111
|
+
* @example
|
|
112
|
+
* ```typescript
|
|
113
|
+
* // In a Fabric chaincode contract class
|
|
114
|
+
* import { FabricContractAdapter } from '@decaf-ts/for-fabric';
|
|
115
|
+
*
|
|
116
|
+
* export class MyContract extends Contract {
|
|
117
|
+
* private adapter = new FabricContractAdapter();
|
|
118
|
+
*
|
|
119
|
+
* @Transaction()
|
|
120
|
+
* async createAsset(ctx: Context, id: string, data: string): Promise<void> {
|
|
121
|
+
* const model = { id, data, timestamp: Date.now() };
|
|
122
|
+
* await this.adapter.create('assets', id, model, {}, { stub: ctx.stub });
|
|
123
|
+
* }
|
|
124
|
+
* }
|
|
125
|
+
* ```
|
|
126
|
+
* @mermaid
|
|
127
|
+
* sequenceDiagram
|
|
128
|
+
* participant Contract
|
|
129
|
+
* participant FabricContractAdapter
|
|
130
|
+
* participant Stub
|
|
131
|
+
* participant StateDB
|
|
132
|
+
*
|
|
133
|
+
* Contract->>FabricContractAdapter: create(tableName, id, model, transient, ctx)
|
|
134
|
+
* FabricContractAdapter->>FabricContractAdapter: Serialize model to JSON
|
|
135
|
+
* FabricContractAdapter->>Stub: putState(id, serializedData)
|
|
136
|
+
* Stub->>StateDB: Write data
|
|
137
|
+
* StateDB-->>Stub: Success
|
|
138
|
+
* Stub-->>FabricContractAdapter: Success
|
|
139
|
+
* FabricContractAdapter-->>Contract: model
|
|
140
|
+
*/
|
|
141
|
+
export class FabricContractAdapter extends CouchDBAdapter {
|
|
142
|
+
getClient() {
|
|
143
|
+
throw new UnsupportedError("Client is not supported in Fabric contracts");
|
|
144
|
+
}
|
|
145
|
+
/**
|
|
146
|
+
* @description Text decoder for converting binary data to strings
|
|
147
|
+
*/
|
|
148
|
+
static { this.textDecoder = new TextDecoder("utf8"); }
|
|
149
|
+
static { this.serializer = new SimpleDeterministicSerializer(); }
|
|
150
|
+
/**
|
|
151
|
+
* @description Creates a logger for a specific chaincode context
|
|
152
|
+
* @summary Returns a ContractLogger instance configured for the current context
|
|
153
|
+
* @param {Ctx} ctx - The Fabric chaincode context
|
|
154
|
+
* @return {ContractLogger} The logger instance
|
|
155
|
+
*/
|
|
156
|
+
logFor(ctx) {
|
|
157
|
+
return Logging.for(FabricContractAdapter, {}, ctx);
|
|
158
|
+
}
|
|
159
|
+
/**
|
|
160
|
+
* @description Gets the repository constructor for this adapter
|
|
161
|
+
* @summary Returns the FabricContractRepository constructor for creating repositories
|
|
162
|
+
* @template M - Type extending Model
|
|
163
|
+
* @return {Constructor<Repository<M, MangoQuery, FabricContractAdapter, FabricContractFlags, FabricContractContext>>} The repository constructor
|
|
164
|
+
*/
|
|
165
|
+
repository() {
|
|
166
|
+
return FabricContractRepository;
|
|
167
|
+
}
|
|
168
|
+
/**
|
|
169
|
+
* @description Creates a new FabricContractAdapter instance
|
|
170
|
+
* @summary Initializes an adapter for interacting with the Fabric state database
|
|
171
|
+
* @param {void} scope - Not used in this adapter
|
|
172
|
+
* @param {string} [alias] - Optional alias for the adapter instance
|
|
173
|
+
*/
|
|
174
|
+
constructor(scope, alias) {
|
|
175
|
+
super(scope, FabricFlavour, alias);
|
|
176
|
+
/**
|
|
177
|
+
* @description Context constructor for this adapter
|
|
178
|
+
* @summary Overrides the base Context constructor with FabricContractContext
|
|
179
|
+
*/
|
|
180
|
+
this.Context = FabricContractContext;
|
|
181
|
+
}
|
|
182
|
+
for(config, ...args) {
|
|
183
|
+
return super.for(config, ...args);
|
|
184
|
+
}
|
|
185
|
+
/**
|
|
186
|
+
* @description Creates a record in the state database
|
|
187
|
+
* @summary Serializes a model and stores it in the Fabric state database
|
|
188
|
+
* @param {string} tableName - The name of the table/collection
|
|
189
|
+
* @param {string | number} id - The record identifier
|
|
190
|
+
* @param {Record<string, any>} model - The record data
|
|
191
|
+
* @param {Record<string, any>} transient - Transient data (not used in this implementation)
|
|
192
|
+
* @param {...any[]} args - Additional arguments, including the chaincode stub and logger
|
|
193
|
+
* @return {Promise<Record<string, any>>} Promise resolving to the created record
|
|
194
|
+
*/
|
|
195
|
+
async create(tableName, id, model, ...args) {
|
|
196
|
+
const { stub, logger } = args.pop();
|
|
197
|
+
const log = logger.for(this.create);
|
|
198
|
+
try {
|
|
199
|
+
log.info(`adding entry to ${tableName} table with pk ${id}`);
|
|
200
|
+
model = await this.putState(stub, id.toString(), model);
|
|
201
|
+
}
|
|
202
|
+
catch (e) {
|
|
203
|
+
throw this.parseError(e);
|
|
204
|
+
}
|
|
205
|
+
return model;
|
|
206
|
+
}
|
|
207
|
+
/**
|
|
208
|
+
* @description Reads a record from the state database
|
|
209
|
+
* @summary Retrieves and deserializes a record from the Fabric state database
|
|
210
|
+
* @param {string} tableName - The name of the table/collection
|
|
211
|
+
* @param {string | number} id - The record identifier
|
|
212
|
+
* @param {...any[]} args - Additional arguments, including the chaincode stub and logger
|
|
213
|
+
* @return {Promise<Record<string, any>>} Promise resolving to the retrieved record
|
|
214
|
+
*/
|
|
215
|
+
async read(tableName, id, ...args) {
|
|
216
|
+
const { stub, logger } = args.pop();
|
|
217
|
+
const log = logger.for(this.read);
|
|
218
|
+
let model;
|
|
219
|
+
try {
|
|
220
|
+
const results = await this.readState(stub, tableName, id.toString());
|
|
221
|
+
if (results.length < 1) {
|
|
222
|
+
log.debug(`No record found for id ${id} in ${tableName} table`);
|
|
223
|
+
throw new NotFoundError(`No record found for id ${id} in ${tableName} table`);
|
|
224
|
+
}
|
|
225
|
+
else if (results.length < 2) {
|
|
226
|
+
log.debug(`No record found for id ${id} in ${tableName} table`);
|
|
227
|
+
model = results.pop();
|
|
228
|
+
}
|
|
229
|
+
else {
|
|
230
|
+
model = this.mergeModels(results);
|
|
231
|
+
}
|
|
232
|
+
}
|
|
233
|
+
catch (e) {
|
|
234
|
+
throw this.parseError(e);
|
|
235
|
+
}
|
|
236
|
+
return model;
|
|
237
|
+
}
|
|
238
|
+
/**
|
|
239
|
+
* @description Updates a record in the state database
|
|
240
|
+
* @summary Serializes a model and updates it in the Fabric state database
|
|
241
|
+
* @param {string} tableName - The name of the table/collection
|
|
242
|
+
* @param {string | number} id - The record identifier
|
|
243
|
+
* @param {Record<string, any>} model - The updated record data
|
|
244
|
+
* @param {Record<string, any>} transient - Transient data (not used in this implementation)
|
|
245
|
+
* @param {...any[]} args - Additional arguments, including the chaincode stub and logger
|
|
246
|
+
* @return {Promise<Record<string, any>>} Promise resolving to the updated record
|
|
247
|
+
*/
|
|
248
|
+
async update(tableName, id, model, ...args) {
|
|
249
|
+
const { stub, logger } = args.pop();
|
|
250
|
+
const log = logger.for(this.update);
|
|
251
|
+
try {
|
|
252
|
+
log.info(`updating entry to ${tableName} table with pk ${id}`);
|
|
253
|
+
model = await this.putState(stub, id.toString(), model);
|
|
254
|
+
}
|
|
255
|
+
catch (e) {
|
|
256
|
+
throw this.parseError(e);
|
|
257
|
+
}
|
|
258
|
+
return model;
|
|
259
|
+
}
|
|
260
|
+
/**
|
|
261
|
+
* @description Deletes a record from the state database
|
|
262
|
+
* @summary Retrieves a record and then removes it from the Fabric state database
|
|
263
|
+
* @param {string} tableName - The name of the table/collection
|
|
264
|
+
* @param {string | number} id - The record identifier to delete
|
|
265
|
+
* @param {...any[]} args - Additional arguments, including the chaincode stub and logger
|
|
266
|
+
* @return {Promise<Record<string, any>>} Promise resolving to the deleted record
|
|
267
|
+
*/
|
|
268
|
+
async delete(tableName, id, ...args) {
|
|
269
|
+
const ctx = args.pop();
|
|
270
|
+
const { stub, logger } = ctx;
|
|
271
|
+
const log = logger.for(this.delete);
|
|
272
|
+
args.push(ctx);
|
|
273
|
+
let model;
|
|
274
|
+
try {
|
|
275
|
+
model = await this.read(tableName, id, ...args);
|
|
276
|
+
log.verbose(`deleting entry with pk ${id} from ${tableName} table`);
|
|
277
|
+
this.deleteState(stub, tableName, id.toString());
|
|
278
|
+
}
|
|
279
|
+
catch (e) {
|
|
280
|
+
throw this.parseError(e);
|
|
281
|
+
}
|
|
282
|
+
return model;
|
|
283
|
+
}
|
|
284
|
+
async deleteState(stub, tableName, id,
|
|
285
|
+
// eslint-disable-next-line @typescript-eslint/no-unused-vars
|
|
286
|
+
...args) {
|
|
287
|
+
const composedKey = stub.createCompositeKey(tableName, [String(id)]);
|
|
288
|
+
await stub.deleteState(composedKey);
|
|
289
|
+
}
|
|
290
|
+
async putState(stub, id, model,
|
|
291
|
+
// eslint-disable-next-line @typescript-eslint/no-unused-vars
|
|
292
|
+
...args) {
|
|
293
|
+
let data;
|
|
294
|
+
try {
|
|
295
|
+
data = Buffer.from(FabricContractAdapter.serializer.serialize(model));
|
|
296
|
+
}
|
|
297
|
+
catch (e) {
|
|
298
|
+
throw new SerializationError(`Failed to serialize record with id ${id}: ${e}`);
|
|
299
|
+
}
|
|
300
|
+
await stub.putState(id.toString(), data);
|
|
301
|
+
return model;
|
|
302
|
+
}
|
|
303
|
+
async readState(stub, tableName, id,
|
|
304
|
+
// eslint-disable-next-line @typescript-eslint/no-unused-vars
|
|
305
|
+
...args) {
|
|
306
|
+
const composedKey = stub.createCompositeKey(tableName, [String(id)]);
|
|
307
|
+
const results = [];
|
|
308
|
+
let res = await stub.getState(composedKey);
|
|
309
|
+
if (res.toString() === "")
|
|
310
|
+
throw new NotFoundError(`Record with id ${id} not found`);
|
|
311
|
+
try {
|
|
312
|
+
res = FabricContractAdapter.serializer.deserialize(res.toString()
|
|
313
|
+
// model.constructor.name
|
|
314
|
+
);
|
|
315
|
+
}
|
|
316
|
+
catch (e) {
|
|
317
|
+
throw new SerializationError(`Failed to parse record: ${e}`);
|
|
318
|
+
}
|
|
319
|
+
results.push(res);
|
|
320
|
+
return results;
|
|
321
|
+
}
|
|
322
|
+
async queryResult(stub, rawInput,
|
|
323
|
+
// eslint-disable-next-line @typescript-eslint/no-unused-vars
|
|
324
|
+
...args) {
|
|
325
|
+
return (await stub.getQueryResult(JSON.stringify(rawInput)));
|
|
326
|
+
}
|
|
327
|
+
async queryResultPaginated(stub, rawInput, limit = 250, skip,
|
|
328
|
+
// eslint-disable-next-line @typescript-eslint/no-unused-vars
|
|
329
|
+
...args) {
|
|
330
|
+
return (await stub.getQueryResultWithPagination(JSON.stringify(rawInput), limit, skip?.toString()));
|
|
331
|
+
}
|
|
332
|
+
mergeModels(results) {
|
|
333
|
+
const extract = (model) => Object.entries(model).reduce((accum, [key, val]) => {
|
|
334
|
+
if (typeof val !== "undefined")
|
|
335
|
+
accum[key] = val;
|
|
336
|
+
return accum;
|
|
337
|
+
}, {});
|
|
338
|
+
let finalModel = results.pop();
|
|
339
|
+
for (const res of results) {
|
|
340
|
+
finalModel = Object.assign({}, extract(finalModel), extract(res));
|
|
341
|
+
}
|
|
342
|
+
return finalModel;
|
|
343
|
+
}
|
|
344
|
+
/**
|
|
345
|
+
* @description Decodes binary data to string
|
|
346
|
+
* @summary Converts a Uint8Array to a string using UTF-8 encoding
|
|
347
|
+
* @param {Uint8Array} buffer - The binary data to decode
|
|
348
|
+
* @return {string} The decoded string
|
|
349
|
+
*/
|
|
350
|
+
decode(buffer) {
|
|
351
|
+
return FabricContractAdapter.textDecoder.decode(buffer);
|
|
352
|
+
}
|
|
353
|
+
/**
|
|
354
|
+
* @description Creates operation flags for Fabric contract operations
|
|
355
|
+
* @summary Merges default flags with Fabric-specific context information
|
|
356
|
+
* @template M - Type extending Model
|
|
357
|
+
* @param {OperationKeys} operation - The operation being performed
|
|
358
|
+
* @param {Constructor<M>} model - The model constructor
|
|
359
|
+
* @param {Partial<FabricContractFlags>} flags - Partial flags to merge with defaults
|
|
360
|
+
* @param {Ctx} ctx - The Fabric chaincode context
|
|
361
|
+
* @return {FabricContractFlags} The merged flags
|
|
362
|
+
*/
|
|
363
|
+
async flags(operation, model, flags, ctx, ...args) {
|
|
364
|
+
return Object.assign(await super.flags(operation, model, flags, ...args), {
|
|
365
|
+
stub: ctx.stub,
|
|
366
|
+
identity: ctx.clientIdentity,
|
|
367
|
+
logger: this.logFor(ctx),
|
|
368
|
+
});
|
|
369
|
+
}
|
|
370
|
+
/**
|
|
371
|
+
* @description Creates an index for a model
|
|
372
|
+
* @summary This method is not implemented for Fabric contracts and returns a resolved promise
|
|
373
|
+
* @template M - Type extending Model
|
|
374
|
+
* @param {Constructor<M>} models - The model constructor
|
|
375
|
+
* @return {Promise<void>} Promise that resolves immediately
|
|
376
|
+
*/
|
|
377
|
+
// eslint-disable-next-line @typescript-eslint/no-unused-vars
|
|
378
|
+
index(models) {
|
|
379
|
+
return Promise.resolve(undefined);
|
|
380
|
+
}
|
|
381
|
+
/**
|
|
382
|
+
* @description Processes results from a state query iterator
|
|
383
|
+
* @summary Iterates through query results and converts them to a structured format
|
|
384
|
+
* @param {Logger} log - Logger instance for debugging
|
|
385
|
+
* @param {Iterators.StateQueryIterator} iterator - The state query iterator
|
|
386
|
+
* @param {boolean} [isHistory=false] - Whether this is a history query
|
|
387
|
+
* @return {Promise<any[]>} Promise resolving to an array of processed results
|
|
388
|
+
* @mermaid
|
|
389
|
+
* sequenceDiagram
|
|
390
|
+
* participant Caller
|
|
391
|
+
* participant ResultIterator
|
|
392
|
+
* participant Iterator
|
|
393
|
+
*
|
|
394
|
+
* Caller->>ResultIterator: resultIterator(log, iterator, isHistory)
|
|
395
|
+
* loop Until done
|
|
396
|
+
* ResultIterator->>Iterator: next()
|
|
397
|
+
* Iterator-->>ResultIterator: { value, done }
|
|
398
|
+
* alt Has value
|
|
399
|
+
* ResultIterator->>ResultIterator: Process value based on isHistory
|
|
400
|
+
* ResultIterator->>ResultIterator: Add to results array
|
|
401
|
+
* end
|
|
402
|
+
* end
|
|
403
|
+
* ResultIterator->>Iterator: close()
|
|
404
|
+
* ResultIterator-->>Caller: allResults
|
|
405
|
+
*/
|
|
406
|
+
async resultIterator(log, iterator, isHistory = false) {
|
|
407
|
+
const allResults = [];
|
|
408
|
+
let res = await iterator.next();
|
|
409
|
+
while (!res.done) {
|
|
410
|
+
if (res.value && res.value.value.toString()) {
|
|
411
|
+
let jsonRes = {};
|
|
412
|
+
log.debug(res.value.value.toString("utf8"));
|
|
413
|
+
if (isHistory /* && isHistory === true*/) {
|
|
414
|
+
jsonRes.TxId = res.value.txId;
|
|
415
|
+
jsonRes.Timestamp = res.value.timestamp;
|
|
416
|
+
try {
|
|
417
|
+
jsonRes.Value = JSON.parse(res.value.value.toString("utf8"));
|
|
418
|
+
}
|
|
419
|
+
catch (err) {
|
|
420
|
+
log.error(err);
|
|
421
|
+
jsonRes.Value = res.value.value.toString("utf8");
|
|
422
|
+
}
|
|
423
|
+
}
|
|
424
|
+
else {
|
|
425
|
+
try {
|
|
426
|
+
jsonRes = JSON.parse(res.value.value.toString("utf8"));
|
|
427
|
+
}
|
|
428
|
+
catch (err) {
|
|
429
|
+
log.error(err);
|
|
430
|
+
jsonRes = res.value.value.toString("utf8");
|
|
431
|
+
}
|
|
432
|
+
}
|
|
433
|
+
allResults.push(jsonRes);
|
|
434
|
+
}
|
|
435
|
+
res = await iterator.next();
|
|
436
|
+
}
|
|
437
|
+
log.debug(`Closing iterator after ${allResults.length} results`);
|
|
438
|
+
iterator.close(); // purposely not await. let iterator close on its own
|
|
439
|
+
return allResults;
|
|
440
|
+
}
|
|
441
|
+
/**
|
|
442
|
+
* @description Executes a raw query against the state database
|
|
443
|
+
* @summary Performs a rich query using CouchDB syntax against the Fabric state database
|
|
444
|
+
* @template R - The return type
|
|
445
|
+
* @param {MangoQuery} rawInput - The Mango Query to execute
|
|
446
|
+
* @param {boolean} docsOnly - Whether to return only documents (not used in this implementation)
|
|
447
|
+
* @param {...any[]} args - Additional arguments, including the chaincode stub and logger
|
|
448
|
+
* @return {Promise<R>} Promise resolving to the query results
|
|
449
|
+
* @mermaid
|
|
450
|
+
* sequenceDiagram
|
|
451
|
+
* participant Caller
|
|
452
|
+
* participant FabricContractAdapter
|
|
453
|
+
* participant Stub
|
|
454
|
+
* participant StateDB
|
|
455
|
+
*
|
|
456
|
+
* Caller->>FabricContractAdapter: raw(rawInput, docsOnly, ctx)
|
|
457
|
+
* FabricContractAdapter->>FabricContractAdapter: Extract limit and skip
|
|
458
|
+
* alt With pagination
|
|
459
|
+
* FabricContractAdapter->>Stub: getQueryResultWithPagination(query, limit, skip)
|
|
460
|
+
* else Without pagination
|
|
461
|
+
* FabricContractAdapter->>Stub: getQueryResult(query)
|
|
462
|
+
* end
|
|
463
|
+
* Stub->>StateDB: Execute query
|
|
464
|
+
* StateDB-->>Stub: Iterator
|
|
465
|
+
* Stub-->>FabricContractAdapter: Iterator
|
|
466
|
+
* FabricContractAdapter->>FabricContractAdapter: resultIterator(log, iterator)
|
|
467
|
+
* FabricContractAdapter-->>Caller: results
|
|
468
|
+
*/
|
|
469
|
+
async raw(rawInput, docsOnly, ...args) {
|
|
470
|
+
const { stub, logger } = args.pop();
|
|
471
|
+
const log = logger.for(this.raw);
|
|
472
|
+
const { skip, limit } = rawInput;
|
|
473
|
+
let iterator;
|
|
474
|
+
if (limit || skip) {
|
|
475
|
+
delete rawInput["limit"];
|
|
476
|
+
delete rawInput["skip"];
|
|
477
|
+
log.debug(`Retrieving paginated iterator: limit: ${limit}/ skip: ${skip}`);
|
|
478
|
+
const response = (await this.queryResultPaginated(stub, rawInput, limit || 250, skip?.toString()));
|
|
479
|
+
iterator = response.iterator;
|
|
480
|
+
}
|
|
481
|
+
else {
|
|
482
|
+
log.debug("Retrieving iterator");
|
|
483
|
+
iterator = (await this.queryResult(stub, rawInput));
|
|
484
|
+
}
|
|
485
|
+
log.debug("Iterator acquired");
|
|
486
|
+
const results = (await this.resultIterator(log, iterator));
|
|
487
|
+
log.debug(`returning {0} results`, `${Array.isArray(results) ? results.length : 1}`);
|
|
488
|
+
return results;
|
|
489
|
+
}
|
|
490
|
+
Statement(ctx) {
|
|
491
|
+
if (!ctx) {
|
|
492
|
+
throw new MissingContextError("Context is required");
|
|
493
|
+
}
|
|
494
|
+
return new FabricStatement(this, ctx);
|
|
495
|
+
}
|
|
496
|
+
async Sequence(options) {
|
|
497
|
+
return new FabricContractSequence(options, this);
|
|
498
|
+
}
|
|
499
|
+
async createAll(tableName, id, model, ...args) {
|
|
500
|
+
if (id.length !== model.length)
|
|
501
|
+
throw new InternalError("Ids and models must have the same length");
|
|
502
|
+
const { logger } = args[args.length - 1];
|
|
503
|
+
const log = logger.for(this.createAll);
|
|
504
|
+
log.info(`Creating ${id.length} entries ${tableName} table`);
|
|
505
|
+
log.debug(`pks: ${id}`);
|
|
506
|
+
return Promise.all(id.map(async (i, index) => {
|
|
507
|
+
return this.create(tableName, i, model[index], ...args);
|
|
508
|
+
}));
|
|
509
|
+
}
|
|
510
|
+
async updateAll(tableName, id, model, ...args) {
|
|
511
|
+
if (id.length !== model.length)
|
|
512
|
+
throw new InternalError("Ids and models must have the same length");
|
|
513
|
+
const { logger } = args[args.length - 1];
|
|
514
|
+
const log = logger.for(this.createAll);
|
|
515
|
+
log.info(`Updating ${id.length} entries ${tableName} table`);
|
|
516
|
+
log.debug(`pks: ${id}`);
|
|
517
|
+
return Promise.all(id.map(async (i, index) => {
|
|
518
|
+
return this.update(tableName, i, model[index], ...args);
|
|
519
|
+
}));
|
|
520
|
+
}
|
|
521
|
+
/**
|
|
522
|
+
*
|
|
523
|
+
* @param model
|
|
524
|
+
* @param {string} pk
|
|
525
|
+
* @param args
|
|
526
|
+
*/
|
|
527
|
+
prepare(model, pk, ...args) {
|
|
528
|
+
const { stub, logger } = args.pop();
|
|
529
|
+
const tableName = args.shift();
|
|
530
|
+
const log = logger.for(this.prepare);
|
|
531
|
+
const split = modelToTransient(model);
|
|
532
|
+
const result = Object.entries(split.model).reduce((accum, [key, val]) => {
|
|
533
|
+
if (typeof val === "undefined")
|
|
534
|
+
return accum;
|
|
535
|
+
const mappedProp = Repository.column(model, key);
|
|
536
|
+
if (this.isReserved(mappedProp))
|
|
537
|
+
throw new InternalError(`Property name ${mappedProp} is reserved`);
|
|
538
|
+
accum[mappedProp] = val;
|
|
539
|
+
return accum;
|
|
540
|
+
}, {});
|
|
541
|
+
if (model[PersistenceKeys.METADATA]) {
|
|
542
|
+
log.silly(`Passing along persistence metadata for ${model[PersistenceKeys.METADATA]}`);
|
|
543
|
+
Object.defineProperty(result, PersistenceKeys.METADATA, {
|
|
544
|
+
enumerable: false,
|
|
545
|
+
writable: false,
|
|
546
|
+
configurable: true,
|
|
547
|
+
value: model[PersistenceKeys.METADATA],
|
|
548
|
+
});
|
|
549
|
+
}
|
|
550
|
+
log.info(`Preparing record for ${tableName} table with pk ${model[pk]}`);
|
|
551
|
+
return {
|
|
552
|
+
record: result,
|
|
553
|
+
id: stub.createCompositeKey(tableName, [String(model[pk])]),
|
|
554
|
+
transient: split.transient,
|
|
555
|
+
};
|
|
556
|
+
}
|
|
557
|
+
revert(obj, clazz, pk, id, transient) {
|
|
558
|
+
const log = this.log.for(this.revert);
|
|
559
|
+
const ob = {};
|
|
560
|
+
ob[pk] = id;
|
|
561
|
+
const m = (typeof clazz === "string" ? Model.build(ob, clazz) : new clazz(ob));
|
|
562
|
+
log.silly(`Rebuilding model ${m.constructor.name} id ${id}`);
|
|
563
|
+
const metadata = obj[PersistenceKeys.METADATA];
|
|
564
|
+
const result = Object.keys(m).reduce((accum, key) => {
|
|
565
|
+
accum[key] = obj[Repository.column(accum, key)];
|
|
566
|
+
return accum;
|
|
567
|
+
}, m);
|
|
568
|
+
if (transient) {
|
|
569
|
+
log.verbose(`re-adding transient properties: ${Object.keys(transient).join(", ")}`);
|
|
570
|
+
Object.entries(transient).forEach(([key, val]) => {
|
|
571
|
+
if (key in result && result[key] !== undefined)
|
|
572
|
+
throw new InternalError(`Transient property ${key} already exists on model ${m.constructor.name}. should be impossible`);
|
|
573
|
+
result[key] = val;
|
|
574
|
+
});
|
|
575
|
+
}
|
|
576
|
+
if (metadata) {
|
|
577
|
+
log.silly(`Passing along ${this.flavour} persistence metadata for ${m.constructor.name} id ${id}: ${metadata}`);
|
|
578
|
+
Object.defineProperty(result, PersistenceKeys.METADATA, {
|
|
579
|
+
enumerable: false,
|
|
580
|
+
configurable: false,
|
|
581
|
+
writable: false,
|
|
582
|
+
value: metadata,
|
|
583
|
+
});
|
|
584
|
+
}
|
|
585
|
+
return result;
|
|
586
|
+
}
|
|
587
|
+
createPrefix(tableName, id, model, ...args) {
|
|
588
|
+
const ctx = args.pop();
|
|
589
|
+
const record = {};
|
|
590
|
+
record[CouchDBKeys.TABLE] = tableName;
|
|
591
|
+
// record[CouchDBKeys.ID] = this.generateId(tableName, id);
|
|
592
|
+
Object.assign(record, model);
|
|
593
|
+
return [tableName, id, record, ctx];
|
|
594
|
+
}
|
|
595
|
+
updatePrefix(tableName, id, model, ...args) {
|
|
596
|
+
const ctx = args.pop();
|
|
597
|
+
const record = {};
|
|
598
|
+
record[CouchDBKeys.TABLE] = tableName;
|
|
599
|
+
// record[CouchDBKeys.ID] = this.generateId(tableName, id);
|
|
600
|
+
Object.assign(record, model);
|
|
601
|
+
return [tableName, id, record, ctx];
|
|
602
|
+
}
|
|
603
|
+
createAllPrefix(tableName, ids, models, ...args) {
|
|
604
|
+
if (ids.length !== models.length)
|
|
605
|
+
throw new InternalError("Ids and models must have the same length");
|
|
606
|
+
const ctx = args.pop();
|
|
607
|
+
const records = ids.map((id, count) => {
|
|
608
|
+
const record = {};
|
|
609
|
+
record[CouchDBKeys.TABLE] = tableName;
|
|
610
|
+
// record[CouchDBKeys.ID] = this.generateId(tableName, id);
|
|
611
|
+
Object.assign(record, models[count]);
|
|
612
|
+
return record;
|
|
613
|
+
});
|
|
614
|
+
return [tableName, ids, records, ctx];
|
|
615
|
+
}
|
|
616
|
+
updateAllPrefix(tableName, ids, models, ...args) {
|
|
617
|
+
if (ids.length !== models.length)
|
|
618
|
+
throw new InternalError("Ids and models must have the same length");
|
|
619
|
+
const ctx = args.pop();
|
|
620
|
+
const records = ids.map((id, count) => {
|
|
621
|
+
const record = {};
|
|
622
|
+
record[CouchDBKeys.TABLE] = tableName;
|
|
623
|
+
Object.assign(record, models[count]);
|
|
624
|
+
return record;
|
|
625
|
+
});
|
|
626
|
+
return [tableName, ids, records, ctx];
|
|
627
|
+
}
|
|
628
|
+
/**
|
|
629
|
+
* @description Static method for decoration overrides
|
|
630
|
+
* @summary Overrides/extends decaf decoration with Fabric-specific functionality
|
|
631
|
+
* @static
|
|
632
|
+
* @override
|
|
633
|
+
* @return {void}
|
|
634
|
+
*/
|
|
635
|
+
static decoration() {
|
|
636
|
+
super.decoration();
|
|
637
|
+
const createdByKey = Repository.key(PersistenceKeys.CREATED_BY);
|
|
638
|
+
const updatedByKey = Repository.key(PersistenceKeys.UPDATED_BY);
|
|
639
|
+
Decoration.flavouredAs(FabricFlavour)
|
|
640
|
+
.for(createdByKey)
|
|
641
|
+
.define(onCreate(createdByOnFabricCreateUpdate), propMetadata(createdByKey, {}))
|
|
642
|
+
.apply();
|
|
643
|
+
Decoration.flavouredAs(FabricFlavour)
|
|
644
|
+
.for(updatedByKey)
|
|
645
|
+
.define(onCreateUpdate(createdByOnFabricCreateUpdate), propMetadata(updatedByKey, {}))
|
|
646
|
+
.apply();
|
|
647
|
+
const pkKey = Repository.key(DBKeys.ID);
|
|
648
|
+
Decoration.flavouredAs(FabricFlavour)
|
|
649
|
+
.for(pkKey)
|
|
650
|
+
.define(index([OrderDirection.ASC, OrderDirection.DSC]), required(), readonly(),
|
|
651
|
+
// type([String.name, Number.name, BigInt.name]),
|
|
652
|
+
propMetadata(pkKey, NumericSequence), onCreate(pkFabricOnCreate, NumericSequence))
|
|
653
|
+
.apply();
|
|
654
|
+
const columnKey = Adapter.key(PersistenceKeys.COLUMN);
|
|
655
|
+
Decoration.flavouredAs(FabricFlavour)
|
|
656
|
+
.for(columnKey)
|
|
657
|
+
.extend(FabricProperty())
|
|
658
|
+
.apply();
|
|
659
|
+
const tableKey = Adapter.key(PersistenceKeys.TABLE);
|
|
660
|
+
Decoration.flavouredAs(FabricFlavour)
|
|
661
|
+
.for(tableKey)
|
|
662
|
+
.extend(function table(obj) {
|
|
663
|
+
// const chain: any[] = [];
|
|
664
|
+
// let current = obj;
|
|
665
|
+
// do {
|
|
666
|
+
// chain.push(current);
|
|
667
|
+
// console.log(`Found class: ${current}`);
|
|
668
|
+
// } while (current && current !== Object.prototype);
|
|
669
|
+
// do {
|
|
670
|
+
// current = chain.pop();
|
|
671
|
+
// console.log(`Applying @Object() to class: ${current}`);
|
|
672
|
+
// //TODO: THIS IS NOT WORKING AND THROWS ERROR
|
|
673
|
+
// // FabricObject()(current);
|
|
674
|
+
// } while (chain.length > 1);
|
|
675
|
+
return FabricObject()(obj);
|
|
676
|
+
})
|
|
677
|
+
.apply();
|
|
678
|
+
const oneToOnekey = Repository.key(PersistenceKeys.ONE_TO_ONE);
|
|
679
|
+
function oneToOneDec(clazz, cascade, populate, joinColumnOpts, fk) {
|
|
680
|
+
const meta = {
|
|
681
|
+
class: clazz.name ? clazz.name : clazz,
|
|
682
|
+
cascade: cascade,
|
|
683
|
+
populate: populate,
|
|
684
|
+
};
|
|
685
|
+
if (joinColumnOpts)
|
|
686
|
+
meta.joinTable = joinColumnOpts;
|
|
687
|
+
if (fk)
|
|
688
|
+
meta.name = fk;
|
|
689
|
+
return apply(prop(PersistenceKeys.RELATIONS), type([
|
|
690
|
+
clazz.name ? clazz.name : clazz,
|
|
691
|
+
String.name,
|
|
692
|
+
Number.name,
|
|
693
|
+
BigInt.name,
|
|
694
|
+
]), onCreate(oneToOneOnCreate, meta), onUpdate(oneToOneOnUpdate, meta), onDelete(oneToOneOnDelete, meta), afterAny(pop, meta), propMetadata(oneToOnekey, meta));
|
|
695
|
+
}
|
|
696
|
+
Decoration.flavouredAs(FabricFlavour)
|
|
697
|
+
.for(oneToOnekey)
|
|
698
|
+
.define({
|
|
699
|
+
decorator: oneToOneDec,
|
|
700
|
+
})
|
|
701
|
+
.apply();
|
|
702
|
+
const oneToManyKey = Repository.key(PersistenceKeys.ONE_TO_MANY);
|
|
703
|
+
function oneToManyDec(clazz, cascade, populate, joinTableOpts, fk) {
|
|
704
|
+
const metadata = {
|
|
705
|
+
class: clazz.name ? clazz.name : clazz,
|
|
706
|
+
cascade: cascade,
|
|
707
|
+
populate: populate,
|
|
708
|
+
};
|
|
709
|
+
if (joinTableOpts)
|
|
710
|
+
metadata.joinTable = joinTableOpts;
|
|
711
|
+
if (fk)
|
|
712
|
+
metadata.name = fk;
|
|
713
|
+
return apply(prop(PersistenceKeys.RELATIONS), list([clazz, String, Number, BigInt]), onCreate(oneToManyOnCreate, metadata), onUpdate(oneToManyOnUpdate, metadata), onDelete(oneToManyOnDelete, metadata), afterAny(pop, metadata), propMetadata(oneToManyKey, metadata));
|
|
714
|
+
}
|
|
715
|
+
Decoration.for(oneToManyKey)
|
|
716
|
+
.define({
|
|
717
|
+
decorator: oneToManyDec,
|
|
718
|
+
})
|
|
719
|
+
.apply();
|
|
720
|
+
}
|
|
721
|
+
}
|
|
722
|
+
FabricContractAdapter.decoration();
|
|
723
|
+
Adapter.setCurrent(FabricFlavour);
|
|
724
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"ContractAdapter.js","sourceRoot":"","sources":["../../../src/contracts/ContractAdapter.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,cAAc,EAAE,WAAW,EAAc,MAAM,uBAAuB,CAAC;AAChF,OAAO,EAEL,UAAU,EACV,IAAI,EACJ,KAAK,EACL,IAAI,EACJ,YAAY,EACZ,QAAQ,EACR,IAAI,GACL,MAAM,gCAAgC,CAAC;AAExC,OAAO,EAAE,qBAAqB,EAAE,6BAA0B;AAC1D,OAAO,EACL,QAAQ,EAER,MAAM,EACN,aAAa,EACb,gBAAgB,EAChB,aAAa,EACb,QAAQ,EACR,cAAc,EACd,QAAQ,EACR,QAAQ,EAER,QAAQ,EACR,kBAAkB,GACnB,MAAM,yBAAyB,CAAC;AACjC,OAAO,EAEL,MAAM,IAAI,YAAY,EACtB,QAAQ,IAAI,cAAc,GAC3B,MAAM,qBAAqB,CAAC;AAC7B,OAAO,EAAU,OAAO,EAAE,MAAM,mBAAmB,CAAC;AAEpD,OAAO,EACL,cAAc,EACd,eAAe,EAEf,UAAU,EAEV,oBAAoB,EAEpB,gBAAgB,EAChB,KAAK,EACL,eAAe,EACf,OAAO,EAGP,iBAAiB,GAGlB,MAAM,gBAAgB,CAAC;AACxB,OAAO,EAAE,wBAAwB,EAAE,sCAAmC;AAOtE,OAAO,EAAE,eAAe,EAAE,qCAAkC;AAC5D,OAAO,EAAE,sBAAsB,EAAE,oCAAiC;AAClE,OAAO,EAAE,mBAAmB,EAAE,8BAAyB;AACvD,OAAO,EAAE,aAAa,EAAE,iCAA4B;AACpD,OAAO,EAAE,6BAA6B,EAAE,qDAAgD;AACxF,OAAO,EACL,iBAAiB,EACjB,iBAAiB,EACjB,gBAAgB,EAChB,gBAAgB,EAChB,gBAAgB,EAChB,QAAQ,IAAI,GAAG,GAChB,gCAA6B;AAC9B,OAAO,EAAE,KAAK,EAAE,MAAM,sBAAsB,CAAC;AAE7C;;;;;;;;;;;;;;;;;;;;;;;;;GAyBG;AACH,MAAM,CAAC,KAAK,UAAU,6BAA6B,CAMjD,OAAqC,EACrC,IAAO,EACP,GAAY,EACZ,KAAQ;IAER,IAAI,CAAC;QACH,MAAM,IAAI,GAAG,OAAO,CAAC,GAAG,CAAC,gBAAgB,CAAmB,CAAC;QAC7D,KAAK,CAAC,GAAG,CAAC,GAAG,IAAI,CAAC,KAAK,EAAmB,CAAC;QAC3C,6DAA6D;IAC/D,CAAC;IAAC,OAAO,CAAU,EAAE,CAAC;QACpB,MAAM,IAAI,gBAAgB,CACxB,gEAAgE,CACjE,CAAC;IACJ,CAAC;AACH,CAAC;AAED;;;;;;;;;;;;;;;;;;;;;;;;;;;GA2BG;AACH,MAAM,CAAC,KAAK,UAAU,gBAAgB,CAOpC,OAAmB,EACnB,IAAO,EACP,GAAY,EACZ,KAAQ;IAER,IAAI,CAAC,IAAI,CAAC,IAAI,IAAI,KAAK,CAAC,GAAG,CAAC,EAAE,CAAC;QAC7B,OAAO;IACT,CAAC;IAED,MAAM,kBAAkB,GAAG,UACzB,MAAS,EACT,WAAmB,EACnB,KAA+B;QAE/B,MAAM,CAAC,cAAc,CAAC,MAAM,EAAE,WAAW,EAAE;YACzC,UAAU,EAAE,IAAI;YAChB,QAAQ,EAAE,KAAK;YACf,YAAY,EAAE,IAAI;YAClB,KAAK,EAAE,KAAK;SACb,CAAC,CAAC;IACL,CAAC,CAAC;IACF,IAAI,CAAC,IAAI,CAAC,IAAI;QAAE,IAAI,CAAC,IAAI,GAAG,oBAAoB,CAAC,KAAK,EAAE,IAAI,CAAC,CAAC;IAC9D,IAAI,QAAgC,CAAC;IACrC,IAAI,CAAC;QACH,QAAQ,GAAG,CAAC,MAAM,IAAI,CAAC,OAAO,CAAC,QAAQ,CAAC,IAAI,CAAC,CAA2B,CAAC;IAC3E,CAAC;IAAC,OAAO,CAAM,EAAE,CAAC;QAChB,MAAM,IAAI,aAAa,CACrB,kCAAkC,IAAI,CAAC,IAAI,KAAK,CAAC,EAAE,CACpD,CAAC;IACJ,CAAC;IAED,MAAM,IAAI,GAAG,MAAM,QAAQ,CAAC,IAAI,CAAC,OAAgC,CAAC,CAAC;IACnE,kBAAkB,CAAC,KAAK,EAAE,GAAa,EAAE,IAAI,CAAC,CAAC;AACjD,CAAC;AAED;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;GAoCG;AACH,MAAM,OAAO,qBAAsB,SAAQ,cAK1C;IACoB,SAAS;QAC1B,MAAM,IAAI,gBAAgB,CAAC,6CAA6C,CAAC,CAAC;IAC5E,CAAC;IACD;;OAEG;aACY,gBAAW,GAAG,IAAI,WAAW,CAAC,MAAM,CAAC,AAA1B,CAA2B;aAE3B,eAAU,GAAG,IAAI,6BAA6B,EAAE,AAAtC,CAAuC;IAE3E;;;;;OAKG;IACI,MAAM,CAAC,GAAQ;QACpB,OAAO,OAAO,CAAC,GAAG,CAAC,qBAAqB,EAAE,EAAE,EAAE,GAAG,CAAmB,CAAC;IACvE,CAAC;IAQD;;;;;OAKG;IACM,UAAU;QASjB,OAAO,wBAAwB,CAAC;IAClC,CAAC;IAED;;;;;OAKG;IACH,YAAY,KAAW,EAAE,KAAc;QACrC,KAAK,CAAC,KAAK,EAAE,aAAa,EAAE,KAAK,CAAC,CAAC;QA/BrC;;;WAGG;QACM,YAAO,GAAG,qBAAqB,CAAC;IA4BzC,CAAC;IAEQ,GAAG,CAAC,MAAoB,EAAE,GAAG,IAAS;QAC7C,OAAO,KAAK,CAAC,GAAG,CAAC,MAAM,EAAE,GAAG,IAAI,CAAC,CAAC;IACpC,CAAC;IAED;;;;;;;;;OASG;IACM,KAAK,CAAC,MAAM,CACnB,SAAiB,EACjB,EAAmB,EACnB,KAA0B,EAC1B,GAAG,IAAW;QAEd,MAAM,EAAE,IAAI,EAAE,MAAM,EAAE,GAAG,IAAI,CAAC,GAAG,EAAE,CAAC;QACpC,MAAM,GAAG,GAAG,MAAM,CAAC,GAAG,CAAC,IAAI,CAAC,MAAM,CAAC,CAAC;QAEpC,IAAI,CAAC;YACH,GAAG,CAAC,IAAI,CAAC,mBAAmB,SAAS,kBAAkB,EAAE,EAAE,CAAC,CAAC;YAC7D,KAAK,GAAG,MAAM,IAAI,CAAC,QAAQ,CAAC,IAAI,EAAE,EAAE,CAAC,QAAQ,EAAE,EAAE,KAAK,CAAC,CAAC;QAC1D,CAAC;QAAC,OAAO,CAAU,EAAE,CAAC;YACpB,MAAM,IAAI,CAAC,UAAU,CAAC,CAAU,CAAC,CAAC;QACpC,CAAC;QAED,OAAO,KAAK,CAAC;IACf,CAAC;IAED;;;;;;;OAOG;IACM,KAAK,CAAC,IAAI,CACjB,SAAiB,EACjB,EAAmB,EACnB,GAAG,IAAW;QAEd,MAAM,EAAE,IAAI,EAAE,MAAM,EAAE,GAAG,IAAI,CAAC,GAAG,EAAE,CAAC;QACpC,MAAM,GAAG,GAAG,MAAM,CAAC,GAAG,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC;QAElC,IAAI,KAA0B,CAAC;QAC/B,IAAI,CAAC;YACH,MAAM,OAAO,GAAG,MAAM,IAAI,CAAC,SAAS,CAAC,IAAI,EAAE,SAAS,EAAE,EAAE,CAAC,QAAQ,EAAE,CAAC,CAAC;YAErE,IAAI,OAAO,CAAC,MAAM,GAAG,CAAC,EAAE,CAAC;gBACvB,GAAG,CAAC,KAAK,CAAC,0BAA0B,EAAE,OAAO,SAAS,QAAQ,CAAC,CAAC;gBAChE,MAAM,IAAI,aAAa,CACrB,0BAA0B,EAAE,OAAO,SAAS,QAAQ,CACrD,CAAC;YACJ,CAAC;iBAAM,IAAI,OAAO,CAAC,MAAM,GAAG,CAAC,EAAE,CAAC;gBAC9B,GAAG,CAAC,KAAK,CAAC,0BAA0B,EAAE,OAAO,SAAS,QAAQ,CAAC,CAAC;gBAChE,KAAK,GAAG,OAAO,CAAC,GAAG,EAAyB,CAAC;YAC/C,CAAC;iBAAM,CAAC;gBACN,KAAK,GAAG,IAAI,CAAC,WAAW,CAAC,OAAO,CAAC,CAAC;YACpC,CAAC;QACH,CAAC;QAAC,OAAO,CAAU,EAAE,CAAC;YACpB,MAAM,IAAI,CAAC,UAAU,CAAC,CAAU,CAAC,CAAC;QACpC,CAAC;QAED,OAAO,KAAK,CAAC;IACf,CAAC;IAED;;;;;;;;;OASG;IACM,KAAK,CAAC,MAAM,CACnB,SAAiB,EACjB,EAAmB,EACnB,KAA0B,EAC1B,GAAG,IAAW;QAEd,MAAM,EAAE,IAAI,EAAE,MAAM,EAAE,GAAG,IAAI,CAAC,GAAG,EAAE,CAAC;QACpC,MAAM,GAAG,GAAG,MAAM,CAAC,GAAG,CAAC,IAAI,CAAC,MAAM,CAAC,CAAC;QAEpC,IAAI,CAAC;YACH,GAAG,CAAC,IAAI,CAAC,qBAAqB,SAAS,kBAAkB,EAAE,EAAE,CAAC,CAAC;YAC/D,KAAK,GAAG,MAAM,IAAI,CAAC,QAAQ,CAAC,IAAI,EAAE,EAAE,CAAC,QAAQ,EAAE,EAAE,KAAK,CAAC,CAAC;QAC1D,CAAC;QAAC,OAAO,CAAU,EAAE,CAAC;YACpB,MAAM,IAAI,CAAC,UAAU,CAAC,CAAU,CAAC,CAAC;QACpC,CAAC;QAED,OAAO,KAAK,CAAC;IACf,CAAC;IAED;;;;;;;OAOG;IACH,KAAK,CAAC,MAAM,CACV,SAAiB,EACjB,EAAmB,EACnB,GAAG,IAAW;QAEd,MAAM,GAAG,GAAG,IAAI,CAAC,GAAG,EAAE,CAAC;QACvB,MAAM,EAAE,IAAI,EAAE,MAAM,EAAE,GAAG,GAAG,CAAC;QAC7B,MAAM,GAAG,GAAG,MAAM,CAAC,GAAG,CAAC,IAAI,CAAC,MAAM,CAAC,CAAC;QAEpC,IAAI,CAAC,IAAI,CAAC,GAAG,CAAC,CAAC;QAEf,IAAI,KAA0B,CAAC;QAC/B,IAAI,CAAC;YACH,KAAK,GAAG,MAAM,IAAI,CAAC,IAAI,CAAC,SAAS,EAAE,EAAE,EAAE,GAAG,IAAI,CAAC,CAAC;YAChD,GAAG,CAAC,OAAO,CAAC,0BAA0B,EAAE,SAAS,SAAS,QAAQ,CAAC,CAAC;YACpE,IAAI,CAAC,WAAW,CAAC,IAAI,EAAE,SAAS,EAAE,EAAE,CAAC,QAAQ,EAAE,CAAC,CAAC;QACnD,CAAC;QAAC,OAAO,CAAU,EAAE,CAAC;YACpB,MAAM,IAAI,CAAC,UAAU,CAAC,CAAU,CAAC,CAAC;QACpC,CAAC;QAED,OAAO,KAAK,CAAC;IACf,CAAC;IAES,KAAK,CAAC,WAAW,CACzB,IAAmB,EACnB,SAAiB,EACjB,EAAU;IACV,6DAA6D;IAC7D,GAAG,IAAW;QAEd,MAAM,WAAW,GAAG,IAAI,CAAC,kBAAkB,CAAC,SAAS,EAAE,CAAC,MAAM,CAAC,EAAE,CAAC,CAAC,CAAC,CAAC;QACrE,MAAM,IAAI,CAAC,WAAW,CAAC,WAAW,CAAC,CAAC;IACtC,CAAC;IAES,KAAK,CAAC,QAAQ,CACtB,IAAmB,EACnB,EAAU,EACV,KAA0B;IAC1B,6DAA6D;IAC7D,GAAG,IAAW;QAEd,IAAI,IAAY,CAAC;QAEjB,IAAI,CAAC;YACH,IAAI,GAAG,MAAM,CAAC,IAAI,CAChB,qBAAqB,CAAC,UAAU,CAAC,SAAS,CAAC,KAAc,CAAC,CAC3D,CAAC;QACJ,CAAC;QAAC,OAAO,CAAU,EAAE,CAAC;YACpB,MAAM,IAAI,kBAAkB,CAC1B,sCAAsC,EAAE,KAAK,CAAC,EAAE,CACjD,CAAC;QACJ,CAAC;QACD,MAAM,IAAI,CAAC,QAAQ,CAAC,EAAE,CAAC,QAAQ,EAAE,EAAE,IAAI,CAAC,CAAC;QAEzC,OAAO,KAAK,CAAC;IACf,CAAC;IAES,KAAK,CAAC,SAAS,CACvB,IAAmB,EACnB,SAAiB,EACjB,EAAU;IACV,6DAA6D;IAC7D,GAAG,IAAW;QAEd,MAAM,WAAW,GAAG,IAAI,CAAC,kBAAkB,CAAC,SAAS,EAAE,CAAC,MAAM,CAAC,EAAE,CAAC,CAAC,CAAC,CAAC;QACrE,MAAM,OAAO,GAAU,EAAE,CAAC;QAE1B,IAAI,GAAG,GAAiC,MAAM,IAAI,CAAC,QAAQ,CAAC,WAAW,CAAC,CAAC;QAEzE,IAAI,GAAG,CAAC,QAAQ,EAAE,KAAK,EAAE;YACvB,MAAM,IAAI,aAAa,CAAC,kBAAkB,EAAE,YAAY,CAAC,CAAC;QAE5D,IAAI,CAAC;YACH,GAAG,GAAG,qBAAqB,CAAC,UAAU,CAAC,WAAW,CAChD,GAAG,CAAC,QAAQ,EAAE;YACd,yBAAyB;aAC1B,CAAC;QACJ,CAAC;QAAC,OAAO,CAAU,EAAE,CAAC;YACpB,MAAM,IAAI,kBAAkB,CAAC,2BAA2B,CAAC,EAAE,CAAC,CAAC;QAC/D,CAAC;QAED,OAAO,CAAC,IAAI,CAAC,GAAG,CAAC,CAAC;QAElB,OAAO,OAAO,CAAC;IACjB,CAAC;IAES,KAAK,CAAC,WAAW,CACzB,IAAmB,EACnB,QAAa;IACb,6DAA6D;IAC7D,GAAG,IAAW;QAEd,OAAO,CAAC,MAAM,IAAI,CAAC,cAAc,CAC/B,IAAI,CAAC,SAAS,CAAC,QAAQ,CAAC,CACzB,CAAiC,CAAC;IACrC,CAAC;IAES,KAAK,CAAC,oBAAoB,CAClC,IAAmB,EACnB,QAAa,EACb,QAAgB,GAAG,EACnB,IAAa;IACb,6DAA6D;IAC7D,GAAG,IAAW;QAEd,OAAO,CAAC,MAAM,IAAI,CAAC,4BAA4B,CAC7C,IAAI,CAAC,SAAS,CAAC,QAAQ,CAAC,EACxB,KAAK,EACL,IAAI,EAAE,QAAQ,EAAE,CACjB,CAAqD,CAAC;IACzD,CAAC;IAES,WAAW,CAAC,OAA8B;QAClD,MAAM,OAAO,GAAG,CAAC,KAA0B,EAAE,EAAE,CAC7C,MAAM,CAAC,OAAO,CAAC,KAAK,CAAC,CAAC,MAAM,CAAC,CAAC,KAA0B,EAAE,CAAC,GAAG,EAAE,GAAG,CAAC,EAAE,EAAE;YACtE,IAAI,OAAO,GAAG,KAAK,WAAW;gBAAE,KAAK,CAAC,GAAG,CAAC,GAAG,GAAG,CAAC;YACjD,OAAO,KAAK,CAAC;QACf,CAAC,EAAE,EAAE,CAAC,CAAC;QAET,IAAI,UAAU,GAAwB,OAAO,CAAC,GAAG,EAAyB,CAAC;QAE3E,KAAK,MAAM,GAAG,IAAI,OAAO,EAAE,CAAC;YAC1B,UAAU,GAAG,MAAM,CAAC,MAAM,CAAC,EAAE,EAAE,OAAO,CAAC,UAAU,CAAC,EAAE,OAAO,CAAC,GAAG,CAAC,CAAC,CAAC;QACpE,CAAC;QAED,OAAO,UAAU,CAAC;IACpB,CAAC;IAED;;;;;OAKG;IACO,MAAM,CAAC,MAAkB;QACjC,OAAO,qBAAqB,CAAC,WAAW,CAAC,MAAM,CAAC,MAAM,CAAC,CAAC;IAC1D,CAAC;IAED;;;;;;;;;OASG;IACgB,KAAK,CAAC,KAAK,CAC5B,SAAwB,EACxB,KAAqB,EACrB,KAAmC,EACnC,GAAQ,EACR,GAAG,IAAW;QAEd,OAAO,MAAM,CAAC,MAAM,CAAC,MAAM,KAAK,CAAC,KAAK,CAAC,SAAS,EAAE,KAAK,EAAE,KAAK,EAAE,GAAG,IAAI,CAAC,EAAE;YACxE,IAAI,EAAE,GAAG,CAAC,IAAI;YACd,QAAQ,EAAE,GAAG,CAAC,cAAc;YAC5B,MAAM,EAAE,IAAI,CAAC,MAAM,CAAC,GAAG,CAAC;SACzB,CAAC,CAAC;IACL,CAAC;IAED;;;;;;OAMG;IACH,6DAA6D;IACnD,KAAK,CAAI,MAAsB;QACvC,OAAO,OAAO,CAAC,OAAO,CAAC,SAAS,CAAC,CAAC;IACpC,CAAC;IAED;;;;;;;;;;;;;;;;;;;;;;;;OAwBG;IACO,KAAK,CAAC,cAAc,CAC5B,GAAW,EACX,QAAsC,EACtC,SAAS,GAAG,KAAK;QAEjB,MAAM,UAAU,GAAG,EAAE,CAAC;QACtB,IAAI,GAAG,GAAkC,MAAM,QAAQ,CAAC,IAAI,EAAE,CAAC;QAC/D,OAAO,CAAC,GAAG,CAAC,IAAI,EAAE,CAAC;YACjB,IAAI,GAAG,CAAC,KAAK,IAAI,GAAG,CAAC,KAAK,CAAC,KAAK,CAAC,QAAQ,EAAE,EAAE,CAAC;gBAC5C,IAAI,OAAO,GAAQ,EAAE,CAAC;gBACtB,GAAG,CAAC,KAAK,CAAC,GAAG,CAAC,KAAK,CAAC,KAAK,CAAC,QAAQ,CAAC,MAAM,CAAC,CAAC,CAAC;gBAC5C,IAAI,SAAS,CAAC,0BAA0B,EAAE,CAAC;oBACzC,OAAO,CAAC,IAAI,GAAG,GAAG,CAAC,KAAK,CAAC,IAAI,CAAC;oBAC9B,OAAO,CAAC,SAAS,GAAG,GAAG,CAAC,KAAK,CAAC,SAAS,CAAC;oBACxC,IAAI,CAAC;wBACH,OAAO,CAAC,KAAK,GAAG,IAAI,CAAC,KAAK,CAAC,GAAG,CAAC,KAAK,CAAC,KAAK,CAAC,QAAQ,CAAC,MAAM,CAAC,CAAC,CAAC;oBAC/D,CAAC;oBAAC,OAAO,GAAQ,EAAE,CAAC;wBAClB,GAAG,CAAC,KAAK,CAAC,GAAG,CAAC,CAAC;wBACf,OAAO,CAAC,KAAK,GAAG,GAAG,CAAC,KAAK,CAAC,KAAK,CAAC,QAAQ,CAAC,MAAM,CAAC,CAAC;oBACnD,CAAC;gBACH,CAAC;qBAAM,CAAC;oBACN,IAAI,CAAC;wBACH,OAAO,GAAG,IAAI,CAAC,KAAK,CAAC,GAAG,CAAC,KAAK,CAAC,KAAK,CAAC,QAAQ,CAAC,MAAM,CAAC,CAAC,CAAC;oBACzD,CAAC;oBAAC,OAAO,GAAQ,EAAE,CAAC;wBAClB,GAAG,CAAC,KAAK,CAAC,GAAG,CAAC,CAAC;wBACf,OAAO,GAAG,GAAG,CAAC,KAAK,CAAC,KAAK,CAAC,QAAQ,CAAC,MAAM,CAAC,CAAC;oBAC7C,CAAC;gBACH,CAAC;gBACD,UAAU,CAAC,IAAI,CAAC,OAAO,CAAC,CAAC;YAC3B,CAAC;YACD,GAAG,GAAG,MAAM,QAAQ,CAAC,IAAI,EAAE,CAAC;QAC9B,CAAC;QACD,GAAG,CAAC,KAAK,CAAC,0BAA0B,UAAU,CAAC,MAAM,UAAU,CAAC,CAAC;QACjE,QAAQ,CAAC,KAAK,EAAE,CAAC,CAAC,qDAAqD;QACvE,OAAO,UAAU,CAAC;IACpB,CAAC;IAED;;;;;;;;;;;;;;;;;;;;;;;;;;;OA2BG;IACH,KAAK,CAAC,GAAG,CACP,QAAoB,EACpB,QAAiB,EACjB,GAAG,IAAW;QAEd,MAAM,EAAE,IAAI,EAAE,MAAM,EAAE,GAAG,IAAI,CAAC,GAAG,EAAE,CAAC;QACpC,MAAM,GAAG,GAAG,MAAM,CAAC,GAAG,CAAC,IAAI,CAAC,GAAG,CAAC,CAAC;QACjC,MAAM,EAAE,IAAI,EAAE,KAAK,EAAE,GAAG,QAAQ,CAAC;QACjC,IAAI,QAAsC,CAAC;QAC3C,IAAI,KAAK,IAAI,IAAI,EAAE,CAAC;YAClB,OAAO,QAAQ,CAAC,OAAO,CAAC,CAAC;YACzB,OAAO,QAAQ,CAAC,MAAM,CAAC,CAAC;YACxB,GAAG,CAAC,KAAK,CACP,yCAAyC,KAAK,WAAW,IAAI,EAAE,CAChE,CAAC;YACF,MAAM,QAAQ,GACZ,CAAC,MAAM,IAAI,CAAC,oBAAoB,CAC9B,IAAI,EACJ,QAAQ,EACR,KAAK,IAAI,GAAG,EACX,IAAY,EAAE,QAAQ,EAAE,CAC1B,CAAqD,CAAC;YACzD,QAAQ,GAAG,QAAQ,CAAC,QAAQ,CAAC;QAC/B,CAAC;aAAM,CAAC;YACN,GAAG,CAAC,KAAK,CAAC,qBAAqB,CAAC,CAAC;YACjC,QAAQ,GAAG,CAAC,MAAM,IAAI,CAAC,WAAW,CAChC,IAAI,EACJ,QAAQ,CACT,CAAiC,CAAC;QACrC,CAAC;QACD,GAAG,CAAC,KAAK,CAAC,mBAAmB,CAAC,CAAC;QAE/B,MAAM,OAAO,GAAG,CAAC,MAAM,IAAI,CAAC,cAAc,CAAC,GAAG,EAAE,QAAQ,CAAC,CAAM,CAAC;QAChE,GAAG,CAAC,KAAK,CACP,uBAAuB,EACvB,GAAG,KAAK,CAAC,OAAO,CAAC,OAAO,CAAC,CAAC,CAAC,CAAC,OAAO,CAAC,MAAM,CAAC,CAAC,CAAC,CAAC,EAAE,CACjD,CAAC;QACF,OAAO,OAAO,CAAC;IACjB,CAAC;IAEQ,SAAS,CAChB,GAA2B;QAE3B,IAAI,CAAC,GAAG,EAAE,CAAC;YACT,MAAM,IAAI,mBAAmB,CAAC,qBAAqB,CAAC,CAAC;QACvD,CAAC;QACD,OAAO,IAAI,eAAe,CAAC,IAAI,EAAE,GAAG,CAAC,CAAC;IACxC,CAAC;IAEQ,KAAK,CAAC,QAAQ,CAAC,OAAwB;QAC9C,OAAO,IAAI,sBAAsB,CAAC,OAAO,EAAE,IAAI,CAAC,CAAC;IACnD,CAAC;IAEQ,KAAK,CAAC,SAAS,CACtB,SAAiB,EACjB,EAAuB,EACvB,KAA4B,EAC5B,GAAG,IAAW;QAEd,IAAI,EAAE,CAAC,MAAM,KAAK,KAAK,CAAC,MAAM;YAC5B,MAAM,IAAI,aAAa,CAAC,0CAA0C,CAAC,CAAC;QAEtE,MAAM,EAAE,MAAM,EAAE,GAAG,IAAI,CAAC,IAAI,CAAC,MAAM,GAAG,CAAC,CAA0B,CAAC;QAClE,MAAM,GAAG,GAAG,MAAM,CAAC,GAAG,CAAC,IAAI,CAAC,SAAS,CAAC,CAAC;QACvC,GAAG,CAAC,IAAI,CAAC,YAAY,EAAE,CAAC,MAAM,YAAY,SAAS,QAAQ,CAAC,CAAC;QAC7D,GAAG,CAAC,KAAK,CAAC,QAAQ,EAAE,EAAE,CAAC,CAAC;QAExB,OAAO,OAAO,CAAC,GAAG,CAChB,EAAE,CAAC,GAAG,CAAC,KAAK,EAAE,CAAC,EAAE,KAAK,EAAE,EAAE;YACxB,OAAO,IAAI,CAAC,MAAM,CAAC,SAAS,EAAE,CAAC,EAAE,KAAK,CAAC,KAAK,CAAC,EAAE,GAAG,IAAI,CAAC,CAAC;QAC1D,CAAC,CAAC,CACH,CAAC;IACJ,CAAC;IAEQ,KAAK,CAAC,SAAS,CACtB,SAAiB,EACjB,EAAuB,EACvB,KAA4B,EAC5B,GAAG,IAAW;QAEd,IAAI,EAAE,CAAC,MAAM,KAAK,KAAK,CAAC,MAAM;YAC5B,MAAM,IAAI,aAAa,CAAC,0CAA0C,CAAC,CAAC;QAEtE,MAAM,EAAE,MAAM,EAAE,GAAG,IAAI,CAAC,IAAI,CAAC,MAAM,GAAG,CAAC,CAA0B,CAAC;QAElE,MAAM,GAAG,GAAG,MAAM,CAAC,GAAG,CAAC,IAAI,CAAC,SAAS,CAAC,CAAC;QACvC,GAAG,CAAC,IAAI,CAAC,YAAY,EAAE,CAAC,MAAM,YAAY,SAAS,QAAQ,CAAC,CAAC;QAC7D,GAAG,CAAC,KAAK,CAAC,QAAQ,EAAE,EAAE,CAAC,CAAC;QAExB,OAAO,OAAO,CAAC,GAAG,CAChB,EAAE,CAAC,GAAG,CAAC,KAAK,EAAE,CAAC,EAAE,KAAK,EAAE,EAAE;YACxB,OAAO,IAAI,CAAC,MAAM,CAAC,SAAS,EAAE,CAAC,EAAE,KAAK,CAAC,KAAK,CAAC,EAAE,GAAG,IAAI,CAAC,CAAC;QAC1D,CAAC,CAAC,CACH,CAAC;IACJ,CAAC;IAED;;;;;OAKG;IACM,OAAO,CACd,KAAQ,EACR,EAAW,EACX,GAAG,IAAW;QAMd,MAAM,EAAE,IAAI,EAAE,MAAM,EAAE,GAAG,IAAI,CAAC,GAAG,EAAE,CAAC;QACpC,MAAM,SAAS,GAAG,IAAI,CAAC,KAAK,EAAE,CAAC;QAC/B,MAAM,GAAG,GAAG,MAAM,CAAC,GAAG,CAAC,IAAI,CAAC,OAAO,CAAC,CAAC;QAErC,MAAM,KAAK,GAAG,gBAAgB,CAAC,KAAK,CAAC,CAAC;QACtC,MAAM,MAAM,GAAG,MAAM,CAAC,OAAO,CAAC,KAAK,CAAC,KAAK,CAAC,CAAC,MAAM,CAC/C,CAAC,KAA0B,EAAE,CAAC,GAAG,EAAE,GAAG,CAAC,EAAE,EAAE;YACzC,IAAI,OAAO,GAAG,KAAK,WAAW;gBAAE,OAAO,KAAK,CAAC;YAC7C,MAAM,UAAU,GAAG,UAAU,CAAC,MAAM,CAAC,KAAK,EAAE,GAAG,CAAC,CAAC;YACjD,IAAI,IAAI,CAAC,UAAU,CAAC,UAAU,CAAC;gBAC7B,MAAM,IAAI,aAAa,CAAC,iBAAiB,UAAU,cAAc,CAAC,CAAC;YACrE,KAAK,CAAC,UAAU,CAAC,GAAG,GAAG,CAAC;YACxB,OAAO,KAAK,CAAC;QACf,CAAC,EACD,EAAE,CACH,CAAC;QACF,IAAK,KAAa,CAAC,eAAe,CAAC,QAAQ,CAAC,EAAE,CAAC;YAC7C,GAAG,CAAC,KAAK,CACP,0CAA2C,KAAa,CAAC,eAAe,CAAC,QAAQ,CAAC,EAAE,CACrF,CAAC;YACF,MAAM,CAAC,cAAc,CAAC,MAAM,EAAE,eAAe,CAAC,QAAQ,EAAE;gBACtD,UAAU,EAAE,KAAK;gBACjB,QAAQ,EAAE,KAAK;gBACf,YAAY,EAAE,IAAI;gBAClB,KAAK,EAAG,KAAa,CAAC,eAAe,CAAC,QAAQ,CAAC;aAChD,CAAC,CAAC;QACL,CAAC;QAED,GAAG,CAAC,IAAI,CAAC,wBAAwB,SAAS,kBAAkB,KAAK,CAAC,EAAE,CAAC,EAAE,CAAC,CAAC;QAEzE,OAAO;YACL,MAAM,EAAE,MAAM;YACd,EAAE,EAAE,IAAI,CAAC,kBAAkB,CAAC,SAAS,EAAE,CAAC,MAAM,CAAC,KAAK,CAAC,EAAE,CAAC,CAAC,CAAC,CAAC;YAC3D,SAAS,EAAE,KAAK,CAAC,SAAS;SAC3B,CAAC;IACJ,CAAC;IAEQ,MAAM,CACb,GAAwB,EACxB,KAA8B,EAC9B,EAAW,EACX,EAAmB,EACnB,SAA+B;QAE/B,MAAM,GAAG,GAAG,IAAI,CAAC,GAAG,CAAC,GAAG,CAAC,IAAI,CAAC,MAAM,CAAC,CAAC;QACtC,MAAM,EAAE,GAAwB,EAAE,CAAC;QACnC,EAAE,CAAC,EAAY,CAAC,GAAG,EAAE,CAAC;QACtB,MAAM,CAAC,GAAG,CACR,OAAO,KAAK,KAAK,QAAQ,CAAC,CAAC,CAAC,KAAK,CAAC,KAAK,CAAC,EAAE,EAAE,KAAK,CAAC,CAAC,CAAC,CAAC,IAAI,KAAK,CAAC,EAAE,CAAC,CAC9D,CAAC;QACP,GAAG,CAAC,KAAK,CAAC,oBAAoB,CAAC,CAAC,WAAW,CAAC,IAAI,OAAO,EAAE,EAAE,CAAC,CAAC;QAC7D,MAAM,QAAQ,GAAG,GAAG,CAAC,eAAe,CAAC,QAAQ,CAAC,CAAC;QAC/C,MAAM,MAAM,GAAG,MAAM,CAAC,IAAI,CAAC,CAAC,CAAC,CAAC,MAAM,CAAC,CAAC,KAAQ,EAAE,GAAG,EAAE,EAAE;YACpD,KAA6B,CAAC,GAAG,CAAC,GAAG,GAAG,CAAC,UAAU,CAAC,MAAM,CAAC,KAAK,EAAE,GAAG,CAAC,CAAC,CAAC;YACzE,OAAO,KAAK,CAAC;QACf,CAAC,EAAE,CAAC,CAAC,CAAC;QAEN,IAAI,SAAS,EAAE,CAAC;YACd,GAAG,CAAC,OAAO,CACT,mCAAmC,MAAM,CAAC,IAAI,CAAC,SAAS,CAAC,CAAC,IAAI,CAAC,IAAI,CAAC,EAAE,CACvE,CAAC;YACF,MAAM,CAAC,OAAO,CAAC,SAAS,CAAC,CAAC,OAAO,CAAC,CAAC,CAAC,GAAG,EAAE,GAAG,CAAC,EAAE,EAAE;gBAC/C,IAAI,GAAG,IAAI,MAAM,IAAK,MAAc,CAAC,GAAG,CAAC,KAAK,SAAS;oBACrD,MAAM,IAAI,aAAa,CACrB,sBAAsB,GAAG,4BAA4B,CAAC,CAAC,WAAW,CAAC,IAAI,wBAAwB,CAChG,CAAC;gBACJ,MAAM,CAAC,GAAc,CAAC,GAAG,GAAG,CAAC;YAC/B,CAAC,CAAC,CAAC;QACL,CAAC;QAED,IAAI,QAAQ,EAAE,CAAC;YACb,GAAG,CAAC,KAAK,CACP,iBAAiB,IAAI,CAAC,OAAO,6BAA6B,CAAC,CAAC,WAAW,CAAC,IAAI,OAAO,EAAE,KAAK,QAAQ,EAAE,CACrG,CAAC;YACF,MAAM,CAAC,cAAc,CAAC,MAAM,EAAE,eAAe,CAAC,QAAQ,EAAE;gBACtD,UAAU,EAAE,KAAK;gBACjB,YAAY,EAAE,KAAK;gBACnB,QAAQ,EAAE,KAAK;gBACf,KAAK,EAAE,QAAQ;aAChB,CAAC,CAAC;QACL,CAAC;QAED,OAAO,MAAM,CAAC;IAChB,CAAC;IAEQ,YAAY,CACnB,SAAiB,EACjB,EAAmB,EACnB,KAA0B,EAC1B,GAAG,IAAW;QAEd,MAAM,GAAG,GAA0B,IAAI,CAAC,GAAG,EAAE,CAAC;QAC9C,MAAM,MAAM,GAAwB,EAAE,CAAC;QACvC,MAAM,CAAC,WAAW,CAAC,KAAK,CAAC,GAAG,SAAS,CAAC;QACtC,2DAA2D;QAC3D,MAAM,CAAC,MAAM,CAAC,MAAM,EAAE,KAAK,CAAC,CAAC;QAC7B,OAAO,CAAC,SAAS,EAAE,EAAE,EAAE,MAAM,EAAE,GAAG,CAAC,CAAC;IACtC,CAAC;IAEQ,YAAY,CACnB,SAAiB,EACjB,EAAmB,EACnB,KAA0B,EAC1B,GAAG,IAAW;QAEd,MAAM,GAAG,GAA0B,IAAI,CAAC,GAAG,EAAE,CAAC;QAC9C,MAAM,MAAM,GAAwB,EAAE,CAAC;QACvC,MAAM,CAAC,WAAW,CAAC,KAAK,CAAC,GAAG,SAAS,CAAC;QACtC,2DAA2D;QAC3D,MAAM,CAAC,MAAM,CAAC,MAAM,EAAE,KAAK,CAAC,CAAC;QAC7B,OAAO,CAAC,SAAS,EAAE,EAAE,EAAE,MAAM,EAAE,GAAG,CAAC,CAAC;IACtC,CAAC;IAEkB,eAAe,CAChC,SAAiB,EACjB,GAAwB,EACxB,MAA6B,EAC7B,GAAG,IAAW;QAEd,IAAI,GAAG,CAAC,MAAM,KAAK,MAAM,CAAC,MAAM;YAC9B,MAAM,IAAI,aAAa,CAAC,0CAA0C,CAAC,CAAC;QAEtE,MAAM,GAAG,GAA0B,IAAI,CAAC,GAAG,EAAE,CAAC;QAE9C,MAAM,OAAO,GAAG,GAAG,CAAC,GAAG,CAAC,CAAC,EAAE,EAAE,KAAK,EAAE,EAAE;YACpC,MAAM,MAAM,GAAwB,EAAE,CAAC;YACvC,MAAM,CAAC,WAAW,CAAC,KAAK,CAAC,GAAG,SAAS,CAAC;YACtC,2DAA2D;YAC3D,MAAM,CAAC,MAAM,CAAC,MAAM,EAAE,MAAM,CAAC,KAAK,CAAC,CAAC,CAAC;YACrC,OAAO,MAAM,CAAC;QAChB,CAAC,CAAC,CAAC;QACH,OAAO,CAAC,SAAS,EAAE,GAAG,EAAE,OAAO,EAAE,GAAU,CAAC,CAAC;IAC/C,CAAC;IAEkB,eAAe,CAChC,SAAiB,EACjB,GAAwB,EACxB,MAA6B,EAC7B,GAAG,IAAW;QAEd,IAAI,GAAG,CAAC,MAAM,KAAK,MAAM,CAAC,MAAM;YAC9B,MAAM,IAAI,aAAa,CAAC,0CAA0C,CAAC,CAAC;QAEtE,MAAM,GAAG,GAA0B,IAAI,CAAC,GAAG,EAAE,CAAC;QAE9C,MAAM,OAAO,GAAG,GAAG,CAAC,GAAG,CAAC,CAAC,EAAE,EAAE,KAAK,EAAE,EAAE;YACpC,MAAM,MAAM,GAAwB,EAAE,CAAC;YACvC,MAAM,CAAC,WAAW,CAAC,KAAK,CAAC,GAAG,SAAS,CAAC;YACtC,MAAM,CAAC,MAAM,CAAC,MAAM,EAAE,MAAM,CAAC,KAAK,CAAC,CAAC,CAAC;YACrC,OAAO,MAAM,CAAC;QAChB,CAAC,CAAC,CAAC;QACH,OAAO,CAAC,SAAS,EAAE,GAAG,EAAE,OAAO,EAAE,GAAU,CAAC,CAAC;IAC/C,CAAC;IAED;;;;;;OAMG;IACH,MAAM,CAAU,UAAU;QACxB,KAAK,CAAC,UAAU,EAAE,CAAC;QACnB,MAAM,YAAY,GAAG,UAAU,CAAC,GAAG,CAAC,eAAe,CAAC,UAAU,CAAC,CAAC;QAChE,MAAM,YAAY,GAAG,UAAU,CAAC,GAAG,CAAC,eAAe,CAAC,UAAU,CAAC,CAAC;QAChE,UAAU,CAAC,WAAW,CAAC,aAAa,CAAC;aAClC,GAAG,CAAC,YAAY,CAAC;aACjB,MAAM,CACL,QAAQ,CAAC,6BAA6B,CAAC,EACvC,YAAY,CAAC,YAAY,EAAE,EAAE,CAAC,CAC/B;aACA,KAAK,EAAE,CAAC;QAEX,UAAU,CAAC,WAAW,CAAC,aAAa,CAAC;aAClC,GAAG,CAAC,YAAY,CAAC;aACjB,MAAM,CACL,cAAc,CAAC,6BAA6B,CAAC,EAC7C,YAAY,CAAC,YAAY,EAAE,EAAE,CAAC,CAC/B;aACA,KAAK,EAAE,CAAC;QAEX,MAAM,KAAK,GAAG,UAAU,CAAC,GAAG,CAAC,MAAM,CAAC,EAAE,CAAC,CAAC;QACxC,UAAU,CAAC,WAAW,CAAC,aAAa,CAAC;aAClC,GAAG,CAAC,KAAK,CAAC;aACV,MAAM,CACL,KAAK,CAAC,CAAC,cAAc,CAAC,GAAG,EAAE,cAAc,CAAC,GAAG,CAAC,CAAC,EAC/C,QAAQ,EAAE,EACV,QAAQ,EAAE;QACV,iDAAiD;QACjD,YAAY,CAAC,KAAK,EAAE,eAAe,CAAC,EACpC,QAAQ,CAAC,gBAAgB,EAAE,eAAe,CAAC,CAC5C;aACA,KAAK,EAAE,CAAC;QAEX,MAAM,SAAS,GAAG,OAAO,CAAC,GAAG,CAAC,eAAe,CAAC,MAAM,CAAC,CAAC;QACtD,UAAU,CAAC,WAAW,CAAC,aAAa,CAAC;aAClC,GAAG,CAAC,SAAS,CAAC;aACd,MAAM,CAAC,cAAc,EAAE,CAAC;aACxB,KAAK,EAAE,CAAC;QAEX,MAAM,QAAQ,GAAG,OAAO,CAAC,GAAG,CAAC,eAAe,CAAC,KAAK,CAAC,CAAC;QACpD,UAAU,CAAC,WAAW,CAAC,aAAa,CAAC;aAClC,GAAG,CAAC,QAAQ,CAAC;aACb,MAAM,CAAC,SAAS,KAAK,CAAC,GAAQ;YAC7B,2BAA2B;YAE3B,qBAAqB;YAErB,OAAO;YACP,yBAAyB;YACzB,4CAA4C;YAC5C,qDAAqD;YAErD,OAAO;YACP,2BAA2B;YAC3B,4DAA4D;YAC5D,iDAAiD;YACjD,gCAAgC;YAChC,8BAA8B;YAE9B,OAAO,YAAY,EAAE,CAAC,GAAG,CAAC,CAAC;QAC7B,CAAC,CAAC;aACD,KAAK,EAAE,CAAC;QAEX,MAAM,WAAW,GAAG,UAAU,CAAC,GAAG,CAAC,eAAe,CAAC,UAAU,CAAC,CAAC;QAE/D,SAAS,WAAW,CAClB,KAA8C,EAC9C,OAAwB,EACxB,QAAiB,EACjB,cAAkC,EAClC,EAAW;YAEX,MAAM,IAAI,GAAsB;gBAC9B,KAAK,EAAE,KAAK,CAAC,IAAI,CAAC,CAAC,CAAC,KAAK,CAAC,IAAI,CAAC,CAAC,CAAE,KAAa;gBAC/C,OAAO,EAAE,OAAO;gBAChB,QAAQ,EAAE,QAAQ;aACnB,CAAC;YACF,IAAI,cAAc;gBAAE,IAAI,CAAC,SAAS,GAAG,cAAc,CAAC;YACpD,IAAI,EAAE;gBAAE,IAAI,CAAC,IAAI,GAAG,EAAE,CAAC;YACvB,OAAO,KAAK,CACV,IAAI,CAAC,eAAe,CAAC,SAAS,CAAC,EAC/B,IAAI,CAAC;gBACH,KAAK,CAAC,IAAI,CAAC,CAAC,CAAC,KAAK,CAAC,IAAI,CAAC,CAAC,CAAE,KAAa;gBACxC,MAAM,CAAC,IAAI;gBACX,MAAM,CAAC,IAAI;gBACX,MAAM,CAAC,IAAI;aACZ,CAAC,EACF,QAAQ,CAAC,gBAAgB,EAAE,IAAI,CAAC,EAChC,QAAQ,CAAC,gBAAgB,EAAE,IAAI,CAAC,EAChC,QAAQ,CAAC,gBAAgB,EAAE,IAAI,CAAC,EAChC,QAAQ,CAAC,GAAG,EAAE,IAAI,CAAC,EACnB,YAAY,CAAC,WAAW,EAAE,IAAI,CAAC,CAChC,CAAC;QACJ,CAAC;QAED,UAAU,CAAC,WAAW,CAAC,aAAa,CAAC;aAClC,GAAG,CAAC,WAAW,CAAC;aAChB,MAAM,CAAC;YACN,SAAS,EAAE,WAAW;SACvB,CAAC;aACD,KAAK,EAAE,CAAC;QAEX,MAAM,YAAY,GAAG,UAAU,CAAC,GAAG,CAAC,eAAe,CAAC,WAAW,CAAC,CAAC;QAEjE,SAAS,YAAY,CACnB,KAA8C,EAC9C,OAAwB,EACxB,QAAiB,EACjB,aAAkE,EAClE,EAAW;YAEX,MAAM,QAAQ,GAAsB;gBAClC,KAAK,EAAE,KAAK,CAAC,IAAI,CAAC,CAAC,CAAC,KAAK,CAAC,IAAI,CAAC,CAAC,CAAE,KAAa;gBAC/C,OAAO,EAAE,OAAO;gBAChB,QAAQ,EAAE,QAAQ;aACnB,CAAC;YACF,IAAI,aAAa;gBAAE,QAAQ,CAAC,SAAS,GAAG,aAAa,CAAC;YACtD,IAAI,EAAE;gBAAE,QAAQ,CAAC,IAAI,GAAG,EAAE,CAAC;YAC3B,OAAO,KAAK,CACV,IAAI,CAAC,eAAe,CAAC,SAAS,CAAC,EAC/B,IAAI,CAAC,CAAC,KAAuB,EAAE,MAAM,EAAE,MAAM,EAAE,MAAM,CAAC,CAAC,EACvD,QAAQ,CAAC,iBAAiB,EAAE,QAAQ,CAAC,EACrC,QAAQ,CAAC,iBAAiB,EAAE,QAAQ,CAAC,EACrC,QAAQ,CAAC,iBAAiB,EAAE,QAAQ,CAAC,EACrC,QAAQ,CAAC,GAAG,EAAE,QAAQ,CAAC,EACvB,YAAY,CAAC,YAAY,EAAE,QAAQ,CAAC,CACrC,CAAC;QACJ,CAAC;QAED,UAAU,CAAC,GAAG,CAAC,YAAY,CAAC;aACzB,MAAM,CAAC;YACN,SAAS,EAAE,YAAY;SACxB,CAAC;aACD,KAAK,EAAE,CAAC;IACb,CAAC;;AAGH,qBAAqB,CAAC,UAAU,EAAE,CAAC;AACnC,OAAO,CAAC,UAAU,CAAC,aAAa,CAAC,CAAC","sourcesContent":["import { CouchDBAdapter, CouchDBKeys, MangoQuery } from \"@decaf-ts/for-couchdb\";\nimport {\n  Constructor,\n  Decoration,\n  list,\n  Model,\n  prop,\n  propMetadata,\n  required,\n  type,\n} from \"@decaf-ts/decorator-validation\";\nimport { FabricContractFlags } from \"./types\";\nimport { FabricContractContext } from \"./ContractContext\";\nimport {\n  afterAny,\n  Context,\n  DBKeys,\n  InternalError,\n  modelToTransient,\n  NotFoundError,\n  onCreate,\n  onCreateUpdate,\n  onDelete,\n  onUpdate,\n  OperationKeys,\n  readonly,\n  SerializationError,\n} from \"@decaf-ts/db-decorators\";\nimport {\n  Context as Ctx,\n  Object as FabricObject,\n  Property as FabricProperty,\n} from \"fabric-contract-api\";\nimport { Logger, Logging } from \"@decaf-ts/logging\";\nimport { ContractLogger } from \"./logging\";\nimport {\n  OrderDirection,\n  PersistenceKeys,\n  RelationsMetadata,\n  Repository,\n  Sequence,\n  sequenceNameForModel,\n  SequenceOptions,\n  UnsupportedError,\n  index,\n  NumericSequence,\n  Adapter,\n  CascadeMetadata,\n  JoinColumnOptions,\n  oneToManyOnUpdate,\n  JoinTableOptions,\n  JoinTableMultipleColumnsOptions,\n} from \"@decaf-ts/core\";\nimport { FabricContractRepository } from \"./FabricContractRepository\";\nimport {\n  ChaincodeStub,\n  ClientIdentity,\n  Iterators,\n  StateQueryResponse,\n} from \"fabric-shim-api\";\nimport { FabricStatement } from \"./FabricContractStatement\";\nimport { FabricContractSequence } from \"./FabricContractSequence\";\nimport { MissingContextError } from \"../shared/errors\";\nimport { FabricFlavour } from \"../shared/constants\";\nimport { SimpleDeterministicSerializer } from \"../shared/SimpleDeterministicSerializer\";\nimport {\n  oneToManyOnCreate,\n  oneToManyOnDelete,\n  oneToOneOnCreate,\n  oneToOneOnDelete,\n  oneToOneOnUpdate,\n  populate as pop,\n} from \"./FabricConstruction\";\nimport { apply } from \"@decaf-ts/reflection\";\n\n/**\n * @description Sets the creator or updater field in a model based on the user in the context\n * @summary Callback function used in decorators to automatically set the created_by or updated_by fields\n * with the username from the context when a document is created or updated\n * @template M - Type extending Model\n * @template R - Type extending NanoRepository<M>\n * @template V - Type extending RelationsMetadata\n * @param {R} this - The repository instance\n * @param {FabricContractContext} context - The operation context containing user information\n * @param {V} data - The relation metadata\n * @param {string} key - The property key to set with the username\n * @param {M} model - The model instance being created or updated\n * @return {Promise<void>} A promise that resolves when the operation is complete\n * @function createdByOnFabricCreateUpdate\n * @memberOf module:fabric.contracts\n * @mermaid\n * sequenceDiagram\n *   participant F as createdByOnNanoCreateUpdate\n *   participant C as Context\n *   participant M as Model\n *   F->>C: get(\"user\")\n *   C-->>F: user object\n *   F->>M: set key to user.name\n *   Note over F: If no user in context\n *   F-->>F: throw UnsupportedError\n */\nexport async function createdByOnFabricCreateUpdate<\n  M extends Model,\n  R extends FabricContractRepository<M>,\n  V extends RelationsMetadata,\n>(\n  this: R,\n  context: Context<FabricContractFlags>,\n  data: V,\n  key: keyof M,\n  model: M\n): Promise<void> {\n  try {\n    const user = context.get(\"clientIdentity\") as ClientIdentity;\n    model[key] = user.getID() as M[typeof key];\n    // eslint-disable-next-line @typescript-eslint/no-unused-vars\n  } catch (e: unknown) {\n    throw new UnsupportedError(\n      \"No User found in context. Please provide a user in the context\"\n    );\n  }\n}\n\n/**\n * @description Primary key auto-assignment callback for Fabric models\n * @summary Generates and assigns a primary key value to the specified model property using a Fabric-backed sequence when the model is created. If the sequence name is not provided in options, it is derived from the model via sequenceNameForModel. The assigned key is defined as non-writable and enumerable.\n * @template M - Type extending Model for the target instance\n * @template R - Type extending FabricContractRepository for repository context\n * @template V - Type extending SequenceOptions to configure sequence behavior\n * @template F - Type extending FabricContractFlags for contextual flags\n * @param {R} this - The repository instance invoking the callback\n * @param {FabricContractContext} context - Fabric contract context containing invocation metadata\n * @param {V} data - Sequence options used to configure or locate the sequence\n * @param {string} key - The primary key property name to assign on the model\n * @param {M} model - The model instance to receive the generated primary key\n * @return {Promise<void>} Resolves when the key is assigned or when no action is required\n * @function pkFabricOnCreate\n * @memberOf module:for-fabric.contracts\n * @mermaid\n * sequenceDiagram\n *   participant R as Repository\n *   participant C as Context<F>\n *   participant S as FabricContractDBSequence\n *   participant M as Model\n *   R->>R: derive sequence name if missing\n *   R->>S: adapter.Sequence(options)\n *   S-->>R: sequence instance\n *   R->>S: next(context)\n *   S-->>R: next value\n *   R->>M: define non-writable primary key\n */\nexport async function pkFabricOnCreate<\n  M extends Model,\n  R extends FabricContractRepository<M>,\n  V extends SequenceOptions,\n  F extends FabricContractFlags,\n>(\n  this: R,\n  context: Context<F>,\n  data: V,\n  key: keyof M,\n  model: M\n): Promise<void> {\n  if (!data.type || model[key]) {\n    return;\n  }\n\n  const setPrimaryKeyValue = function <M extends Model>(\n    target: M,\n    propertyKey: string,\n    value: string | number | bigint\n  ) {\n    Object.defineProperty(target, propertyKey, {\n      enumerable: true,\n      writable: false,\n      configurable: true,\n      value: value,\n    });\n  };\n  if (!data.name) data.name = sequenceNameForModel(model, \"pk\");\n  let sequence: FabricContractSequence;\n  try {\n    sequence = (await this.adapter.Sequence(data)) as FabricContractSequence;\n  } catch (e: any) {\n    throw new InternalError(\n      `Failed to instantiate Sequence ${data.name}: ${e}`\n    );\n  }\n\n  const next = await sequence.next(context as FabricContractContext);\n  setPrimaryKeyValue(model, key as string, next);\n}\n\n/**\n * @description Adapter for Hyperledger Fabric chaincode state database operations\n * @summary Provides a CouchDB-like interface for interacting with the Fabric state database from within a chaincode contract\n * @template void - No configuration needed for contract adapter\n * @template FabricContractFlags - Flags specific to Fabric contract operations\n * @template FabricContractContext - Context type for Fabric contract operations\n * @class FabricContractAdapter\n * @example\n * ```typescript\n * // In a Fabric chaincode contract class\n * import { FabricContractAdapter } from '@decaf-ts/for-fabric';\n *\n * export class MyContract extends Contract {\n *   private adapter = new FabricContractAdapter();\n *\n *   @Transaction()\n *   async createAsset(ctx: Context, id: string, data: string): Promise<void> {\n *     const model = { id, data, timestamp: Date.now() };\n *     await this.adapter.create('assets', id, model, {}, { stub: ctx.stub });\n *   }\n * }\n * ```\n * @mermaid\n * sequenceDiagram\n *   participant Contract\n *   participant FabricContractAdapter\n *   participant Stub\n *   participant StateDB\n *\n *   Contract->>FabricContractAdapter: create(tableName, id, model, transient, ctx)\n *   FabricContractAdapter->>FabricContractAdapter: Serialize model to JSON\n *   FabricContractAdapter->>Stub: putState(id, serializedData)\n *   Stub->>StateDB: Write data\n *   StateDB-->>Stub: Success\n *   Stub-->>FabricContractAdapter: Success\n *   FabricContractAdapter-->>Contract: model\n */\nexport class FabricContractAdapter extends CouchDBAdapter<\n  any,\n  void,\n  FabricContractFlags,\n  FabricContractContext\n> {\n  protected override getClient(): void {\n    throw new UnsupportedError(\"Client is not supported in Fabric contracts\");\n  }\n  /**\n   * @description Text decoder for converting binary data to strings\n   */\n  private static textDecoder = new TextDecoder(\"utf8\");\n\n  protected static readonly serializer = new SimpleDeterministicSerializer();\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: Ctx): ContractLogger {\n    return Logging.for(FabricContractAdapter, {}, ctx) as ContractLogger;\n  }\n\n  /**\n   * @description Context constructor for this adapter\n   * @summary Overrides the base Context constructor with FabricContractContext\n   */\n  override Context = FabricContractContext;\n\n  /**\n   * @description Gets the repository constructor for this adapter\n   * @summary Returns the FabricContractRepository constructor for creating repositories\n   * @template M - Type extending Model\n   * @return {Constructor<Repository<M, MangoQuery, FabricContractAdapter, FabricContractFlags, FabricContractContext>>} The repository constructor\n   */\n  override repository<M extends Model>(): Constructor<\n    Repository<\n      M,\n      MangoQuery,\n      Adapter<any, any, MangoQuery, FabricContractFlags, FabricContractContext>,\n      FabricContractFlags,\n      FabricContractContext\n    >\n  > {\n    return FabricContractRepository;\n  }\n\n  /**\n   * @description Creates a new FabricContractAdapter instance\n   * @summary Initializes an adapter for interacting with the Fabric state database\n   * @param {void} scope - Not used in this adapter\n   * @param {string} [alias] - Optional alias for the adapter instance\n   */\n  constructor(scope: void, alias?: string) {\n    super(scope, FabricFlavour, alias);\n  }\n\n  override for(config: Partial<any>, ...args: any): typeof this {\n    return super.for(config, ...args);\n  }\n\n  /**\n   * @description Creates a record in the state database\n   * @summary Serializes a model and stores it in the Fabric state database\n   * @param {string} tableName - The name of the table/collection\n   * @param {string | number} id - The record identifier\n   * @param {Record<string, any>} model - The record data\n   * @param {Record<string, any>} transient - Transient data (not used in this implementation)\n   * @param {...any[]} args - Additional arguments, including the chaincode stub and logger\n   * @return {Promise<Record<string, any>>} Promise resolving to the created record\n   */\n  override async create(\n    tableName: string,\n    id: string | number,\n    model: Record<string, any>,\n    ...args: any[]\n  ): Promise<Record<string, any>> {\n    const { stub, logger } = args.pop();\n    const log = logger.for(this.create);\n\n    try {\n      log.info(`adding entry to ${tableName} table with pk ${id}`);\n      model = await this.putState(stub, id.toString(), model);\n    } catch (e: unknown) {\n      throw this.parseError(e as Error);\n    }\n\n    return model;\n  }\n\n  /**\n   * @description Reads a record from the state database\n   * @summary Retrieves and deserializes a record from the Fabric state database\n   * @param {string} tableName - The name of the table/collection\n   * @param {string | number} id - The record identifier\n   * @param {...any[]} args - Additional arguments, including the chaincode stub and logger\n   * @return {Promise<Record<string, any>>} Promise resolving to the retrieved record\n   */\n  override async read(\n    tableName: string,\n    id: string | number,\n    ...args: any[]\n  ): Promise<Record<string, any>> {\n    const { stub, logger } = args.pop();\n    const log = logger.for(this.read);\n\n    let model: Record<string, any>;\n    try {\n      const results = await this.readState(stub, tableName, id.toString());\n\n      if (results.length < 1) {\n        log.debug(`No record found for id ${id} in ${tableName} table`);\n        throw new NotFoundError(\n          `No record found for id ${id} in ${tableName} table`\n        );\n      } else if (results.length < 2) {\n        log.debug(`No record found for id ${id} in ${tableName} table`);\n        model = results.pop() as Record<string, any>;\n      } else {\n        model = this.mergeModels(results);\n      }\n    } catch (e: unknown) {\n      throw this.parseError(e as Error);\n    }\n\n    return model;\n  }\n\n  /**\n   * @description Updates a record in the state database\n   * @summary Serializes a model and updates it in the Fabric state database\n   * @param {string} tableName - The name of the table/collection\n   * @param {string | number} id - The record identifier\n   * @param {Record<string, any>} model - The updated record data\n   * @param {Record<string, any>} transient - Transient data (not used in this implementation)\n   * @param {...any[]} args - Additional arguments, including the chaincode stub and logger\n   * @return {Promise<Record<string, any>>} Promise resolving to the updated record\n   */\n  override async update(\n    tableName: string,\n    id: string | number,\n    model: Record<string, any>,\n    ...args: any[]\n  ): Promise<Record<string, any>> {\n    const { stub, logger } = args.pop();\n    const log = logger.for(this.update);\n\n    try {\n      log.info(`updating entry to ${tableName} table with pk ${id}`);\n      model = await this.putState(stub, id.toString(), model);\n    } catch (e: unknown) {\n      throw this.parseError(e as Error);\n    }\n\n    return model;\n  }\n\n  /**\n   * @description Deletes a record from the state database\n   * @summary Retrieves a record and then removes it from the Fabric state database\n   * @param {string} tableName - The name of the table/collection\n   * @param {string | number} id - The record identifier to delete\n   * @param {...any[]} args - Additional arguments, including the chaincode stub and logger\n   * @return {Promise<Record<string, any>>} Promise resolving to the deleted record\n   */\n  async delete(\n    tableName: string,\n    id: string | number,\n    ...args: any[]\n  ): Promise<Record<string, any>> {\n    const ctx = args.pop();\n    const { stub, logger } = ctx;\n    const log = logger.for(this.delete);\n\n    args.push(ctx);\n\n    let model: Record<string, any>;\n    try {\n      model = await this.read(tableName, id, ...args);\n      log.verbose(`deleting entry with pk ${id} from ${tableName} table`);\n      this.deleteState(stub, tableName, id.toString());\n    } catch (e: unknown) {\n      throw this.parseError(e as Error);\n    }\n\n    return model;\n  }\n\n  protected async deleteState(\n    stub: ChaincodeStub,\n    tableName: string,\n    id: string,\n    // eslint-disable-next-line @typescript-eslint/no-unused-vars\n    ...args: any[]\n  ) {\n    const composedKey = stub.createCompositeKey(tableName, [String(id)]);\n    await stub.deleteState(composedKey);\n  }\n\n  protected async putState(\n    stub: ChaincodeStub,\n    id: string,\n    model: Record<string, any>,\n    // eslint-disable-next-line @typescript-eslint/no-unused-vars\n    ...args: any[]\n  ) {\n    let data: Buffer;\n\n    try {\n      data = Buffer.from(\n        FabricContractAdapter.serializer.serialize(model as Model)\n      );\n    } catch (e: unknown) {\n      throw new SerializationError(\n        `Failed to serialize record with id ${id}: ${e}`\n      );\n    }\n    await stub.putState(id.toString(), data);\n\n    return model;\n  }\n\n  protected async readState(\n    stub: ChaincodeStub,\n    tableName: string,\n    id: string,\n    // eslint-disable-next-line @typescript-eslint/no-unused-vars\n    ...args: any[]\n  ) {\n    const composedKey = stub.createCompositeKey(tableName, [String(id)]);\n    const results: any[] = [];\n\n    let res: Buffer | Record<string, any> = await stub.getState(composedKey);\n\n    if (res.toString() === \"\")\n      throw new NotFoundError(`Record with id ${id} not found`);\n\n    try {\n      res = FabricContractAdapter.serializer.deserialize(\n        res.toString()\n        // model.constructor.name\n      );\n    } catch (e: unknown) {\n      throw new SerializationError(`Failed to parse record: ${e}`);\n    }\n\n    results.push(res);\n\n    return results;\n  }\n\n  protected async queryResult(\n    stub: ChaincodeStub,\n    rawInput: any,\n    // eslint-disable-next-line @typescript-eslint/no-unused-vars\n    ...args: any[]\n  ): Promise<Iterators.StateQueryIterator> {\n    return (await stub.getQueryResult(\n      JSON.stringify(rawInput)\n    )) as Iterators.StateQueryIterator;\n  }\n\n  protected async queryResultPaginated(\n    stub: ChaincodeStub,\n    rawInput: any,\n    limit: number = 250,\n    skip?: number,\n    // eslint-disable-next-line @typescript-eslint/no-unused-vars\n    ...args: any[]\n  ): Promise<StateQueryResponse<Iterators.StateQueryIterator>> {\n    return (await stub.getQueryResultWithPagination(\n      JSON.stringify(rawInput),\n      limit,\n      skip?.toString()\n    )) as StateQueryResponse<Iterators.StateQueryIterator>;\n  }\n\n  protected mergeModels(results: Record<string, any>[]): Record<string, any> {\n    const extract = (model: Record<string, any>) =>\n      Object.entries(model).reduce((accum: Record<string, any>, [key, val]) => {\n        if (typeof val !== \"undefined\") accum[key] = val;\n        return accum;\n      }, {});\n\n    let finalModel: Record<string, any> = results.pop() as Record<string, any>;\n\n    for (const res of results) {\n      finalModel = Object.assign({}, extract(finalModel), extract(res));\n    }\n\n    return finalModel;\n  }\n\n  /**\n   * @description Decodes binary data to string\n   * @summary Converts a Uint8Array to a string using UTF-8 encoding\n   * @param {Uint8Array} buffer - The binary data to decode\n   * @return {string} The decoded string\n   */\n  protected decode(buffer: Uint8Array) {\n    return FabricContractAdapter.textDecoder.decode(buffer);\n  }\n\n  /**\n   * @description Creates operation flags for Fabric contract operations\n   * @summary Merges default flags with Fabric-specific context information\n   * @template M - Type extending Model\n   * @param {OperationKeys} operation - The operation being performed\n   * @param {Constructor<M>} model - The model constructor\n   * @param {Partial<FabricContractFlags>} flags - Partial flags to merge with defaults\n   * @param {Ctx} ctx - The Fabric chaincode context\n   * @return {FabricContractFlags} The merged flags\n   */\n  protected override async flags<M extends Model>(\n    operation: OperationKeys,\n    model: Constructor<M>,\n    flags: Partial<FabricContractFlags>,\n    ctx: Ctx,\n    ...args: any[]\n  ): Promise<FabricContractFlags> {\n    return Object.assign(await super.flags(operation, model, flags, ...args), {\n      stub: ctx.stub,\n      identity: ctx.clientIdentity,\n      logger: this.logFor(ctx),\n    });\n  }\n\n  /**\n   * @description Creates an index for a model\n   * @summary This method is not implemented for Fabric contracts and returns a resolved promise\n   * @template M - Type extending Model\n   * @param {Constructor<M>} models - The model constructor\n   * @return {Promise<void>} Promise that resolves immediately\n   */\n  // eslint-disable-next-line @typescript-eslint/no-unused-vars\n  protected index<M>(models: Constructor<M>): Promise<void> {\n    return Promise.resolve(undefined);\n  }\n\n  /**\n   * @description Processes results from a state query iterator\n   * @summary Iterates through query results and converts them to a structured format\n   * @param {Logger} log - Logger instance for debugging\n   * @param {Iterators.StateQueryIterator} iterator - The state query iterator\n   * @param {boolean} [isHistory=false] - Whether this is a history query\n   * @return {Promise<any[]>} Promise resolving to an array of processed results\n   * @mermaid\n   * sequenceDiagram\n   *   participant Caller\n   *   participant ResultIterator\n   *   participant Iterator\n   *\n   *   Caller->>ResultIterator: resultIterator(log, iterator, isHistory)\n   *   loop Until done\n   *     ResultIterator->>Iterator: next()\n   *     Iterator-->>ResultIterator: { value, done }\n   *     alt Has value\n   *       ResultIterator->>ResultIterator: Process value based on isHistory\n   *       ResultIterator->>ResultIterator: Add to results array\n   *     end\n   *   end\n   *   ResultIterator->>Iterator: close()\n   *   ResultIterator-->>Caller: allResults\n   */\n  protected async resultIterator(\n    log: Logger,\n    iterator: Iterators.StateQueryIterator,\n    isHistory = false\n  ) {\n    const allResults = [];\n    let res: { value: any; done: boolean } = await iterator.next();\n    while (!res.done) {\n      if (res.value && res.value.value.toString()) {\n        let jsonRes: any = {};\n        log.debug(res.value.value.toString(\"utf8\"));\n        if (isHistory /* && isHistory === true*/) {\n          jsonRes.TxId = res.value.txId;\n          jsonRes.Timestamp = res.value.timestamp;\n          try {\n            jsonRes.Value = JSON.parse(res.value.value.toString(\"utf8\"));\n          } catch (err: any) {\n            log.error(err);\n            jsonRes.Value = res.value.value.toString(\"utf8\");\n          }\n        } else {\n          try {\n            jsonRes = JSON.parse(res.value.value.toString(\"utf8\"));\n          } catch (err: any) {\n            log.error(err);\n            jsonRes = res.value.value.toString(\"utf8\");\n          }\n        }\n        allResults.push(jsonRes);\n      }\n      res = await iterator.next();\n    }\n    log.debug(`Closing iterator after ${allResults.length} results`);\n    iterator.close(); // purposely not await. let iterator close on its own\n    return allResults;\n  }\n\n  /**\n   * @description Executes a raw query against the state database\n   * @summary Performs a rich query using CouchDB syntax against the Fabric state database\n   * @template R - The return type\n   * @param {MangoQuery} rawInput - The Mango Query to execute\n   * @param {boolean} docsOnly - Whether to return only documents (not used in this implementation)\n   * @param {...any[]} args - Additional arguments, including the chaincode stub and logger\n   * @return {Promise<R>} Promise resolving to the query results\n   * @mermaid\n   * sequenceDiagram\n   *   participant Caller\n   *   participant FabricContractAdapter\n   *   participant Stub\n   *   participant StateDB\n   *\n   *   Caller->>FabricContractAdapter: raw(rawInput, docsOnly, ctx)\n   *   FabricContractAdapter->>FabricContractAdapter: Extract limit and skip\n   *   alt With pagination\n   *     FabricContractAdapter->>Stub: getQueryResultWithPagination(query, limit, skip)\n   *   else Without pagination\n   *     FabricContractAdapter->>Stub: getQueryResult(query)\n   *   end\n   *   Stub->>StateDB: Execute query\n   *   StateDB-->>Stub: Iterator\n   *   Stub-->>FabricContractAdapter: Iterator\n   *   FabricContractAdapter->>FabricContractAdapter: resultIterator(log, iterator)\n   *   FabricContractAdapter-->>Caller: results\n   */\n  async raw<R>(\n    rawInput: MangoQuery,\n    docsOnly: boolean,\n    ...args: any[]\n  ): Promise<R> {\n    const { stub, logger } = args.pop();\n    const log = logger.for(this.raw);\n    const { skip, limit } = rawInput;\n    let iterator: Iterators.StateQueryIterator;\n    if (limit || skip) {\n      delete rawInput[\"limit\"];\n      delete rawInput[\"skip\"];\n      log.debug(\n        `Retrieving paginated iterator: limit: ${limit}/ skip: ${skip}`\n      );\n      const response: StateQueryResponse<Iterators.StateQueryIterator> =\n        (await this.queryResultPaginated(\n          stub,\n          rawInput,\n          limit || 250,\n          (skip as any)?.toString()\n        )) as StateQueryResponse<Iterators.StateQueryIterator>;\n      iterator = response.iterator;\n    } else {\n      log.debug(\"Retrieving iterator\");\n      iterator = (await this.queryResult(\n        stub,\n        rawInput\n      )) as Iterators.StateQueryIterator;\n    }\n    log.debug(\"Iterator acquired\");\n\n    const results = (await this.resultIterator(log, iterator)) as R;\n    log.debug(\n      `returning {0} results`,\n      `${Array.isArray(results) ? results.length : 1}`\n    );\n    return results;\n  }\n\n  override Statement<M extends Model>(\n    ctx?: FabricContractContext\n  ): FabricStatement<M, any> {\n    if (!ctx) {\n      throw new MissingContextError(\"Context is required\");\n    }\n    return new FabricStatement(this, ctx);\n  }\n\n  override async Sequence(options: SequenceOptions): Promise<Sequence> {\n    return new FabricContractSequence(options, this);\n  }\n\n  override async createAll(\n    tableName: string,\n    id: (string | number)[],\n    model: Record<string, any>[],\n    ...args: any[]\n  ): Promise<Record<string, any>[]> {\n    if (id.length !== model.length)\n      throw new InternalError(\"Ids and models must have the same length\");\n\n    const { logger } = args[args.length - 1] as FabricContractContext;\n    const log = logger.for(this.createAll);\n    log.info(`Creating ${id.length} entries ${tableName} table`);\n    log.debug(`pks: ${id}`);\n\n    return Promise.all(\n      id.map(async (i, index) => {\n        return this.create(tableName, i, model[index], ...args);\n      })\n    );\n  }\n\n  override async updateAll(\n    tableName: string,\n    id: string[] | number[],\n    model: Record<string, any>[],\n    ...args: any[]\n  ): Promise<Record<string, any>[]> {\n    if (id.length !== model.length)\n      throw new InternalError(\"Ids and models must have the same length\");\n\n    const { logger } = args[args.length - 1] as FabricContractContext;\n\n    const log = logger.for(this.createAll);\n    log.info(`Updating ${id.length} entries ${tableName} table`);\n    log.debug(`pks: ${id}`);\n\n    return Promise.all(\n      id.map(async (i, index) => {\n        return this.update(tableName, i, model[index], ...args);\n      })\n    );\n  }\n\n  /**\n   *\n   * @param model\n   * @param {string} pk\n   * @param args\n   */\n  override prepare<M extends Model>(\n    model: M,\n    pk: keyof M,\n    ...args: any[]\n  ): {\n    record: Record<string, any>;\n    id: string;\n    transient?: Record<string, any>;\n  } {\n    const { stub, logger } = args.pop();\n    const tableName = args.shift();\n    const log = logger.for(this.prepare);\n\n    const split = modelToTransient(model);\n    const result = Object.entries(split.model).reduce(\n      (accum: Record<string, any>, [key, val]) => {\n        if (typeof val === \"undefined\") return accum;\n        const mappedProp = Repository.column(model, key);\n        if (this.isReserved(mappedProp))\n          throw new InternalError(`Property name ${mappedProp} is reserved`);\n        accum[mappedProp] = val;\n        return accum;\n      },\n      {}\n    );\n    if ((model as any)[PersistenceKeys.METADATA]) {\n      log.silly(\n        `Passing along persistence metadata for ${(model as any)[PersistenceKeys.METADATA]}`\n      );\n      Object.defineProperty(result, PersistenceKeys.METADATA, {\n        enumerable: false,\n        writable: false,\n        configurable: true,\n        value: (model as any)[PersistenceKeys.METADATA],\n      });\n    }\n\n    log.info(`Preparing record for ${tableName} table with pk ${model[pk]}`);\n\n    return {\n      record: result,\n      id: stub.createCompositeKey(tableName, [String(model[pk])]),\n      transient: split.transient,\n    };\n  }\n\n  override revert<M extends Model>(\n    obj: Record<string, any>,\n    clazz: string | Constructor<M>,\n    pk: keyof M,\n    id: string | number,\n    transient?: Record<string, any>\n  ): M {\n    const log = this.log.for(this.revert);\n    const ob: Record<string, any> = {};\n    ob[pk as string] = id;\n    const m = (\n      typeof clazz === \"string\" ? Model.build(ob, clazz) : new clazz(ob)\n    ) as M;\n    log.silly(`Rebuilding model ${m.constructor.name} id ${id}`);\n    const metadata = obj[PersistenceKeys.METADATA];\n    const result = Object.keys(m).reduce((accum: M, key) => {\n      (accum as Record<string, any>)[key] = obj[Repository.column(accum, key)];\n      return accum;\n    }, m);\n\n    if (transient) {\n      log.verbose(\n        `re-adding transient properties: ${Object.keys(transient).join(\", \")}`\n      );\n      Object.entries(transient).forEach(([key, val]) => {\n        if (key in result && (result as any)[key] !== undefined)\n          throw new InternalError(\n            `Transient property ${key} already exists on model ${m.constructor.name}. should be impossible`\n          );\n        result[key as keyof M] = val;\n      });\n    }\n\n    if (metadata) {\n      log.silly(\n        `Passing along ${this.flavour} persistence metadata for ${m.constructor.name} id ${id}: ${metadata}`\n      );\n      Object.defineProperty(result, PersistenceKeys.METADATA, {\n        enumerable: false,\n        configurable: false,\n        writable: false,\n        value: metadata,\n      });\n    }\n\n    return result;\n  }\n\n  override createPrefix(\n    tableName: string,\n    id: string | number,\n    model: Record<string, any>,\n    ...args: any[]\n  ) {\n    const ctx: FabricContractContext = args.pop();\n    const record: Record<string, any> = {};\n    record[CouchDBKeys.TABLE] = tableName;\n    // record[CouchDBKeys.ID] = this.generateId(tableName, id);\n    Object.assign(record, model);\n    return [tableName, id, record, ctx];\n  }\n\n  override updatePrefix(\n    tableName: string,\n    id: string | number,\n    model: Record<string, any>,\n    ...args: any[]\n  ): (string | number | Record<string, any>)[] {\n    const ctx: FabricContractContext = args.pop();\n    const record: Record<string, any> = {};\n    record[CouchDBKeys.TABLE] = tableName;\n    // record[CouchDBKeys.ID] = this.generateId(tableName, id);\n    Object.assign(record, model);\n    return [tableName, id, record, ctx];\n  }\n\n  protected override createAllPrefix(\n    tableName: string,\n    ids: string[] | number[],\n    models: Record<string, any>[],\n    ...args: any[]\n  ): (string | string[] | number[] | Record<string, any>[])[] {\n    if (ids.length !== models.length)\n      throw new InternalError(\"Ids and models must have the same length\");\n\n    const ctx: FabricContractContext = args.pop();\n\n    const records = ids.map((id, count) => {\n      const record: Record<string, any> = {};\n      record[CouchDBKeys.TABLE] = tableName;\n      // record[CouchDBKeys.ID] = this.generateId(tableName, id);\n      Object.assign(record, models[count]);\n      return record;\n    });\n    return [tableName, ids, records, ctx as any];\n  }\n\n  protected override updateAllPrefix(\n    tableName: string,\n    ids: string[] | number[],\n    models: Record<string, any>[],\n    ...args: any[]\n  ) {\n    if (ids.length !== models.length)\n      throw new InternalError(\"Ids and models must have the same length\");\n\n    const ctx: FabricContractContext = args.pop();\n\n    const records = ids.map((id, count) => {\n      const record: Record<string, any> = {};\n      record[CouchDBKeys.TABLE] = tableName;\n      Object.assign(record, models[count]);\n      return record;\n    });\n    return [tableName, ids, records, ctx as any];\n  }\n\n  /**\n   * @description Static method for decoration overrides\n   * @summary Overrides/extends decaf decoration with Fabric-specific functionality\n   * @static\n   * @override\n   * @return {void}\n   */\n  static override decoration(): void {\n    super.decoration();\n    const createdByKey = Repository.key(PersistenceKeys.CREATED_BY);\n    const updatedByKey = Repository.key(PersistenceKeys.UPDATED_BY);\n    Decoration.flavouredAs(FabricFlavour)\n      .for(createdByKey)\n      .define(\n        onCreate(createdByOnFabricCreateUpdate),\n        propMetadata(createdByKey, {})\n      )\n      .apply();\n\n    Decoration.flavouredAs(FabricFlavour)\n      .for(updatedByKey)\n      .define(\n        onCreateUpdate(createdByOnFabricCreateUpdate),\n        propMetadata(updatedByKey, {})\n      )\n      .apply();\n\n    const pkKey = Repository.key(DBKeys.ID);\n    Decoration.flavouredAs(FabricFlavour)\n      .for(pkKey)\n      .define(\n        index([OrderDirection.ASC, OrderDirection.DSC]),\n        required(),\n        readonly(),\n        // type([String.name, Number.name, BigInt.name]),\n        propMetadata(pkKey, NumericSequence),\n        onCreate(pkFabricOnCreate, NumericSequence)\n      )\n      .apply();\n\n    const columnKey = Adapter.key(PersistenceKeys.COLUMN);\n    Decoration.flavouredAs(FabricFlavour)\n      .for(columnKey)\n      .extend(FabricProperty())\n      .apply();\n\n    const tableKey = Adapter.key(PersistenceKeys.TABLE);\n    Decoration.flavouredAs(FabricFlavour)\n      .for(tableKey)\n      .extend(function table(obj: any) {\n        // const chain: any[] = [];\n\n        // let current = obj;\n\n        // do {\n        //   chain.push(current);\n        //   console.log(`Found class: ${current}`);\n        // } while (current && current !== Object.prototype);\n\n        // do {\n        //   current = chain.pop();\n        //   console.log(`Applying @Object() to class: ${current}`);\n        //   //TODO: THIS IS NOT WORKING AND THROWS ERROR\n        //   // FabricObject()(current);\n        // } while (chain.length > 1);\n\n        return FabricObject()(obj);\n      })\n      .apply();\n\n    const oneToOnekey = Repository.key(PersistenceKeys.ONE_TO_ONE);\n\n    function oneToOneDec<M extends Model>(\n      clazz: Constructor<M> | (() => Constructor<M>),\n      cascade: CascadeMetadata,\n      populate: boolean,\n      joinColumnOpts?: JoinColumnOptions,\n      fk?: string\n    ) {\n      const meta: RelationsMetadata = {\n        class: clazz.name ? clazz.name : (clazz as any),\n        cascade: cascade,\n        populate: populate,\n      };\n      if (joinColumnOpts) meta.joinTable = joinColumnOpts;\n      if (fk) meta.name = fk;\n      return apply(\n        prop(PersistenceKeys.RELATIONS),\n        type([\n          clazz.name ? clazz.name : (clazz as any),\n          String.name,\n          Number.name,\n          BigInt.name,\n        ]),\n        onCreate(oneToOneOnCreate, meta),\n        onUpdate(oneToOneOnUpdate, meta),\n        onDelete(oneToOneOnDelete, meta),\n        afterAny(pop, meta),\n        propMetadata(oneToOnekey, meta)\n      );\n    }\n\n    Decoration.flavouredAs(FabricFlavour)\n      .for(oneToOnekey)\n      .define({\n        decorator: oneToOneDec,\n      })\n      .apply();\n\n    const oneToManyKey = Repository.key(PersistenceKeys.ONE_TO_MANY);\n\n    function oneToManyDec<M extends Model>(\n      clazz: Constructor<M> | (() => Constructor<M>),\n      cascade: CascadeMetadata,\n      populate: boolean,\n      joinTableOpts?: JoinTableOptions | JoinTableMultipleColumnsOptions,\n      fk?: string\n    ) {\n      const metadata: RelationsMetadata = {\n        class: clazz.name ? clazz.name : (clazz as any),\n        cascade: cascade,\n        populate: populate,\n      };\n      if (joinTableOpts) metadata.joinTable = joinTableOpts;\n      if (fk) metadata.name = fk;\n      return apply(\n        prop(PersistenceKeys.RELATIONS),\n        list([clazz as Constructor<M>, String, Number, BigInt]),\n        onCreate(oneToManyOnCreate, metadata),\n        onUpdate(oneToManyOnUpdate, metadata),\n        onDelete(oneToManyOnDelete, metadata),\n        afterAny(pop, metadata),\n        propMetadata(oneToManyKey, metadata)\n      );\n    }\n\n    Decoration.for(oneToManyKey)\n      .define({\n        decorator: oneToManyDec,\n      })\n      .apply();\n  }\n}\n\nFabricContractAdapter.decoration();\nAdapter.setCurrent(FabricFlavour);\n"]}
|