Source: mod.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 { getPsks } from "./keys.js";
  7. import { duskToLux } from "./crypto.js";
  8. import { getBalance } from "./balance.js";
  9. import { transfer } from "./contracts/transfer.js";
  10. import { sync, stakeInfo } from "./node.js";
  11. import { stake, unstake, withdrawReward } from "./contracts/stake.js";
  12. import { history } from "./history.js";
  13. import { clearDB } from "./db.js";
  14. import { getNetworkBlockHeight } from "./graphql.js";
  15. import { wasmbytecode, exu } from "../deps.js";
  16. /**
  17. * Construct gas configuration from this class
  18. *
  19. * @class Gas
  20. * @type {Object}
  21. * @property {number} limit The gas limit of the wallet, default is 2900000000
  22. * @property {number} price The gas price of the wallet, default is 1
  23. */
  24. export class Gas {
  25. static DEFAULT_LIMIT = 2_900_000_000;
  26. static DEFAULT_PRICE = 1;
  27. limit = NaN;
  28. price = NaN;
  29. // Passing null/undefined/0 or negative values will set the default value for price and limit
  30. constructor({ limit, price } = {}) {
  31. this.limit = Math.max(limit, 0) || Gas.DEFAULT_LIMIT;
  32. this.price = Math.max(price, 0) || Gas.DEFAULT_PRICE;
  33. Object.freeze(this);
  34. }
  35. }
  36. /**
  37. * Construct a wallet from this function, this function will load the web assembly into the buffer
  38. * and instantiate it, it will block until the web assembly is loaded
  39. *
  40. * @class Wallet
  41. * @type {Object}
  42. * @property {Uint8Array} seed The seed of the wallet
  43. */
  44. export class Wallet {
  45. constructor(seed) {
  46. this.wasm = new exu.Module(wasmbytecode);
  47. this.seed = seed;
  48. }
  49. /**
  50. * Get balance
  51. * @param {string} psk - bs58 encoded public spend key of the user we want to
  52. * @returns {Promise<BalanceInfo>} The balance info
  53. * @memberof Wallet
  54. */
  55. getBalance(psk) {
  56. return getBalance(this.wasm, this.seed, psk);
  57. }
  58. /**
  59. * Get psks for the seed
  60. * @returns {Promise<Array<string>>} psks Psks of the first 21 address for the seed
  61. */
  62. getPsks() {
  63. return getPsks(this.wasm, this.seed);
  64. }
  65. /**
  66. * Sync the wallet
  67. *
  68. * @param {SyncOptions} [options] Options for the sync
  69. *
  70. * @returns {Promise} promise that resolves after the sync is complete
  71. */
  72. sync(options = {}) {
  73. return sync(this.wasm, this.seed, options);
  74. }
  75. /**
  76. * Transfer Dusk from sender psk to receiver psk
  77. * @param {string} sender bs58 encoded Psk to send the dusk from
  78. * @param {string} receiver bs68 encoded psk of the address who will receive the Dusk
  79. * @param {number} amount Amount of DUSK to send
  80. * @param {Gas} [gas] Gas settings for the transfer transaction
  81. * @returns {Promise} promise that resolves after the transfer is accepted into blockchain
  82. */
  83. transfer(sender, receiver, amount, gas = new Gas()) {
  84. return transfer(
  85. this.wasm,
  86. this.seed,
  87. sender,
  88. receiver,
  89. amount,
  90. gas.limit,
  91. gas.price,
  92. );
  93. }
  94. /**
  95. * Stake Dusk from the provided psk, refund to the same psk
  96. * @param {string} staker bs58 encoded Psk to stake from
  97. * @param {number} amount Amount of dusk to stake
  98. * @param {Gas} [gas] Gas settings for the stake transaction
  99. * @returns {Promise} promise that resolves after the stake is accepted into blockchain
  100. */
  101. async stake(staker, amount, gas = new Gas()) {
  102. const minStake = 1000;
  103. const index = (await this.getPsks()).indexOf(staker);
  104. if (amount < minStake) {
  105. throw new Error(`Stake amount needs to be above a ${minStake} dusk`);
  106. }
  107. if (index === -1) {
  108. throw new Error("Staker psk not found");
  109. }
  110. const bal = await this.getBalance(staker);
  111. if (bal.value < minStake) {
  112. throw new Error(
  113. `Balance needs to be greater than min stake amount of ${minStake}`,
  114. );
  115. } else {
  116. return stake(
  117. this.wasm,
  118. this.seed,
  119. index,
  120. staker,
  121. amount,
  122. gas.limit,
  123. gas.price,
  124. );
  125. }
  126. }
  127. /**
  128. * Fetches the info of the stake if the person has staked
  129. * @param {string} psk bs58 encoded Psk of the staker
  130. * @returns {Promise<StakeInfo>} The stake info
  131. */
  132. async stakeInfo(psk) {
  133. const index = (await this.getPsks()).indexOf(psk);
  134. if (index < 0) {
  135. throw new Error("Staker psk not found");
  136. }
  137. const info = await stakeInfo(this.wasm, this.seed, index);
  138. if (info.amount) {
  139. info["amount"] = await duskToLux(this.wasm, info.amount);
  140. }
  141. if (info.reward) {
  142. info["reward"] = await duskToLux(this.wasm, info.reward);
  143. }
  144. return info;
  145. }
  146. /**
  147. * Unstake dusk from the provided psk, refund to the same psk
  148. * @param {string} unstaker bs58 encoded psk to unstake from
  149. * @param {Gas} [gas] Gas settings for the unstake transaction
  150. * @returns {Promise} promise that resolves after the unstake is accepted into blockchain
  151. */
  152. async unstake(unstaker, gas = new Gas()) {
  153. const index = (await this.getPsks()).indexOf(unstaker);
  154. if (index === -1) {
  155. throw new Error("psk not found");
  156. }
  157. return unstake(this.wasm, this.seed, index, unstaker, gas.limit, gas.price);
  158. }
  159. /**
  160. * Withdraw reward
  161. * @param {string} unstaker bs58 encoded psk to unstake from
  162. * @param {Gas} [gas] Gas settings for the withdrawReward transaction
  163. * @returns {Promise} promise that resolves after the unstake is accepted into blockchain
  164. */
  165. async withdrawReward(psk, gas = new Gas()) {
  166. const index = (await this.getPsks()).indexOf(psk);
  167. return withdrawReward(this.wasm, this.seed, index, gas.limit, gas.price);
  168. }
  169. /**
  170. * Get the history of the wallet
  171. *
  172. * @param {string} psk - bs58 encoded public spend key of the user we want to fetch the history of
  173. * @returns {Array<TxData>} The history of the wallet
  174. */
  175. history(psk) {
  176. return history(this.wasm, this.seed, psk);
  177. }
  178. /**
  179. * Reset the state indexedb db and localStorage
  180. * @returns {Promise} promise that resolves after the db is reset
  181. */
  182. reset() {
  183. return clearDB();
  184. }
  185. /**
  186. * Get the network block height
  187. * @returns {Promise<number>} The network block height
  188. */
  189. static get networkBlockHeight() {
  190. return getNetworkBlockHeight();
  191. }
  192. }