@subwallet/extension-base 1.1.28-0 → 1.1.28-beta.1

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.
Files changed (292) hide show
  1. package/background/KoniTypes.d.ts +92 -68
  2. package/background/KoniTypes.js +16 -13
  3. package/background/errors/TransactionError.js +9 -0
  4. package/cjs/background/KoniTypes.js +17 -16
  5. package/cjs/background/errors/TransactionError.js +9 -0
  6. package/cjs/constants/index.js +9 -3
  7. package/cjs/koni/api/dotsama/balance.js +464 -0
  8. package/cjs/koni/api/nft/config.js +33 -23
  9. package/cjs/koni/api/nft/index.js +14 -0
  10. package/cjs/koni/api/nft/nft.js +1 -22
  11. package/cjs/koni/api/nft/ordinal_nft/constants.js +21 -0
  12. package/cjs/koni/api/nft/ordinal_nft/index.js +121 -0
  13. package/cjs/koni/api/nft/ordinal_nft/utils.js +41 -0
  14. package/cjs/koni/api/staking/bonding/amplitude.js +17 -15
  15. package/cjs/koni/api/staking/bonding/astar.js +11 -10
  16. package/cjs/koni/api/staking/bonding/index.js +4 -1
  17. package/cjs/koni/api/staking/bonding/paraChain.js +25 -23
  18. package/cjs/koni/api/staking/bonding/relayChain.js +48 -45
  19. package/cjs/koni/api/staking/bonding/utils.js +104 -86
  20. package/cjs/koni/api/staking/index.js +6 -5
  21. package/cjs/koni/api/staking/paraChain.js +6 -5
  22. package/cjs/koni/api/staking/relayChain.js +3 -2
  23. package/cjs/koni/api/yield/helper/utils.js +46 -0
  24. package/cjs/koni/background/cron.js +2 -0
  25. package/cjs/koni/background/handlers/Extension.js +323 -69
  26. package/cjs/koni/background/handlers/State.js +57 -4
  27. package/cjs/koni/background/handlers/index.js +4 -2
  28. package/cjs/koni/background/subscription.js +89 -19
  29. package/cjs/packageInfo.js +1 -1
  30. package/cjs/services/campaign-service/index.js +9 -6
  31. package/cjs/services/chain-service/constants.js +2 -16
  32. package/cjs/services/chain-service/utils.js +7 -1
  33. package/cjs/services/earning-service/constants/chains.js +29 -0
  34. package/cjs/services/earning-service/constants/index.js +27 -0
  35. package/cjs/services/earning-service/constants/step.js +18 -0
  36. package/cjs/services/earning-service/handlers/base.js +202 -0
  37. package/cjs/services/earning-service/handlers/index.js +60 -0
  38. package/cjs/services/earning-service/handlers/lending/base.js +20 -0
  39. package/cjs/services/earning-service/handlers/lending/index.js +13 -0
  40. package/cjs/services/earning-service/handlers/lending/interlay.js +168 -0
  41. package/cjs/services/earning-service/handlers/liquid-staking/acala.js +196 -0
  42. package/cjs/services/earning-service/handlers/liquid-staking/base.js +39 -0
  43. package/cjs/services/earning-service/handlers/liquid-staking/bifrost.js +257 -0
  44. package/cjs/services/earning-service/handlers/liquid-staking/index.js +34 -0
  45. package/cjs/services/earning-service/handlers/liquid-staking/parallel.js +180 -0
  46. package/cjs/services/earning-service/handlers/liquid-staking/stella-swap.js +373 -0
  47. package/cjs/services/earning-service/handlers/native-staking/amplitude.js +359 -0
  48. package/cjs/services/earning-service/handlers/native-staking/astar.js +426 -0
  49. package/cjs/services/earning-service/handlers/native-staking/base-para.js +137 -0
  50. package/cjs/services/earning-service/handlers/native-staking/base.js +109 -0
  51. package/cjs/services/earning-service/handlers/native-staking/index.js +34 -0
  52. package/cjs/services/earning-service/handlers/native-staking/para-chain.js +361 -0
  53. package/cjs/services/earning-service/handlers/native-staking/relay-chain.js +512 -0
  54. package/cjs/services/earning-service/handlers/nomination-pool/index.js +501 -0
  55. package/cjs/services/earning-service/handlers/special.js +387 -0
  56. package/cjs/services/earning-service/service.js +327 -0
  57. package/cjs/services/earning-service/utils/index.js +111 -0
  58. package/cjs/services/event-service/index.js +4 -0
  59. package/cjs/services/keyring-service/index.js +14 -2
  60. package/cjs/services/migration-service/scripts/DeleteEarningData.js +21 -0
  61. package/cjs/services/migration-service/scripts/EnableEarningChains.js +21 -0
  62. package/cjs/services/migration-service/scripts/MigrateEthProvider.js +17 -0
  63. package/cjs/services/migration-service/scripts/MigrateLedgerAccount.js +1 -1
  64. package/cjs/services/migration-service/scripts/MigratePioneerProvider.js +17 -0
  65. package/cjs/services/migration-service/scripts/MigrateProvider.js +29 -0
  66. package/cjs/services/migration-service/scripts/MigrateTransactionHistory.js +1 -1
  67. package/cjs/services/migration-service/scripts/index.js +4 -0
  68. package/cjs/services/mint-campaign-service/campaigns/UnlockDotCampaign.js +149 -0
  69. package/cjs/services/mint-campaign-service/campaigns/index.js +13 -0
  70. package/cjs/services/mint-campaign-service/constants.js +11 -0
  71. package/cjs/services/mint-campaign-service/index.js +18 -0
  72. package/cjs/services/notification-service/NotificationService.js +1 -1
  73. package/cjs/services/storage-service/DatabaseService.js +47 -0
  74. package/cjs/services/storage-service/databases/index.js +4 -0
  75. package/cjs/services/storage-service/db-stores/NominatorMetadata.js +3 -3
  76. package/cjs/services/storage-service/db-stores/YieldPoolStore.js +36 -0
  77. package/cjs/services/storage-service/db-stores/YieldPositionStore.js +65 -0
  78. package/cjs/services/storage-service/index.js +241 -0
  79. package/cjs/services/transaction-service/event-parser/index.js +58 -0
  80. package/cjs/services/transaction-service/helpers/index.js +3 -1
  81. package/cjs/services/transaction-service/index.js +180 -74
  82. package/cjs/services/transaction-service/utils.js +1 -0
  83. package/cjs/types/campaigns/index.js +16 -0
  84. package/cjs/types/campaigns/unlock-dot.js +1 -0
  85. package/cjs/types/index.js +44 -0
  86. package/cjs/types/ordinal.js +1 -0
  87. package/cjs/types/transaction.js +1 -0
  88. package/cjs/types/yield/actions/index.js +27 -0
  89. package/cjs/types/yield/actions/join/index.js +38 -0
  90. package/cjs/types/yield/actions/join/step.js +37 -0
  91. package/cjs/types/yield/actions/join/submit.js +1 -0
  92. package/cjs/types/yield/actions/join/validate.js +16 -0
  93. package/cjs/types/yield/actions/others.js +1 -0
  94. package/cjs/types/yield/index.js +27 -0
  95. package/cjs/types/yield/info/account/index.js +49 -0
  96. package/cjs/types/yield/info/account/info.js +1 -0
  97. package/cjs/types/yield/info/account/reward.js +1 -0
  98. package/cjs/types/yield/info/account/target.js +32 -0
  99. package/cjs/types/yield/info/account/unstake.js +27 -0
  100. package/cjs/types/yield/info/base.js +34 -0
  101. package/cjs/types/yield/info/chain/index.js +27 -0
  102. package/cjs/types/yield/info/chain/info.js +1 -0
  103. package/cjs/types/yield/info/chain/target.js +1 -0
  104. package/cjs/types/yield/info/index.js +49 -0
  105. package/cjs/types/yield/info/pallet.js +15 -0
  106. package/cjs/types.js +1 -0
  107. package/cjs/utils/address.js +34 -0
  108. package/cjs/utils/fetchStaticData.js +2 -1
  109. package/cjs/utils/index.js +89 -1
  110. package/cjs/utils/keyring.js +57 -0
  111. package/cjs/utils/object.js +12 -0
  112. package/constants/index.d.ts +2 -0
  113. package/constants/index.js +2 -0
  114. package/koni/api/nft/config.js +33 -23
  115. package/koni/api/nft/index.js +15 -1
  116. package/koni/api/nft/nft.js +2 -23
  117. package/koni/api/nft/ordinal_nft/constants.d.ts +9 -0
  118. package/koni/api/nft/ordinal_nft/constants.js +12 -0
  119. package/koni/api/nft/ordinal_nft/index.d.ts +8 -0
  120. package/koni/api/nft/ordinal_nft/index.js +114 -0
  121. package/koni/api/nft/ordinal_nft/utils.d.ts +2 -0
  122. package/koni/api/nft/ordinal_nft/utils.js +33 -0
  123. package/koni/api/staking/bonding/amplitude.js +11 -9
  124. package/koni/api/staking/bonding/astar.js +9 -8
  125. package/koni/api/staking/bonding/index.d.ts +1 -1
  126. package/koni/api/staking/bonding/index.js +5 -1
  127. package/koni/api/staking/bonding/paraChain.js +12 -10
  128. package/koni/api/staking/bonding/relayChain.d.ts +2 -2
  129. package/koni/api/staking/bonding/relayChain.js +33 -30
  130. package/koni/api/staking/bonding/utils.d.ts +15 -38
  131. package/koni/api/staking/bonding/utils.js +85 -69
  132. package/koni/api/staking/index.js +2 -1
  133. package/koni/api/staking/paraChain.js +7 -6
  134. package/koni/api/staking/relayChain.js +4 -3
  135. package/koni/api/yield/helper/utils.d.ts +10 -0
  136. package/koni/api/yield/helper/utils.js +32 -0
  137. package/koni/background/cron.js +2 -0
  138. package/koni/background/handlers/Extension.d.ts +14 -1
  139. package/koni/background/handlers/Extension.js +282 -30
  140. package/koni/background/handlers/State.d.ts +18 -2
  141. package/koni/background/handlers/State.js +56 -4
  142. package/koni/background/handlers/index.js +4 -2
  143. package/koni/background/subscription.d.ts +5 -1
  144. package/koni/background/subscription.js +91 -21
  145. package/package.json +311 -8
  146. package/packageInfo.js +1 -1
  147. package/services/campaign-service/index.js +9 -6
  148. package/services/chain-service/constants.d.ts +0 -11
  149. package/services/chain-service/constants.js +1 -14
  150. package/services/chain-service/utils.d.ts +1 -0
  151. package/services/chain-service/utils.js +5 -1
  152. package/services/earning-service/constants/abis/compound_finance_v2_abi.json +1235 -0
  153. package/services/earning-service/constants/abis/st_liquid_token_abi.json +1355 -0
  154. package/services/earning-service/constants/chains.d.ts +14 -0
  155. package/services/earning-service/constants/chains.js +21 -0
  156. package/services/earning-service/constants/index.d.ts +2 -0
  157. package/services/earning-service/constants/index.js +5 -0
  158. package/services/earning-service/constants/step.d.ts +3 -0
  159. package/services/earning-service/constants/step.js +10 -0
  160. package/services/earning-service/handlers/base.d.ts +105 -0
  161. package/services/earning-service/handlers/base.js +195 -0
  162. package/services/earning-service/handlers/index.d.ts +5 -0
  163. package/services/earning-service/handlers/index.js +8 -0
  164. package/services/earning-service/handlers/lending/base.d.ts +6 -0
  165. package/services/earning-service/handlers/lending/base.js +12 -0
  166. package/services/earning-service/handlers/lending/index.d.ts +1 -0
  167. package/services/earning-service/handlers/lending/index.js +4 -0
  168. package/services/earning-service/handlers/lending/interlay.d.ts +22 -0
  169. package/services/earning-service/handlers/lending/interlay.js +160 -0
  170. package/services/earning-service/handlers/liquid-staking/acala.d.ts +27 -0
  171. package/services/earning-service/handlers/liquid-staking/acala.js +188 -0
  172. package/services/earning-service/handlers/liquid-staking/base.d.ts +8 -0
  173. package/services/earning-service/handlers/liquid-staking/base.js +31 -0
  174. package/services/earning-service/handlers/liquid-staking/bifrost.d.ts +44 -0
  175. package/services/earning-service/handlers/liquid-staking/bifrost.js +246 -0
  176. package/services/earning-service/handlers/liquid-staking/index.d.ts +4 -0
  177. package/services/earning-service/handlers/liquid-staking/index.js +7 -0
  178. package/services/earning-service/handlers/liquid-staking/parallel.d.ts +27 -0
  179. package/services/earning-service/handlers/liquid-staking/parallel.js +172 -0
  180. package/services/earning-service/handlers/liquid-staking/stella-swap.d.ts +36 -0
  181. package/services/earning-service/handlers/liquid-staking/stella-swap.js +363 -0
  182. package/services/earning-service/handlers/native-staking/amplitude.d.ts +22 -0
  183. package/services/earning-service/handlers/native-staking/amplitude.js +350 -0
  184. package/services/earning-service/handlers/native-staking/astar.d.ts +18 -0
  185. package/services/earning-service/handlers/native-staking/astar.js +416 -0
  186. package/services/earning-service/handlers/native-staking/base-para.d.ts +11 -0
  187. package/services/earning-service/handlers/native-staking/base-para.js +129 -0
  188. package/services/earning-service/handlers/native-staking/base.d.ts +19 -0
  189. package/services/earning-service/handlers/native-staking/base.js +101 -0
  190. package/services/earning-service/handlers/native-staking/index.d.ts +4 -0
  191. package/services/earning-service/handlers/native-staking/index.js +7 -0
  192. package/services/earning-service/handlers/native-staking/para-chain.d.ts +15 -0
  193. package/services/earning-service/handlers/native-staking/para-chain.js +353 -0
  194. package/services/earning-service/handlers/native-staking/relay-chain.d.ts +21 -0
  195. package/services/earning-service/handlers/native-staking/relay-chain.js +503 -0
  196. package/services/earning-service/handlers/nomination-pool/index.d.ts +33 -0
  197. package/services/earning-service/handlers/nomination-pool/index.js +492 -0
  198. package/services/earning-service/handlers/special.d.ts +56 -0
  199. package/services/earning-service/handlers/special.js +379 -0
  200. package/services/earning-service/service.d.ts +36 -0
  201. package/services/earning-service/service.js +320 -0
  202. package/services/earning-service/utils/index.d.ts +14 -0
  203. package/services/earning-service/utils/index.js +100 -0
  204. package/services/event-service/index.d.ts +1 -0
  205. package/services/event-service/index.js +4 -0
  206. package/services/event-service/types.d.ts +8 -0
  207. package/services/keyring-service/index.d.ts +2 -1
  208. package/services/keyring-service/index.js +14 -2
  209. package/services/migration-service/scripts/DeleteEarningData.d.ts +4 -0
  210. package/services/migration-service/scripts/DeleteEarningData.js +13 -0
  211. package/services/migration-service/scripts/EnableEarningChains.d.ts +4 -0
  212. package/services/migration-service/scripts/EnableEarningChains.js +13 -0
  213. package/services/migration-service/scripts/MigrateLedgerAccount.js +1 -1
  214. package/services/migration-service/scripts/MigrateTransactionHistory.js +1 -1
  215. package/services/migration-service/scripts/index.js +4 -0
  216. package/services/mint-campaign-service/campaigns/UnlockDotCampaign.d.ts +13 -0
  217. package/services/mint-campaign-service/campaigns/UnlockDotCampaign.js +139 -0
  218. package/services/mint-campaign-service/campaigns/index.d.ts +1 -0
  219. package/services/mint-campaign-service/campaigns/index.js +4 -0
  220. package/services/mint-campaign-service/constants.d.ts +1 -0
  221. package/services/mint-campaign-service/constants.js +4 -0
  222. package/services/mint-campaign-service/index.d.ts +7 -0
  223. package/services/mint-campaign-service/index.js +11 -0
  224. package/services/notification-service/NotificationService.js +1 -1
  225. package/services/storage-service/DatabaseService.d.ts +17 -1
  226. package/services/storage-service/DatabaseService.js +47 -0
  227. package/services/storage-service/databases/index.d.ts +3 -1
  228. package/services/storage-service/databases/index.js +4 -0
  229. package/services/storage-service/db-stores/NominatorMetadata.js +3 -3
  230. package/services/storage-service/db-stores/YieldPoolStore.d.ts +10 -0
  231. package/services/storage-service/db-stores/YieldPoolStore.js +28 -0
  232. package/services/storage-service/db-stores/YieldPositionStore.d.ts +9 -0
  233. package/services/storage-service/db-stores/YieldPositionStore.js +57 -0
  234. package/services/subscan-service/index.d.ts +2 -2
  235. package/services/transaction-service/event-parser/index.d.ts +3 -1
  236. package/services/transaction-service/event-parser/index.js +57 -1
  237. package/services/transaction-service/helpers/index.js +3 -1
  238. package/services/transaction-service/index.d.ts +4 -15
  239. package/services/transaction-service/index.js +178 -72
  240. package/services/transaction-service/types.d.ts +2 -0
  241. package/services/transaction-service/utils.js +1 -0
  242. package/types/campaigns/index.d.ts +1 -0
  243. package/types/campaigns/index.js +4 -0
  244. package/types/campaigns/unlock-dot.d.ts +71 -0
  245. package/types/campaigns/unlock-dot.js +1 -0
  246. package/types/index.d.ts +5 -0
  247. package/types/index.js +5 -1
  248. package/types/ordinal.d.ts +69 -0
  249. package/types/ordinal.js +1 -0
  250. package/types/transaction.d.ts +3 -0
  251. package/types/transaction.js +1 -0
  252. package/types/yield/actions/index.d.ts +2 -0
  253. package/types/yield/actions/index.js +5 -0
  254. package/types/yield/actions/join/index.d.ts +3 -0
  255. package/types/yield/actions/join/index.js +6 -0
  256. package/types/yield/actions/join/step.d.ts +75 -0
  257. package/types/yield/actions/join/step.js +34 -0
  258. package/types/yield/actions/join/submit.d.ts +53 -0
  259. package/types/yield/actions/join/submit.js +1 -0
  260. package/types/yield/actions/join/validate.d.ts +18 -0
  261. package/types/yield/actions/join/validate.js +10 -0
  262. package/types/yield/actions/others.d.ts +60 -0
  263. package/types/yield/actions/others.js +1 -0
  264. package/types/yield/index.d.ts +2 -0
  265. package/types/yield/index.js +5 -0
  266. package/types/yield/info/account/index.d.ts +4 -0
  267. package/types/yield/info/account/index.js +7 -0
  268. package/types/yield/info/account/info.d.ts +112 -0
  269. package/types/yield/info/account/info.js +1 -0
  270. package/types/yield/info/account/reward.d.ts +41 -0
  271. package/types/yield/info/account/reward.js +1 -0
  272. package/types/yield/info/account/target.d.ts +43 -0
  273. package/types/yield/info/account/target.js +27 -0
  274. package/types/yield/info/account/unstake.d.ts +31 -0
  275. package/types/yield/info/account/unstake.js +22 -0
  276. package/types/yield/info/base.d.ts +28 -0
  277. package/types/yield/info/base.js +28 -0
  278. package/types/yield/info/chain/index.d.ts +2 -0
  279. package/types/yield/info/chain/index.js +5 -0
  280. package/types/yield/info/chain/info.d.ts +190 -0
  281. package/types/yield/info/chain/info.js +1 -0
  282. package/types/yield/info/chain/target.d.ts +33 -0
  283. package/types/yield/info/chain/target.js +1 -0
  284. package/types/yield/info/index.d.ts +4 -0
  285. package/types/yield/info/index.js +7 -0
  286. package/types/yield/info/pallet.d.ts +143 -0
  287. package/types/yield/info/pallet.js +9 -0
  288. package/utils/fetchStaticData.js +2 -1
  289. package/utils/index.d.ts +5 -0
  290. package/utils/index.js +53 -1
  291. package/utils/object.d.ts +1 -0
  292. package/utils/object.js +6 -0
@@ -0,0 +1,14 @@
1
+ export declare const _STAKING_CHAIN_GROUP: {
2
+ relay: string[];
3
+ para: string[];
4
+ astar: string[];
5
+ amplitude: string[];
6
+ kilt: string[];
7
+ nominationPool: string[];
8
+ bifrost: string[];
9
+ aleph: string[];
10
+ ternoa: string[];
11
+ liquidStaking: string[];
12
+ lending: string[];
13
+ };
14
+ export declare const ST_LIQUID_TOKEN_ABI: Record<string, any>;
@@ -0,0 +1,21 @@
1
+ // Copyright 2019-2022 @subwallet/extension-base
2
+ // SPDX-License-Identifier: Apache-2.0
3
+
4
+ export const _STAKING_CHAIN_GROUP = {
5
+ relay: ['polkadot', 'kusama', 'aleph', 'polkadex', 'ternoa', 'ternoa_alphanet', 'alephTest', 'polkadexTest', 'westend', 'kate', 'edgeware', 'creditcoin', 'vara_network'],
6
+ para: ['moonbeam', 'moonriver', 'moonbase', 'turing', 'turingStaging', 'bifrost', 'bifrost_testnet', 'calamari_test', 'calamari'],
7
+ astar: ['astar', 'shiden', 'shibuya'],
8
+ amplitude: ['amplitude', 'amplitude_test', 'kilt', 'kilt_peregrine', 'pendulum'],
9
+ // amplitude and kilt only share some common logic
10
+ kilt: ['kilt', 'kilt_peregrine'],
11
+ nominationPool: ['polkadot', 'kusama', 'westend', 'alephTest', 'aleph', 'kate', 'vara_network'],
12
+ bifrost: ['bifrost', 'bifrost_testnet'],
13
+ aleph: ['aleph', 'alephTest'],
14
+ // A0 has distinct tokenomics
15
+ ternoa: ['ternoa', 'ternoa_alphanet'],
16
+ liquidStaking: ['bifrost_dot', 'acala', 'parallel', 'moonbeam'],
17
+ lending: ['interlay']
18
+ };
19
+
20
+ // eslint-disable-next-line @typescript-eslint/no-var-requires,@typescript-eslint/no-unsafe-assignment
21
+ export const ST_LIQUID_TOKEN_ABI = require("./abis/st_liquid_token_abi.json");
@@ -0,0 +1,2 @@
1
+ export * from './chains';
2
+ export * from './step';
@@ -0,0 +1,5 @@
1
+ // Copyright 2019-2022 @subwallet/extension-base
2
+ // SPDX-License-Identifier: Apache-2.0
3
+
4
+ export * from "./chains.js";
5
+ export * from "./step.js";
@@ -0,0 +1,3 @@
1
+ import { YieldStepDetail } from '@subwallet/extension-base/types';
2
+ export declare const DEFAULT_YIELD_FIRST_STEP: YieldStepDetail;
3
+ export declare const fakeAddress = "15MLn9YQaHZ4GMkhK3qXqR5iGGSdULyJ995ctjeBgFRseyi6";
@@ -0,0 +1,10 @@
1
+ // Copyright 2019-2022 @subwallet/extension-base
2
+ // SPDX-License-Identifier: Apache-2.0
3
+
4
+ import { YieldStepType } from '@subwallet/extension-base/types';
5
+ export const DEFAULT_YIELD_FIRST_STEP = {
6
+ id: 0,
7
+ name: 'Fill information',
8
+ type: YieldStepType.DEFAULT
9
+ };
10
+ export const fakeAddress = '15MLn9YQaHZ4GMkhK3qXqR5iGGSdULyJ995ctjeBgFRseyi6';
@@ -0,0 +1,105 @@
1
+ import { _ChainAsset, _ChainInfo } from '@subwallet/chain-list/types';
2
+ import { TransactionError } from '@subwallet/extension-base/background/errors/TransactionError';
3
+ import { ChainType, ExtrinsicType } from '@subwallet/extension-base/background/KoniTypes';
4
+ import KoniState from '@subwallet/extension-base/koni/background/handlers/State';
5
+ import { _EvmApi, _SubstrateApi } from '@subwallet/extension-base/services/chain-service/types';
6
+ import { EarningRewardItem, HandleYieldStepData, OptimalYieldPath, OptimalYieldPathParams, StakeCancelWithdrawalParams, SubmitYieldJoinData, TransactionData, UnstakingInfo, YieldPoolInfo, YieldPoolTarget, YieldPoolType, YieldPositionInfo, YieldStepBaseInfo, YieldTokenBaseInfo } from '@subwallet/extension-base/types';
7
+ /**
8
+ * @class BasePoolHandler
9
+ * @description Base pool handler
10
+ * */
11
+ export default abstract class BasePoolHandler {
12
+ /** Koni state */
13
+ protected readonly state: KoniState;
14
+ /** Pool's chain */
15
+ readonly chain: string;
16
+ /** Pool's logo */
17
+ protected _logo: string;
18
+ /** Pool's slug */
19
+ abstract slug: string;
20
+ /** Pool's type */
21
+ abstract type: YieldPoolType;
22
+ /** Pool's name */
23
+ protected abstract name: string;
24
+ /** Pool's short name */
25
+ protected abstract shortName: string;
26
+ /** Pool's description */
27
+ protected abstract description: string;
28
+ /** Pool's transaction type */
29
+ readonly transactionChainType: ChainType;
30
+ /**
31
+ * @constructor
32
+ * @param {KoniState} state - Koni state
33
+ * @param {string} chain - Pool's chain
34
+ * */
35
+ protected constructor(state: KoniState, chain: string);
36
+ get logo(): string;
37
+ get group(): string;
38
+ protected get substrateApi(): _SubstrateApi;
39
+ protected get evmApi(): _EvmApi;
40
+ get chainInfo(): _ChainInfo;
41
+ protected get nativeToken(): _ChainAsset;
42
+ protected get defaultInfo(): Pick<YieldPoolInfo, 'name' | 'group' | 'chain' | 'slug' | 'shortName' | 'logo'>;
43
+ /** Can mint when haven't enough native token (use input token for fee) */
44
+ get isPoolSupportAlternativeFee(): boolean;
45
+ getPoolInfo(): Promise<YieldPoolInfo | undefined>;
46
+ getPoolPosition(address: string): Promise<YieldPositionInfo | undefined>;
47
+ /** Subscribe pool info */
48
+ abstract subscribePoolInfo(callback: (data: YieldPoolInfo) => void): Promise<VoidFunction>;
49
+ /** Subscribe pool position */
50
+ abstract subscribePoolPosition(useAddresses: string[], callback: (rs: YieldPositionInfo) => void): Promise<VoidFunction>;
51
+ /** Get pool reward */
52
+ abstract getPoolReward(useAddresses: string[], callback: (rs: EarningRewardItem) => void): Promise<VoidFunction>;
53
+ /** Get pool target */
54
+ abstract getPoolTargets(): Promise<YieldPoolTarget[]>;
55
+ /**
56
+ * @function firstStepFee
57
+ * */
58
+ protected get firstStepFee(): YieldTokenBaseInfo;
59
+ /**
60
+ * @function defaultSubmitStep
61
+ * @description Default submit step data
62
+ * */
63
+ protected abstract get defaultSubmitStep(): YieldStepBaseInfo;
64
+ /**
65
+ * @async
66
+ * @function getTokenApproveStep
67
+ * @param {OptimalYieldPathParams} params - base param to join pool
68
+ * @description Generate token approve step data
69
+ * */
70
+ protected getTokenApproveStep(params: OptimalYieldPathParams): Promise<YieldStepBaseInfo | undefined>;
71
+ /**
72
+ * @async
73
+ * @function getXcmStep
74
+ * @param {OptimalYieldPathParams} params - base param to join pool
75
+ * @description Generate token approve step data
76
+ * */
77
+ protected getXcmStep(params: OptimalYieldPathParams): Promise<YieldStepBaseInfo | undefined>;
78
+ /**
79
+ * @async
80
+ * @function getSubmitStep
81
+ * @param {OptimalYieldPathParams} params - base param to join pool
82
+ * @description Generate token approve step data
83
+ * */
84
+ protected abstract getSubmitStep(params: OptimalYieldPathParams): Promise<YieldStepBaseInfo>;
85
+ /** Generate the optimal steps to join pool */
86
+ generateOptimalPath(params: OptimalYieldPathParams): Promise<OptimalYieldPath>;
87
+ /** Validate param to join the pool */
88
+ abstract validateYieldJoin(data: SubmitYieldJoinData, path: OptimalYieldPath): Promise<TransactionError[]>;
89
+ /** Create `transaction` to join the pool step-by-step */
90
+ abstract handleYieldJoin(data: SubmitYieldJoinData, path: OptimalYieldPath, currentStep: number): Promise<HandleYieldStepData>;
91
+ /** Validate param to leave the pool */
92
+ abstract validateYieldLeave(amount: string, address: string, fastLeave: boolean, selectedTarget?: string): Promise<TransactionError[]>;
93
+ /** Create `transaction` to leave the pool normal (default unstake) */
94
+ protected abstract handleYieldUnstake(amount: string, address: string, selectedTarget?: string): Promise<[ExtrinsicType, TransactionData]>;
95
+ /** Create `transaction` to leave the pool fast (swap token) */
96
+ protected abstract handleYieldRedeem(amount: string, address: string, selectedTarget?: string): Promise<[ExtrinsicType, TransactionData]>;
97
+ /** Create `transaction` to leave the pool */
98
+ handleYieldLeave(fastLeave: boolean, amount: string, address: string, selectedTarget?: string): Promise<[ExtrinsicType, TransactionData]>;
99
+ /** Create `transaction` to withdraw unstaked amount */
100
+ abstract handleYieldWithdraw(address: string, unstakingInfo: UnstakingInfo): Promise<TransactionData>;
101
+ /** Create `transaction` to cancel unstake */
102
+ abstract handleYieldCancelUnstake(params: StakeCancelWithdrawalParams): Promise<TransactionData>;
103
+ /** Create `transaction` to claim reward */
104
+ abstract handleYieldClaimReward(address: string, bondReward?: boolean): Promise<TransactionData>;
105
+ }
@@ -0,0 +1,195 @@
1
+ // Copyright 2019-2022 @subwallet/extension-base
2
+ // SPDX-License-Identifier: Apache-2.0
3
+
4
+ import { ChainType } from '@subwallet/extension-base/background/KoniTypes';
5
+ import { DEFAULT_YIELD_FIRST_STEP } from '@subwallet/extension-base/services/earning-service/constants';
6
+ /**
7
+ * @class BasePoolHandler
8
+ * @description Base pool handler
9
+ * */
10
+ export default class BasePoolHandler {
11
+ /** Koni state */
12
+
13
+ /** Pool's chain */
14
+
15
+ /** Pool's logo */
16
+
17
+ /** Pool's slug */
18
+
19
+ /** Pool's type */
20
+
21
+ /** Pool's name */
22
+
23
+ /** Pool's short name */
24
+
25
+ /** Pool's description */
26
+
27
+ /** Pool's transaction type */
28
+ transactionChainType = ChainType.SUBSTRATE;
29
+
30
+ /**
31
+ * @constructor
32
+ * @param {KoniState} state - Koni state
33
+ * @param {string} chain - Pool's chain
34
+ * */
35
+ constructor(state, chain) {
36
+ this.state = state;
37
+ this.chain = chain;
38
+ this._logo = chain;
39
+ }
40
+ get logo() {
41
+ return this._logo;
42
+ }
43
+ get group() {
44
+ const groupSlug = this.nativeToken.multiChainAsset;
45
+ return groupSlug || this.nativeToken.slug;
46
+ }
47
+ get substrateApi() {
48
+ return this.state.getSubstrateApi(this.chain);
49
+ }
50
+ get evmApi() {
51
+ return this.state.getEvmApi(this.chain);
52
+ }
53
+ get chainInfo() {
54
+ return this.state.getChainInfo(this.chain);
55
+ }
56
+ get nativeToken() {
57
+ return this.state.getNativeTokenInfo(this.chain);
58
+ }
59
+ get defaultInfo() {
60
+ return {
61
+ name: this.name,
62
+ shortName: this.shortName,
63
+ group: this.group,
64
+ logo: this.logo,
65
+ chain: this.chain,
66
+ slug: this.slug
67
+ };
68
+ }
69
+
70
+ /** Can mint when haven't enough native token (use input token for fee) */
71
+ get isPoolSupportAlternativeFee() {
72
+ return false;
73
+ }
74
+ async getPoolInfo() {
75
+ return this.state.dbService.getYieldPool(this.slug);
76
+ }
77
+ async getPoolPosition(address) {
78
+ return this.state.dbService.getYieldPositionByAddressAndSlug(address, this.slug);
79
+ }
80
+
81
+ /* Subscribe data */
82
+
83
+ /** Subscribe pool info */
84
+
85
+ /* Subscribe data */
86
+
87
+ /* Join action */
88
+
89
+ /* Generate steps */
90
+
91
+ /**
92
+ * @function firstStepFee
93
+ * */
94
+ get firstStepFee() {
95
+ return {
96
+ slug: ''
97
+ };
98
+ }
99
+
100
+ /**
101
+ * @function defaultSubmitStep
102
+ * @description Default submit step data
103
+ * */
104
+
105
+ /**
106
+ * @async
107
+ * @function getTokenApproveStep
108
+ * @param {OptimalYieldPathParams} params - base param to join pool
109
+ * @description Generate token approve step data
110
+ * */
111
+ async getTokenApproveStep(params) {
112
+ return Promise.resolve(undefined);
113
+ }
114
+
115
+ /**
116
+ * @async
117
+ * @function getXcmStep
118
+ * @param {OptimalYieldPathParams} params - base param to join pool
119
+ * @description Generate token approve step data
120
+ * */
121
+ async getXcmStep(params) {
122
+ return Promise.resolve(undefined);
123
+ }
124
+
125
+ /**
126
+ * @async
127
+ * @function getSubmitStep
128
+ * @param {OptimalYieldPathParams} params - base param to join pool
129
+ * @description Generate token approve step data
130
+ * */
131
+
132
+ /** Generate the optimal steps to join pool */
133
+ async generateOptimalPath(params) {
134
+ const result = {
135
+ totalFee: [this.firstStepFee],
136
+ steps: [DEFAULT_YIELD_FIRST_STEP]
137
+ };
138
+ try {
139
+ const stepFunctions = [this.getTokenApproveStep, this.getXcmStep, this.getSubmitStep];
140
+ for (const stepFunction of stepFunctions) {
141
+ const step = await stepFunction(params);
142
+ if (step) {
143
+ const [info, fee] = step;
144
+ result.steps.push({
145
+ id: result.steps.length,
146
+ ...info
147
+ });
148
+ result.totalFee.push(fee);
149
+ }
150
+ }
151
+ return result;
152
+ } catch (e) {
153
+ const errorMessage = e.message;
154
+ if (errorMessage.includes('network')) {
155
+ result.connectionError = errorMessage.split(' ')[0];
156
+ }
157
+
158
+ /* Submit step */
159
+
160
+ const [step, fee] = this.defaultSubmitStep;
161
+ result.steps.push({
162
+ id: result.steps.length,
163
+ ...step
164
+ });
165
+ result.totalFee.push(fee);
166
+
167
+ /* Submit step */
168
+
169
+ return result;
170
+ }
171
+ }
172
+
173
+ /* Generate steps */
174
+
175
+ /* Validate */
176
+
177
+ /** Validate param to join the pool */
178
+
179
+ /** Create `transaction` to leave the pool */
180
+ async handleYieldLeave(fastLeave, amount, address, selectedTarget) {
181
+ if (fastLeave) {
182
+ return this.handleYieldRedeem(amount, address, selectedTarget);
183
+ } else {
184
+ return this.handleYieldUnstake(amount, address, selectedTarget);
185
+ }
186
+ }
187
+
188
+ /* Leave action */
189
+
190
+ /* Other actions */
191
+
192
+ /** Create `transaction` to withdraw unstaked amount */
193
+
194
+ /* Other actions */
195
+ }
@@ -0,0 +1,5 @@
1
+ export { default as BasePoolHandler } from './base';
2
+ export { default as NominationPoolHandler } from './nomination-pool';
3
+ export * from './lending';
4
+ export * from './liquid-staking';
5
+ export * from './native-staking';
@@ -0,0 +1,8 @@
1
+ // Copyright 2019-2022 @subwallet/extension-base
2
+ // SPDX-License-Identifier: Apache-2.0
3
+
4
+ export { default as BasePoolHandler } from "./base.js";
5
+ export { default as NominationPoolHandler } from "./nomination-pool/index.js";
6
+ export * from "./lending/index.js";
7
+ export * from "./liquid-staking/index.js";
8
+ export * from "./native-staking/index.js";
@@ -0,0 +1,6 @@
1
+ import { LendingYieldPoolInfo, YieldPoolType } from '@subwallet/extension-base/types';
2
+ import BaseSpecialStakingPoolHandler from '../special';
3
+ export default abstract class BaseLendingPoolHandler extends BaseSpecialStakingPoolHandler {
4
+ readonly type = YieldPoolType.LENDING;
5
+ abstract getPoolStat(): Promise<LendingYieldPoolInfo>;
6
+ }
@@ -0,0 +1,12 @@
1
+ // Copyright 2019-2022 @subwallet/extension-base
2
+ // SPDX-License-Identifier: Apache-2.0
3
+
4
+ import { YieldPoolType } from '@subwallet/extension-base/types';
5
+ import BaseSpecialStakingPoolHandler from "../special.js";
6
+ export default class BaseLendingPoolHandler extends BaseSpecialStakingPoolHandler {
7
+ type = YieldPoolType.LENDING;
8
+
9
+ /* Subscribe pool info */
10
+
11
+ /* Subscribe pool info */
12
+ }
@@ -0,0 +1 @@
1
+ export { default as InterlayLendingPoolHandler } from './interlay';
@@ -0,0 +1,4 @@
1
+ // Copyright 2019-2022 @subwallet/extension-base
2
+ // SPDX-License-Identifier: Apache-2.0
3
+
4
+ export { default as InterlayLendingPoolHandler } from "./interlay.js";
@@ -0,0 +1,22 @@
1
+ import { ExtrinsicType } from '@subwallet/extension-base/background/KoniTypes';
2
+ import KoniState from '@subwallet/extension-base/koni/background/handlers/State';
3
+ import { BaseYieldStepDetail, HandleYieldStepData, LendingYieldPoolInfo, OptimalYieldPath, OptimalYieldPathParams, SubmitYieldJoinData, TransactionData, YieldPositionInfo, YieldTokenBaseInfo } from '@subwallet/extension-base/types';
4
+ import BaseLendingPoolHandler from './base';
5
+ export default class InterlayLendingPoolHandler extends BaseLendingPoolHandler {
6
+ protected readonly description: string;
7
+ protected readonly name: string;
8
+ protected readonly shortName: string;
9
+ protected readonly altInputAsset: string;
10
+ protected readonly derivativeAssets: string[];
11
+ protected readonly inputAsset: string;
12
+ protected readonly rewardAssets: string[];
13
+ protected readonly feeAssets: string[];
14
+ slug: string;
15
+ constructor(state: KoniState, chain: string);
16
+ getPoolStat(): Promise<LendingYieldPoolInfo>;
17
+ subscribePoolPosition(useAddresses: string[], resultCallback: (rs: YieldPositionInfo) => void): Promise<VoidFunction>;
18
+ get submitJoinStepInfo(): BaseYieldStepDetail;
19
+ getSubmitStepFee(params: OptimalYieldPathParams): Promise<YieldTokenBaseInfo>;
20
+ handleSubmitStep(data: SubmitYieldJoinData, path: OptimalYieldPath): Promise<HandleYieldStepData>;
21
+ handleYieldRedeem(amount: string, address: string, selectedTarget?: string): Promise<[ExtrinsicType, TransactionData]>;
22
+ }
@@ -0,0 +1,160 @@
1
+ // Copyright 2019-2022 @subwallet/extension-base
2
+ // SPDX-License-Identifier: Apache-2.0
3
+
4
+ import { TransactionError } from '@subwallet/extension-base/background/errors/TransactionError';
5
+ import { BasicTxErrorType, ExtrinsicType } from '@subwallet/extension-base/background/KoniTypes';
6
+ import { _getTokenOnChainInfo } from '@subwallet/extension-base/services/chain-service/utils';
7
+ import { EarningStatus, YieldStepType } from '@subwallet/extension-base/types';
8
+ import BN from 'bn.js';
9
+ import { BN_ZERO } from '@polkadot/util';
10
+ import { fakeAddress } from "../../constants/index.js";
11
+ import BaseLendingPoolHandler from "./base.js";
12
+ export default class InterlayLendingPoolHandler extends BaseLendingPoolHandler {
13
+ altInputAsset = 'polkadot-NATIVE-DOT';
14
+ derivativeAssets = ['interlay-LOCAL-qDOT'];
15
+ inputAsset = 'interlay-LOCAL-DOT';
16
+ rewardAssets = ['interlay-LOCAL-DOT'];
17
+ feeAssets = ['interlay-NATIVE-INTR', 'interlay-LOCAL-DOT'];
18
+ constructor(state, chain) {
19
+ super(state, chain);
20
+ const chainInfo = this.chainInfo;
21
+ this.slug = `DOT___lending___${chain}`;
22
+ this.name = `${chainInfo.name} Lending`;
23
+ this.shortName = chainInfo.name.replaceAll(' Relay Chain', '');
24
+ this.description = 'Earn rewards by lending DOT';
25
+ }
26
+
27
+ /* Subscribe pool info */
28
+
29
+ async getPoolStat() {
30
+ const substrateApi = await this.substrateApi.isReady;
31
+ const inputTokenSlug = this.inputAsset;
32
+ const inputTokenInfo = this.state.getAssetBySlug(inputTokenSlug);
33
+ const _exchangeRate = await substrateApi.api.query.loans.exchangeRate(_getTokenOnChainInfo(inputTokenInfo));
34
+ const exchangeRate = _exchangeRate.toPrimitive();
35
+ const decimals = 10 ** 18;
36
+ return {
37
+ ...this.defaultInfo,
38
+ description: this.description,
39
+ type: this.type,
40
+ metadata: {
41
+ ...this.baseMetadata,
42
+ isAvailable: true,
43
+ allowCancelUnstaking: false,
44
+ assetEarning: [{
45
+ slug: this.rewardAssets[0],
46
+ apr: 1.29,
47
+ exchangeRate: exchangeRate / decimals
48
+ }],
49
+ maxCandidatePerFarmer: 1,
50
+ maxWithdrawalRequestPerFarmer: 1,
51
+ minJoinPool: '10000000000',
52
+ minWithdrawal: '0',
53
+ totalApr: 1.29,
54
+ tvl: '291890000000000'
55
+ }
56
+ };
57
+ }
58
+
59
+ /* Subscribe pool info */
60
+
61
+ /* Subscribe pool position */
62
+
63
+ async subscribePoolPosition(useAddresses, resultCallback) {
64
+ let cancel = false;
65
+ const substrateApi = this.substrateApi;
66
+ await substrateApi.isReady;
67
+ const derivativeTokenSlug = this.derivativeAssets[0];
68
+ const derivativeTokenInfo = this.state.getAssetBySlug(derivativeTokenSlug);
69
+ const unsub = await substrateApi.api.query.tokens.accounts.multi(useAddresses.map(address => [address, _getTokenOnChainInfo(derivativeTokenInfo)]), _balances => {
70
+ if (cancel) {
71
+ unsub();
72
+ return;
73
+ }
74
+ const balances = _balances;
75
+ for (let i = 0; i < balances.length; i++) {
76
+ const balanceItem = balances[i];
77
+ const address = useAddresses[i];
78
+ const bnTotalBalance = balanceItem.free || BN_ZERO;
79
+ const totalBalance = bnTotalBalance.toString();
80
+ const result = {
81
+ ...this.defaultInfo,
82
+ type: this.type,
83
+ address,
84
+ balanceToken: this.inputAsset,
85
+ totalStake: totalBalance,
86
+ activeStake: totalBalance,
87
+ unstakeBalance: '0',
88
+ status: EarningStatus.EARNING_REWARD,
89
+ derivativeToken: derivativeTokenSlug,
90
+ isBondedBefore: bnTotalBalance.gt(BN_ZERO),
91
+ nominations: [],
92
+ unstakings: []
93
+ };
94
+ resultCallback(result);
95
+ }
96
+ });
97
+ return () => {
98
+ cancel = true;
99
+ unsub();
100
+ };
101
+ }
102
+
103
+ /* Subscribe pool position */
104
+
105
+ /* Join pool action */
106
+
107
+ get submitJoinStepInfo() {
108
+ return {
109
+ name: 'Mint qDOT',
110
+ type: YieldStepType.MINT_QDOT
111
+ };
112
+ }
113
+ async getSubmitStepFee(params) {
114
+ const poolOriginSubstrateApi = await this.substrateApi.isReady;
115
+ const inputTokenSlug = this.inputAsset;
116
+ const inputTokenInfo = this.state.getAssetBySlug(inputTokenSlug);
117
+ const defaultFeeTokenSlug = this.feeAssets[0];
118
+ const _mintFeeInfo = await poolOriginSubstrateApi.api.tx.loans.mint(_getTokenOnChainInfo(inputTokenInfo), params.amount).paymentInfo(fakeAddress);
119
+ const mintFeeInfo = _mintFeeInfo.toPrimitive();
120
+ return {
121
+ amount: mintFeeInfo.partialFee.toString(),
122
+ slug: defaultFeeTokenSlug
123
+ };
124
+ }
125
+ async handleSubmitStep(data, path) {
126
+ const substrateApi = await this.substrateApi.isReady;
127
+ const inputTokenSlug = this.inputAsset;
128
+ const inputTokenInfo = this.state.getAssetBySlug(inputTokenSlug);
129
+ const extrinsic = substrateApi.api.tx.loans.mint(_getTokenOnChainInfo(inputTokenInfo), data.amount);
130
+ return {
131
+ txChain: this.chain,
132
+ extrinsicType: ExtrinsicType.MINT_QDOT,
133
+ extrinsic,
134
+ txData: data,
135
+ transferNativeAmount: '0'
136
+ };
137
+ }
138
+
139
+ /* Join pool action */
140
+
141
+ /* Leave pool action */
142
+
143
+ async handleYieldRedeem(amount, address, selectedTarget) {
144
+ const substrateApi = await this.substrateApi.isReady;
145
+ // @ts-ignore
146
+ const inputTokenSlug = this.inputAsset;
147
+ const inputTokenInfo = this.state.getAssetBySlug(inputTokenSlug);
148
+ const yieldPositionInfo = await this.getPoolPosition(address);
149
+ if (!yieldPositionInfo) {
150
+ return Promise.reject(new TransactionError(BasicTxErrorType.INVALID_PARAMS));
151
+ }
152
+ const bnAmount = new BN(amount);
153
+ const bnActiveBalance = new BN(yieldPositionInfo.activeStake);
154
+ const redeemAll = bnAmount.eq(bnActiveBalance);
155
+ const extrinsic = !redeemAll ? substrateApi.api.tx.loans.redeem(_getTokenOnChainInfo(inputTokenInfo), amount) : substrateApi.api.tx.loans.redeemAll(_getTokenOnChainInfo(inputTokenInfo));
156
+ return [ExtrinsicType.REDEEM_QDOT, extrinsic];
157
+ }
158
+
159
+ /* Leave pool action */
160
+ }
@@ -0,0 +1,27 @@
1
+ import { ExtrinsicType } from '@subwallet/extension-base/background/KoniTypes';
2
+ import KoniState from '@subwallet/extension-base/koni/background/handlers/State';
3
+ import { BaseYieldStepDetail, HandleYieldStepData, LiquidYieldPoolInfo, OptimalYieldPath, OptimalYieldPathParams, SubmitYieldJoinData, TransactionData, YieldPositionInfo, YieldTokenBaseInfo } from '@subwallet/extension-base/types';
4
+ import BaseLiquidStakingPoolHandler from './base';
5
+ export default class AcalaLiquidStakingPoolHandler extends BaseLiquidStakingPoolHandler {
6
+ protected readonly description: string;
7
+ protected readonly name: string;
8
+ protected readonly shortName: string;
9
+ protected readonly altInputAsset: string;
10
+ protected readonly derivativeAssets: string[];
11
+ protected readonly inputAsset: string;
12
+ protected readonly rewardAssets: string[];
13
+ protected readonly feeAssets: string[];
14
+ /** @inner */
15
+ protected readonly minAmountPercent = 0.98;
16
+ /** @inner */
17
+ protected readonly allowDefaultUnstake = true;
18
+ slug: string;
19
+ constructor(state: KoniState, chain: string);
20
+ getPoolStat(): Promise<LiquidYieldPoolInfo>;
21
+ subscribePoolPosition(useAddresses: string[], resultCallback: (rs: YieldPositionInfo) => void): Promise<VoidFunction>;
22
+ get submitJoinStepInfo(): BaseYieldStepDetail;
23
+ getSubmitStepFee(params: OptimalYieldPathParams): Promise<YieldTokenBaseInfo>;
24
+ handleSubmitStep(data: SubmitYieldJoinData, path: OptimalYieldPath): Promise<HandleYieldStepData>;
25
+ handleYieldRedeem(amount: string, address: string, selectedTarget?: string): Promise<[ExtrinsicType, TransactionData]>;
26
+ handleYieldUnstake(amount: string, address: string, selectedTarget?: string): Promise<[ExtrinsicType, TransactionData]>;
27
+ }