UNPKG

125 kBJavaScriptView Raw
1"use strict";
2Object.defineProperty(exports, "__esModule", { value: true });
3const tslib_1 = require("tslib");
4const client_common_1 = require("@neo-one/client-common");
5const node_core_1 = require("@neo-one/node-core");
6const utils_1 = require("@neo-one/utils");
7const bn_js_1 = require("bn.js");
8const lodash_1 = tslib_1.__importDefault(require("lodash"));
9const errors_1 = require("./errors");
10const getValidators_1 = require("./getValidators");
11const StorageCache_1 = require("./StorageCache");
12const wrapExecuteScripts_1 = require("./wrapExecuteScripts");
13class WriteBatchBlockchain {
14 constructor(options) {
15 this.settings = options.settings;
16 this.currentBlockInternal = options.currentBlock;
17 this.currentHeaderInternal = options.currentHeader;
18 this.storage = options.storage;
19 this.vm = options.vm;
20 this.getValidators = options.getValidators;
21 const output = new StorageCache_1.OutputStorageCache(() => this.storage.output);
22 this.caches = {
23 account: new StorageCache_1.ReadAllAddUpdateDeleteStorageCache({
24 name: 'account',
25 readAllStorage: () => this.storage.account,
26 update: (value, update) => value.update(update),
27 getKeyFromValue: (value) => ({ hash: value.hash }),
28 getKeyString: (key) => client_common_1.common.uInt160ToString(key.hash),
29 createAddChange: (value) => ({ type: 'account', value }),
30 createDeleteChange: (key) => ({ type: 'account', key }),
31 }),
32 accountUnspent: new StorageCache_1.ReadGetAllAddDeleteStorageCache({
33 name: 'accountUnspent',
34 readGetAllStorage: () => this.storage.accountUnspent,
35 getKeyFromValue: (value) => ({ hash: value.hash, input: value.input }),
36 getKeyString: (key) => `${client_common_1.common.uInt160ToString(key.hash)}:${client_common_1.common.uInt256ToString(key.input.hash)}:${key.input.index}`,
37 matchesPartialKey: (value, key) => client_common_1.common.uInt160Equal(value.hash, key.hash),
38 createAddChange: (value) => ({ type: 'accountUnspent', value }),
39 createDeleteChange: (key) => ({ type: 'accountUnspent', key }),
40 }),
41 accountUnclaimed: new StorageCache_1.ReadGetAllAddDeleteStorageCache({
42 name: 'accountUnclaimed',
43 readGetAllStorage: () => this.storage.accountUnclaimed,
44 getKeyFromValue: (value) => ({ hash: value.hash, input: value.input }),
45 getKeyString: (key) => `${client_common_1.common.uInt160ToString(key.hash)}:${client_common_1.common.uInt256ToString(key.input.hash)}:${key.input.index}`,
46 matchesPartialKey: (value, key) => client_common_1.common.uInt160Equal(value.hash, key.hash),
47 createAddChange: (value) => ({ type: 'accountUnclaimed', value }),
48 createDeleteChange: (key) => ({ type: 'accountUnclaimed', key }),
49 }),
50 action: new StorageCache_1.ReadGetAllAddStorageCache({
51 name: 'action',
52 readGetAllStorage: () => this.storage.action,
53 getKeyFromValue: (value) => ({
54 index: value.index,
55 }),
56 getKeyString: (key) => key.index.toString(10),
57 matchesPartialKey: (value, key) => (key.indexStart === undefined || value.index.gte(key.indexStart)) &&
58 (key.indexStop === undefined || value.index.lte(key.indexStop)),
59 createAddChange: (value) => ({ type: 'action', value }),
60 }),
61 asset: new StorageCache_1.ReadAddUpdateStorageCache({
62 name: 'asset',
63 readStorage: () => this.storage.asset,
64 update: (value, update) => value.update(update),
65 getKeyFromValue: (value) => ({ hash: value.hash }),
66 getKeyString: (key) => client_common_1.common.uInt256ToString(key.hash),
67 createAddChange: (value) => ({ type: 'asset', value }),
68 }),
69 block: new StorageCache_1.BlockLikeStorageCache({
70 name: 'block',
71 readStorage: () => ({
72 get: this.storage.block.get,
73 tryGet: this.storage.block.tryGet,
74 }),
75 createAddChange: (value) => ({ type: 'block', value }),
76 }),
77 blockData: new StorageCache_1.ReadAddStorageCache({
78 name: 'blockData',
79 readStorage: () => this.storage.blockData,
80 getKeyFromValue: (value) => ({ hash: value.hash }),
81 getKeyString: (key) => client_common_1.common.uInt256ToString(key.hash),
82 createAddChange: (value) => ({ type: 'blockData', value }),
83 }),
84 header: new StorageCache_1.BlockLikeStorageCache({
85 name: 'header',
86 readStorage: () => ({
87 get: this.storage.header.get,
88 tryGet: this.storage.header.tryGet,
89 }),
90 createAddChange: (value) => ({ type: 'header', value }),
91 }),
92 transaction: new StorageCache_1.ReadAddStorageCache({
93 name: 'transaction',
94 readStorage: () => this.storage.transaction,
95 getKeyFromValue: (value) => ({ hash: value.hash }),
96 getKeyString: (key) => client_common_1.common.uInt256ToString(key.hash),
97 createAddChange: (value) => ({ type: 'transaction', value }),
98 onAdd: async (value) => {
99 await Promise.all(value.outputs.map(async (out, index) => output.add({ hash: value.hash, index, output: out })));
100 },
101 allowDupes: true,
102 }),
103 transactionData: new StorageCache_1.ReadAddUpdateStorageCache({
104 name: 'transactionData',
105 readStorage: () => this.storage.transactionData,
106 update: (value, update) => value.update(update),
107 getKeyFromValue: (value) => ({ hash: value.hash }),
108 getKeyString: (key) => client_common_1.common.uInt256ToString(key.hash),
109 createAddChange: (value) => ({ type: 'transactionData', value }),
110 allowDupes: true,
111 }),
112 output,
113 contract: new StorageCache_1.ReadAddDeleteStorageCache({
114 name: 'contract',
115 readStorage: () => this.storage.contract,
116 getKeyFromValue: (value) => ({ hash: value.hash }),
117 getKeyString: (key) => client_common_1.common.uInt160ToString(key.hash),
118 createAddChange: (value) => ({ type: 'contract', value }),
119 createDeleteChange: (key) => ({ type: 'contract', key }),
120 }),
121 storageItem: new StorageCache_1.ReadGetAllAddUpdateDeleteStorageCache({
122 name: 'storageItem',
123 readGetAllStorage: () => this.storage.storageItem,
124 update: (value, update) => value.update(update),
125 getKeyFromValue: (value) => ({
126 hash: value.hash,
127 key: value.key,
128 }),
129 getKeyString: (key) => `${client_common_1.common.uInt160ToString(key.hash)}:${key.key.toString('hex')}`,
130 matchesPartialKey: (value, key) => (key.hash === undefined || client_common_1.common.uInt160Equal(value.hash, key.hash)) &&
131 (key.prefix === undefined || key.prefix.every((byte, idx) => value.key[idx] === byte)),
132 createAddChange: (value) => ({ type: 'storageItem', value }),
133 createDeleteChange: (key) => ({ type: 'storageItem', key }),
134 }),
135 validator: new StorageCache_1.ReadAllAddUpdateDeleteStorageCache({
136 name: 'validator',
137 readAllStorage: () => this.storage.validator,
138 getKeyFromValue: (value) => ({ publicKey: value.publicKey }),
139 getKeyString: (key) => client_common_1.common.ecPointToString(key.publicKey),
140 createAddChange: (value) => ({ type: 'validator', value }),
141 update: (value, update) => value.update(update),
142 createDeleteChange: (key) => ({ type: 'validator', key }),
143 }),
144 invocationData: new StorageCache_1.ReadAddStorageCache({
145 name: 'invocationData',
146 readStorage: () => this.storage.invocationData,
147 getKeyFromValue: (value) => ({ hash: value.hash }),
148 getKeyString: (key) => client_common_1.common.uInt256ToString(key.hash),
149 createAddChange: (value) => ({ type: 'invocationData', value }),
150 }),
151 validatorsCount: new StorageCache_1.ReadAddUpdateMetadataStorageCache({
152 name: 'validatorsCount',
153 readStorage: () => this.storage.validatorsCount,
154 createAddChange: (value) => ({ type: 'validatorsCount', value }),
155 update: (value, update) => value.update(update),
156 }),
157 };
158 this.account = this.caches.account;
159 this.accountUnspent = this.caches.accountUnspent;
160 this.accountUnclaimed = this.caches.accountUnclaimed;
161 this.action = this.caches.action;
162 this.asset = this.caches.asset;
163 this.block = this.caches.block;
164 this.blockData = this.caches.blockData;
165 this.header = this.caches.header;
166 this.transaction = this.caches.transaction;
167 this.transactionData = this.caches.transactionData;
168 this.output = this.caches.output;
169 this.contract = this.caches.contract;
170 this.storageItem = this.caches.storageItem;
171 this.validator = this.caches.validator;
172 this.invocationData = this.caches.invocationData;
173 this.validatorsCount = this.caches.validatorsCount;
174 }
175 get currentBlock() {
176 if (this.currentBlockInternal === undefined) {
177 throw new errors_1.GenesisBlockNotRegisteredError();
178 }
179 return this.currentBlockInternal;
180 }
181 get currentBlockIndex() {
182 return this.currentBlockInternal === undefined ? 0 : this.currentBlockInternal.index;
183 }
184 get currentHeader() {
185 if (this.currentHeaderInternal === undefined) {
186 throw new errors_1.GenesisBlockNotRegisteredError();
187 }
188 return this.currentHeaderInternal;
189 }
190 getChangeSet() {
191 return Object.values(this.caches).reduce((acc, cache) => acc.concat(cache.getChangeSet()), []);
192 }
193 getTrackedChangeSet() {
194 return Object.values(this.caches).reduce((acc, cache) => acc.concat(cache.getTrackedChangeSet()), []);
195 }
196 async persistBlock(block) {
197 const [maybePrevBlockData, outputContractsList] = await Promise.all([
198 block.index === 0 ? Promise.resolve(undefined) : this.blockData.get({ hash: block.previousHash }),
199 Promise.all([
200 ...new Set(block.transactions.reduce((acc, transaction) => acc.concat(transaction.outputs.map((output) => client_common_1.common.uInt160ToString(output.address))), [])),
201 ].map(async (hash) => this.contract.tryGet({ hash: client_common_1.common.stringToUInt160(hash) }))),
202 this.block.add(block),
203 this.header.add(block.header),
204 ]);
205 const prevBlockData = maybePrevBlockData === undefined
206 ? {
207 lastGlobalTransactionIndex: client_common_1.utils.NEGATIVE_ONE,
208 lastGlobalActionIndex: client_common_1.utils.NEGATIVE_ONE,
209 systemFee: client_common_1.utils.ZERO,
210 }
211 : {
212 lastGlobalTransactionIndex: maybePrevBlockData.lastGlobalTransactionIndex,
213 lastGlobalActionIndex: maybePrevBlockData.lastGlobalActionIndex,
214 systemFee: maybePrevBlockData.systemFee,
215 };
216 const outputContracts = {};
217 outputContractsList.filter(utils_1.utils.notNull).forEach((outputContract) => {
218 outputContracts[outputContract.hashHex] = outputContract;
219 });
220 const [utxo, rest] = lodash_1.default.partition(block.transactions.map((transaction, idx) => [idx, transaction]), ([idx, transaction]) => ((transaction.type === node_core_1.TransactionType.Claim && transaction instanceof node_core_1.ClaimTransaction) ||
221 (transaction.type === node_core_1.TransactionType.Contract && transaction instanceof node_core_1.ContractTransaction) ||
222 (transaction.type === node_core_1.TransactionType.Miner && transaction instanceof node_core_1.MinerTransaction)) &&
223 !transaction.outputs.some((output) => outputContracts[client_common_1.common.uInt160ToString(output.address)] !== undefined));
224 const [globalActionIndex] = await Promise.all([
225 rest.length > 0
226 ? this.persistTransactions(block, rest, prevBlockData.lastGlobalTransactionIndex, prevBlockData.lastGlobalActionIndex)
227 : Promise.resolve(prevBlockData.lastGlobalActionIndex),
228 utxo.length > 0
229 ?
230 this.persistUTXOTransactions(block, utxo, prevBlockData.lastGlobalTransactionIndex)
231 : Promise.resolve(),
232 ]);
233 await this.blockData.add(new node_core_1.BlockData({
234 hash: block.hash,
235 lastGlobalTransactionIndex: prevBlockData.lastGlobalTransactionIndex.add(new bn_js_1.BN(block.transactions.length)),
236 lastGlobalActionIndex: globalActionIndex,
237 systemFee: prevBlockData.systemFee.add(block.getSystemFee({
238 getOutput: this.output.get,
239 governingToken: this.settings.governingToken,
240 utilityToken: this.settings.utilityToken,
241 fees: this.settings.fees,
242 registerValidatorFee: this.settings.registerValidatorFee,
243 })),
244 }));
245 }
246 async persistUTXOTransactions(block, transactions, lastGlobalTransactionIndex) {
247 const inputs = [];
248 const claims = [];
249 const outputWithInputs = [];
250 for (const idxAndTransaction of transactions) {
251 const transaction = idxAndTransaction[1];
252 inputs.push(...transaction.inputs);
253 if (transaction.type === node_core_1.TransactionType.Claim && transaction instanceof node_core_1.ClaimTransaction) {
254 claims.push(...transaction.claims);
255 }
256 outputWithInputs.push(...this.getOutputWithInput(transaction));
257 }
258 await Promise.all([
259 Promise.all(transactions.map(async ([idx, transaction]) => this.transaction.add(transaction))),
260 Promise.all(transactions.map(async ([idx, transaction]) => this.transactionData.add(new node_core_1.TransactionData({
261 hash: transaction.hash,
262 startHeight: block.index,
263 blockHash: block.hash,
264 index: idx,
265 globalIndex: lastGlobalTransactionIndex.add(new bn_js_1.BN(idx + 1)),
266 })))),
267 this.updateAccounts(inputs, claims, outputWithInputs),
268 this.updateCoins(inputs, claims, block),
269 ]);
270 }
271 async persistTransactions(block, transactions, lastGlobalTransactionIndex, lastGlobalActionIndex) {
272 let globalActionIndex = lastGlobalActionIndex.add(client_common_1.utils.ONE);
273 for (const [idx, transaction] of transactions) {
274 globalActionIndex = await this.persistTransaction(block, transaction, idx, lastGlobalTransactionIndex, globalActionIndex);
275 }
276 return globalActionIndex.sub(client_common_1.utils.ONE);
277 }
278 async persistTransaction(block, transactionIn, transactionIndex, lastGlobalTransactionIndex, globalActionIndexIn) {
279 let globalActionIndex = globalActionIndexIn;
280 const transaction = transactionIn;
281 const claims = transaction.type === node_core_1.TransactionType.Claim && transaction instanceof node_core_1.ClaimTransaction ? transaction.claims : [];
282 let accountChanges = {};
283 let validatorChanges = {};
284 let validatorsCountChanges = [];
285 if (transaction.type === node_core_1.TransactionType.State && transaction instanceof node_core_1.StateTransaction) {
286 ({ accountChanges, validatorChanges, validatorsCountChanges } = await getValidators_1.getDescriptorChanges({
287 transactions: [transaction],
288 getAccount: async (hash) => this.account.tryGet({ hash }).then((account) => (account === undefined ? new node_core_1.Account({ hash }) : account)),
289 governingTokenHash: this.settings.governingToken.hashHex,
290 }));
291 }
292 await Promise.all([
293 this.transaction.add(transaction),
294 this.transactionData.add(new node_core_1.TransactionData({
295 hash: transaction.hash,
296 blockHash: block.hash,
297 startHeight: block.index,
298 index: transactionIndex,
299 globalIndex: lastGlobalTransactionIndex.add(new bn_js_1.BN(transactionIndex + 1)),
300 })),
301 this.updateAccounts(transaction.inputs, claims, this.getOutputWithInput(transaction), accountChanges),
302 this.updateCoins(transaction.inputs, claims, block),
303 this.processStateTransaction(validatorChanges, validatorsCountChanges),
304 ]);
305 if (transaction.type === node_core_1.TransactionType.Register && transaction instanceof node_core_1.RegisterTransaction) {
306 await this.asset.add(new node_core_1.Asset({
307 hash: transaction.hash,
308 type: transaction.asset.type,
309 name: transaction.asset.name,
310 amount: transaction.asset.amount,
311 precision: transaction.asset.precision,
312 owner: transaction.asset.owner,
313 admin: transaction.asset.admin,
314 issuer: transaction.asset.admin,
315 expiration: this.currentBlockIndex + 2 * 2000000,
316 isFrozen: false,
317 }));
318 }
319 else if (transaction.type === node_core_1.TransactionType.Issue && transaction instanceof node_core_1.IssueTransaction) {
320 const results = await Promise.all(Object.entries(transaction.getTransactionResults({
321 getOutput: this.output.get,
322 })));
323 await Promise.all(results.map(async ([assetHex, value]) => {
324 const hash = client_common_1.common.stringToUInt256(assetHex);
325 const asset = await this.asset.get({ hash });
326 await this.asset.update(asset, {
327 available: asset.available.add(value.neg()),
328 });
329 }));
330 }
331 else if (transaction.type === node_core_1.TransactionType.Enrollment && transaction instanceof node_core_1.EnrollmentTransaction) {
332 await this.validator.add(new node_core_1.Validator({
333 publicKey: transaction.publicKey,
334 }));
335 }
336 else if (transaction.type === node_core_1.TransactionType.Publish && transaction instanceof node_core_1.PublishTransaction) {
337 const contract = await this.contract.tryGet({
338 hash: transaction.contract.hash,
339 });
340 if (contract === undefined) {
341 await this.contract.add(transaction.contract);
342 }
343 }
344 else if (transaction.type === node_core_1.TransactionType.Invocation && transaction instanceof node_core_1.InvocationTransaction) {
345 const temporaryBlockchain = new WriteBatchBlockchain({
346 settings: this.settings,
347 currentBlock: this.currentBlockInternal,
348 currentHeader: this.currentHeader,
349 storage: this,
350 vm: this.vm,
351 getValidators: this.getValidators,
352 });
353 const migratedContractHashes = [];
354 const voteUpdates = [];
355 const actions = [];
356 const result = await wrapExecuteScripts_1.wrapExecuteScripts(async () => this.vm.executeScripts({
357 scripts: [{ code: transaction.script }],
358 blockchain: temporaryBlockchain,
359 scriptContainer: {
360 type: node_core_1.ScriptContainerType.Transaction,
361 value: transaction,
362 },
363 triggerType: node_core_1.TriggerType.Application,
364 action: {
365 blockIndex: block.index,
366 blockHash: block.hash,
367 transactionIndex,
368 transactionHash: transaction.hash,
369 },
370 gas: transaction.gas,
371 listeners: {
372 onLog: ({ message, scriptHash }) => {
373 actions.push(new node_core_1.LogAction({
374 index: globalActionIndex,
375 scriptHash,
376 message,
377 }));
378 globalActionIndex = globalActionIndex.add(client_common_1.utils.ONE);
379 },
380 onNotify: ({ args, scriptHash }) => {
381 actions.push(new node_core_1.NotificationAction({
382 index: globalActionIndex,
383 scriptHash,
384 args,
385 }));
386 globalActionIndex = globalActionIndex.add(client_common_1.utils.ONE);
387 },
388 onMigrateContract: ({ from, to }) => {
389 migratedContractHashes.push([from, to]);
390 },
391 onSetVotes: ({ address, votes }) => {
392 voteUpdates.push([address, votes]);
393 },
394 },
395 persistingBlock: block,
396 }));
397 const addActionsPromise = Promise.all(actions.map(async (action) => this.action.add(action)));
398 if (result instanceof node_core_1.InvocationResultSuccess) {
399 const assetChangeSet = temporaryBlockchain.asset.getChangeSet();
400 const assetHash = assetChangeSet
401 .map((change) => change.type === 'add' && change.change.type === 'asset' ? change.change.value.hash : undefined)
402 .find((value) => value !== undefined);
403 const contractsChangeSet = temporaryBlockchain.contract.getChangeSet();
404 const contractHashes = contractsChangeSet
405 .map((change) => change.type === 'add' && change.change.type === 'contract' ? change.change.value.hash : undefined)
406 .filter(utils_1.utils.notNull);
407 const deletedContractHashes = contractsChangeSet
408 .map((change) => change.type === 'delete' && change.change.type === 'contract' ? change.change.key.hash : undefined)
409 .filter(utils_1.utils.notNull);
410 const storageChanges = temporaryBlockchain.storageItem
411 .getChangeSet()
412 .map((change) => {
413 const addChange = change.type === 'add' && change.change.type === 'storageItem'
414 ? { value: change.change.value, subType: change.subType }
415 : undefined;
416 if (addChange !== undefined) {
417 const options = {
418 hash: addChange.value.hash,
419 key: addChange.value.key,
420 value: addChange.value.value,
421 };
422 return addChange.subType === 'add' ? new node_core_1.StorageChangeAdd(options) : new node_core_1.StorageChangeModify(options);
423 }
424 const deleteChange = change.type === 'delete' && change.change.type === 'storageItem' ? change.change.key : undefined;
425 if (deleteChange !== undefined) {
426 return new node_core_1.StorageChangeDelete(deleteChange);
427 }
428 return undefined;
429 })
430 .filter(utils_1.utils.notNull);
431 temporaryBlockchain.getTrackedChangeSet().forEach((change) => {
432 this.caches[change.type].addTrackedChange(change.key, change.value);
433 });
434 await Promise.all([
435 this.invocationData.add(new node_core_1.InvocationData({
436 hash: transaction.hash,
437 assetHash,
438 contractHashes,
439 deletedContractHashes,
440 migratedContractHashes,
441 voteUpdates,
442 blockIndex: block.index,
443 transactionIndex,
444 actionIndexStart: globalActionIndexIn,
445 actionIndexStop: globalActionIndex,
446 result,
447 storageChanges,
448 })),
449 addActionsPromise,
450 ]);
451 }
452 else {
453 await Promise.all([
454 this.invocationData.add(new node_core_1.InvocationData({
455 hash: transaction.hash,
456 assetHash: undefined,
457 contractHashes: [],
458 deletedContractHashes: [],
459 migratedContractHashes: [],
460 voteUpdates: [],
461 blockIndex: block.index,
462 transactionIndex,
463 actionIndexStart: globalActionIndexIn,
464 actionIndexStop: globalActionIndex,
465 result,
466 storageChanges: [],
467 })),
468 addActionsPromise,
469 ]);
470 }
471 }
472 return globalActionIndex;
473 }
474 async processStateTransaction(validatorChanges, validatorsCountChanges) {
475 const validatorsCount = await this.validatorsCount.tryGet();
476 const validatorsCountVotes = validatorsCount === undefined ? [] : [...validatorsCount.votes];
477 for (const [index, value] of validatorsCountChanges.entries()) {
478 validatorsCountVotes[index] = value;
479 }
480 await Promise.all([
481 Promise.all(Object.entries(validatorChanges).map(async ([publicKeyHex, { registered, votes }]) => {
482 const publicKey = client_common_1.common.hexToECPoint(publicKeyHex);
483 const validator = await this.validator.tryGet({ publicKey });
484 if (validator === undefined) {
485 await this.validator.add(new node_core_1.Validator({
486 publicKey,
487 registered,
488 votes,
489 }));
490 }
491 else if (((registered !== undefined && !registered) || (registered === undefined && !validator.registered)) &&
492 ((votes !== undefined && votes.eq(client_common_1.utils.ZERO)) || (votes === undefined && validator.votes.eq(client_common_1.utils.ZERO)))) {
493 await this.validator.delete({ publicKey: validator.publicKey });
494 }
495 else {
496 await this.validator.update(validator, { votes, registered });
497 }
498 })),
499 validatorsCount === undefined
500 ? this.validatorsCount.add(new node_core_1.ValidatorsCount({
501 votes: validatorsCountVotes,
502 }))
503 : (async () => {
504 await this.validatorsCount.update(validatorsCount, {
505 votes: validatorsCountVotes,
506 });
507 })(),
508 ]);
509 }
510 async updateAccounts(inputs, claims, outputs, accountChanges = {}) {
511 const [inputOutputs, claimOutputs] = await Promise.all([
512 this.getInputOutputs(inputs),
513 this.getInputOutputs(claims),
514 ]);
515 const addressValues = Object.entries(lodash_1.default.groupBy(inputOutputs
516 .map(({ output }) => [output.address, output.asset, output.value.neg()])
517 .concat(outputs.map(({ output }) => [output.address, output.asset, output.value])), ([address]) => client_common_1.common.uInt160ToHex(address)));
518 const addressSpent = this.groupByAddress(inputOutputs);
519 const addressClaimed = lodash_1.default.mapValues(this.groupByAddress(claimOutputs), (values) => values.map(({ input }) => input));
520 const addressOutputs = lodash_1.default.groupBy(outputs, (output) => client_common_1.common.uInt160ToHex(output.output.address));
521 await Promise.all(addressValues.map(async ([address, values]) => {
522 const spent = addressSpent[address];
523 const claimed = addressClaimed[address];
524 const outs = addressOutputs[address];
525 const changes = accountChanges[address];
526 await this.updateAccount(client_common_1.common.hexToUInt160(address), values.map(([_address, asset, value]) => [asset, value]), spent === undefined ? [] : spent, claimed === undefined ? [] : claimed, outs === undefined ? [] : outs, changes === undefined ? [] : changes);
527 }));
528 }
529 getOutputWithInput(transaction) {
530 return transaction.outputs.map((output, index) => ({
531 output,
532 input: new node_core_1.Input({ hash: transaction.hash, index }),
533 }));
534 }
535 async getInputOutputs(inputs) {
536 return Promise.all(inputs.map(async (input) => {
537 const output = await this.output.get(input);
538 return { input, output };
539 }));
540 }
541 groupByAddress(inputOutputs) {
542 return lodash_1.default.groupBy(inputOutputs, ({ output }) => client_common_1.common.uInt160ToHex(output.address));
543 }
544 async updateAccount(address, values, spent, claimed, outputs, votes) {
545 const account = await this.account.tryGet({ hash: address });
546 const balances = values.reduce((acc, [asset, value]) => {
547 const key = client_common_1.common.uInt256ToHex(asset);
548 if (acc[key] === undefined) {
549 acc[key] = client_common_1.utils.ZERO;
550 }
551 acc[key] = acc[key].add(value);
552 return acc;
553 }, account === undefined
554 ? {}
555 : Object.entries(account.balances).reduce((acc, [key, value]) => {
556 if (value === undefined) {
557 return Object.assign({}, acc, { [key]: client_common_1.utils.ZERO });
558 }
559 return Object.assign({}, acc, { [key]: value });
560 }, {}));
561 const promises = [];
562 promises.push(...spent.map(async ({ input }) => this.accountUnspent.delete({
563 hash: address,
564 input,
565 })));
566 promises.push(...outputs.map(async ({ input }) => this.accountUnspent.add(new node_core_1.AccountUnspent({ hash: address, input }))));
567 promises.push(...claimed.map(async (input) => this.accountUnclaimed.delete({
568 hash: address,
569 input,
570 })));
571 promises.push(...spent
572 .filter(({ output }) => client_common_1.common.uInt256Equal(output.asset, this.settings.governingToken.hash))
573 .map(async ({ input }) => this.accountUnclaimed.add(new node_core_1.AccountUnclaimed({ hash: address, input }))));
574 if (account === undefined) {
575 promises.push(this.account.add(new node_core_1.Account({
576 hash: address,
577 balances,
578 votes,
579 })));
580 }
581 else {
582 promises.push(this.account.update(account, { balances, votes }).then(async (newAccount) => {
583 if (newAccount.isDeletable()) {
584 await this.account.delete({ hash: address });
585 }
586 }));
587 }
588 await Promise.all(promises);
589 }
590 async updateCoins(inputs, claims, block) {
591 const inputClaims = inputs
592 .map((input) => ({ type: 'input', input, hash: input.hash }))
593 .concat(claims.map((input) => ({ type: 'claim', input, hash: input.hash })));
594 const hashInputClaims = Object.entries(lodash_1.default.groupBy(inputClaims, ({ hash }) => client_common_1.common.uInt256ToHex(hash)));
595 await Promise.all(hashInputClaims.map(async ([hash, values]) => this.updateCoin(client_common_1.common.hexToUInt256(hash), values, block)));
596 }
597 async updateCoin(hash, inputClaims, block) {
598 const spentCoins = await this.transactionData.get({ hash });
599 const endHeights = Object.assign({}, spentCoins.endHeights);
600 const claimed = Object.assign({}, spentCoins.claimed);
601 for (const inputClaim of inputClaims) {
602 if (inputClaim.type === 'input') {
603 endHeights[inputClaim.input.index] = block.index;
604 }
605 else {
606 claimed[inputClaim.input.index] = true;
607 }
608 }
609 await this.transactionData.update(spentCoins, {
610 endHeights,
611 claimed,
612 });
613 }
614}
615exports.WriteBatchBlockchain = WriteBatchBlockchain;
616
617//# sourceMappingURL=data:application/json;charset=utf8;base64,{"version":3,"sources":["WriteBatchBlockchain.ts"],"names":[],"mappings":"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","file":"neo-one-node-blockchain/src/WriteBatchBlockchain.js","sourcesContent":["// tslint:disable no-array-mutation no-object-mutation\nimport { common, ECPoint, UInt160, UInt256, utils } from '@neo-one/client-common';\nimport {\n  Account,\n  AccountKey,\n  AccountUnclaimed,\n  AccountUnclaimedKey,\n  AccountUnclaimedsKey,\n  AccountUnspent,\n  AccountUnspentKey,\n  AccountUnspentsKey,\n  AccountUpdate,\n  Action,\n  ActionKey,\n  ActionsKey,\n  Asset,\n  AssetKey,\n  AssetUpdate,\n  Block,\n  BlockchainStorage,\n  BlockData,\n  BlockDataKey,\n  ChangeSet,\n  ClaimTransaction,\n  Contract,\n  ContractKey,\n  ContractTransaction,\n  EnrollmentTransaction,\n  Header,\n  Input,\n  InvocationData,\n  InvocationDataKey,\n  InvocationResultSuccess,\n  InvocationTransaction,\n  IssueTransaction,\n  LogAction,\n  MinerTransaction,\n  NotificationAction,\n  Output,\n  PublishTransaction,\n  RegisterTransaction,\n  ScriptContainerType,\n  StateTransaction,\n  StorageChangeAdd,\n  StorageChangeDelete,\n  StorageChangeModify,\n  StorageItem,\n  StorageItemKey,\n  StorageItemsKey,\n  StorageItemUpdate,\n  Transaction,\n  TransactionData,\n  TransactionDataKey,\n  TransactionDataUpdate,\n  TransactionKey,\n  TransactionType,\n  TriggerType,\n  Validator,\n  ValidatorKey,\n  ValidatorsCount,\n  ValidatorsCountUpdate,\n  ValidatorUpdate,\n  VM,\n  WriteBlockchain,\n} from '@neo-one/node-core';\nimport { utils as commonUtils } from '@neo-one/utils';\nimport { BN } from 'bn.js';\nimport _ from 'lodash';\nimport { GenesisBlockNotRegisteredError } from './errors';\nimport { AccountChanges, getDescriptorChanges, ValidatorChanges, ValidatorsCountChanges } from './getValidators';\nimport {\n  BlockLikeStorageCache,\n  OutputStorageCache,\n  ReadAddDeleteStorageCache,\n  ReadAddStorageCache,\n  ReadAddUpdateMetadataStorageCache,\n  ReadAddUpdateStorageCache,\n  ReadAllAddUpdateDeleteStorageCache,\n  ReadGetAllAddDeleteStorageCache,\n  ReadGetAllAddStorageCache,\n  ReadGetAllAddUpdateDeleteStorageCache,\n  TrackedChangeSet,\n} from './StorageCache';\nimport { wrapExecuteScripts } from './wrapExecuteScripts';\n\ninterface WriteBatchBlockchainOptions {\n  readonly settings: WriteBlockchain['settings'];\n  readonly currentBlock: WriteBlockchain['currentBlock'] | undefined;\n  readonly currentHeader: WriteBlockchain['currentHeader'] | undefined;\n  readonly storage: BlockchainStorage;\n  readonly vm: VM;\n  readonly getValidators: WriteBlockchain['getValidators'];\n}\n\ninterface Caches {\n  readonly account: ReadAllAddUpdateDeleteStorageCache<AccountKey, Account, AccountUpdate>;\n  readonly accountUnspent: ReadGetAllAddDeleteStorageCache<AccountUnspentKey, AccountUnspentsKey, AccountUnspent>;\n  readonly accountUnclaimed: ReadGetAllAddDeleteStorageCache<\n    AccountUnclaimedKey,\n    AccountUnclaimedsKey,\n    AccountUnclaimed\n  >;\n  readonly action: ReadGetAllAddStorageCache<ActionKey, ActionsKey, Action>;\n  readonly asset: ReadAddUpdateStorageCache<AssetKey, Asset, AssetUpdate>;\n  readonly block: BlockLikeStorageCache<Block>;\n  readonly blockData: ReadAddStorageCache<BlockDataKey, BlockData>;\n  readonly header: BlockLikeStorageCache<Header>;\n  readonly transaction: ReadAddStorageCache<TransactionKey, Transaction>;\n  readonly transactionData: ReadAddUpdateStorageCache<TransactionDataKey, TransactionData, TransactionDataUpdate>;\n  readonly output: OutputStorageCache;\n  readonly contract: ReadAddDeleteStorageCache<ContractKey, Contract>;\n  readonly storageItem: ReadGetAllAddUpdateDeleteStorageCache<\n    StorageItemKey,\n    StorageItemsKey,\n    StorageItem,\n    StorageItemUpdate\n  >;\n  readonly validator: ReadAllAddUpdateDeleteStorageCache<ValidatorKey, Validator, ValidatorUpdate>;\n  readonly invocationData: ReadAddStorageCache<InvocationDataKey, InvocationData>;\n  readonly validatorsCount: ReadAddUpdateMetadataStorageCache<ValidatorsCount, ValidatorsCountUpdate>;\n}\n\ninterface InputClaim {\n  readonly type: 'claim' | 'input';\n  readonly hash: UInt256;\n  readonly input: Input;\n}\n\ninterface OutputWithInput {\n  readonly output: Output;\n  readonly input: Input;\n}\n\nexport class WriteBatchBlockchain {\n  public readonly settings: WriteBlockchain['settings'];\n  public readonly account: ReadAllAddUpdateDeleteStorageCache<AccountKey, Account, AccountUpdate>;\n  public readonly accountUnspent: ReadGetAllAddDeleteStorageCache<\n    AccountUnspentKey,\n    AccountUnspentsKey,\n    AccountUnspent\n  >;\n  public readonly accountUnclaimed: ReadGetAllAddDeleteStorageCache<\n    AccountUnclaimedKey,\n    AccountUnclaimedsKey,\n    AccountUnclaimed\n  >;\n  public readonly action: ReadGetAllAddStorageCache<ActionKey, ActionsKey, Action>;\n  public readonly asset: ReadAddUpdateStorageCache<AssetKey, Asset, AssetUpdate>;\n  public readonly block: BlockLikeStorageCache<Block>;\n  public readonly blockData: ReadAddStorageCache<BlockDataKey, BlockData>;\n  public readonly header: BlockLikeStorageCache<Header>;\n  public readonly transaction: ReadAddStorageCache<TransactionKey, Transaction>;\n  public readonly transactionData: ReadAddUpdateStorageCache<\n    TransactionDataKey,\n    TransactionData,\n    TransactionDataUpdate\n  >;\n  public readonly output: OutputStorageCache;\n  public readonly contract: ReadAddDeleteStorageCache<ContractKey, Contract>;\n  public readonly storageItem: ReadGetAllAddUpdateDeleteStorageCache<\n    StorageItemKey,\n    StorageItemsKey,\n    StorageItem,\n    StorageItemUpdate\n  >;\n  public readonly validator: ReadAllAddUpdateDeleteStorageCache<ValidatorKey, Validator, ValidatorUpdate>;\n  public readonly invocationData: ReadAddStorageCache<InvocationDataKey, InvocationData>;\n  public readonly validatorsCount: ReadAddUpdateMetadataStorageCache<ValidatorsCount, ValidatorsCountUpdate>;\n  public readonly getValidators: WriteBlockchain['getValidators'];\n  private readonly currentBlockInternal: WriteBlockchain['currentBlock'] | undefined;\n  private readonly currentHeaderInternal: WriteBlockchain['currentHeader'] | undefined;\n  private readonly storage: BlockchainStorage;\n  private readonly vm: VM;\n  private readonly caches: Caches;\n\n  public constructor(options: WriteBatchBlockchainOptions) {\n    this.settings = options.settings;\n    this.currentBlockInternal = options.currentBlock;\n    this.currentHeaderInternal = options.currentHeader;\n    this.storage = options.storage;\n    this.vm = options.vm;\n    this.getValidators = options.getValidators;\n\n    const output = new OutputStorageCache(() => this.storage.output);\n    this.caches = {\n      account: new ReadAllAddUpdateDeleteStorageCache({\n        name: 'account',\n        readAllStorage: () => this.storage.account,\n        update: (value, update) => value.update(update),\n        getKeyFromValue: (value) => ({ hash: value.hash }),\n        getKeyString: (key) => common.uInt160ToString(key.hash),\n        createAddChange: (value) => ({ type: 'account', value }),\n        createDeleteChange: (key) => ({ type: 'account', key }),\n      }),\n      accountUnspent: new ReadGetAllAddDeleteStorageCache({\n        name: 'accountUnspent',\n        readGetAllStorage: () => this.storage.accountUnspent,\n        getKeyFromValue: (value) => ({ hash: value.hash, input: value.input }),\n        getKeyString: (key) =>\n          `${common.uInt160ToString(key.hash)}:${common.uInt256ToString(key.input.hash)}:${key.input.index}`,\n        matchesPartialKey: (value, key) => common.uInt160Equal(value.hash, key.hash),\n        createAddChange: (value) => ({ type: 'accountUnspent', value }),\n        createDeleteChange: (key) => ({ type: 'accountUnspent', key }),\n      }),\n      accountUnclaimed: new ReadGetAllAddDeleteStorageCache({\n        name: 'accountUnclaimed',\n        readGetAllStorage: () => this.storage.accountUnclaimed,\n        getKeyFromValue: (value) => ({ hash: value.hash, input: value.input }),\n        getKeyString: (key) =>\n          `${common.uInt160ToString(key.hash)}:${common.uInt256ToString(key.input.hash)}:${key.input.index}`,\n        matchesPartialKey: (value, key) => common.uInt160Equal(value.hash, key.hash),\n        createAddChange: (value) => ({ type: 'accountUnclaimed', value }),\n        createDeleteChange: (key) => ({ type: 'accountUnclaimed', key }),\n      }),\n      action: new ReadGetAllAddStorageCache({\n        name: 'action',\n        readGetAllStorage: () => this.storage.action,\n        getKeyFromValue: (value) => ({\n          index: value.index,\n        }),\n        getKeyString: (key) => key.index.toString(10),\n        matchesPartialKey: (value, key) =>\n          (key.indexStart === undefined || value.index.gte(key.indexStart)) &&\n          (key.indexStop === undefined || value.index.lte(key.indexStop)),\n        createAddChange: (value) => ({ type: 'action', value }),\n      }),\n      asset: new ReadAddUpdateStorageCache({\n        name: 'asset',\n        readStorage: () => this.storage.asset,\n        update: (value, update) => value.update(update),\n        getKeyFromValue: (value) => ({ hash: value.hash }),\n        getKeyString: (key) => common.uInt256ToString(key.hash),\n        createAddChange: (value) => ({ type: 'asset', value }),\n      }),\n      block: new BlockLikeStorageCache({\n        name: 'block',\n        readStorage: () => ({\n          get: this.storage.block.get,\n          tryGet: this.storage.block.tryGet,\n        }),\n        createAddChange: (value) => ({ type: 'block', value }),\n      }),\n      blockData: new ReadAddStorageCache({\n        name: 'blockData',\n        readStorage: () => this.storage.blockData,\n        getKeyFromValue: (value) => ({ hash: value.hash }),\n        getKeyString: (key) => common.uInt256ToString(key.hash),\n        createAddChange: (value) => ({ type: 'blockData', value }),\n      }),\n      header: new BlockLikeStorageCache({\n        name: 'header',\n        readStorage: () => ({\n          get: this.storage.header.get,\n          tryGet: this.storage.header.tryGet,\n        }),\n        createAddChange: (value) => ({ type: 'header', value }),\n      }),\n      transaction: new ReadAddStorageCache({\n        name: 'transaction',\n        readStorage: () => this.storage.transaction,\n        getKeyFromValue: (value) => ({ hash: value.hash }),\n        getKeyString: (key) => common.uInt256ToString(key.hash),\n        createAddChange: (value) => ({ type: 'transaction', value }),\n        onAdd: async (value) => {\n          await Promise.all(\n            value.outputs.map(async (out, index) => output.add({ hash: value.hash, index, output: out })),\n          );\n        },\n        allowDupes: true,\n      }),\n      transactionData: new ReadAddUpdateStorageCache({\n        name: 'transactionData',\n        readStorage: () => this.storage.transactionData,\n        update: (value, update) => value.update(update),\n        getKeyFromValue: (value) => ({ hash: value.hash }),\n        getKeyString: (key) => common.uInt256ToString(key.hash),\n        createAddChange: (value) => ({ type: 'transactionData', value }),\n        allowDupes: true,\n      }),\n      output,\n      contract: new ReadAddDeleteStorageCache({\n        name: 'contract',\n        readStorage: () => this.storage.contract,\n        getKeyFromValue: (value) => ({ hash: value.hash }),\n        getKeyString: (key) => common.uInt160ToString(key.hash),\n        createAddChange: (value) => ({ type: 'contract', value }),\n        createDeleteChange: (key) => ({ type: 'contract', key }),\n      }),\n      storageItem: new ReadGetAllAddUpdateDeleteStorageCache({\n        name: 'storageItem',\n        readGetAllStorage: () => this.storage.storageItem,\n        update: (value, update) => value.update(update),\n        getKeyFromValue: (value) => ({\n          hash: value.hash,\n          key: value.key,\n        }),\n        getKeyString: (key) => `${common.uInt160ToString(key.hash)}:${key.key.toString('hex')}`,\n        matchesPartialKey: (value, key) =>\n          (key.hash === undefined || common.uInt160Equal(value.hash, key.hash)) &&\n          (key.prefix === undefined || key.prefix.every((byte, idx) => value.key[idx] === byte)),\n        createAddChange: (value) => ({ type: 'storageItem', value }),\n        createDeleteChange: (key) => ({ type: 'storageItem', key }),\n      }),\n      validator: new ReadAllAddUpdateDeleteStorageCache({\n        name: 'validator',\n        readAllStorage: () => this.storage.validator,\n        getKeyFromValue: (value) => ({ publicKey: value.publicKey }),\n        getKeyString: (key) => common.ecPointToString(key.publicKey),\n        createAddChange: (value) => ({ type: 'validator', value }),\n        update: (value, update) => value.update(update),\n        createDeleteChange: (key) => ({ type: 'validator', key }),\n      }),\n      invocationData: new ReadAddStorageCache({\n        name: 'invocationData',\n        readStorage: () => this.storage.invocationData,\n        getKeyFromValue: (value) => ({ hash: value.hash }),\n        getKeyString: (key) => common.uInt256ToString(key.hash),\n        createAddChange: (value) => ({ type: 'invocationData', value }),\n      }),\n      validatorsCount: new ReadAddUpdateMetadataStorageCache({\n        name: 'validatorsCount',\n        readStorage: () => this.storage.validatorsCount,\n        createAddChange: (value) => ({ type: 'validatorsCount', value }),\n        update: (value, update) => value.update(update),\n      }),\n    };\n\n    this.account = this.caches.account;\n    this.accountUnspent = this.caches.accountUnspent;\n    this.accountUnclaimed = this.caches.accountUnclaimed;\n    this.action = this.caches.action;\n    this.asset = this.caches.asset;\n    this.block = this.caches.block;\n    this.blockData = this.caches.blockData;\n    this.header = this.caches.header;\n    this.transaction = this.caches.transaction;\n    this.transactionData = this.caches.transactionData;\n    this.output = this.caches.output;\n    this.contract = this.caches.contract;\n    this.storageItem = this.caches.storageItem;\n    this.validator = this.caches.validator;\n    this.invocationData = this.caches.invocationData;\n    this.validatorsCount = this.caches.validatorsCount;\n  }\n\n  public get currentBlock(): Block {\n    if (this.currentBlockInternal === undefined) {\n      throw new GenesisBlockNotRegisteredError();\n    }\n\n    return this.currentBlockInternal;\n  }\n\n  public get currentBlockIndex(): number {\n    return this.currentBlockInternal === undefined ? 0 : this.currentBlockInternal.index;\n  }\n\n  public get currentHeader(): Header {\n    if (this.currentHeaderInternal === undefined) {\n      throw new GenesisBlockNotRegisteredError();\n    }\n\n    return this.currentHeaderInternal;\n  }\n\n  public getChangeSet(): ChangeSet {\n    return Object.values(this.caches).reduce<ChangeSet>((acc, cache) => acc.concat(cache.getChangeSet()), []);\n  }\n\n  // tslint:disable-next-line no-any\n  public getTrackedChangeSet(): TrackedChangeSet<any, any, any> {\n    // tslint:disable-next-line no-any\n    return Object.values(this.caches).reduce<TrackedChangeSet<any, any, any>>(\n      (acc, cache) => acc.concat(cache.getTrackedChangeSet()),\n      [],\n    );\n  }\n\n  public async persistBlock(block: Block): Promise<void> {\n    const [maybePrevBlockData, outputContractsList] = await Promise.all([\n      block.index === 0 ? Promise.resolve(undefined) : this.blockData.get({ hash: block.previousHash }),\n      Promise.all(\n        [\n          ...new Set(\n            block.transactions.reduce<string[]>(\n              (acc, transaction) =>\n                acc.concat(transaction.outputs.map((output) => common.uInt160ToString(output.address))),\n\n              [],\n            ),\n          ),\n        ].map(async (hash) => this.contract.tryGet({ hash: common.stringToUInt160(hash) })),\n      ),\n\n      this.block.add(block),\n      this.header.add(block.header),\n    ]);\n\n    const prevBlockData =\n      maybePrevBlockData === undefined\n        ? {\n            lastGlobalTransactionIndex: utils.NEGATIVE_ONE,\n            lastGlobalActionIndex: utils.NEGATIVE_ONE,\n            systemFee: utils.ZERO,\n          }\n        : {\n            lastGlobalTransactionIndex: maybePrevBlockData.lastGlobalTransactionIndex,\n            lastGlobalActionIndex: maybePrevBlockData.lastGlobalActionIndex,\n            systemFee: maybePrevBlockData.systemFee,\n          };\n\n    const outputContracts: { [key: string]: Contract | undefined } = {};\n    outputContractsList.filter(commonUtils.notNull).forEach((outputContract) => {\n      outputContracts[outputContract.hashHex] = outputContract;\n    });\n\n    const [utxo, rest] = _.partition(\n      block.transactions.map<[number, Transaction]>((transaction, idx) => [idx, transaction]),\n      // tslint:disable-next-line no-unused\n      ([idx, transaction]) =>\n        ((transaction.type === TransactionType.Claim && transaction instanceof ClaimTransaction) ||\n          (transaction.type === TransactionType.Contract && transaction instanceof ContractTransaction) ||\n          (transaction.type === TransactionType.Miner && transaction instanceof MinerTransaction)) &&\n        !transaction.outputs.some((output) => outputContracts[common.uInt160ToString(output.address)] !== undefined),\n    );\n\n    const [globalActionIndex] = await Promise.all([\n      rest.length > 0\n        ? this.persistTransactions(\n            block,\n            // tslint:disable-next-line no-any\n            rest as any,\n            prevBlockData.lastGlobalTransactionIndex,\n            prevBlockData.lastGlobalActionIndex,\n          )\n        : Promise.resolve(prevBlockData.lastGlobalActionIndex),\n      utxo.length > 0\n        ? // tslint:disable-next-line no-any\n          this.persistUTXOTransactions(block, utxo as any, prevBlockData.lastGlobalTransactionIndex)\n        : Promise.resolve(),\n    ]);\n\n    await this.blockData.add(\n      new BlockData({\n        hash: block.hash,\n        lastGlobalTransactionIndex: prevBlockData.lastGlobalTransactionIndex.add(new BN(block.transactions.length)),\n\n        lastGlobalActionIndex: globalActionIndex,\n        systemFee: prevBlockData.systemFee.add(\n          block.getSystemFee({\n            getOutput: this.output.get,\n            governingToken: this.settings.governingToken,\n            utilityToken: this.settings.utilityToken,\n            fees: this.settings.fees,\n            registerValidatorFee: this.settings.registerValidatorFee,\n          }),\n        ),\n      }),\n    );\n  }\n\n  private async persistUTXOTransactions(\n    block: Block,\n    transactions: ReadonlyArray<readonly [number, (ContractTransaction | ClaimTransaction | MinerTransaction)]>,\n    lastGlobalTransactionIndex: BN,\n  ): Promise<void> {\n    const inputs = [];\n    const claims = [];\n    const outputWithInputs = [];\n    // tslint:disable-next-line no-unused no-loop-statement no-dead-store\n    for (const idxAndTransaction of transactions) {\n      const transaction = idxAndTransaction[1];\n      inputs.push(...transaction.inputs);\n      if (transaction.type === TransactionType.Claim && transaction instanceof ClaimTransaction) {\n        claims.push(...transaction.claims);\n      }\n      outputWithInputs.push(...this.getOutputWithInput(transaction));\n    }\n    await Promise.all([\n      Promise.all(\n        // tslint:disable-next-line no-unused\n        transactions.map(async ([idx, transaction]) => this.transaction.add(transaction)),\n      ),\n      Promise.all(\n        transactions.map(async ([idx, transaction]) =>\n          this.transactionData.add(\n            new TransactionData({\n              hash: transaction.hash,\n              startHeight: block.index,\n              blockHash: block.hash,\n              index: idx,\n              globalIndex: lastGlobalTransactionIndex.add(new BN(idx + 1)),\n            }),\n          ),\n        ),\n      ),\n\n      this.updateAccounts(inputs, claims, outputWithInputs),\n      this.updateCoins(inputs, claims, block),\n    ]);\n  }\n\n  private async persistTransactions(\n    block: Block,\n    transactions: ReadonlyArray<readonly [number, Transaction]>,\n    lastGlobalTransactionIndex: BN,\n    lastGlobalActionIndex: BN,\n  ): Promise<BN> {\n    let globalActionIndex = lastGlobalActionIndex.add(utils.ONE);\n    // tslint:disable-next-line no-loop-statement\n    for (const [idx, transaction] of transactions) {\n      globalActionIndex = await this.persistTransaction(\n        block,\n        transaction,\n        idx,\n        lastGlobalTransactionIndex,\n        globalActionIndex,\n      );\n    }\n\n    return globalActionIndex.sub(utils.ONE);\n  }\n\n  private async persistTransaction(\n    block: Block,\n    transactionIn: Transaction,\n    transactionIndex: number,\n    lastGlobalTransactionIndex: BN,\n    globalActionIndexIn: BN,\n  ): Promise<BN> {\n    let globalActionIndex = globalActionIndexIn;\n    const transaction = transactionIn;\n    const claims =\n      transaction.type === TransactionType.Claim && transaction instanceof ClaimTransaction ? transaction.claims : [];\n    let accountChanges = {};\n    let validatorChanges = {};\n    let validatorsCountChanges: ValidatorsCountChanges = [];\n    if (transaction.type === TransactionType.State && transaction instanceof StateTransaction) {\n      ({ accountChanges, validatorChanges, validatorsCountChanges } = await getDescriptorChanges({\n        transactions: [transaction],\n        getAccount: async (hash) =>\n          this.account.tryGet({ hash }).then((account) => (account === undefined ? new Account({ hash }) : account)),\n\n        governingTokenHash: this.settings.governingToken.hashHex,\n      }));\n    }\n    await Promise.all([\n      this.transaction.add(transaction),\n      this.transactionData.add(\n        new TransactionData({\n          hash: transaction.hash,\n          blockHash: block.hash,\n          startHeight: block.index,\n          index: transactionIndex,\n          globalIndex: lastGlobalTransactionIndex.add(new BN(transactionIndex + 1)),\n        }),\n      ),\n\n      this.updateAccounts(transaction.inputs, claims, this.getOutputWithInput(transaction), accountChanges),\n\n      this.updateCoins(transaction.inputs, claims, block),\n      this.processStateTransaction(validatorChanges, validatorsCountChanges),\n    ]);\n\n    if (transaction.type === TransactionType.Register && transaction instanceof RegisterTransaction) {\n      await this.asset.add(\n        new Asset({\n          hash: transaction.hash,\n          type: transaction.asset.type,\n          name: transaction.asset.name,\n          amount: transaction.asset.amount,\n          precision: transaction.asset.precision,\n          owner: transaction.asset.owner,\n          admin: transaction.asset.admin,\n          issuer: transaction.asset.admin,\n          expiration: this.currentBlockIndex + 2 * 2000000,\n          isFrozen: false,\n        }),\n      );\n    } else if (transaction.type === TransactionType.Issue && transaction instanceof IssueTransaction) {\n      const results = await Promise.all(\n        Object.entries(\n          transaction.getTransactionResults({\n            getOutput: this.output.get,\n          }),\n        ),\n      );\n\n      await Promise.all(\n        results.map(async ([assetHex, value]) => {\n          const hash = common.stringToUInt256(assetHex);\n          const asset = await this.asset.get({ hash });\n          await this.asset.update(asset, {\n            available: asset.available.add(value.neg()),\n          });\n        }),\n      );\n    } else if (transaction.type === TransactionType.Enrollment && transaction instanceof EnrollmentTransaction) {\n      await this.validator.add(\n        new Validator({\n          publicKey: transaction.publicKey,\n        }),\n      );\n    } else if (transaction.type === TransactionType.Publish && transaction instanceof PublishTransaction) {\n      const contract = await this.contract.tryGet({\n        hash: transaction.contract.hash,\n      });\n\n      if (contract === undefined) {\n        await this.contract.add(transaction.contract);\n      }\n    } else if (transaction.type === TransactionType.Invocation && transaction instanceof InvocationTransaction) {\n      const temporaryBlockchain = new WriteBatchBlockchain({\n        settings: this.settings,\n        currentBlock: this.currentBlockInternal,\n        currentHeader: this.currentHeader,\n        // tslint:disable-next-line no-any\n        storage: this as any,\n        vm: this.vm,\n        getValidators: this.getValidators,\n      });\n\n      const migratedContractHashes: Array<readonly [UInt160, UInt160]> = [];\n      const voteUpdates: Array<readonly [UInt160, ReadonlyArray<ECPoint>]> = [];\n      const actions: Array<NotificationAction | LogAction> = [];\n      const result = await wrapExecuteScripts(async () =>\n        this.vm.executeScripts({\n          scripts: [{ code: transaction.script }],\n          blockchain: temporaryBlockchain,\n          scriptContainer: {\n            type: ScriptContainerType.Transaction,\n            value: transaction,\n          },\n\n          triggerType: TriggerType.Application,\n          action: {\n            blockIndex: block.index,\n            blockHash: block.hash,\n            transactionIndex,\n            transactionHash: transaction.hash,\n          },\n\n          gas: transaction.gas,\n          listeners: {\n            onLog: ({ message, scriptHash }) => {\n              actions.push(\n                new LogAction({\n                  index: globalActionIndex,\n                  scriptHash,\n                  message,\n                }),\n              );\n\n              globalActionIndex = globalActionIndex.add(utils.ONE);\n            },\n            onNotify: ({ args, scriptHash }) => {\n              actions.push(\n                new NotificationAction({\n                  index: globalActionIndex,\n                  scriptHash,\n                  args,\n                }),\n              );\n\n              globalActionIndex = globalActionIndex.add(utils.ONE);\n            },\n            onMigrateContract: ({ from, to }) => {\n              migratedContractHashes.push([from, to] as const);\n            },\n            onSetVotes: ({ address, votes }) => {\n              voteUpdates.push([address, votes] as const);\n            },\n          },\n\n          persistingBlock: block,\n        }),\n      );\n\n      const addActionsPromise = Promise.all(actions.map(async (action) => this.action.add(action)));\n\n      if (result instanceof InvocationResultSuccess) {\n        const assetChangeSet = temporaryBlockchain.asset.getChangeSet();\n        const assetHash = assetChangeSet\n          .map((change) =>\n            change.type === 'add' && change.change.type === 'asset' ? change.change.value.hash : undefined,\n          )\n          .find((value) => value !== undefined);\n\n        const contractsChangeSet = temporaryBlockchain.contract.getChangeSet();\n        const contractHashes = contractsChangeSet\n          .map((change) =>\n            change.type === 'add' && change.change.type === 'contract' ? change.change.value.hash : undefined,\n          )\n          .filter(commonUtils.notNull);\n\n        const deletedContractHashes = contractsChangeSet\n          .map((change) =>\n            change.type === 'delete' && change.change.type === 'contract' ? change.change.key.hash : undefined,\n          )\n          .filter(commonUtils.notNull);\n\n        const storageChanges = temporaryBlockchain.storageItem\n          .getChangeSet()\n          .map((change) => {\n            const addChange =\n              change.type === 'add' && change.change.type === 'storageItem'\n                ? { value: change.change.value, subType: change.subType }\n                : undefined;\n            if (addChange !== undefined) {\n              const options = {\n                hash: addChange.value.hash,\n                key: addChange.value.key,\n                value: addChange.value.value,\n              };\n\n              return addChange.subType === 'add' ? new StorageChangeAdd(options) : new StorageChangeModify(options);\n            }\n\n            const deleteChange =\n              change.type === 'delete' && change.change.type === 'storageItem' ? change.change.key : undefined;\n            if (deleteChange !== undefined) {\n              return new StorageChangeDelete(deleteChange);\n            }\n\n            return undefined;\n          })\n          .filter(commonUtils.notNull);\n\n        temporaryBlockchain.getTrackedChangeSet().forEach((change) => {\n          this.caches[change.type as keyof Caches].addTrackedChange(change.key, change.value);\n        });\n        await Promise.all([\n          this.invocationData.add(\n            new InvocationData({\n              hash: transaction.hash,\n              assetHash,\n              contractHashes,\n              deletedContractHashes,\n              migratedContractHashes,\n              voteUpdates,\n              blockIndex: block.index,\n              transactionIndex,\n              actionIndexStart: globalActionIndexIn,\n              actionIndexStop: globalActionIndex,\n              result,\n              storageChanges,\n            }),\n          ),\n          addActionsPromise,\n        ]);\n      } else {\n        await Promise.all([\n          this.invocationData.add(\n            new InvocationData({\n              hash: transaction.hash,\n              assetHash: undefined,\n              contractHashes: [],\n              deletedContractHashes: [],\n              migratedContractHashes: [],\n              voteUpdates: [],\n              blockIndex: block.index,\n              transactionIndex,\n              actionIndexStart: globalActionIndexIn,\n              actionIndexStop: globalActionIndex,\n              result,\n              storageChanges: [],\n            }),\n          ),\n          addActionsPromise,\n        ]);\n      }\n    }\n\n    return globalActionIndex;\n  }\n\n  private async processStateTransaction(\n    validatorChanges: ValidatorChanges,\n    validatorsCountChanges: ValidatorsCountChanges,\n  ): Promise<void> {\n    const validatorsCount = await this.validatorsCount.tryGet();\n    const validatorsCountVotes = validatorsCount === undefined ? [] : [...validatorsCount.votes];\n    // tslint:disable-next-line no-loop-statement\n    for (const [index, value] of validatorsCountChanges.entries()) {\n      validatorsCountVotes[index] = value;\n    }\n\n    await Promise.all([\n      Promise.all(\n        Object.entries(validatorChanges).map(async ([publicKeyHex, { registered, votes }]) => {\n          const publicKey = common.hexToECPoint(publicKeyHex);\n          const validator = await this.validator.tryGet({ publicKey });\n          if (validator === undefined) {\n            await this.validator.add(\n              new Validator({\n                publicKey,\n                registered,\n                votes,\n              }),\n            );\n          } else if (\n            ((registered !== undefined && !registered) || (registered === undefined && !validator.registered)) &&\n            ((votes !== undefined && votes.eq(utils.ZERO)) || (votes === undefined && validator.votes.eq(utils.ZERO)))\n          ) {\n            await this.validator.delete({ publicKey: validator.publicKey });\n          } else {\n            await this.validator.update(validator, { votes, registered });\n          }\n        }),\n      ),\n      validatorsCount === undefined\n        ? this.validatorsCount.add(\n            new ValidatorsCount({\n              votes: validatorsCountVotes,\n            }),\n          )\n        : (async () => {\n            await this.validatorsCount.update(validatorsCount, {\n              votes: validatorsCountVotes,\n            });\n          })(),\n    ]);\n  }\n\n  private async updateAccounts(\n    inputs: readonly Input[],\n    claims: readonly Input[],\n    outputs: readonly OutputWithInput[],\n    accountChanges: AccountChanges = {},\n  ): Promise<void> {\n    const [inputOutputs, claimOutputs] = await Promise.all([\n      this.getInputOutputs(inputs),\n      this.getInputOutputs(claims),\n    ]);\n\n    const addressValues = Object.entries(\n      _.groupBy(\n        inputOutputs\n          .map<[UInt160, UInt256, BN]>(({ output }) => [output.address, output.asset, output.value.neg()])\n          .concat(outputs.map<[UInt160, UInt256, BN]>(({ output }) => [output.address, output.asset, output.value])),\n        ([address]) => common.uInt160ToHex(address),\n      ),\n    );\n\n    const addressSpent = this.groupByAddress(inputOutputs);\n    const addressClaimed = _.mapValues(this.groupByAddress(claimOutputs), (values) => values.map(({ input }) => input));\n\n    const addressOutputs = _.groupBy(outputs, (output) => common.uInt160ToHex(output.output.address));\n\n    await Promise.all(\n      addressValues.map(async ([address, values]) => {\n        const spent = addressSpent[address] as readonly OutputWithInput[] | undefined;\n        const claimed = addressClaimed[address] as readonly Input[] | undefined;\n        const outs = addressOutputs[address] as readonly OutputWithInput[] | undefined;\n        const changes = accountChanges[address] as readonly ECPoint[] | undefined;\n        await this.updateAccount(\n          common.hexToUInt160(address),\n          // tslint:disable-next-line no-unused\n          values.map<readonly [UInt256, BN]>(([_address, asset, value]) => [asset, value] as const),\n          spent === undefined ? [] : spent,\n          claimed === undefined ? [] : claimed,\n          outs === undefined ? [] : outs,\n          changes === undefined ? [] : changes,\n        );\n      }),\n    );\n  }\n\n  private getOutputWithInput(transaction: Transaction): readonly OutputWithInput[] {\n    return transaction.outputs.map((output, index) => ({\n      output,\n      input: new Input({ hash: transaction.hash, index }),\n    }));\n  }\n\n  private async getInputOutputs(\n    inputs: readonly Input[],\n  ): Promise<\n    ReadonlyArray<{\n      readonly input: Input;\n      readonly output: Output;\n    }>\n  > {\n    return Promise.all(\n      inputs.map(async (input) => {\n        const output = await this.output.get(input);\n\n        return { input, output };\n      }),\n    );\n  }\n\n  private groupByAddress(\n    inputOutputs: readonly OutputWithInput[],\n  ): { readonly [key: string]: readonly OutputWithInput[] } {\n    return _.groupBy(inputOutputs, ({ output }) => common.uInt160ToHex(output.address));\n  }\n\n  private async updateAccount(\n    address: UInt160,\n    values: ReadonlyArray<readonly [UInt256, BN]>,\n    spent: readonly OutputWithInput[],\n    claimed: readonly Input[],\n    outputs: readonly OutputWithInput[],\n    votes: readonly ECPoint[],\n  ): Promise<void> {\n    const account = await this.account.tryGet({ hash: address });\n\n    const balances = values.reduce<{ [asset: string]: BN }>(\n      (acc, [asset, value]) => {\n        const key = common.uInt256ToHex(asset);\n        if ((acc[key] as BN | undefined) === undefined) {\n          acc[key] = utils.ZERO;\n        }\n        acc[key] = acc[key].add(value);\n\n        return acc;\n      },\n      account === undefined\n        ? {}\n        : Object.entries(account.balances).reduce<{ [asset: string]: BN }>((acc, [key, value]) => {\n            if (value === undefined) {\n              return {\n                ...acc,\n                [key]: utils.ZERO,\n              };\n            }\n\n            return {\n              ...acc,\n              [key]: value,\n            };\n          }, {}),\n    );\n\n    const promises = [];\n    promises.push(\n      ...spent.map(async ({ input }) =>\n        this.accountUnspent.delete({\n          hash: address,\n          input,\n        }),\n      ),\n    );\n\n    promises.push(\n      ...outputs.map(async ({ input }) => this.accountUnspent.add(new AccountUnspent({ hash: address, input }))),\n    );\n\n    promises.push(\n      ...claimed.map(async (input) =>\n        this.accountUnclaimed.delete({\n          hash: address,\n          input,\n        }),\n      ),\n    );\n\n    promises.push(\n      ...spent\n        .filter(({ output }) => common.uInt256Equal(output.asset, this.settings.governingToken.hash))\n        .map(async ({ input }) => this.accountUnclaimed.add(new AccountUnclaimed({ hash: address, input }))),\n    );\n\n    if (account === undefined) {\n      promises.push(\n        this.account.add(\n          new Account({\n            hash: address,\n            balances,\n            votes,\n          }),\n        ),\n      );\n    } else {\n      promises.push(\n        this.account.update(account, { balances, votes }).then(async (newAccount) => {\n          if (newAccount.isDeletable()) {\n            await this.account.delete({ hash: address });\n          }\n        }),\n      );\n    }\n\n    await Promise.all(promises);\n  }\n\n  private async updateCoins(inputs: readonly Input[], claims: readonly Input[], block: Block): Promise<void> {\n    const inputClaims = inputs\n      .map<InputClaim>((input) => ({ type: 'input', input, hash: input.hash }))\n      .concat(claims.map<InputClaim>((input) => ({ type: 'claim', input, hash: input.hash })));\n\n    const hashInputClaims = Object.entries(_.groupBy(inputClaims, ({ hash }) => common.uInt256ToHex(hash)));\n\n    await Promise.all(\n      hashInputClaims.map(async ([hash, values]) => this.updateCoin(common.hexToUInt256(hash), values, block)),\n    );\n  }\n\n  private async updateCoin(hash: UInt256, inputClaims: readonly InputClaim[], block: Block): Promise<void> {\n    const spentCoins = await this.transactionData.get({ hash });\n    const endHeights = { ...spentCoins.endHeights };\n    const claimed = { ...spentCoins.claimed };\n    // tslint:disable-next-line no-loop-statement\n    for (const inputClaim of inputClaims) {\n      if (inputClaim.type === 'input') {\n        endHeights[inputClaim.input.index] = block.index;\n      } else {\n        claimed[inputClaim.input.index] = true;\n      }\n    }\n\n    await this.transactionData.update(spentCoins, {\n      endHeights,\n      claimed,\n    });\n  }\n}\n"]}