@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,257 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
3
|
+
exports.FabricCrudContract = void 0;
|
|
4
|
+
const ContractAdapter_1 = require("./../ContractAdapter.cjs");
|
|
5
|
+
const fabric_contract_api_1 = require("fabric-contract-api");
|
|
6
|
+
const core_1 = require("@decaf-ts/core");
|
|
7
|
+
const DeterministicSerializer_1 = require("./../../shared/DeterministicSerializer.cjs");
|
|
8
|
+
const logging_1 = require("@decaf-ts/logging");
|
|
9
|
+
const private_data_1 = require("./../private-data.cjs");
|
|
10
|
+
const ContractPrivateDataAdapter_1 = require("./../ContractPrivateDataAdapter.cjs");
|
|
11
|
+
/**
|
|
12
|
+
* @description Base contract class for CRUD operations in Fabric chaincode
|
|
13
|
+
* @summary Provides standard create, read, update, and delete operations for models in Fabric chaincode
|
|
14
|
+
* @template M - Type extending Model
|
|
15
|
+
* @class FabricCrudContract
|
|
16
|
+
* @extends {Contract}
|
|
17
|
+
* @example
|
|
18
|
+
* ```typescript
|
|
19
|
+
* // Define a model
|
|
20
|
+
* @table('assets')
|
|
21
|
+
* class Asset extends Model {
|
|
22
|
+
* @id()
|
|
23
|
+
* id: string;
|
|
24
|
+
*
|
|
25
|
+
* @property()
|
|
26
|
+
* data: string;
|
|
27
|
+
* }
|
|
28
|
+
*
|
|
29
|
+
* // Create a contract that extends FabricCrudContract
|
|
30
|
+
* export class AssetContract extends FabricCrudContract<Asset> {
|
|
31
|
+
* constructor() {
|
|
32
|
+
* super('AssetContract', Asset);
|
|
33
|
+
* }
|
|
34
|
+
*
|
|
35
|
+
* // Add custom methods as needed
|
|
36
|
+
* async getAssetHistory(ctx: Context, id: string): Promise<any[]> {
|
|
37
|
+
* // Custom implementation
|
|
38
|
+
* }
|
|
39
|
+
* }
|
|
40
|
+
* ```
|
|
41
|
+
* @mermaid
|
|
42
|
+
* sequenceDiagram
|
|
43
|
+
* participant Client
|
|
44
|
+
* participant Contract
|
|
45
|
+
* participant Repository
|
|
46
|
+
* participant Adapter
|
|
47
|
+
* participant StateDB
|
|
48
|
+
*
|
|
49
|
+
* Client->>Contract: create(ctx, model)
|
|
50
|
+
* Contract->>Repository: repository(ctx)
|
|
51
|
+
* Contract->>Repository: create(model, ctx)
|
|
52
|
+
* Repository->>Adapter: create(tableName, id, record, transient, ctx)
|
|
53
|
+
* Adapter->>StateDB: putState(id, serializedData)
|
|
54
|
+
* StateDB-->>Adapter: Success
|
|
55
|
+
* Adapter-->>Repository: record
|
|
56
|
+
* Repository-->>Contract: model
|
|
57
|
+
* Contract-->>Client: model
|
|
58
|
+
*/
|
|
59
|
+
class FabricCrudContract extends fabric_contract_api_1.Contract {
|
|
60
|
+
static { this.serializer = new DeterministicSerializer_1.DeterministicSerializer(); }
|
|
61
|
+
/**
|
|
62
|
+
* @description Creates a new FabricCrudContract instance
|
|
63
|
+
* @summary Initializes a contract with a name and model class
|
|
64
|
+
* @param {string} name - The name of the contract
|
|
65
|
+
* @param {Constructor<M>} clazz - The model constructor
|
|
66
|
+
*/
|
|
67
|
+
constructor(name, clazz) {
|
|
68
|
+
super(name);
|
|
69
|
+
this.clazz = clazz;
|
|
70
|
+
this.initialized = false;
|
|
71
|
+
FabricCrudContract.adapter = this.getAdapter(clazz);
|
|
72
|
+
this.repo = core_1.Repository.forModel(clazz, FabricCrudContract.adapter.alias);
|
|
73
|
+
}
|
|
74
|
+
getAdapter(clazz) {
|
|
75
|
+
const instance = new clazz();
|
|
76
|
+
if ((0, private_data_1.isModelPrivate)(instance)) {
|
|
77
|
+
const pvt = (0, private_data_1.modelToPrivate)(instance);
|
|
78
|
+
const collections = Object.keys(pvt.private);
|
|
79
|
+
return new ContractPrivateDataAdapter_1.FabricContractPrivateDataAdapter(undefined, "fabric-private-data-adapter", collections);
|
|
80
|
+
}
|
|
81
|
+
else {
|
|
82
|
+
return new ContractAdapter_1.FabricContractAdapter(undefined, "fabric-public-data-adapter");
|
|
83
|
+
}
|
|
84
|
+
}
|
|
85
|
+
/**
|
|
86
|
+
* @description Creates a logger for a specific chaincode context
|
|
87
|
+
* @summary Returns a ContractLogger instance configured for the current context
|
|
88
|
+
* @param {Ctx} ctx - The Fabric chaincode context
|
|
89
|
+
* @return {ContractLogger} The logger instance
|
|
90
|
+
*/
|
|
91
|
+
logFor(ctx) {
|
|
92
|
+
return logging_1.Logging.for(FabricCrudContract.name, {}, ctx);
|
|
93
|
+
}
|
|
94
|
+
/**
|
|
95
|
+
* @description Creates a single model in the state database
|
|
96
|
+
* @summary Delegates to the repository's create method
|
|
97
|
+
* @param {Ctx} ctx - The Fabric chaincode context
|
|
98
|
+
* @param {M} model - The model to create
|
|
99
|
+
* @param {...any[]} args - Additional arguments
|
|
100
|
+
* @return {Promise<M>} Promise resolving to the created model
|
|
101
|
+
*/
|
|
102
|
+
async create(ctx, model, ...args) {
|
|
103
|
+
const log = this.logFor(ctx).for(this.create);
|
|
104
|
+
if (typeof model === "string")
|
|
105
|
+
model = this.deserialize(model);
|
|
106
|
+
log.info(`Creating model: ${JSON.stringify(model)}`);
|
|
107
|
+
const transient = this.getTransientData(ctx);
|
|
108
|
+
log.info(`Merging transient data...`);
|
|
109
|
+
model = this.repo.merge(model, transient);
|
|
110
|
+
return this.repo.create(model, ctx, ...args);
|
|
111
|
+
}
|
|
112
|
+
/**
|
|
113
|
+
* @description Reads a single model from the state database
|
|
114
|
+
* @summary Delegates to the repository's read method
|
|
115
|
+
* @param {Ctx} ctx - The Fabric chaincode context
|
|
116
|
+
* @param {string | number} key - The key of the model to read
|
|
117
|
+
* @param {...any[]} args - Additional arguments
|
|
118
|
+
* @return {Promise<M>} Promise resolving to the retrieved model
|
|
119
|
+
*/
|
|
120
|
+
async read(ctx, key, ...args) {
|
|
121
|
+
const log = this.logFor(ctx).for(this.read);
|
|
122
|
+
log.info(`reading entry with pk ${key} `);
|
|
123
|
+
return this.repo.read(key, ctx, ...args);
|
|
124
|
+
}
|
|
125
|
+
getTransientData(ctx) {
|
|
126
|
+
const transientMap = ctx.stub.getTransient();
|
|
127
|
+
let transient = {};
|
|
128
|
+
if (transientMap.has(this.repo.tableName)) {
|
|
129
|
+
transient = JSON.parse(transientMap.get(this.repo.tableName)?.toString("utf8"));
|
|
130
|
+
}
|
|
131
|
+
return transient;
|
|
132
|
+
}
|
|
133
|
+
/**
|
|
134
|
+
* @description Updates a single model in the state database
|
|
135
|
+
* @summary Delegates to the repository's update method
|
|
136
|
+
* @param {Ctx} ctx - The Fabric chaincode context
|
|
137
|
+
* @param {M} model - The model to update
|
|
138
|
+
* @param {...any[]} args - Additional arguments
|
|
139
|
+
* @return {Promise<M>} Promise resolving to the updated model
|
|
140
|
+
*/
|
|
141
|
+
async update(ctx, model, ...args) {
|
|
142
|
+
const log = this.logFor(ctx).for(this.update);
|
|
143
|
+
if (typeof model === "string")
|
|
144
|
+
model = this.deserialize(model);
|
|
145
|
+
log.info(`Updating model: ${JSON.stringify(model)}`);
|
|
146
|
+
const transient = this.getTransientData(ctx);
|
|
147
|
+
log.info(`Merging transient data...`);
|
|
148
|
+
model = this.repo.merge(model, transient);
|
|
149
|
+
return this.repo.update(model, ctx, ...args);
|
|
150
|
+
}
|
|
151
|
+
/**
|
|
152
|
+
* @description Deletes a single model from the state database
|
|
153
|
+
* @summary Delegates to the repository's delete method
|
|
154
|
+
* @param {Ctx} ctx - The Fabric chaincode context
|
|
155
|
+
* @param {string | number} key - The key of the model to delete
|
|
156
|
+
* @param {...any[]} args - Additional arguments
|
|
157
|
+
* @return {Promise<M>} Promise resolving to the deleted model
|
|
158
|
+
*/
|
|
159
|
+
async delete(ctx, key, ...args) {
|
|
160
|
+
const log = this.logFor(ctx).for(this.delete);
|
|
161
|
+
log.info(`deleting entry with pk ${key} `);
|
|
162
|
+
return this.repo.delete(String(key), ctx, ...args);
|
|
163
|
+
}
|
|
164
|
+
/**
|
|
165
|
+
* @description Deletes multiple models from the state database
|
|
166
|
+
* @summary Delegates to the repository's deleteAll method
|
|
167
|
+
* @param {string[] | number[]} keys - The keys of the models to delete
|
|
168
|
+
* @param {Ctx} ctx - The Fabric chaincode context
|
|
169
|
+
* @param {...any[]} args - Additional arguments
|
|
170
|
+
* @return {Promise<M[]>} Promise resolving to the deleted models
|
|
171
|
+
*/
|
|
172
|
+
async deleteAll(ctx, keys, ...args) {
|
|
173
|
+
if (typeof keys === "string")
|
|
174
|
+
keys = JSON.parse(keys);
|
|
175
|
+
return this.repo.deleteAll(keys, ctx, ...args);
|
|
176
|
+
}
|
|
177
|
+
/**
|
|
178
|
+
* @description Reads multiple models from the state database
|
|
179
|
+
* @summary Delegates to the repository's readAll method
|
|
180
|
+
* @param {Ctx} ctx - The Fabric chaincode context
|
|
181
|
+
* @param {string[] | number[]} keys - The keys of the models to read
|
|
182
|
+
* @param {...any[]} args - Additional arguments
|
|
183
|
+
* @return {Promise<M[]>} Promise resolving to the retrieved models
|
|
184
|
+
*/
|
|
185
|
+
async readAll(ctx, keys, ...args) {
|
|
186
|
+
if (typeof keys === "string")
|
|
187
|
+
keys = JSON.parse(keys);
|
|
188
|
+
return this.repo.readAll(keys, ctx, ...args);
|
|
189
|
+
}
|
|
190
|
+
/**
|
|
191
|
+
* @description Updates multiple models in the state database
|
|
192
|
+
* @summary Delegates to the repository's updateAll method
|
|
193
|
+
* @param {Ctx} ctx - The Fabric chaincode context
|
|
194
|
+
* @param {M[]} models - The models to update
|
|
195
|
+
* @param {...any[]} args - Additional arguments
|
|
196
|
+
* @return {Promise<M[]>} Promise resolving to the updated models
|
|
197
|
+
*/
|
|
198
|
+
async updateAll(ctx, models, ...args) {
|
|
199
|
+
const log = this.logFor(ctx).for(this.updateAll);
|
|
200
|
+
if (typeof models === "string")
|
|
201
|
+
models = JSON.parse(models)
|
|
202
|
+
.map((m) => this.deserialize(m))
|
|
203
|
+
.map((m) => new this.clazz(m));
|
|
204
|
+
log.info(`updating ${models.length} entries to the table`);
|
|
205
|
+
return this.repo.updateAll(models, ctx, ...args);
|
|
206
|
+
}
|
|
207
|
+
/**
|
|
208
|
+
* @description Executes a raw query against the state database
|
|
209
|
+
* @summary Delegates to the repository's raw method
|
|
210
|
+
* @param {Ctx} ctx - The Fabric chaincode context
|
|
211
|
+
* @param {any} rawInput - The query to execute
|
|
212
|
+
* @param {boolean} docsOnly - Whether to return only documents
|
|
213
|
+
* @param {...any[]} args - Additional arguments
|
|
214
|
+
* @return {Promise<any>} Promise resolving to the query results
|
|
215
|
+
*/
|
|
216
|
+
async raw(ctx, rawInput, docsOnly, ...args) {
|
|
217
|
+
if (typeof rawInput === "string")
|
|
218
|
+
rawInput = JSON.parse(rawInput);
|
|
219
|
+
return this.repo.raw(rawInput, docsOnly, ctx, ...args);
|
|
220
|
+
}
|
|
221
|
+
serialize(model) {
|
|
222
|
+
return FabricCrudContract.serializer.serialize(model);
|
|
223
|
+
}
|
|
224
|
+
deserialize(str) {
|
|
225
|
+
return FabricCrudContract.serializer.deserialize(str);
|
|
226
|
+
}
|
|
227
|
+
async init(ctx) {
|
|
228
|
+
const log = this.logFor(ctx).for(this.init);
|
|
229
|
+
log.info(`Running contract initialization...`);
|
|
230
|
+
this.initialized = true;
|
|
231
|
+
log.info(`Contract initialization completed.`);
|
|
232
|
+
}
|
|
233
|
+
async healthcheck(ctx) {
|
|
234
|
+
const log = this.logFor(ctx).for(this.healthcheck);
|
|
235
|
+
log.info(`Running Healthcheck: ${this.initialized}...`);
|
|
236
|
+
return { healthcheck: this.initialized };
|
|
237
|
+
}
|
|
238
|
+
/**
|
|
239
|
+
* @description Creates multiple models in the state database
|
|
240
|
+
* @summary Delegates to the repository's createAll method
|
|
241
|
+
* @param {Ctx} ctx - The Fabric chaincode context
|
|
242
|
+
* @param {M[]} models - The models to create
|
|
243
|
+
* @param {...any[]} args - Additional arguments
|
|
244
|
+
* @return {Promise<M[]>} Promise resolving to the created models
|
|
245
|
+
*/
|
|
246
|
+
async createAll(ctx, models, ...args) {
|
|
247
|
+
const log = this.logFor(ctx).for(this.createAll);
|
|
248
|
+
if (typeof models === "string")
|
|
249
|
+
models = JSON.parse(models)
|
|
250
|
+
.map((m) => this.deserialize(m))
|
|
251
|
+
.map((m) => new this.clazz(m));
|
|
252
|
+
log.info(`adding ${models.length} entries to the table`);
|
|
253
|
+
return this.repo.createAll(models, ctx, ...args);
|
|
254
|
+
}
|
|
255
|
+
}
|
|
256
|
+
exports.FabricCrudContract = FabricCrudContract;
|
|
257
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"crud-contract.js","sourceRoot":"","sources":["../../../src/contracts/crud/crud-contract.ts"],"names":[],"mappings":"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","sourcesContent":["import { FabricContractAdapter } from \"../ContractAdapter\";\nimport { Context, Contract, Context as Ctx } from \"fabric-contract-api\";\nimport { Constructor, Model, Serializer } from \"@decaf-ts/decorator-validation\";\nimport { Repository } from \"@decaf-ts/core\";\nimport { FabricContractRepository } from \"../FabricContractRepository\";\nimport { DeterministicSerializer } from \"../../shared/DeterministicSerializer\";\nimport { MangoQuery } from \"@decaf-ts/for-couchdb\";\nimport { Checkable, healthcheck } from \"../../shared/interfaces/Checkable\";\nimport { ContractLogger } from \"../logging\";\nimport { Logging } from \"@decaf-ts/logging\";\nimport { isModelPrivate, modelToPrivate } from \"../private-data\";\nimport { FabricContractPrivateDataAdapter } from \"../ContractPrivateDataAdapter\";\n\n/**\n * @description Base contract class for CRUD operations in Fabric chaincode\n * @summary Provides standard create, read, update, and delete operations for models in Fabric chaincode\n * @template M - Type extending Model\n * @class FabricCrudContract\n * @extends {Contract}\n * @example\n * ```typescript\n * // Define a model\n * @table('assets')\n * class Asset extends Model {\n *   @id()\n *   id: string;\n *\n *   @property()\n *   data: string;\n * }\n *\n * // Create a contract that extends FabricCrudContract\n * export class AssetContract extends FabricCrudContract<Asset> {\n *   constructor() {\n *     super('AssetContract', Asset);\n *   }\n *\n *   // Add custom methods as needed\n *   async getAssetHistory(ctx: Context, id: string): Promise<any[]> {\n *     // Custom implementation\n *   }\n * }\n * ```\n * @mermaid\n * sequenceDiagram\n *   participant Client\n *   participant Contract\n *   participant Repository\n *   participant Adapter\n *   participant StateDB\n *\n *   Client->>Contract: create(ctx, model)\n *   Contract->>Repository: repository(ctx)\n *   Contract->>Repository: create(model, ctx)\n *   Repository->>Adapter: create(tableName, id, record, transient, ctx)\n *   Adapter->>StateDB: putState(id, serializedData)\n *   StateDB-->>Adapter: Success\n *   Adapter-->>Repository: record\n *   Repository-->>Contract: model\n *   Contract-->>Client: model\n */\nexport abstract class FabricCrudContract<M extends Model>\n  extends Contract\n  implements Checkable\n{\n  /**\n   * @description Shared adapter instance for all contract instances\n   */\n  protected static adapter: FabricContractAdapter;\n\n  protected readonly repo: FabricContractRepository<M>;\n\n  protected static readonly serializer = new DeterministicSerializer();\n\n  protected initialized: boolean = false;\n\n  /**\n   * @description Creates a new FabricCrudContract instance\n   * @summary Initializes a contract with a name and model class\n   * @param {string} name - The name of the contract\n   * @param {Constructor<M>} clazz - The model constructor\n   */\n  protected constructor(\n    name: string,\n    protected readonly clazz: Constructor<M>\n  ) {\n    super(name);\n    FabricCrudContract.adapter = this.getAdapter(clazz);\n    this.repo = Repository.forModel(clazz, FabricCrudContract.adapter.alias);\n  }\n\n  protected getAdapter(clazz: Constructor<M>): FabricContractAdapter {\n    const instance = new clazz();\n\n    if (isModelPrivate(instance)) {\n      const pvt = modelToPrivate(instance);\n      const collections = Object.keys(pvt.private!);\n\n      return new FabricContractPrivateDataAdapter(\n        undefined,\n        \"fabric-private-data-adapter\",\n        collections\n      );\n    } else {\n      return new FabricContractAdapter(undefined, \"fabric-public-data-adapter\");\n    }\n  }\n\n  /**\n   * @description Creates a logger for a specific chaincode context\n   * @summary Returns a ContractLogger instance configured for the current context\n   * @param {Ctx} ctx - The Fabric chaincode context\n   * @return {ContractLogger} The logger instance\n   */\n  public logFor(ctx: Context): ContractLogger {\n    return Logging.for(FabricCrudContract.name, {}, ctx) as ContractLogger;\n  }\n\n  /**\n   * @description Creates a single model in the state database\n   * @summary Delegates to the repository's create method\n   * @param {Ctx} ctx - The Fabric chaincode context\n   * @param {M} model - The model to create\n   * @param {...any[]} args - Additional arguments\n   * @return {Promise<M>} Promise resolving to the created model\n   */\n  async create(\n    ctx: Ctx,\n    model: string | M,\n    ...args: any[]\n  ): Promise<string | M> {\n    const log = this.logFor(ctx).for(this.create);\n\n    if (typeof model === \"string\") model = this.deserialize<M>(model) as any;\n\n    log.info(`Creating model: ${JSON.stringify(model)}`);\n\n    const transient = this.getTransientData(ctx);\n\n    log.info(`Merging transient data...`);\n    model = (this.repo as any).merge(model as M, transient);\n\n    return this.repo.create(model as unknown as M, ctx, ...args);\n  }\n\n  /**\n   * @description Reads a single model from the state database\n   * @summary Delegates to the repository's read method\n   * @param {Ctx} ctx - The Fabric chaincode context\n   * @param {string | number} key - The key of the model to read\n   * @param {...any[]} args - Additional arguments\n   * @return {Promise<M>} Promise resolving to the retrieved model\n   */\n  async read(\n    ctx: Ctx,\n    key: string | number,\n    ...args: any[]\n  ): Promise<M | string> {\n    const log = this.logFor(ctx).for(this.read);\n\n    log.info(`reading entry with pk ${key} `);\n\n    return this.repo.read(key, ctx, ...args);\n  }\n\n  protected getTransientData(ctx: Ctx): any {\n    const transientMap = ctx.stub.getTransient();\n    let transient: any = {};\n\n    if (transientMap.has((this.repo as any).tableName)) {\n      transient = JSON.parse(\n        (transientMap.get((this.repo as any).tableName) as Buffer)?.toString(\n          \"utf8\"\n        ) as string\n      );\n    }\n\n    return transient;\n  }\n\n  /**\n   * @description Updates a single model in the state database\n   * @summary Delegates to the repository's update method\n   * @param {Ctx} ctx - The Fabric chaincode context\n   * @param {M} model - The model to update\n   * @param {...any[]} args - Additional arguments\n   * @return {Promise<M>} Promise resolving to the updated model\n   */\n  async update(\n    ctx: Ctx,\n    model: string | M,\n    ...args: any[]\n  ): Promise<string | M> {\n    const log = this.logFor(ctx).for(this.update);\n\n    if (typeof model === \"string\") model = this.deserialize<M>(model) as any;\n\n    log.info(`Updating model: ${JSON.stringify(model)}`);\n\n    const transient = this.getTransientData(ctx);\n\n    log.info(`Merging transient data...`);\n    model = (this.repo as any).merge(model as M, transient);\n\n    return this.repo.update(model as unknown as M, ctx, ...args);\n  }\n\n  /**\n   * @description Deletes a single model from the state database\n   * @summary Delegates to the repository's delete method\n   * @param {Ctx} ctx - The Fabric chaincode context\n   * @param {string | number} key - The key of the model to delete\n   * @param {...any[]} args - Additional arguments\n   * @return {Promise<M>} Promise resolving to the deleted model\n   */\n  async delete(\n    ctx: Ctx,\n    key: string | number,\n    ...args: any[]\n  ): Promise<M | string> {\n    const log = this.logFor(ctx).for(this.delete);\n    log.info(`deleting entry with pk ${key} `);\n    return this.repo.delete(String(key), ctx, ...args);\n  }\n\n  /**\n   * @description Deletes multiple models from the state database\n   * @summary Delegates to the repository's deleteAll method\n   * @param {string[] | number[]} keys - The keys of the models to delete\n   * @param {Ctx} ctx - The Fabric chaincode context\n   * @param {...any[]} args - Additional arguments\n   * @return {Promise<M[]>} Promise resolving to the deleted models\n   */\n  async deleteAll(\n    ctx: Ctx,\n    keys: string | string[] | number[],\n    ...args: any[]\n  ): Promise<M[] | string> {\n    if (typeof keys === \"string\") keys = JSON.parse(keys) as string[];\n    return this.repo.deleteAll(keys, ctx, ...args);\n  }\n\n  /**\n   * @description Reads multiple models from the state database\n   * @summary Delegates to the repository's readAll method\n   * @param {Ctx} ctx - The Fabric chaincode context\n   * @param {string[] | number[]} keys - The keys of the models to read\n   * @param {...any[]} args - Additional arguments\n   * @return {Promise<M[]>} Promise resolving to the retrieved models\n   */\n  async readAll(\n    ctx: Ctx,\n    keys: string | string[] | number[],\n    ...args: any[]\n  ): Promise<M[] | string> {\n    if (typeof keys === \"string\") keys = JSON.parse(keys) as string[];\n    return this.repo.readAll(keys, ctx, ...args);\n  }\n\n  /**\n   * @description Updates multiple models in the state database\n   * @summary Delegates to the repository's updateAll method\n   * @param {Ctx} ctx - The Fabric chaincode context\n   * @param {M[]} models - The models to update\n   * @param {...any[]} args - Additional arguments\n   * @return {Promise<M[]>} Promise resolving to the updated models\n   */\n  async updateAll(\n    ctx: Ctx,\n    models: string | M[],\n    ...args: any[]\n  ): Promise<string | M[]> {\n    const log = this.logFor(ctx).for(this.updateAll);\n    if (typeof models === \"string\")\n      models = (JSON.parse(models) as [])\n        .map((m) => this.deserialize(m))\n        .map((m) => new this.clazz(m)) as any;\n\n    log.info(`updating ${models.length} entries to the table`);\n    return this.repo.updateAll(models as unknown as M[], ctx, ...args);\n  }\n\n  /**\n   * @description Executes a raw query against the state database\n   * @summary Delegates to the repository's raw method\n   * @param {Ctx} ctx - The Fabric chaincode context\n   * @param {any} rawInput - The query to execute\n   * @param {boolean} docsOnly - Whether to return only documents\n   * @param {...any[]} args - Additional arguments\n   * @return {Promise<any>} Promise resolving to the query results\n   */\n  async raw(\n    ctx: Ctx,\n    rawInput: MangoQuery | string,\n    docsOnly: boolean,\n    ...args: any[]\n  ): Promise<any | string> {\n    if (typeof rawInput === \"string\")\n      rawInput = JSON.parse(rawInput) as MangoQuery;\n    return this.repo.raw(rawInput, docsOnly, ctx, ...args);\n  }\n\n  protected serialize(model: M): string {\n    return FabricCrudContract.serializer.serialize(model);\n  }\n\n  protected deserialize<M extends Model>(str: string): M {\n    return (\n      FabricCrudContract.serializer as unknown as Serializer<M>\n    ).deserialize(str);\n  }\n\n  protected async init(ctx: Ctx): Promise<void> {\n    const log = this.logFor(ctx).for(this.init);\n    log.info(`Running contract initialization...`);\n    this.initialized = true;\n    log.info(`Contract initialization completed.`);\n  }\n\n  async healthcheck(ctx: Ctx): Promise<string | healthcheck> {\n    const log = this.logFor(ctx).for(this.healthcheck);\n    log.info(`Running Healthcheck: ${this.initialized}...`);\n    return { healthcheck: this.initialized };\n  }\n\n  /**\n   * @description Creates multiple models in the state database\n   * @summary Delegates to the repository's createAll method\n   * @param {Ctx} ctx - The Fabric chaincode context\n   * @param {M[]} models - The models to create\n   * @param {...any[]} args - Additional arguments\n   * @return {Promise<M[]>} Promise resolving to the created models\n   */\n  async createAll(\n    ctx: Ctx,\n    models: string | M[],\n    ...args: any[]\n  ): Promise<string | M[]> {\n    const log = this.logFor(ctx).for(this.createAll);\n\n    if (typeof models === \"string\")\n      models = (JSON.parse(models) as [])\n        .map((m) => this.deserialize(m))\n        .map((m) => new this.clazz(m)) as any;\n\n    log.info(`adding ${models.length} entries to the table`);\n    return this.repo.createAll(models as unknown as M[], ctx, ...args);\n  }\n}\n"]}
|
|
@@ -0,0 +1,168 @@
|
|
|
1
|
+
import { FabricContractAdapter } from "../ContractAdapter";
|
|
2
|
+
import { Context, Contract, Context as Ctx } from "fabric-contract-api";
|
|
3
|
+
import { Constructor, Model } from "@decaf-ts/decorator-validation";
|
|
4
|
+
import { FabricContractRepository } from "../FabricContractRepository";
|
|
5
|
+
import { DeterministicSerializer } from "../../shared/DeterministicSerializer";
|
|
6
|
+
import { MangoQuery } from "@decaf-ts/for-couchdb";
|
|
7
|
+
import { Checkable, healthcheck } from "../../shared/interfaces/Checkable";
|
|
8
|
+
import { ContractLogger } from "../logging";
|
|
9
|
+
/**
|
|
10
|
+
* @description Base contract class for CRUD operations in Fabric chaincode
|
|
11
|
+
* @summary Provides standard create, read, update, and delete operations for models in Fabric chaincode
|
|
12
|
+
* @template M - Type extending Model
|
|
13
|
+
* @class FabricCrudContract
|
|
14
|
+
* @extends {Contract}
|
|
15
|
+
* @example
|
|
16
|
+
* ```typescript
|
|
17
|
+
* // Define a model
|
|
18
|
+
* @table('assets')
|
|
19
|
+
* class Asset extends Model {
|
|
20
|
+
* @id()
|
|
21
|
+
* id: string;
|
|
22
|
+
*
|
|
23
|
+
* @property()
|
|
24
|
+
* data: string;
|
|
25
|
+
* }
|
|
26
|
+
*
|
|
27
|
+
* // Create a contract that extends FabricCrudContract
|
|
28
|
+
* export class AssetContract extends FabricCrudContract<Asset> {
|
|
29
|
+
* constructor() {
|
|
30
|
+
* super('AssetContract', Asset);
|
|
31
|
+
* }
|
|
32
|
+
*
|
|
33
|
+
* // Add custom methods as needed
|
|
34
|
+
* async getAssetHistory(ctx: Context, id: string): Promise<any[]> {
|
|
35
|
+
* // Custom implementation
|
|
36
|
+
* }
|
|
37
|
+
* }
|
|
38
|
+
* ```
|
|
39
|
+
* @mermaid
|
|
40
|
+
* sequenceDiagram
|
|
41
|
+
* participant Client
|
|
42
|
+
* participant Contract
|
|
43
|
+
* participant Repository
|
|
44
|
+
* participant Adapter
|
|
45
|
+
* participant StateDB
|
|
46
|
+
*
|
|
47
|
+
* Client->>Contract: create(ctx, model)
|
|
48
|
+
* Contract->>Repository: repository(ctx)
|
|
49
|
+
* Contract->>Repository: create(model, ctx)
|
|
50
|
+
* Repository->>Adapter: create(tableName, id, record, transient, ctx)
|
|
51
|
+
* Adapter->>StateDB: putState(id, serializedData)
|
|
52
|
+
* StateDB-->>Adapter: Success
|
|
53
|
+
* Adapter-->>Repository: record
|
|
54
|
+
* Repository-->>Contract: model
|
|
55
|
+
* Contract-->>Client: model
|
|
56
|
+
*/
|
|
57
|
+
export declare abstract class FabricCrudContract<M extends Model> extends Contract implements Checkable {
|
|
58
|
+
protected readonly clazz: Constructor<M>;
|
|
59
|
+
/**
|
|
60
|
+
* @description Shared adapter instance for all contract instances
|
|
61
|
+
*/
|
|
62
|
+
protected static adapter: FabricContractAdapter;
|
|
63
|
+
protected readonly repo: FabricContractRepository<M>;
|
|
64
|
+
protected static readonly serializer: DeterministicSerializer<Model<false>>;
|
|
65
|
+
protected initialized: boolean;
|
|
66
|
+
/**
|
|
67
|
+
* @description Creates a new FabricCrudContract instance
|
|
68
|
+
* @summary Initializes a contract with a name and model class
|
|
69
|
+
* @param {string} name - The name of the contract
|
|
70
|
+
* @param {Constructor<M>} clazz - The model constructor
|
|
71
|
+
*/
|
|
72
|
+
protected constructor(name: string, clazz: Constructor<M>);
|
|
73
|
+
protected getAdapter(clazz: Constructor<M>): FabricContractAdapter;
|
|
74
|
+
/**
|
|
75
|
+
* @description Creates a logger for a specific chaincode context
|
|
76
|
+
* @summary Returns a ContractLogger instance configured for the current context
|
|
77
|
+
* @param {Ctx} ctx - The Fabric chaincode context
|
|
78
|
+
* @return {ContractLogger} The logger instance
|
|
79
|
+
*/
|
|
80
|
+
logFor(ctx: Context): ContractLogger;
|
|
81
|
+
/**
|
|
82
|
+
* @description Creates a single model in the state database
|
|
83
|
+
* @summary Delegates to the repository's create method
|
|
84
|
+
* @param {Ctx} ctx - The Fabric chaincode context
|
|
85
|
+
* @param {M} model - The model to create
|
|
86
|
+
* @param {...any[]} args - Additional arguments
|
|
87
|
+
* @return {Promise<M>} Promise resolving to the created model
|
|
88
|
+
*/
|
|
89
|
+
create(ctx: Ctx, model: string | M, ...args: any[]): Promise<string | M>;
|
|
90
|
+
/**
|
|
91
|
+
* @description Reads a single model from the state database
|
|
92
|
+
* @summary Delegates to the repository's read method
|
|
93
|
+
* @param {Ctx} ctx - The Fabric chaincode context
|
|
94
|
+
* @param {string | number} key - The key of the model to read
|
|
95
|
+
* @param {...any[]} args - Additional arguments
|
|
96
|
+
* @return {Promise<M>} Promise resolving to the retrieved model
|
|
97
|
+
*/
|
|
98
|
+
read(ctx: Ctx, key: string | number, ...args: any[]): Promise<M | string>;
|
|
99
|
+
protected getTransientData(ctx: Ctx): any;
|
|
100
|
+
/**
|
|
101
|
+
* @description Updates a single model in the state database
|
|
102
|
+
* @summary Delegates to the repository's update method
|
|
103
|
+
* @param {Ctx} ctx - The Fabric chaincode context
|
|
104
|
+
* @param {M} model - The model to update
|
|
105
|
+
* @param {...any[]} args - Additional arguments
|
|
106
|
+
* @return {Promise<M>} Promise resolving to the updated model
|
|
107
|
+
*/
|
|
108
|
+
update(ctx: Ctx, model: string | M, ...args: any[]): Promise<string | M>;
|
|
109
|
+
/**
|
|
110
|
+
* @description Deletes a single model from the state database
|
|
111
|
+
* @summary Delegates to the repository's delete method
|
|
112
|
+
* @param {Ctx} ctx - The Fabric chaincode context
|
|
113
|
+
* @param {string | number} key - The key of the model to delete
|
|
114
|
+
* @param {...any[]} args - Additional arguments
|
|
115
|
+
* @return {Promise<M>} Promise resolving to the deleted model
|
|
116
|
+
*/
|
|
117
|
+
delete(ctx: Ctx, key: string | number, ...args: any[]): Promise<M | string>;
|
|
118
|
+
/**
|
|
119
|
+
* @description Deletes multiple models from the state database
|
|
120
|
+
* @summary Delegates to the repository's deleteAll method
|
|
121
|
+
* @param {string[] | number[]} keys - The keys of the models to delete
|
|
122
|
+
* @param {Ctx} ctx - The Fabric chaincode context
|
|
123
|
+
* @param {...any[]} args - Additional arguments
|
|
124
|
+
* @return {Promise<M[]>} Promise resolving to the deleted models
|
|
125
|
+
*/
|
|
126
|
+
deleteAll(ctx: Ctx, keys: string | string[] | number[], ...args: any[]): Promise<M[] | string>;
|
|
127
|
+
/**
|
|
128
|
+
* @description Reads multiple models from the state database
|
|
129
|
+
* @summary Delegates to the repository's readAll method
|
|
130
|
+
* @param {Ctx} ctx - The Fabric chaincode context
|
|
131
|
+
* @param {string[] | number[]} keys - The keys of the models to read
|
|
132
|
+
* @param {...any[]} args - Additional arguments
|
|
133
|
+
* @return {Promise<M[]>} Promise resolving to the retrieved models
|
|
134
|
+
*/
|
|
135
|
+
readAll(ctx: Ctx, keys: string | string[] | number[], ...args: any[]): Promise<M[] | string>;
|
|
136
|
+
/**
|
|
137
|
+
* @description Updates multiple models in the state database
|
|
138
|
+
* @summary Delegates to the repository's updateAll method
|
|
139
|
+
* @param {Ctx} ctx - The Fabric chaincode context
|
|
140
|
+
* @param {M[]} models - The models to update
|
|
141
|
+
* @param {...any[]} args - Additional arguments
|
|
142
|
+
* @return {Promise<M[]>} Promise resolving to the updated models
|
|
143
|
+
*/
|
|
144
|
+
updateAll(ctx: Ctx, models: string | M[], ...args: any[]): Promise<string | M[]>;
|
|
145
|
+
/**
|
|
146
|
+
* @description Executes a raw query against the state database
|
|
147
|
+
* @summary Delegates to the repository's raw method
|
|
148
|
+
* @param {Ctx} ctx - The Fabric chaincode context
|
|
149
|
+
* @param {any} rawInput - The query to execute
|
|
150
|
+
* @param {boolean} docsOnly - Whether to return only documents
|
|
151
|
+
* @param {...any[]} args - Additional arguments
|
|
152
|
+
* @return {Promise<any>} Promise resolving to the query results
|
|
153
|
+
*/
|
|
154
|
+
raw(ctx: Ctx, rawInput: MangoQuery | string, docsOnly: boolean, ...args: any[]): Promise<any | string>;
|
|
155
|
+
protected serialize(model: M): string;
|
|
156
|
+
protected deserialize<M extends Model>(str: string): M;
|
|
157
|
+
protected init(ctx: Ctx): Promise<void>;
|
|
158
|
+
healthcheck(ctx: Ctx): Promise<string | healthcheck>;
|
|
159
|
+
/**
|
|
160
|
+
* @description Creates multiple models in the state database
|
|
161
|
+
* @summary Delegates to the repository's createAll method
|
|
162
|
+
* @param {Ctx} ctx - The Fabric chaincode context
|
|
163
|
+
* @param {M[]} models - The models to create
|
|
164
|
+
* @param {...any[]} args - Additional arguments
|
|
165
|
+
* @return {Promise<M[]>} Promise resolving to the created models
|
|
166
|
+
*/
|
|
167
|
+
createAll(ctx: Ctx, models: string | M[], ...args: any[]): Promise<string | M[]>;
|
|
168
|
+
}
|
|
@@ -0,0 +1,19 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
var __createBinding = (this && this.__createBinding) || (Object.create ? (function(o, m, k, k2) {
|
|
3
|
+
if (k2 === undefined) k2 = k;
|
|
4
|
+
var desc = Object.getOwnPropertyDescriptor(m, k);
|
|
5
|
+
if (!desc || ("get" in desc ? !m.__esModule : desc.writable || desc.configurable)) {
|
|
6
|
+
desc = { enumerable: true, get: function() { return m[k]; } };
|
|
7
|
+
}
|
|
8
|
+
Object.defineProperty(o, k2, desc);
|
|
9
|
+
}) : (function(o, m, k, k2) {
|
|
10
|
+
if (k2 === undefined) k2 = k;
|
|
11
|
+
o[k2] = m[k];
|
|
12
|
+
}));
|
|
13
|
+
var __exportStar = (this && this.__exportStar) || function(m, exports) {
|
|
14
|
+
for (var p in m) if (p !== "default" && !Object.prototype.hasOwnProperty.call(exports, p)) __createBinding(exports, m, p);
|
|
15
|
+
};
|
|
16
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
17
|
+
__exportStar(require("./crud-contract.cjs"), exports);
|
|
18
|
+
__exportStar(require("./serialized-crud-contract.cjs"), exports);
|
|
19
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi9zcmMvY29udHJhY3RzL2NydWQvaW5kZXgudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6Ijs7Ozs7Ozs7Ozs7Ozs7OztBQUFBLHNEQUFnQztBQUNoQyxpRUFBMkMiLCJzb3VyY2VzQ29udGVudCI6WyJleHBvcnQgKiBmcm9tIFwiLi9jcnVkLWNvbnRyYWN0XCI7XG5leHBvcnQgKiBmcm9tIFwiLi9zZXJpYWxpemVkLWNydWQtY29udHJhY3RcIjtcbiJdfQ==
|