Search in sources :

Example 1 with BlindVote

use of bisq.core.dao.governance.blindvote.BlindVote in project bisq-core by bisq-network.

the class VoteResultService method getDecryptedBallotsWithMeritsSet.

private Set<DecryptedBallotsWithMerits> getDecryptedBallotsWithMeritsSet(int chainHeight) {
    // We want all voteRevealTxOutputs which are in current cycle we are processing.
    return bsqStateService.getVoteRevealOpReturnTxOutputs().stream().filter(txOutput -> periodService.isTxInCorrectCycle(txOutput.getTxId(), chainHeight)).map(txOutput -> {
        // TODO make method
        byte[] opReturnData = txOutput.getOpReturnData();
        String voteRevealTxId = txOutput.getTxId();
        Optional<Tx> optionalVoteRevealTx = bsqStateService.getTx(voteRevealTxId);
        if (!optionalVoteRevealTx.isPresent()) {
            log.error("optionalVoteRevealTx is not present. voteRevealTxId={}", voteRevealTxId);
            // TODO throw exception
            return null;
        }
        Tx voteRevealTx = optionalVoteRevealTx.get();
        try {
            // TODO maybe verify version in opReturn
            byte[] hashOfBlindVoteList = VoteResultConsensus.getHashOfBlindVoteList(opReturnData);
            SecretKey secretKey = VoteResultConsensus.getSecretKey(opReturnData);
            TxOutput blindVoteStakeOutput = VoteResultConsensus.getConnectedBlindVoteStakeOutput(voteRevealTx, bsqStateService);
            long blindVoteStake = blindVoteStakeOutput.getValue();
            Tx blindVoteTx = VoteResultConsensus.getBlindVoteTx(blindVoteStakeOutput, bsqStateService, periodService, chainHeight);
            String blindVoteTxId = blindVoteTx.getId();
            // Here we deal with eventual consistency of the p2p network data!
            // TODO make more clear we are in p2p domain now
            List<BlindVote> blindVoteList = BlindVoteConsensus.getSortedBlindVoteListOfCycle(blindVoteListService);
            Optional<BlindVote> optionalBlindVote = blindVoteList.stream().filter(blindVote -> blindVote.getTxId().equals(blindVoteTxId)).findAny();
            if (optionalBlindVote.isPresent()) {
                BlindVote blindVote = optionalBlindVote.get();
                VoteWithProposalTxIdList voteWithProposalTxIdList = VoteResultConsensus.decryptVotes(blindVote.getEncryptedVotes(), secretKey);
                MeritList meritList = MeritConsensus.decryptMeritList(blindVote.getEncryptedMeritList(), secretKey);
                // We lookup for the proposals we have in our local list which match the txId from the
                // voteWithProposalTxIdList and create a ballot list with the proposal and the vote from
                // the voteWithProposalTxIdList
                BallotList ballotList = createBallotList(voteWithProposalTxIdList);
                return new DecryptedBallotsWithMerits(hashOfBlindVoteList, voteRevealTxId, blindVoteTxId, blindVoteStake, ballotList, meritList);
            } else {
                // TODO handle recovering
                log.warn("We have a blindVoteTx but we do not have the corresponding blindVote in our local list.\n" + "That can happen if the blindVote item was not properly broadcast. We will go on " + "and see if that blindVote was part of the majority data view. If so we should " + "recover the missing blind vote by a request to our peers. blindVoteTxId={}", blindVoteTxId);
                return null;
            }
        } catch (MissingBallotException e) {
            // TODO handle case that we are missing proposals
            log.error("We are missing proposals to create the vote result: " + e.toString());
            return null;
        } catch (Throwable e) {
            log.error("Could not create DecryptedBallotsWithMerits: " + e.toString());
            return null;
        }
    }).filter(Objects::nonNull).collect(Collectors.toSet());
}
Also used : VoteWithProposalTxId(bisq.core.dao.governance.blindvote.VoteWithProposalTxId) Arrays(java.util.Arrays) Utilities(bisq.common.util.Utilities) Vote(bisq.core.dao.governance.ballot.vote.Vote) P2PDataStorage(bisq.network.p2p.storage.P2PDataStorage) Preconditions.checkArgument(com.google.common.base.Preconditions.checkArgument) BlindVoteListService(bisq.core.dao.governance.blindvote.BlindVoteListService) Map(java.util.Map) VoteRevealConsensus(bisq.core.dao.governance.votereveal.VoteRevealConsensus) ChangeParamProposal(bisq.core.dao.governance.proposal.param.ChangeParamProposal) BlindVote(bisq.core.dao.governance.blindvote.BlindVote) CompensationProposal(bisq.core.dao.governance.proposal.compensation.CompensationProposal) VoteRevealService(bisq.core.dao.governance.votereveal.VoteRevealService) BondedRolesService(bisq.core.dao.governance.role.BondedRolesService) BallotListService(bisq.core.dao.governance.ballot.BallotListService) Proposal(bisq.core.dao.governance.proposal.Proposal) PeriodService(bisq.core.dao.state.period.PeriodService) Set(java.util.Set) ConfiscateBond(bisq.core.dao.state.governance.ConfiscateBond) Collectors(java.util.stream.Collectors) ProposalListPresentation(bisq.core.dao.governance.proposal.ProposalListPresentation) Objects(java.util.Objects) List(java.util.List) Slf4j(lombok.extern.slf4j.Slf4j) IssuanceService(bisq.core.dao.governance.voteresult.issuance.IssuanceService) Optional(java.util.Optional) BallotList(bisq.core.dao.governance.ballot.BallotList) ObservableList(javafx.collections.ObservableList) SecretKey(javax.crypto.SecretKey) Getter(lombok.Getter) BondedRoleProposal(bisq.core.dao.governance.proposal.role.BondedRoleProposal) TxOutput(bisq.core.dao.state.blockchain.TxOutput) MeritConsensus(bisq.core.dao.governance.merit.MeritConsensus) FXCollections(javafx.collections.FXCollections) HashMap(java.util.HashMap) Block(bisq.core.dao.state.blockchain.Block) ArrayList(java.util.ArrayList) Value(lombok.Value) Inject(javax.inject.Inject) HashSet(java.util.HashSet) ParamChange(bisq.core.dao.state.governance.ParamChange) BondedRole(bisq.core.dao.governance.role.BondedRole) Nullable(javax.annotation.Nullable) MeritList(bisq.core.dao.governance.merit.MeritList) ConfiscateBondProposal(bisq.core.dao.governance.proposal.confiscatebond.ConfiscateBondProposal) Ballot(bisq.core.dao.governance.ballot.Ballot) Tx(bisq.core.dao.state.blockchain.Tx) BsqStateService(bisq.core.dao.state.BsqStateService) DaoSetupService(bisq.core.dao.DaoSetupService) BlindVoteConsensus(bisq.core.dao.governance.blindvote.BlindVoteConsensus) VoteWithProposalTxIdList(bisq.core.dao.governance.blindvote.VoteWithProposalTxIdList) BsqStateListener(bisq.core.dao.state.BsqStateListener) Comparator(java.util.Comparator) DaoPhase(bisq.core.dao.state.period.DaoPhase) TxOutput(bisq.core.dao.state.blockchain.TxOutput) Tx(bisq.core.dao.state.blockchain.Tx) VoteWithProposalTxIdList(bisq.core.dao.governance.blindvote.VoteWithProposalTxIdList) MeritList(bisq.core.dao.governance.merit.MeritList) BlindVote(bisq.core.dao.governance.blindvote.BlindVote) BallotList(bisq.core.dao.governance.ballot.BallotList) SecretKey(javax.crypto.SecretKey)

Example 2 with BlindVote

use of bisq.core.dao.governance.blindvote.BlindVote in project bisq-core by bisq-network.

the class VoteRevealService method revealVote.

private void revealVote(MyVote myVote, int chainHeight) throws IOException, WalletException, InsufficientMoneyException, TransactionVerificationException, VoteRevealException {
    // We collect all valid blind vote items we received via the p2p network.
    // It might be that different nodes have a different collection of those items.
    // To ensure we get a consensus of the data for later calculating the result we will put a hash of each
    // voters  blind vote collection into the opReturn data and check for a majority at issuance time.
    // The voters "vote" with their stake at the reveal tx for their version of the blind vote collection.
    // TODO make more clear by using param like here:
    /* List<BlindVote> blindVotes = BlindVoteConsensus.getSortedBlindVoteListOfCycle(blindVoteListService);
         VoteRevealConsensus.getHashOfBlindVoteList(blindVotes);*/
    byte[] hashOfBlindVoteList = getHashOfBlindVoteList();
    log.info("Sha256Ripemd160 hash of hashOfBlindVoteList " + Utilities.bytesAsHexString(hashOfBlindVoteList));
    byte[] opReturnData = VoteRevealConsensus.getOpReturnData(hashOfBlindVoteList, myVote.getSecretKey());
    // We search for my unspent stake output.
    // myVote is already tested if it is in current cycle at maybeRevealVotes
    // We expect that the blind vote tx and stake output is available. If not we throw an exception.
    TxOutput stakeTxOutput = bsqStateService.getUnspentBlindVoteStakeTxOutputs().stream().filter(txOutput -> txOutput.getTxId().equals(myVote.getTxId())).findFirst().orElseThrow(() -> new VoteRevealException("stakeTxOutput is not found for myVote.", myVote));
    // TODO is phase check needed and done in parser still?
    if (periodService.isTxInCorrectCycle(stakeTxOutput.getTxId(), chainHeight)) {
        Transaction voteRevealTx = getVoteRevealTx(stakeTxOutput, opReturnData);
        log.info("voteRevealTx={}", voteRevealTx);
        publishTx(voteRevealTx);
        // TODO add comment...
        // We don't want to wait for a successful broadcast to avoid issues if the broadcast succeeds delayed or at
        // next startup but the tx was actually broadcasted.
        myVoteListService.applyRevealTxId(myVote, voteRevealTx.getHashAsString());
        // Just for additional resilience we republish our blind votes
        final List<BlindVote> sortedBlindVoteListOfCycle = BlindVoteConsensus.getSortedBlindVoteListOfCycle(blindVoteListService);
        rePublishBlindVotePayloadList(sortedBlindVoteListOfCycle);
    } else {
        final String msg = "Tx of stake out put is not in our cycle. That must not happen.";
        log.error("{}. chainHeight={},  blindVoteTxId()={}", msg, chainHeight, myVote.getTxId());
        voteRevealExceptions.add(new VoteRevealException(msg, stakeTxOutput.getTxId()));
    }
}
Also used : TxOutput(bisq.core.dao.state.blockchain.TxOutput) Transaction(org.bitcoinj.core.Transaction) BlindVote(bisq.core.dao.governance.blindvote.BlindVote)

Aggregations

BlindVote (bisq.core.dao.governance.blindvote.BlindVote)2 TxOutput (bisq.core.dao.state.blockchain.TxOutput)2 Utilities (bisq.common.util.Utilities)1 DaoSetupService (bisq.core.dao.DaoSetupService)1 Ballot (bisq.core.dao.governance.ballot.Ballot)1 BallotList (bisq.core.dao.governance.ballot.BallotList)1 BallotListService (bisq.core.dao.governance.ballot.BallotListService)1 Vote (bisq.core.dao.governance.ballot.vote.Vote)1 BlindVoteConsensus (bisq.core.dao.governance.blindvote.BlindVoteConsensus)1 BlindVoteListService (bisq.core.dao.governance.blindvote.BlindVoteListService)1 VoteWithProposalTxId (bisq.core.dao.governance.blindvote.VoteWithProposalTxId)1 VoteWithProposalTxIdList (bisq.core.dao.governance.blindvote.VoteWithProposalTxIdList)1 MeritConsensus (bisq.core.dao.governance.merit.MeritConsensus)1 MeritList (bisq.core.dao.governance.merit.MeritList)1 Proposal (bisq.core.dao.governance.proposal.Proposal)1 ProposalListPresentation (bisq.core.dao.governance.proposal.ProposalListPresentation)1 CompensationProposal (bisq.core.dao.governance.proposal.compensation.CompensationProposal)1 ConfiscateBondProposal (bisq.core.dao.governance.proposal.confiscatebond.ConfiscateBondProposal)1 ChangeParamProposal (bisq.core.dao.governance.proposal.param.ChangeParamProposal)1 BondedRoleProposal (bisq.core.dao.governance.proposal.role.BondedRoleProposal)1