EOS钱包开发:买入卖出RAM
华胥
发表于 2022-12-6 18:51:38
94
0
0
现在我们能获取到了基于测试网络的账号的网络资源数据,现在我们就介绍如何交易RAM。; S" C- B4 Q Q' j6 B; g
交易RAM的前提是我们需要知道它的价格,即买1KB内存需要花费多少EOS,卖1个EOS能获取多少内存。这就需要我们去链上查询数据库获取相应的实时的数据,再进行计算拿到单价。下面我们来一步步实现,最后展示在项目中的相应源码。+ R3 }( f5 B- }) y" r7 W6 S9 \
一、获取全网RAM数量4 m: b- V5 w4 |) R# a
全网RAM数量的数据是存在数据库中的,需要使用eosjs库的getTableRows方法获取,或者RPC的chain/get_table_rows接口获取。下面会有eosjs进行交易,所以本章我们统一使用eosjs访问数据,可以到官网查看eosjs-api接口文档。
首先访问主网中RAM的数量。
Eos = require('eosjs'); Y+ W4 i! Q3 ]+ w7 Q" y! @
eosconfig = {$ {! W0 n& j* o+ y9 s
httpEndpoint: 'https://node1.zbeos.com',0 @) u5 a+ d. F" N9 E$ w4 ?( F
chainId: 'aca376f206b8fc25a6ed44dbdc66547c36c6c33e3a119ffbeaef943642f0e906',
// keyProvider: privatekeyList, // WIF string or array of keys..' J$ u2 [& H3 Z- |; R
expireInSeconds: 60,% [7 N/ j3 q' x2 Z
broadcast: true,
verbose: false, // API activity1 M1 s8 m4 O0 X+ Y9 Y
sign: true
},
eos = Eos(eosconfig); q5 F. o: ^5 Z0 i4 u; Y
eos.getTableRows(true,"eosio","eosio","global").then(result => {
console.log(result)
})) a5 A! t. N" c$ p4 e" W$ w
输出如下
{ rows:7 W) _' |0 A! B! C
[ { max_block_net_usage: 1048576,
target_block_net_usage_pct: 1000,3 c- |2 P& j) v
max_transaction_net_usage: 524288,/ n% y" Z. G$ o% @3 {* X
base_per_transaction_net_usage: 12,
net_usage_leeway: 500,0 k4 D' R1 J& q- L4 Z2 v+ t
context_free_discount_net_usage_num: 20,7 T0 k$ ]6 R7 S
context_free_discount_net_usage_den: 100,
max_block_cpu_usage: 200000,. @6 }$ y" E6 p8 Q
target_block_cpu_usage_pct: 2000,
max_transaction_cpu_usage: 150000,
min_transaction_cpu_usage: 100,
max_transaction_lifetime: 3600,
deferred_trx_expiration_window: 600,
max_transaction_delay: 3888000,
max_inline_action_size: 4096,% m& v8 g' {0 X# u
max_inline_action_depth: 4,# i( L b; Q5 H4 `3 I& _
max_authority_depth: 6,
max_ram_size: '88414727168',5 P# p& @6 J1 @" N
total_ram_bytes_reserved: '49777052734',) u2 A v) O8 N5 K: q) b
total_ram_stake: '26261375993',! L% `+ [" a0 z% t% F- @* g* M
last_producer_schedule_update: '2018-11-05T06:57:51.000',0 [. O" U% I# f3 F+ G
last_pervote_bucket_fill: '1541399606000000',2 `% z, Q1 t: q1 y1 I3 r9 c
pervote_bucket: 219209992,; b) I+ {( T0 n0 `! }# D
perblock_bucket: 36804696,
total_unpaid_blocks: 95761,
total_activated_stake: '4035546740321',
thresh_activated_stake_time: '1529505892000000',4 C8 M6 |- v# O4 Y a1 |$ {# i9 N8 B
last_producer_schedule_size: 21,2 p& Z2 Q# A3 x' w- j6 u
total_producer_vote_weight: '24124687214114213888.00000000000000000',
last_name_close: '2018-11-05T02:08:26.000' } ],$ ?- T( x( c, Z$ k8 l
more: false }8 c8 G2 N# j1 e H
其中“max_ram_size“字段的数据就是全网RAM的数量,单位是bytes,那么转换为G,则结果为:88402589696/(1024*1024*1024)=82.34263134002686 G。获取方式如下:# N. w2 C d2 p) e
eos.getTableRows(true,"eosio","eosio","global").then(result => {2 R4 { M H! b3 U7 U; j% @" d8 Z- R8 k
//ram总量,bytes转为G, c. j3 ^ {. ]" i' R
let ramTotal = result.rows[0].max_ram_size / (1024*1024*1024) % @5 B4 p n1 G8 H7 y
})
二、获取RAM使用量、计算RAM价格
与获取RAM总量一样,我们使用eosjs库的getTableRows方法获取主网中RAM的使用量、消耗RAM的EOS数量,最后通过Bancor算法计算RAM价格。+ k3 z' b9 i! `
eosconfig = {
httpEndpoint: 'https://node1.zbeos.com',
chainId: 'aca376f206b8fc25a6ed44dbdc66547c36c6c33e3a119ffbeaef943642f0e906',
// keyProvider: privatekeyList, // WIF string or array of keys..$ ~" Y* Z1 X1 m/ d
expireInSeconds: 60,# d* y& g+ ]$ T
broadcast: true,$ p0 S$ B, F& F% F* s
verbose: false, // API activity
sign: true
},: C% ?# s9 d) r9 t6 x! O" g9 \
eos = Eos(eosconfig), n0 O W6 R$ E; \ G! H
eos.getTableRows(true, "eosio", "eosio", "rammarket").then(ramData => {7 A2 l6 ]1 Y9 w; {" ~
console.log(ramData)) L4 B1 f) s( s; k3 Z# X x
})
输出如下2 I7 D' |# t5 k( x# N* I9 i V+ A1 t/ @
% b, p. {& L& ?; }0 T
其中“base“字段里的“balance”数据就是全网RAM的使用量,单位是bytes。 R: ~# p+ M2 V% ]
“quote“字段里的“balance”数据就是消耗RAM的EOS数量。
下面根据Bancor算法算出RAM的价格5 h( ~- k) S3 i; I8 ]5 U! o
ramPrice = quote.balance / (base.balance / 1024)
Bancor协议是底层货币协议,通过区块链技术和智能合约为加密数字货币提供连续流动性和实现异步价格。6 a$ X; C6 a9 g. U
所以ramPrice的获取方式如下:9 N! U6 Y' h+ q7 ]% C
eos.getTableRows(true, "eosio", "eosio", "rammarket").then(ramData => {
console.log(ramData)
//RAM消耗的EOS数量. g9 v4 r2 y6 C q/ I' n; c4 [" h
let eosAmount = ramData.rows[0].quote.balance.split(" ")[0];: k& H' {7 s$ e
//RAM使用量
let ramAmount = ramData.rows[0].base.balance.split(" ")[0] / 1024;2 @$ }/ z; [7 r ?
//RAM价格! G3 X! Q6 v) ]6 F; D0 ^, K
let ramPriceWithEOS = eosAmount / ramAmount' M* X( g4 _5 B+ p7 [! c. L3 W
console.log(eosAmount, ramAmount, ramPriceWithEOS);
})
三、买入RAM
买入RAM的API其实我们已经使用过了,在创建账号的时候我们为别人创建的账号就需要为它购买一定的RAM,现在我们在来单独调用购买的API。由于现在需要交易,所以会使用资源,因此我们重新配置一下eosconfig切换到测试网络进行测试。
eosconfig = {
httpEndpoint:"http://jungle.cryptolions.io:18888",+ |' p, S' @, k) y. x, b
chainId: "038f4b0fc8ff18a4f0842a8f0564611f6e96e8535901dd45e43ac8691a1c4dca", // 32 byte (64 char) hex string* }' |* y& o( _3 z6 X; g' x/ C2 o
keyProvider: ["5HqCj7sg4K2xZ1KD5sSH38kuJkKGqfE1wSGiLL3M599ajacVmTs"], 7 F9 v' W' ]7 H2 u" [4 u( V
expireInSeconds: 60,
broadcast: true,
verbose: false, // API activity: y8 B1 {" y; G P
sign: true) V# b; S. q' x8 w* J" M
},2 H' @0 R1 {/ Q2 P) A8 X6 Y6 O% q
eos = Eos(eosconfig)
eos.getTableRows(true, "eosio", "eosio", "rammarket").then(ramData => {
console.log(ramData)7 C7 @ `5 j. F) q( I
//RAM消耗的EOS数量
let eosAmount = ramData.rows[0].quote.balance.split(" ")[0];$ B+ g0 {* o; w# u! ]: ]9 _. X
//RAM使用量
let ramAmount = ramData.rows[0].base.balance.split(" ")[0] / 1024;
//RAM价格' W. o7 w8 Z$ Q8 I z
let ramPriceWithEOS = eosAmount / ramAmount
console.log(eosAmount, ramAmount, ramPriceWithEOS);
})
let account = "lixutest1111"
let ramAmount = 8*10240 c' ?2 U0 G5 p m
async function buyRAM() {
result = await eos.transaction(tr => {
tr.buyrambytes({
payer: account,
receiver: account,! S7 j6 t m& h0 i, X% k
bytes: ramAmount
})# g* V- j. D0 p6 X. s6 Q
})/ p8 _" J) Y0 A& @6 u. A( V( b
console.log(result)
}
buyRAM(): m5 }3 Y1 o. D, {
在运行之前先记录下来账号“lixutest1111”的网络资源数据。: M+ ]. m( `0 j+ x
" q* o3 y+ t$ p9 P1 \
运行后看到价格是0.07576784159120188 EOS/KB+ i- a* m+ i$ u
" l. v: u- s. V4 @: p) r
代码中是买了8KB的RAM,那我们计算一下,需要花费多少EOS,其实很简单) ]8 H; \) N2 k! r+ D8 \% z) _
8KB * 0.07576784159120188 EOS/KB = 0.6061EOS4 a" W$ q( r4 `0 S6 J
那我们再计算一下账号“lixutest1111”现在的EOS可用余额和RAM总量是多少
78.7959 - 0.6061 = 78.1898EOS
13.30 + 8 = 21.30KB _3 v, x9 ?9 D4 Y, t
计算好以后我们再看看现在账号“lixutest1111”的在链上的资源数据是多少。
/ @, @$ D: G# G5 |$ C$ F' \' v
可以发现计算的结果和链上的结果有了一定的偏差,这是因为RAM价格会随着市场波动而变化,所以导致结果有一点点偏差,这是不可避免的。另外我们在购买RAM的时候,需要输入的是EOS数量,然后根据RAM价格转换成需要购买的RAM大小,再进行交易。
四、卖出RAM
卖出RAM需要用到另外一个方法 sellram# Z8 M! _: |. i" l; a
eosconfig = {
httpEndpoint: "http://jungle.cryptolions.io:18888",7 \4 p6 q/ L3 M4 Q. h b( Q/ m
chainId: "038f4b0fc8ff18a4f0842a8f0564611f6e96e8535901dd45e43ac8691a1c4dca", // 32 byte (64 char) hex string. n9 }. E- [& ^& @
keyProvider: ["5HqCj7sg4K2xZ1KD5sSH38kuJkKGqfE1wSGiLL3M599ajacVmTs"],
expireInSeconds: 60,& h4 b3 y" M5 k/ J
broadcast: true,
verbose: false, // API activity
sign: true" J( ^" E$ N/ S2 O
},
eos = Eos(eosconfig)
eos.getTableRows(true, "eosio", "eosio", "rammarket").then(ramData => {
console.log(ramData)
//RAM消耗的EOS数量
let eosAmount = ramData.rows[0].quote.balance.split(" ")[0];
//RAM使用量
let ramAmount = ramData.rows[0].base.balance.split(" ")[0] / 1024;8 |, K+ G# a% W7 B S3 L/ @6 \ `
//RAM价格# R, M g6 J w! q) U% {
let ramPriceWithEOS = eosAmount / ramAmount
console.log(eosAmount, ramAmount, ramPriceWithEOS);1 s( t: s, {! `( K* o: A: [: U& O
})
let account = "lixutest1111"9 Z D4 y/ x* X5 @" i( v, L' z
let ramAmount = 8*1024
async function sellRAM() {/ e3 F+ g1 m) y; E9 t, b" a
result = await eos.transaction(tr => {, K+ n" C/ G/ i/ f
tr.sellram({ Y" J4 U, V) g; R. M
account: account,
bytes: ramAmount% C0 z/ X% C; f
})
})
console.log(result)
}
sellRAM()8 d2 e' T* h0 ~8 ]6 a, `
运行后看到价格是0.0757692783033557 EOS/KB,确实与上一次获取到的价格有点偏差。0 Y; ]4 b# W1 O( v4 }
这里就不花时间作计算了,大家可以在根据价格与出售量计算在做比较。重新查看账号“lixutest1111”的EOS数量多了0.6,RAM内存减少了8KB。; q1 g1 R* f! O+ |8 F
。, P1 j4 Y9 Q$ V [; {4 L6 X
五、项目源码8 d+ r5 H; Z( a( ]2 }# ]% v
1. controllers/netResource.js
在controllers文件夹下新建netResource.js文件,实现获取全局RAM数据与买入卖出RAM的功能。
let { success, fail } = require("../utils/myUtils")
let myUtils = require("../utils/myUtils"): t" E7 A- C4 F
let walletModel = require("../models/wallet")& J6 c2 x. Q& k1 B0 _
async function getRamInfo() {
ramData = await eos.getTableRows(true, "eosio", "eosio", "rammarket")8 B) |0 X0 @) H2 E+ D! k! Z
//RAM消耗的EOS数量2 @8 M4 D1 u# D8 ^
let eosAmount = ramData.rows[0].quote.balance.split(" ")[0];. o& V ?) p4 t# A' L) F; j
//RAM使用量
let ramAmount = ramData.rows[0].base.balance.split(" ")[0] / 1024;
//RAM价格
let ramPriceWithEOS = eosAmount / ramAmount) L8 x$ b! F& G6 u6 v8 _/ r$ V
console.log(eosAmount, ramAmount, ramPriceWithEOS);
return {# Z5 g( B& s9 m
ramUsed: ramAmount / (1024 * 1024),% }- p& M- `5 t' N! D, _, [
ramPrice: ramPriceWithEOS,) M' ]$ T1 H0 b2 |) A
}2 c! y- {- w+ }: l- `3 M
}) f% a* e9 d: V1 n& M
module.exports = {
netResourceGetRAMInfo: async (ctx) => {: S8 L, k2 y M, b- M2 D
console.log(ctx.request.body)
let { wallet, password } = ctx.request.body
//获取钱包里面所有的私钥配置EOSJS
let privatekeyList = await walletModel.getWalletPrivatekeyList(wallet, password). v2 w/ M U- V. A* V8 [! I
eos = myUtils.getEOSJS(privatekeyList)
let ramData = await eos.getTableRows(true, "eosio", "eosio", "global")( I- K4 m" Y3 n) _# G$ ~( h; C
//ram总量,bytes转为G
let ramTotal = ramData.rows[0].max_ram_size / (1024 * 1024 * 1024)
console.log(ramTotal);
let ramInfo = await getRamInfo()7 I% a2 o6 V/ G& ]3 m$ y: r2 i
ctx.body = success({/ s1 p$ c3 l/ z& F4 e- Q
ramPrice: ramInfo.ramPrice,
ramTotal: ramTotal,
ramAvailable: ramTotal - ramInfo.ramUsed,
})
},
netResourceTransactionRAM: async (ctx) => {
console.log(ctx.request.body)
let { amount, transaction_type, account, wallet, password } = ctx.request.body3 R5 }8 j' u- x, K, V
//获取钱包里面所有的私钥配置EOSJS
let privatekeyList = await walletModel.getWalletPrivatekeyList(wallet, password)
eos = myUtils.getEOSJS(privatekeyList)+ z! K3 a+ C) y; U3 H: ^
let result
if (transaction_type == '1') {
//买RAM
console.log("买RAM")+ W( M W+ d6 H3 D6 h, r! |
let ramInfo = await getRamInfo()
let ramAmount = parseInt((amount / ramInfo.ramPrice) * 1024)' D2 n. s( H1 B& w0 C, S% e Z
console.log("ramAmount:", ramAmount)3 O1 j. `/ K' D# ?5 g5 x4 [
result = await eos.transaction(tr => {
tr.buyrambytes({
payer: account,/ I, T5 B: x& W8 W
receiver: account,
bytes: ramAmount
})/ |( o/ K6 u+ C8 E
})
} else {
//卖RAM" B+ I( p9 W( o* P$ { ?
console.log("卖RAM")2 g/ f, ~) P5 y$ Q. f
let ramAmount = parseInt(amount*1024)$ I4 H- J# I+ s. {/ z
result = await eos.transaction(tr => {
tr.sellram({
account: account,
bytes: ramAmount ' p0 |# O+ F9 S) b$ X0 y
})) ~' C" V9 l) A F: K; g/ @
})
}
console.log("data:", result)9 G; _% Y2 X7 _- l5 O% M( E
if (result.broadcast) { ~% ?7 j" t2 T1 R" u, h$ W
ctx.body = success("ok")
} else {; @ h5 s, ?0 \% S3 ?
ctx.body = fail("error")9 G' d4 l; z' R& w1 h6 c
}9 c& f4 f y2 U" L' b5 D! w
}
}
2. router/router.js/ x, C9 y: @7 V
将获取全局RAM数据与买入卖出RAM的接口绑定到路由。2 p4 _4 i5 H/ K7 w3 F, P
......8 ~1 w9 z8 {1 ^# A+ B0 d
let netresourceController = require("../controllers/netResource")
//网络资源
router.post("/net_resource/ram/info", netresourceController.netResourceGetRAMInfo)
router.post("/net_resource/ram/transaction", netresourceController.netResourceTransactionRAM)
3. views/netResource.html/ C5 ~9 x& R0 _0 `6 D: o+ J3 R2 X
编辑views文件夹下的netResource.html文件,实现RAM交易的表单显示。4 } U+ T' g& A3 T6 Z
......+ G4 X, A9 O: @+ A" Z% D& o' f
/ v: Y* C1 u2 F4 d$ X7 R1 E
买入
卖出4 G7 g! ?0 @- G' K: }6 M# M* E" D
全网内存:
内存价格: EOS/KB
数量:9 j- h, Q$ O5 g# e+ O8 K5 F/ `
* {0 |$ h" `6 O" ?' D
" ^) J9 p! z/ y- ]; g
买入4 d4 {" `3 Z: }/ G; j
5 u9 C L* a, ?7 c' A
......# c3 F# u: t2 _1 ~' `
4. static/js/netResource.js
对全局RAM数据与买入卖出RAM的表单进行网络请求处理与页面渲染。0 G! D8 k, Q; o& p8 K
$(document).ready(function () {5 T0 H- I) p1 [2 i8 \2 y
......
$("input[name=account][hidden=hidden]").val(currentAccount)9 x/ R- s1 [. e' v" e- E
let currentwallet = localStorage.getItem("currentwallet")
let walletPassword = localStorage.getItem(currentwallet)
$("input[name=wallet][hidden=hidden]").val(currentwallet)% s8 f+ m% _ [; y; x0 |
$("input[name=password][hidden=hidden]").val(walletPassword), p7 F# ]; ^# L; E
$("input[name=transaction_type]").change(function () {
if (this.value == 1) {) J: ?% E3 s4 X V
$("#ram-transaction-button").text("买入")
$("input[name=amount]").attr({"placeholder":"请输入EOS数量"})
} else {: N& c1 o3 m+ A# B0 k
$("#ram-transaction-button").text("卖出")
$("input[name=amount]").attr({"placeholder":"请输入RAM(KB)数量"})
}9 Y2 j# W8 {$ c; m% q4 g
})) f* B( F: z. E% g7 N' s! W
//ram全局数据3 m6 g& W0 X0 k* k+ ^! B" s" v
$.post("/net_resource/ram/info", { "account": currentAccount }, function (res, status) {: R, @1 b. u' {2 t1 V" p& E; b
console.log(status + JSON.stringify(res))
if (res.code == 0) {
$("#ram-total").text(res.data.ramAvailable.toFixed(2) + " GB / " + res.data.ramTotal.toFixed(2) + "GB")
$("#ram-price").text(res.data.ramPrice.toFixed(6))
}
})
//交易RAM8 X# m' i3 F$ p" a: a% f
$("#ram-transaction").validate({
rules: {
amount: {required: true,},8 U2 S+ n8 B0 o$ o
},
messages: {
name: {required: "请输入要交易的数量",},
},7 z6 V$ P* d+ ~7 p: ]1 L
submitHandler: function (form) {
$(form).ajaxSubmit({+ ^0 V* t! c1 u5 s
url: "/net_resource/ram/transaction",
type: "post",
dataType: "json",) r _; m+ @. Q1 o5 g( ]. e
success: function (res, status) {
console.log(status + JSON.stringify(res))
if (res.code == 0) {
alert("交易成功")
location.reload() ' `0 z7 U$ e, W# k; d, {5 L" E
} else {( N3 c( e5 U1 c7 _- c5 C& b6 ?2 V( {
alert("交易失败")' S) o k) R3 l. X( T- T8 G* p' a
}/ W& w9 I& f9 V& W
},$ Q4 S' Y" }" z3 W$ A. o
error: function (res, status) {
console.log(status + JSON.stringify(res)) q3 w3 y/ k! A& e3 q5 C
alert(res.data)
}+ v( C. \/ V. s; a0 T
});
}
})! W! _+ U' k; C' U5 N8 g5 e. B
})
b) r3 M' T$ ^7 ]0 g, T
成为第一个吐槽的人