java使用spring boot和web3j开发以太坊应用。
朋友一起走
发表于 2022-11-13 23:50:49
102
1
0
于是以太坊来了。这是一个Vitarik Buterin提供的去中心化的平台,可以通过脚本语言创建开发应用。它的想法是从比特币获得的,并由新的叫Ether即以太坊币的加密数字币驱动。今天,以太币是继比特币之后的第二大加密数字货币。以太坊技术的核心是EVM(以太坊虚拟机),它可以被视为类似于Java虚拟机,而且用一种完全去中心化的节点网络。基于java世界实现以太坊交易我们使用web3j库。这是一个轻量级的、响应式、类型安全的java和Android库结合了以太坊区块链节点。更多的细节可以在这里找到web3j.io中文版。6 c# v# r: h5 d! E
1.本地运行3 d s0 W/ n! Q+ J$ n8 K
虽然有许多针对区块链文章,但以太坊相关的网络内容中不容易找到一个解决方案描述如何准备在本地机器使用实例运行以太坊。值得一提的是,一般有两种最基本的客户端可以使用:Geth和Parity。原来,我们可以很容易地在本地使用Docker容器运行节点。默认情况下,连接节点的以太坊主网络(公有链)。或者,你可以将它连接到测试网络或Rinkeby网络。但开始最好的选择就是运行在设置了开发参数(–dev)的开发模式下,并在Docker容器中运行命令。7 U& J* `4 k" b0 ?% m
下面的命令启动Docker容器开发模式在端口8545调用以太坊RPC API。
$ docker run -d --name ethereum -p 8545:8545 -p 30303:30303 ethereum/client-go --rpc --rpcaddr "0.0.0.0" --rpcapi="db,eth,net,web3,personal" --rpccorsdomain "*" --dev
在开发模式中运行该容器时,一个非常好的消息是,在默认的测试帐户上有大量的Ether。在这种情况下,你不必挖掘任何Ether,便能够开始测试。超级棒!现在,让我们创建一些其他的测试帐号,并做一些检查。为了实现这一点,我们需要在容器内部运行Geth的交互式JavaScript控制台。
$ docker exec -it ethereum geth attach ipc:/tmp/geth.ipc$ T$ g3 P' u: |. |! p
2.以太坊节点使用JavaScript控制台管理
运行JavaScript控制台可以方便显示默认帐户(Coinbase),所有可用的账户及其余额清单。这里的屏幕显示我的以太坊结果。
现在,我们必须创建一些测试帐号。我们可以通过调用personal.newAccount(password)函数来实现这一点。在创建必需的帐户之后,我们可以使用JavaScript控制台执行一些测试交易,并将一些资金从基础帐户转移到新创建的帐户。下面是用于创建帐户和执行交易的命令。
' N, G: t8 S7 [8 F& v2 |$ t3 S
3.系统体系结构' ] X, ?" I' f, k! \7 `. K
我们的demo系统的体系结构非常简单。不用想复杂的事情,只是告诉大家如何发送交易到geth节点和接收交易收据。而transaction-service发送新交易到以太坊节点,bonus-service节点监听传入的交易。然后每10笔交易发送者的帐户收到一次奖金(bonus)。这里的图表说明了一个我们的demo的系统架构。0 N, \& {* ]8 Y
4.spring boot应用程序使用web3j. [, y7 n [/ p6 O# ~; l4 K/ {$ R
我想现在我们清楚了我们到底想做什么。所以,让我们来进行实施。首先,我们应该包括所有必需的依赖项,以便能够在Spring boot应用程序中使用web3j库。幸运的是,有一个starter可以使用。: y# g' M+ F; Z- s k/ t' g1 I
( C- [# |/ _3 @. u! C1 Y
org.web3j8 i( M3 q, i: x5 p5 J
web3j-spring-boot-starter! O& @' c% q3 U2 I6 j# z) y0 p
1.6.0. W/ L( Y( `9 b0 R6 I6 A7 t
) F& w; C& u2 s$ _
因为我们在Docker容器运行以太坊客户端需要改变客户端的自动默认配置的web3j的调用地址。$ Q+ R. D o& l2 X3 N3 |8 ]
spring:
application:& `9 v, b' B4 y7 {3 c( C
name: transaction-service
server:- a) r' G# E" Q1 X( r
port: ${PORT:8090}( M2 {' y: U. G
web3j:8 R" I9 t5 z7 r! C
client-address: http://192.168.99.100:8545
5.构建应用- K6 l& j/ Q$ F! k& u
如果我们将web3j starter包含到项目依赖项中,需要的是自动装载web3j bean。web3j负责向Geth客户端节点发送交易。它用交易哈希接收响应,不管是节点接受或由于错误被拒绝。在创建交易对象时,重要的是将gas限制最小值设置为21000。如果发送较低的值,则可能会收到错误信息:intrinsic gas too low。* b6 W6 j) ~ ?/ M; P
@Service0 Z! }" w" h2 @* _& c1 S+ |% w+ A
public class BlockchainService {
( N9 G1 ^8 j/ R N t% Q, l: {
private static final Logger LOGGER = LoggerFactory.getLogger(BlockchainService.class);
@Autowired
Web3j web3j;
public BlockchainTransaction process(BlockchainTransaction trx) throws IOException {
EthAccounts accounts = web3j.ethAccounts().send();
EthGetTransactionCount transactionCount = web3j.ethGetTransactionCount(accounts.getAccounts().get(trx.getFromId()), DefaultBlockParameterName.LATEST).send();0 n7 c: u. g, l5 S% @+ b' h+ |! K
Transaction transaction = Transaction.createEtherTransaction(accounts.getAccounts().get(trx.getFromId()), transactionCount.getTransactionCount(), BigInteger.valueOf(trx.getValue()), BigInteger.valueOf(21_000), accounts.getAccounts().get(trx.getToId()),BigInteger.valueOf(trx.getValue()));; _2 s, ?, O: G9 E( D+ ~/ R
EthSendTransaction response = web3j.ethSendTransaction(transaction).send();
if (response.getError() != null) {
trx.setAccepted(false);
return trx;0 g) E1 h) ?4 h2 g' @, l
}4 C/ W P5 \; M" h
trx.setAccepted(true);0 l8 i: \8 C9 g0 O4 o1 r
String txHash = response.getTransactionHash();
LOGGER.info("Tx hash: {}", txHash);. x* `! M4 b0 y
trx.setId(txHash);
EthGetTransactionReceipt receipt = web3j.ethGetTransactionReceipt(txHash).send();- H) q6 X% m; T
if (receipt.getTransactionReceipt().isPresent()) {
LOGGER.info("Tx receipt: {}", receipt.getTransactionReceipt().get().getCumulativeGasUsed().intValue());9 \3 _9 X, [7 p" M, Q
}
return trx;
}
}
@Service从上面的代码看由控制器调用。POST方法的需要BlockchainTransaction对象作为参数。你可以发送发件人ID,接收人ID和交易金额。发送者和接收者ID是通过eth.account[index]可查询的。! C; H, c* h) i! V H
@RestController3 S9 X( {3 b, {7 d
public class BlockchainController {9 v s$ B! h9 i
1 D2 e& G7 y2 G$ z( q; d
@Autowired+ e, m( I4 ?6 u: [) E O' ?* o7 h
BlockchainService service;2 M$ ~2 W8 N `( a: A# @. M
@PostMapping("/transaction")* U4 J Q% Y9 C1 z3 }
public BlockchainTransaction execute(@RequestBody BlockchainTransaction transaction) throws NoSuchAlgorithmException, NoSuchProviderException, InvalidAlgorithmParameterException, CipherException, IOException {
return service.process(transaction);- x$ L/ p0 R- b: s8 P+ o
}5 G% A; n8 K/ k8 f# }3 R
( d# H5 _& Z3 {' t1 c: a4 \8 B
}6 p6 T# t$ T. x: H }: H
你可以用下面的命令调用POST方法发送测试交易。
在发送任何交易之前,你应该解锁发送人帐户。
$ curl --header "Content-Type: application/json" --request POST --data '{"fromId":2,"toId":1,"value":3}' http://localhost:8090/transaction# |6 P/ e1 _. a! O
应用程序bonus-service监听由以太坊节点处理的交易。它通过调用web3j.transactionObservable().subscribe(...)方法从web3j库订阅通知消息。它将从该地址每10个交易返回后,发送一次到发送者的帐户。下面是bonus-service中可监听方法的实现。
@Autowired% T' r" t0 Q, A4 y: F- Y9 [) S+ \7 v
Web3j web3j;* Q6 l! R: S: {; M
@PostConstruct1 v/ C9 @% T' D8 K2 d$ P0 q6 \1 r! r, f
public void listen() {% `. d1 V2 v. y7 n
Subscription subscription = web3j.transactionObservable().subscribe(tx -> {0 r$ B1 W' r1 x4 i& K4 F2 { Z; P
LOGGER.info("New tx: id={}, block={}, from={}, to={}, value={}", tx.getHash(), tx.getBlockHash(), tx.getFrom(), tx.getTo(), tx.getValue().intValue());& F" N, D F; q0 a
try {
EthCoinbase coinbase = web3j.ethCoinbase().send(); k7 I6 G) I- G2 s- o% G
EthGetTransactionCount transactionCount = web3j.ethGetTransactionCount(tx.getFrom(), DefaultBlockParameterName.LATEST).send();6 n7 X5 B& R# B( a4 L) X/ m# Y
LOGGER.info("Tx count: {}", transactionCount.getTransactionCount().intValue());; e Q! H2 V5 d+ q' L
if (transactionCount.getTransactionCount().intValue() % 10 == 0) {0 M5 \8 h4 M; n) `7 D/ c1 ?9 v3 W
EthGetTransactionCount tc = web3j.ethGetTransactionCount(coinbase.getAddress(), DefaultBlockParameterName.LATEST).send();
Transaction transaction = Transaction.createEtherTransaction(coinbase.getAddress(), tc.getTransactionCount(), tx.getValue(), BigInteger.valueOf(21_000), tx.getFrom(), tx.getValue());
web3j.ethSendTransaction(transaction).send();' x+ o ]9 [6 n- N+ _1 y
}$ }( H7 a2 D2 h* s1 g8 i* [
} catch (IOException e) {
LOGGER.error("Error getting transactions", e);
}% D4 J- A2 \7 {( p+ q& i* \6 ]( d
});3 j C3 W1 P/ X
LOGGER.info("Subscribed");- q* D$ U3 |0 g2 w/ H' H
总结( ^$ L; Q! z1 r/ ]1 p7 v1 n. z5 G
区块链和数字货币不是容易开始的话题。通过提供完整的脚本语言,以太坊简化了使用区块链进行应用程序开发的难度。使用web3j、spring boot和以太坊geth客户端的docker容器镜像,可以快速启动解决方案,实现区块链技术的本地开发。
如果你想进行本地开发时clone我的库,可以在github上下载源代码。- Q6 P$ A2 z' N) o
如果希望快速进行web3j、java、以太坊开发,那请看我们精心打造的教程:/ r; r2 h7 F4 J* K/ \7 _, z; W
web3j教程,主要是针对java和android程序员进行区块链以太坊开发的web3j详解。
BitMere.com 比特池塘系信息发布平台,比特池塘仅提供信息存储空间服务。
声明:该文观点仅代表作者本人,本文不代表比特池塘立场,且不构成建议,请谨慎对待。
声明:该文观点仅代表作者本人,本文不代表比特池塘立场,且不构成建议,请谨慎对待。
成为第一个吐槽的人