Source: history.js

  1. // This Source Code Form is subject to the terms of the Mozilla Public
  2. // License, v. 2.0. If a copy of the MPL was not distributed with this
  3. // file, You can obtain one at http://mozilla.org/MPL/2.0/.
  4. //
  5. // Copyright (c) DUSK NETWORK. All rights reserved.
  6. import { getAllNotes, insertHistory, getHistory } from "./db.js";
  7. import { call } from "./wasm.js";
  8. import { parseEncodedJSON } from "./encoding.js";
  9. import { txFromBlock } from "./graphql.js";
  10. import { getPsks } from "./keys.js";
  11. import { duskToLux } from "./crypto.js";
  12. /**
  13. * Returns the maximum block height within the list of items given.
  14. * The items are expected to have a `block_height` property.
  15. *
  16. * @param {Array<{block_height: number}>} items
  17. * @returns {number} the maximum block height found, or 0.
  18. *
  19. * It's not possible to use `Math.max` because in certain engine (WebKit)
  20. * it's recursive, and it will exceed the stack if the arguments are a lot.
  21. *
  22. * @see {@link https://stackoverflow.com/questions/42623071/maximum-call-stack-size-exceeded-with-math-min-and-math-max}
  23. */
  24. const maxBlockHeight = (items) =>
  25. items.reduce(
  26. (max, { block_height }) => (block_height > max ? block_height : max),
  27. 0,
  28. );
  29. /**
  30. * @class TxData
  31. * @type {Object}
  32. * @property {number} amount The amount of the transaction
  33. * @property {number} block_height The block_height where the tx is
  34. * @property {string} direction The direction of the tx "In" or "Out"
  35. * @property {number} fee The fee of the tx
  36. * @property {string} id The hash of the tx
  37. * @property {string} tx_type The type of the tx
  38. */
  39. export function TxData(amount, block_height, direction, fee, id, tx_type) {
  40. this.amount = amount;
  41. this.block_height = block_height;
  42. this.direction = direction;
  43. this.fee = fee;
  44. this.id = id;
  45. this.tx_type = tx_type;
  46. }
  47. /**
  48. * Fetch the history of the user
  49. *
  50. * @param {WebAssembly.Exports} wasm
  51. * @param {Uint8Array} seed
  52. * @param {string} psk The psk to get the history for
  53. * @returns {Array<TxData>} The history of the transactions
  54. * @ignore Only called by the Wallet object prototype
  55. */
  56. export async function history(wasm, seed, psk) {
  57. let histData = await getHistory(psk);
  58. const lastInsertedBlockHeight = histData.lastBlockHeight;
  59. histData = histData.history;
  60. const notes = await getAllNotes(psk);
  61. const noteBlockHeights = maxBlockHeight(notes);
  62. if (lastInsertedBlockHeight >= noteBlockHeights) {
  63. return histData;
  64. }
  65. const txData = [];
  66. const noteData = [];
  67. const index = (await getPsks(wasm, seed)).indexOf(psk);
  68. for (const note of notes) {
  69. const blockHeight = note.block_height;
  70. const txs = await txFromBlock(blockHeight);
  71. txData.push({
  72. txs: txs,
  73. block_height: blockHeight,
  74. });
  75. noteData.push({
  76. pos: note.pos,
  77. psk: note.psk,
  78. note: note.note,
  79. nullifier: note.nullifier,
  80. block_height: note.block_height,
  81. });
  82. }
  83. const args = {
  84. seed: Array.from(seed),
  85. index: index,
  86. notes: noteData,
  87. tx_data: txData,
  88. };
  89. const result = await call(wasm, args, "get_history").then(parseEncodedJSON);
  90. const history = await Promise.all(
  91. result.history.map(async (tx) => {
  92. tx.fee = await duskToLux(wasm, parseInt(tx.fee));
  93. return tx;
  94. }),
  95. );
  96. const lastBlockHeight = maxBlockHeight(histData);
  97. const historyData = {
  98. psk: psk,
  99. history: history,
  100. lastBlockHeight: lastBlockHeight,
  101. };
  102. await insertHistory(historyData);
  103. return history;
  104. }