java - 使用 Java SDK Client for Hyperledger Fabric V1.0 的应用程序在调用链码时无限期等待

标签 java blockchain hyperledger hyperledger-fabric

我按照 Building Your First Network 的步骤启动并运行了我的 Hyperledger Fabric V1.0 网络 .

现在我可以创建 channel 安装/实例化/调用/查询链代码等。

现在我正在尝试创建一些 Assets 并使用 Java SDK Client 查询相同的 Assets .

我创建了以下方法来从我的 java 应用程序调用和查询链代码。

void createChannel() throws InvalidArgumentException, TransactionException, IOException, ProposalException{
    Properties ordererProperties = getOrdererProperties("orderer.example.com");
    ordererProperties.put("grpc.NettyChannelBuilderOption.keepAliveTime", new Object[] {5L, TimeUnit.MINUTES});
    ordererProperties.put("grpc.NettyChannelBuilderOption.keepAliveTimeout", new Object[] {8L, TimeUnit.SECONDS});
    Orderer orderer = client.newOrderer("orderer.example.com", "grpcs://192.168.99.100:7050",ordererProperties);

    Properties peerProperties = getPeerProperties("peer0.org1.example.com"); //test properties for peer.. if any.
    if (peerProperties == null) {
        peerProperties = new Properties();
    }
    peerProperties.put("grpc.NettyChannelBuilderOption.maxInboundMessageSize", 9000000);
    Peer peer = client.newPeer("peer0.org1.example.com", "grpcs://192.168.99.100:7051",peerProperties);
    channel = client.newChannel("testchannel");
    channel.addOrderer(orderer);
    channel.addPeer(peer);
    channel.initialize();
}

void creteTransactionalProposal(){
    proposalRequest = client.newTransactionProposalRequest();
    final ChaincodeID chaincodeID = ChaincodeID.newBuilder()
            .setName("asset_test")
            .setVersion("1.0")
            .setPath("github.com/myuser/myfabricrepo/asset_chain")
            .build();

    proposalRequest.setChaincodeID(chaincodeID);
    proposalRequest.setFcn("set");
    proposalRequest.setProposalWaitTime(TimeUnit.SECONDS.toMillis(1));
    proposalRequest.setArgs(new String[]{"a1", "a1_val"});
}

void sendProposal() throws ProposalException, InvalidArgumentException, InterruptedException, ExecutionException{
    final Collection<ProposalResponse> responses = channel.sendTransactionProposal(proposalRequest);
    CompletableFuture<BlockEvent.TransactionEvent> txFuture = channel.sendTransaction(responses, client.getUserContext());
    BlockEvent.TransactionEvent event = txFuture.get();//waiting indefinitely
    System.out.println(event.toString());
    //query();
}

void query() throws InvalidArgumentException, ProposalException{
     final ChaincodeID chaincodeID = ChaincodeID.newBuilder()
                .setName(""asset_test"")
                .setVersion("1.0")
                .setPath("github.com/myuser/myfabricrepo/asset_chain")
                .build();

    QueryByChaincodeRequest queryByChaincodeRequest = client.newQueryProposalRequest();
    queryByChaincodeRequest.setArgs(new String[] {"a1"});
    queryByChaincodeRequest.setFcn("get");
    queryByChaincodeRequest.setChaincodeID(chaincodeID);

    Map<String, byte[]> tm2 = new HashMap<>();
    tm2.put("HyperLedgerFabric", "QueryByChaincodeRequest:JavaSDK".getBytes(UTF_8));
    tm2.put("method", "QueryByChaincodeRequest".getBytes(UTF_8));
    queryByChaincodeRequest.setTransientMap(tm2);

    Collection<ProposalResponse> queryProposals = channel.queryByChaincode(queryByChaincodeRequest, channel.getPeers());
    for (ProposalResponse proposalResponse : queryProposals) {
        if (!proposalResponse.isVerified()
                || proposalResponse.getStatus() != ProposalResponse.Status.SUCCESS) {
            System.out.println("Failed query proposal from peer " + proposalResponse.getPeer().getName() + " status: "
                    + proposalResponse.getStatus() + ". Messages: " + proposalResponse.getMessage()
                    + ". Was verified : " + proposalResponse.isVerified());
        } else {
            String payload = proposalResponse.getProposalResponse().getResponse().getPayload()
                    .toStringUtf8();
            System.out.printf("\nQuery payload of b from peer %s returned %s", proposalResponse.getPeer().getName(),
                    payload);
            //assertEquals(payload, expect);
        }
    }
}

我可以通过调用来创建 Assets

t.creteTransactionalProposal();
t.sendProposal();

但是,即使在事务提交到账本之后,BlockEvent.TransactionEvent event = txFuture.get(); 行也会使应用程序处于无限期等待状态。为什么会有这样的行为?

一旦我强制退出并运行 query() 方法,它就会列出 Assets 。

最佳答案

我遇到了与此类似的问题,网上的许多答案都缺少代码的关键部分 - 将 EventHub 分配给 channel 。我在初始化 channel 之前添加了此内容(在本例中将在 createChannel 方法中),然后我的交易被成功处理:

    channel.addEventHub(client.newEventHub("eventhub0", "grpc://localhost:7053"));

关于java - 使用 Java SDK Client for Hyperledger Fabric V1.0 的应用程序在调用链码时无限期等待,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/45990076/

相关文章:

java - jGraph 中的单端边

c - 当我通过 Cgo 导入 2 个使用 C 包的不同包时,Go 编译返回体系结构 x86_64 错误的重复符号

hyperledger-fabric - 如何在 Hyperledger Composer 中从 CA 获取身份证书

hyperledger-fabric - Hyperledger Composer 发出身份信息但缺少名片

python - 如何通过 API 获取 Cardano 未花费的交易输出(UTxO)?

blockchain - Hyperledger Fabric 1.0 中的属性(property)级隐私

java - Android - 从微调器中选择项目后,如何使微调器显示该项目?

java - 用于处理表单的简单 servlet 或过滤器

java - 反射性能 : quality byte code in JVM

blockchain - IPFS 作为去中心化服务的交易成本