com.openelements.hiero.base.implementation.ProtocolLayerClientImpl Maven / Gradle / Ivy
package com.openelements.hiero.base.implementation;
import com.google.protobuf.ByteString;
import com.hedera.hashgraph.sdk.AccountBalance;
import com.hedera.hashgraph.sdk.AccountBalanceQuery;
import com.hedera.hashgraph.sdk.AccountCreateTransaction;
import com.hedera.hashgraph.sdk.AccountDeleteTransaction;
import com.hedera.hashgraph.sdk.ContractCreateTransaction;
import com.hedera.hashgraph.sdk.ContractDeleteTransaction;
import com.hedera.hashgraph.sdk.ContractExecuteTransaction;
import com.hedera.hashgraph.sdk.ContractFunctionParameters;
import com.hedera.hashgraph.sdk.FileAppendTransaction;
import com.hedera.hashgraph.sdk.FileContentsQuery;
import com.hedera.hashgraph.sdk.FileCreateTransaction;
import com.hedera.hashgraph.sdk.FileDeleteTransaction;
import com.hedera.hashgraph.sdk.FileInfo;
import com.hedera.hashgraph.sdk.FileInfoQuery;
import com.hedera.hashgraph.sdk.FileUpdateTransaction;
import com.hedera.hashgraph.sdk.NftId;
import com.hedera.hashgraph.sdk.PrivateKey;
import com.hedera.hashgraph.sdk.PublicKey;
import com.hedera.hashgraph.sdk.Query;
import com.hedera.hashgraph.sdk.SubscriptionHandle;
import com.hedera.hashgraph.sdk.TokenAssociateTransaction;
import com.hedera.hashgraph.sdk.TokenBurnTransaction;
import com.hedera.hashgraph.sdk.TokenCreateTransaction;
import com.hedera.hashgraph.sdk.TokenMintTransaction;
import com.hedera.hashgraph.sdk.TopicCreateTransaction;
import com.hedera.hashgraph.sdk.TopicDeleteTransaction;
import com.hedera.hashgraph.sdk.TopicMessageQuery;
import com.hedera.hashgraph.sdk.TopicMessageSubmitTransaction;
import com.hedera.hashgraph.sdk.Transaction;
import com.hedera.hashgraph.sdk.TransactionReceipt;
import com.hedera.hashgraph.sdk.TransactionRecord;
import com.hedera.hashgraph.sdk.TransactionResponse;
import com.hedera.hashgraph.sdk.TransferTransaction;
import com.openelements.hiero.base.HieroContext;
import com.openelements.hiero.base.HieroException;
import com.openelements.hiero.base.data.Account;
import com.openelements.hiero.base.data.ContractParam;
import com.openelements.hiero.base.protocol.AccountBalanceRequest;
import com.openelements.hiero.base.protocol.AccountBalanceResponse;
import com.openelements.hiero.base.protocol.AccountCreateRequest;
import com.openelements.hiero.base.protocol.AccountCreateResult;
import com.openelements.hiero.base.protocol.AccountDeleteRequest;
import com.openelements.hiero.base.protocol.AccountDeleteResult;
import com.openelements.hiero.base.protocol.ContractCallRequest;
import com.openelements.hiero.base.protocol.ContractCallResult;
import com.openelements.hiero.base.protocol.ContractCreateRequest;
import com.openelements.hiero.base.protocol.ContractCreateResult;
import com.openelements.hiero.base.protocol.ContractDeleteRequest;
import com.openelements.hiero.base.protocol.ContractDeleteResult;
import com.openelements.hiero.base.protocol.FileAppendRequest;
import com.openelements.hiero.base.protocol.FileAppendResult;
import com.openelements.hiero.base.protocol.FileContentsRequest;
import com.openelements.hiero.base.protocol.FileContentsResponse;
import com.openelements.hiero.base.protocol.FileCreateRequest;
import com.openelements.hiero.base.protocol.FileCreateResult;
import com.openelements.hiero.base.protocol.FileDeleteRequest;
import com.openelements.hiero.base.protocol.FileDeleteResult;
import com.openelements.hiero.base.protocol.FileInfoRequest;
import com.openelements.hiero.base.protocol.FileInfoResponse;
import com.openelements.hiero.base.protocol.FileUpdateRequest;
import com.openelements.hiero.base.protocol.FileUpdateResult;
import com.openelements.hiero.base.protocol.ProtocolLayerClient;
import com.openelements.hiero.base.protocol.TokenAssociateRequest;
import com.openelements.hiero.base.protocol.TokenAssociateResult;
import com.openelements.hiero.base.protocol.TokenBurnRequest;
import com.openelements.hiero.base.protocol.TokenBurnResult;
import com.openelements.hiero.base.protocol.TokenCreateRequest;
import com.openelements.hiero.base.protocol.TokenCreateResult;
import com.openelements.hiero.base.protocol.TokenMintRequest;
import com.openelements.hiero.base.protocol.TokenMintResult;
import com.openelements.hiero.base.protocol.TokenTransferRequest;
import com.openelements.hiero.base.protocol.TokenTransferResult;
import com.openelements.hiero.base.protocol.TopicCreateRequest;
import com.openelements.hiero.base.protocol.TopicCreateResult;
import com.openelements.hiero.base.protocol.TopicDeleteRequest;
import com.openelements.hiero.base.protocol.TopicDeleteResult;
import com.openelements.hiero.base.protocol.TopicMessageRequest;
import com.openelements.hiero.base.protocol.TopicMessageResult;
import com.openelements.hiero.base.protocol.TopicSubmitMessageRequest;
import com.openelements.hiero.base.protocol.TopicSubmitMessageResult;
import com.openelements.hiero.base.protocol.TransactionListener;
import com.openelements.hiero.base.protocol.TransactionType;
import java.util.List;
import java.util.Objects;
import java.util.concurrent.CopyOnWriteArrayList;
import java.util.function.Consumer;
import org.jspecify.annotations.NonNull;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
public class ProtocolLayerClientImpl implements ProtocolLayerClient {
private static final Logger log = LoggerFactory.getLogger(ProtocolLayerClientImpl.class);
public static final int DEFAULT_GAS = 1_000_000;
private final List listeners;
private final HieroContext hieroContext;
public ProtocolLayerClientImpl(@NonNull final HieroContext hieroContext) {
this.hieroContext = Objects.requireNonNull(hieroContext, "hieroContext must not be null");
listeners = new CopyOnWriteArrayList<>();
}
@Override
public AccountBalanceResponse executeAccountBalanceQuery(@NonNull final AccountBalanceRequest request)
throws HieroException {
final AccountBalanceQuery query = new AccountBalanceQuery().setAccountId(request.accountId())
.setQueryPayment(request.queryPayment())
.setMaxQueryPayment(request.maxQueryPayment());
final AccountBalance balance = executeQueryAndWait(query);
return new AccountBalanceResponse(balance.hbars);
}
@Override
public FileContentsResponse executeFileContentsQuery(@NonNull final FileContentsRequest request)
throws HieroException {
final FileContentsQuery query = new FileContentsQuery().setFileId(request.fileId())
.setQueryPayment(request.queryPayment())
.setMaxQueryPayment(request.maxQueryPayment());
final ByteString byteString = executeQueryAndWait(query);
final byte[] bytes = byteString.toByteArray();
return new FileContentsResponse(request.fileId(), bytes);
}
@Override
public FileInfoResponse executeFileInfoQuery(@NonNull final FileInfoRequest request) throws HieroException {
Objects.requireNonNull(request, "request must not be null");
final FileInfoQuery query = new FileInfoQuery().setFileId(request.fileId())
.setQueryPayment(request.queryPayment())
.setMaxQueryPayment(request.maxQueryPayment());
final FileInfo fileInfo = executeQueryAndWait(query);
if (fileInfo.size > Integer.MAX_VALUE) {
throw new HieroException("File size is too large to be represented as an integer");
}
return new FileInfoResponse(request.fileId(), (int) fileInfo.size, fileInfo.isDeleted, fileInfo.expirationTime);
}
@Override
public FileCreateResult executeFileCreateTransaction(@NonNull final FileCreateRequest request)
throws HieroException {
Objects.requireNonNull(request, "request must not be null");
Objects.requireNonNull(request.contents(), "content must not be null");
if (request.contents().length > FileCreateRequest.FILE_CREATE_MAX_SIZE) {
throw new HieroException(
"File contents of 1 transaction must be less than " + FileCreateRequest.FILE_CREATE_MAX_SIZE
+ " bytes. Use FileAppend for larger files.");
}
final FileCreateTransaction transaction = new FileCreateTransaction()
.setMaxTransactionFee(request.maxTransactionFee())
.setTransactionValidDuration(request.transactionValidDuration())
.setContents(request.contents())
.setTransactionMemo(request.fileMemo())
.setKeys(Objects.requireNonNull(hieroContext.getOperatorAccount().publicKey()));
if (request.expirationTime() != null) {
transaction.setExpirationTime(request.expirationTime());
}
final TransactionReceipt receipt = executeTransactionAndWaitOnReceipt(transaction);
return new FileCreateResult(receipt.transactionId, receipt.status, receipt.fileId);
}
@Override
public FileUpdateResult executeFileUpdateRequestTransaction(@NonNull final FileUpdateRequest request)
throws HieroException {
Objects.requireNonNull(request, "request must not be null");
if (request.contents() != null && request.contents().length > FileCreateRequest.FILE_CREATE_MAX_SIZE) {
throw new HieroException(
"File contents of 1 transaction must be less than " + FileCreateRequest.FILE_CREATE_MAX_SIZE
+ " bytes. Use FileAppend for larger files.");
}
final FileUpdateTransaction transaction = new FileUpdateTransaction()
.setMaxTransactionFee(request.maxTransactionFee())
.setTransactionValidDuration(request.transactionValidDuration())
.setFileId(request.fileId())
.setTransactionMemo(request.fileMemo());
if (request.contents() != null) {
transaction.setContents(request.contents());
}
if (request.expirationTime() != null) {
transaction.setExpirationTime(request.expirationTime());
}
final TransactionReceipt receipt = executeTransactionAndWaitOnReceipt(transaction);
return new FileUpdateResult(receipt.transactionId, receipt.status);
}
@Override
public FileAppendResult executeFileAppendRequestTransaction(@NonNull final FileAppendRequest request)
throws HieroException {
Objects.requireNonNull(request, "request must not be null");
Objects.requireNonNull(request.contents(), "content must not be null");
if (request.contents().length > FileCreateRequest.FILE_CREATE_MAX_SIZE) {
throw new HieroException(
"File contents of 1 transaction must be less than " + FileCreateRequest.FILE_CREATE_MAX_SIZE
+ " bytes. Use multiple FileAppend for larger files.");
}
final FileAppendTransaction transaction = new FileAppendTransaction()
.setMaxTransactionFee(request.maxTransactionFee())
.setTransactionValidDuration(request.transactionValidDuration())
.setFileId(request.fileId())
.setContents(request.contents())
.setTransactionMemo(request.fileMemo());
final TransactionReceipt receipt = executeTransactionAndWaitOnReceipt(transaction);
return new FileAppendResult(receipt.transactionId, receipt.status);
}
@Override
public FileDeleteResult executeFileDeleteTransaction(@NonNull final FileDeleteRequest request)
throws HieroException {
final FileDeleteTransaction transaction = new FileDeleteTransaction()
.setMaxTransactionFee(request.maxTransactionFee())
.setTransactionValidDuration(request.transactionValidDuration())
.setFileId(request.fileId());
final TransactionReceipt receipt = executeTransactionAndWaitOnReceipt(transaction);
return new FileDeleteResult(receipt.transactionId, receipt.status);
}
@Override
public ContractCreateResult executeContractCreateTransaction(@NonNull final ContractCreateRequest request)
throws HieroException {
final ContractFunctionParameters constructorParams = createParameters(request.constructorParams());
final ContractCreateTransaction transaction = new ContractCreateTransaction()
.setMaxTransactionFee(request.maxTransactionFee())
.setTransactionValidDuration(request.transactionValidDuration())
.setBytecodeFileId(request.fileId())
.setGas(DEFAULT_GAS)
.setConstructorParameters(constructorParams);
final TransactionReceipt receipt = executeTransactionAndWaitOnReceipt(transaction);
return new ContractCreateResult(receipt.transactionId, receipt.status, receipt.contractId);
}
@Override
public ContractDeleteResult executeContractDeleteTransaction(@NonNull final ContractDeleteRequest request)
throws HieroException {
Objects.requireNonNull(request, "request must not be null");
final ContractDeleteTransaction transaction = new ContractDeleteTransaction()
.setMaxTransactionFee(request.maxTransactionFee())
.setTransactionValidDuration(request.transactionValidDuration())
.setContractId(request.contractId());
if (request.transferFeeToContractId() != null) {
transaction.setTransferContractId(request.transferFeeToContractId());
}
if (request.transferFeeToAccountId() != null) {
transaction.setTransferAccountId(request.transferFeeToAccountId());
}
final TransactionReceipt receipt = executeTransactionAndWaitOnReceipt(transaction);
return new ContractDeleteResult(receipt.transactionId, receipt.status);
}
@Override
@NonNull
public ContractCallResult executeContractCallTransaction(@NonNull final ContractCallRequest request)
throws HieroException {
Objects.requireNonNull(request, "request must not be null");
final ContractFunctionParameters functionParams = createParameters(request.constructorParams());
final ContractExecuteTransaction transaction = new ContractExecuteTransaction()
.setMaxTransactionFee(request.maxTransactionFee())
.setTransactionValidDuration(request.transactionValidDuration())
.setContractId(request.contractId())
.setFunction(request.functionName(), functionParams)
.setGas(DEFAULT_GAS);
final TransactionRecord record = executeTransactionAndWaitOnRecord(transaction);
return new ContractCallResult(record.transactionId, record.receipt.status, record.transactionHash.toByteArray(),
record.consensusTimestamp, record.transactionFee, record.contractFunctionResult);
}
@Override
@NonNull
public AccountCreateResult executeAccountCreateTransaction(@NonNull final AccountCreateRequest request)
throws HieroException {
Objects.requireNonNull(request, "request must not be null");
final PrivateKey privateKey = PrivateKey.generateED25519();
final PublicKey publicKey = privateKey.getPublicKey();
final AccountCreateTransaction transaction = new AccountCreateTransaction()
.setMaxTransactionFee(request.maxTransactionFee())
.setTransactionValidDuration(request.transactionValidDuration())
.setKey(publicKey)
.setInitialBalance(request.initialBalance());
final TransactionRecord record = executeTransactionAndWaitOnRecord(transaction);
final Account newAccount = Account.of(record.receipt.accountId, publicKey, privateKey);
return new AccountCreateResult(record.transactionId, record.receipt.status,
record.transactionHash.toByteArray(), record.consensusTimestamp, record.transactionFee, newAccount);
}
@Override
@NonNull
public AccountDeleteResult executeAccountDeleteTransaction(@NonNull final AccountDeleteRequest request)
throws HieroException {
Objects.requireNonNull(request, "request must not be null");
final AccountDeleteTransaction transaction = new AccountDeleteTransaction()
.setMaxTransactionFee(request.maxTransactionFee())
.setTransactionValidDuration(request.transactionValidDuration())
.setAccountId(request.toDelete().accountId());
if (request.transferFoundsToAccount() != null) {
transaction.setTransferAccountId(request.transferFoundsToAccount().accountId());
sign(transaction, request.toDelete().privateKey(), request.transferFoundsToAccount().privateKey());
} else {
transaction.setTransferAccountId(hieroContext.getOperatorAccount().accountId());
sign(transaction, request.toDelete().privateKey(), hieroContext.getOperatorAccount().privateKey());
}
final TransactionRecord record = executeTransactionAndWaitOnRecord(transaction);
return new AccountDeleteResult(record.transactionId, record.receipt.status,
record.transactionHash.toByteArray(), record.consensusTimestamp, record.transactionFee);
}
public TopicCreateResult executeTopicCreateTransaction(@NonNull final TopicCreateRequest request)
throws HieroException {
Objects.requireNonNull(request, "request must not be null");
Objects.requireNonNull(request.maxTransactionFee(), "maxTransactionFee must not be null");
Objects.requireNonNull(request.transactionValidDuration(), "transactionValidDuration must not be null");
try {
final TopicCreateTransaction transaction = new TopicCreateTransaction()
.setMaxTransactionFee(request.maxTransactionFee())
.setTransactionValidDuration(request.transactionValidDuration());
final TransactionReceipt receipt = executeTransactionAndWaitOnReceipt(transaction);
return new TopicCreateResult(receipt.transactionId, receipt.status, receipt.topicId);
} catch (final Exception e) {
throw new HieroException("Failed to execute create topic transaction", e);
}
}
public TopicDeleteResult executeTopicDeleteTransaction(@NonNull final TopicDeleteRequest request)
throws HieroException {
Objects.requireNonNull(request, "request must not be null");
try {
final TopicDeleteTransaction transaction = new TopicDeleteTransaction()
.setMaxTransactionFee(request.maxTransactionFee())
.setTransactionValidDuration(request.transactionValidDuration())
.setTopicId(request.topicId());
final TransactionReceipt receipt = executeTransactionAndWaitOnReceipt(transaction);
return new TopicDeleteResult(receipt.transactionId, receipt.status);
} catch (final Exception e) {
throw new HieroException("Failed to execute delete topic transaction", e);
}
}
public TopicSubmitMessageResult executeTopicMessageSubmitTransaction(
@NonNull final TopicSubmitMessageRequest request) throws HieroException {
Objects.requireNonNull(request, "request must not be null");
try {
final TopicMessageSubmitTransaction transaction = new TopicMessageSubmitTransaction()
.setMaxTransactionFee(request.maxTransactionFee())
.setTransactionValidDuration(request.transactionValidDuration())
.setTopicId(request.topicId())
.setMessage(request.message());
final TransactionReceipt receipt = executeTransactionAndWaitOnReceipt(transaction);
return new TopicSubmitMessageResult(receipt.transactionId, receipt.status);
} catch (final Exception e) {
throw new HieroException("Failed to execute submit message transaction", e);
}
}
@Override
public TopicMessageResult executeTopicMessageQuery(TopicMessageRequest request) throws HieroException {
Objects.requireNonNull(request, "request must not be null");
try {
final TopicMessageQuery query = new TopicMessageQuery()
.setTopicId(request.topicId());
if (request.startTime() != null) {
query.setStartTime(request.startTime());
}
if (request.endTime() != null) {
query.setEndTime(request.endTime());
}
if (request.limit() >= 0) {
query.setLimit(request.limit());
}
final SubscriptionHandle subscribe = query.subscribe(hieroContext.getClient(), request.subscription());
return new TopicMessageResult();
} catch (final Exception e) {
throw new HieroException("Failed to execute query message transaction", e);
}
}
public TokenCreateResult executeTokenCreateTransaction(@NonNull final TokenCreateRequest request)
throws HieroException {
Objects.requireNonNull(request, "request must not be null");
try {
final TokenCreateTransaction transaction = new TokenCreateTransaction()
.setMaxTransactionFee(request.maxTransactionFee())
.setTransactionValidDuration(request.transactionValidDuration())
.setTokenName(request.name())
.setTokenSymbol(request.symbol())
.setTreasuryAccountId(request.treasuryAccountId())
.setTokenType(request.tokenType())
.setSupplyKey(request.supplyKey());
sign(transaction, request.treasuryKey(), request.supplyKey());
final TransactionReceipt receipt = executeTransactionAndWaitOnReceipt(transaction);
return new TokenCreateResult(receipt.transactionId, receipt.status, receipt.tokenId);
} catch (final Exception e) {
throw new HieroException("Failed to execute create token transaction", e);
}
}
public TokenAssociateResult executeTokenAssociateTransaction(@NonNull final TokenAssociateRequest request)
throws HieroException {
Objects.requireNonNull(request, "request must not be null");
try {
final TokenAssociateTransaction transaction = new TokenAssociateTransaction()
.setMaxTransactionFee(request.maxTransactionFee())
.setTransactionValidDuration(request.transactionValidDuration())
.setTokenIds(List.of(request.tokenId()))
.setAccountId(request.accountId());
sign(transaction, request.accountPrivateKey());
final TransactionReceipt receipt = executeTransactionAndWaitOnReceipt(transaction);
return new TokenAssociateResult(receipt.transactionId, receipt.status);
} catch (final Exception e) {
throw new HieroException("Failed to execute associate token transaction", e);
}
}
public TokenBurnResult executeBurnTokenTransaction(@NonNull final TokenBurnRequest request) throws HieroException {
Objects.requireNonNull(request, "request must not be null");
try {
final TokenBurnTransaction transaction = new TokenBurnTransaction()
.setMaxTransactionFee(request.maxTransactionFee())
.setTransactionValidDuration(request.transactionValidDuration())
.setTokenId(request.tokenId());
if (request.amount() != null) {
transaction.setAmount(request.amount());
} else if (request.serials() != null) {
transaction.setSerials(List.copyOf(request.serials()));
} else {
throw new IllegalArgumentException("either amount or serial must be provided");
}
final TransactionReceipt receipt = executeTransactionAndWaitOnReceipt(transaction);
return new TokenBurnResult(receipt.transactionId, receipt.status);
} catch (final Exception e) {
throw new HieroException("Failed to execute burn token transaction", e);
}
}
public TokenMintResult executeMintTokenTransaction(@NonNull final TokenMintRequest request) throws HieroException {
Objects.requireNonNull(request, "request must not be null");
try {
final TokenMintTransaction transaction = new TokenMintTransaction()
.setMaxTransactionFee(request.maxTransactionFee())
.setTransactionValidDuration(request.transactionValidDuration())
.setTokenId(request.tokenId());
if (request.amount() != null) {
transaction.setAmount(request.amount());
} else if (request.metadata() != null) {
transaction.setMetadata(request.metadata());
} else {
throw new IllegalArgumentException("either amount or metadata must be provided");
}
sign(transaction, request.supplyKey());
final TransactionReceipt receipt = executeTransactionAndWaitOnReceipt(transaction);
return new TokenMintResult(receipt.transactionId, receipt.status, receipt.serials);
} catch (final Exception e) {
throw new HieroException("Failed to execute mint token transaction", e);
}
}
public TokenTransferResult executeTransferTransactionForNft(@NonNull final TokenTransferRequest request)
throws HieroException {
Objects.requireNonNull(request, "request must not be null");
try {
final TransferTransaction transaction = new TransferTransaction()
.setMaxTransactionFee(request.maxTransactionFee())
.setTransactionValidDuration(request.transactionValidDuration());
request.serials().forEach(
serial -> transaction.addNftTransfer(new NftId(request.tokenId(), serial), request.sender(),
request.receiver()));
sign(transaction, request.senderKey());
final TransactionReceipt receipt = executeTransactionAndWaitOnReceipt(transaction);
return new TokenTransferResult(receipt.transactionId, receipt.status);
} catch (final Exception e) {
throw new HieroException("Failed to execute transfer nft transaction", e);
}
}
@NonNull
private > Transaction sign(Transaction transaction, final PrivateKey... keys) {
if (keys != null) {
transaction.freezeWith(hieroContext.getClient());
for (PrivateKey key : keys) {
transaction.sign(key);
}
}
return transaction;
}
@NonNull
private ContractFunctionParameters createParameters(@NonNull final List> params) {
Objects.requireNonNull(params, "params must not be null");
final ContractFunctionParameters constructorParams = new ContractFunctionParameters();
final Consumer consumer = param -> param.supplier()
.addParamToFunctionParameters(param.value(), constructorParams);
params.forEach(consumer);
return constructorParams;
}
@NonNull
private > TransactionReceipt executeTransactionAndWaitOnReceipt(
@NonNull final T transaction) throws HieroException {
Objects.requireNonNull(transaction, "transaction must not be null");
try {
log.debug("Sending transaction of type {}", transaction.getClass().getSimpleName());
final TransactionResponse response = transaction.execute(hieroContext.getClient());
listeners.forEach(listener -> {
try {
listener.transactionSubmitted(TransactionType.ACCOUNT_CREATE, response.transactionId);
} catch (Exception e) {
log.error("Failed to notify listener", e);
}
});
try {
log.debug("Waiting for receipt of transaction '{}' of type {}", response.transactionId,
transaction.getClass().getSimpleName());
final TransactionReceipt receipt = response.getReceipt(hieroContext.getClient());
listeners.forEach(listener -> {
try {
listener.transactionHandled(TransactionType.ACCOUNT_CREATE, response.transactionId,
receipt.status);
} catch (Exception e) {
log.error("Failed to notify listener", e);
}
});
return receipt;
} catch (Exception e) {
throw new HieroException(
"Failed to receive receipt of transaction '" + response.transactionId + "' of type "
+ transaction.getClass(), e);
}
} catch (final Exception e) {
throw new HieroException("Failed to execute transaction of type " + transaction.getClass().getSimpleName(),
e);
}
}
@NonNull
private > TransactionRecord executeTransactionAndWaitOnRecord(@NonNull final T transaction)
throws HieroException {
final TransactionReceipt receipt = executeTransactionAndWaitOnReceipt(transaction);
try {
log.debug("Waiting for record of transaction '{}' of type {}", receipt.transactionId,
transaction.getClass().getSimpleName());
return receipt.transactionId.getRecord(hieroContext.getClient());
} catch (final Exception e) {
throw new HieroException("Failed to receive record of transaction '" + receipt.transactionId + "' of type "
+ transaction.getClass(), e);
}
}
@NonNull
private > R executeQueryAndWait(@NonNull final Q query) throws HieroException {
Objects.requireNonNull(query, "query must not be null");
try {
log.debug("Sending query of type {}", query.getClass().getSimpleName());
return query.execute(hieroContext.getClient());
} catch (Exception e) {
throw new HieroException("Failed to execute query", e);
}
}
@NonNull
@Override
public Runnable addTransactionListener(@NonNull TransactionListener listener) {
listeners.add(listener);
return () -> listeners.remove(listener);
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy