Source: db.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 { Dexie, indexedDB } from "../deps.js";
  7. import { unspentSpentNotes } from "./crypto.js";
  8. import { request, responseBytes } from "./node.js";
  9. import { getNullifiersRkyvSerialized } from "./rkyv.js";
  10. // Set Polyfill
  11. if (globalThis.indexedDB === undefined) {
  12. Dexie.dependencies.indexedDB = indexedDB;
  13. }
  14. /**
  15. * @class NoteData
  16. * @type {Object}
  17. * @property {UInt8Array} note The rkyv serialized note.
  18. * @property {UInt8Array} nullifier The rkyv serialized BlsScalar.
  19. * @property {number} pos The position of the node
  20. * @property {number} block_height The block height of the note
  21. * @property {string} psk The bs58 encoded public spend key of the note
  22. */
  23. export function NoteData(note, psk, pos, nullifier, block_height) {
  24. this.note = note;
  25. this.psk = psk;
  26. this.pos = pos;
  27. this.nullifier = nullifier;
  28. this.block_height = block_height;
  29. }
  30. /**
  31. * @class HistoryData
  32. * @type {Object}
  33. * @property {string} psk The bs58 encoded public spend key of the note
  34. * @property {Array<TxData>} history the tx data
  35. */
  36. export function HistoryData(psk, history) {
  37. this.psk = psk;
  38. this.history = history;
  39. }
  40. /**
  41. * Persist the state of unspent_notes and spent_notes in the indexedDB
  42. * This is called by the sync function
  43. *
  44. * @param {Array<NoteData>} unspent_notes Notes which are not spent
  45. * @param {Array<NoteData>} spent_notes
  46. * @param {number} pos The position we are at right now
  47. * @param {number} blockHeight The block height we are at right now
  48. * @ignore Only called by the sync function
  49. */
  50. export async function insertSpentUnspentNotes(
  51. unspentNotes,
  52. spentNotes,
  53. pos,
  54. blockHeight,
  55. ) {
  56. try {
  57. if (localStorage.getItem("lastPos") == null) {
  58. }
  59. localStorage.setItem("lastPos", Math.max(pos, getLastPos()));
  60. localStorage.setItem(
  61. "blockHeight",
  62. Math.max(blockHeight, getLastBlockHeight()),
  63. );
  64. } catch (e) {
  65. console.warn("Cannot set pos in local storage, the wallet will be slow");
  66. }
  67. const db = initializeState();
  68. await db.unspentNotes
  69. .bulkPut(unspentNotes)
  70. .then(() => db.spentNotes.bulkPut(spentNotes))
  71. .finally(() => db.close());
  72. }
  73. /**
  74. * Fetch unspent notes from the IndexedDB if there are any
  75. *
  76. * @param {string} psk - bs58 encoded public spend key to fetch the unspent notes of
  77. * @returns {Promise<Array<NoteData>>} notes - unspent notes belonging to the psk
  78. * @ignore Only called by the sync function
  79. */
  80. export async function getUnspentNotes(psk) {
  81. const db = initializeState();
  82. const myTable = db.table("unspentNotes");
  83. if (myTable) {
  84. const notes = myTable.filter((note) => note.psk == psk);
  85. const result = await notes.toArray();
  86. return result;
  87. }
  88. }
  89. /**
  90. * Fetch spent notes from the IndexedDB if there are any
  91. *
  92. * @param {string} psk bs58 encoded public spend key to fetch the unspent notes of
  93. * @returns {Array<NoteData>} spent notes of the psk
  94. * @ignore Only called by the sync function
  95. */
  96. export async function getSpentNotes(psk) {
  97. const db = initializeState();
  98. const myTable = db.table("spentNotes");
  99. if (myTable) {
  100. const notes = myTable.filter((note) => note.psk == psk);
  101. const result = await notes.toArray();
  102. return result;
  103. }
  104. }
  105. /**
  106. * Fetch lastPos from the localStorage if there is any 0 by default
  107. *
  108. * @returns {number} lastPos the position where to fetch from
  109. * @ignore Only called by the sync function
  110. */
  111. export const getLastPos = () => parseInt(localStorage.getItem("lastPos")) || 0;
  112. /**
  113. * Fetch last blockheight from the localStorage if there is any 0 by default
  114. *
  115. * @returns {number} blockHeight the last block height
  116. * @ignore Only called by the sync function
  117. */
  118. export const getLastBlockHeight = () =>
  119. parseInt(localStorage.getItem("blockHeight")) || 0;
  120. /**
  121. * Increment the lastPos by 1 if non zero
  122. * @returns {number} lastPos the position where to fetch from
  123. */
  124. export function getNextPos() {
  125. const pos = getLastPos();
  126. return pos === 0 ? pos : pos + 1;
  127. }
  128. /**
  129. * Set the lastPos in the localStorage
  130. * @param {number} position the position to set
  131. */
  132. export function setLastPos(position) {
  133. localStorage.setItem("lastPos", position);
  134. }
  135. /**
  136. * Check if the last position exists in the localstorage
  137. */
  138. export const lastPosExists = () => localStorage.getItem("lastPos") !== null;
  139. /**
  140. * Given bs58 encoded psk, fetch all the spent and unspent notes for that psk
  141. *
  142. * @param {string} psk
  143. * @returns {Array<NoteData>} spent and unspent notes of the psk contactinated
  144. * @ignore Only called by the sync function
  145. */
  146. export async function getAllNotes(psk) {
  147. const db = initializeState();
  148. const unspentNotesTable = db
  149. .table("unspentNotes")
  150. .filter((note) => note.psk == psk);
  151. const spentNotesTable = db
  152. .table("spentNotes")
  153. .filter((note) => note.psk == psk);
  154. const unspent = await unspentNotesTable.toArray();
  155. const spent = await spentNotesTable.toArray();
  156. const concat = spent.concat(unspent);
  157. return concat;
  158. }
  159. /**
  160. * Read all the unspent notes and check if they are spent from the node
  161. * If they are spent then move from unspent to spent
  162. *
  163. * @param {WebAssembly.Exports} wasm
  164. *
  165. * @returns {Promise} Promise object which resolves after the corrected notes are inserted
  166. * @ignore Only called by the sync function
  167. */
  168. export async function correctNotes(wasm) {
  169. // Move the unspent notes to spent notes if they were spent
  170. const unspentNotesNullifiers = [];
  171. const unspentNotesTemp = [];
  172. const unspentNotesPsks = [];
  173. const unspentNotesPos = [];
  174. const unspentNotesBlockHeights = [];
  175. // grab all the unspent notes and put the data of those unspent notes in arrays
  176. const allUnspentNotes = await getAllUnpsentNotes();
  177. for (const unspentNote of await allUnspentNotes) {
  178. unspentNotesNullifiers.push(unspentNote.nullifier);
  179. unspentNotesTemp.push(unspentNote.note);
  180. unspentNotesPsks.push(unspentNote.psk);
  181. unspentNotesPos.push(unspentNote.pos);
  182. unspentNotesBlockHeights.push(unspentNote.block_height);
  183. }
  184. // start the correction of the notes
  185. // get the nullifiers
  186. const unspentNotesNullifiersSerialized = await getNullifiersRkyvSerialized(
  187. wasm,
  188. unspentNotesNullifiers,
  189. );
  190. // Fetch existing nullifiers from the node
  191. const unspentNotesExistingNullifiersBytes = await responseBytes(
  192. await request(
  193. unspentNotesNullifiersSerialized,
  194. "existing_nullifiers",
  195. false,
  196. ),
  197. );
  198. // calculate the unspent and spent notes
  199. // from all the unspent note in the db
  200. // their nullifiers
  201. const correctedNotes = await unspentSpentNotes(
  202. wasm,
  203. unspentNotesTemp,
  204. unspentNotesNullifiers,
  205. unspentNotesBlockHeights,
  206. unspentNotesExistingNullifiersBytes,
  207. unspentNotesPsks,
  208. );
  209. // These are the spent notes which were unspent before
  210. const correctedSpentNotes = Array.from(correctedNotes.spent_notes);
  211. const posToRemove = correctedSpentNotes.map((noteData) => noteData.pos);
  212. return deleteUnspentNotesInsertSpentNotes(posToRemove, correctedSpentNotes);
  213. }
  214. /**
  215. * Insert history data
  216. * @param {HistoryData} historyData
  217. */
  218. export async function insertHistory(historyData) {
  219. const db = initializeHistory();
  220. const existingHistory = await getHistory(historyData.psk);
  221. // remove duplicates
  222. historyData.history = existingHistory.history
  223. .concat(historyData.history)
  224. .filter(
  225. (v, i, a) =>
  226. a.findIndex((v2) => v2.block_height === v.block_height) === i,
  227. );
  228. await db.cache.put(historyData);
  229. }
  230. /**
  231. *
  232. * @param {string} psk
  233. * @returns {HistoryData}
  234. */
  235. export async function getHistory(psk) {
  236. const db = initializeHistory();
  237. const historyData = (await db.cache.get(psk)) ?? new HistoryData(psk, []);
  238. return historyData;
  239. }
  240. /**
  241. * Clears all localstorage inserts and IndexedDB inserts
  242. */
  243. export async function clearDB() {
  244. localStorage.removeItem("lastPos");
  245. localStorage.removeItem("lastPsk");
  246. localStorage.removeItem("blockHeight");
  247. await Dexie.delete("history");
  248. return Dexie.delete("state");
  249. }
  250. /**
  251. * Fetch all unspent notes from the IndexedDB if there are any
  252. * @returns {Promise<Array<NoteData>>} unspent notes of the psk
  253. * @ignore Only called by the sync function
  254. */
  255. async function getAllUnpsentNotes() {
  256. const db = initializeState();
  257. const myTable = db.table("unspentNotes");
  258. if (myTable) {
  259. const result = await myTable.toArray();
  260. return result;
  261. }
  262. }
  263. /**
  264. * Delete unspent notes given their Pos and insert spent notes given data
  265. * We want to move notes from unspent to spent when they are used in a tx
  266. *
  267. * @param {Array<number>} unspentNotesPos - ids of the unspent notes to delete
  268. * @param {Array<NoteData>} spentNotes - spent notes to insert
  269. * @ignore Only called by the sync function
  270. */
  271. async function deleteUnspentNotesInsertSpentNotes(unspentNotesPos, spentNotes) {
  272. const db = initializeState();
  273. const unspentNotesTable = db.table("unspentNotes");
  274. if (unspentNotesTable) {
  275. await unspentNotesTable.bulkDelete(unspentNotesPos);
  276. }
  277. const spentNotesTable = db.table("spentNotes");
  278. if (spentNotesTable) {
  279. return spentNotesTable.bulkPut(spentNotes);
  280. }
  281. }
  282. function initializeState() {
  283. const db = new Dexie("state");
  284. db.version(1).stores({
  285. // Added a autoincremented id for good practice
  286. // if we need to index it in future
  287. unspentNotes: "pos,psk,nullifier",
  288. spentNotes: "pos,psk,nullifier",
  289. });
  290. return db;
  291. }
  292. function initializeHistory() {
  293. const db = new Dexie("history");
  294. db.version(1).stores({
  295. cache: "&psk",
  296. });
  297. return db;
  298. }