All Downloads are FREE. Search and download functionalities are using the official Maven repository.

network.nerve.base.data.Transaction Maven / Gradle / Ivy

There is a newer version: 1.2.5
Show newest version
/*
 * MIT License
 *
 * Copyright (c) 2017-2019 nuls.io
 *
 * Permission is hereby granted, free of charge, to any person obtaining a copy
 * of this software and associated documentation files (the "Software"), to deal
 * in the Software without restriction, including without limitation the rights
 * to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
 * copies of the Software, and to permit persons to whom the Software is
 * furnished to do so, subject to the following conditions:
 *
 * The above copyright notice and this permission notice shall be included in all
 * copies or substantial portions of the Software.
 *
 * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
 * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
 * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
 * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
 * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
 * OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
 * SOFTWARE.
 *
 */
package network.nerve.base.data;

import network.nerve.base.basic.AddressTool;
import network.nerve.base.basic.NulsByteBuffer;
import network.nerve.base.basic.NulsOutputStreamBuffer;
import network.nerve.core.constant.CommonCodeConstanst;
import network.nerve.core.constant.ToolsConstant;
import network.nerve.core.constant.TxStatusEnum;
import network.nerve.core.crypto.HexUtil;
import network.nerve.core.crypto.UnsafeByteArrayOutputStream;
import network.nerve.core.exception.NulsException;
import network.nerve.core.log.Log;
import network.nerve.core.model.DateUtils;
import network.nerve.core.parse.SerializeUtils;

import java.io.ByteArrayOutputStream;
import java.io.IOException;
import java.io.UnsupportedEncodingException;
import java.math.BigInteger;
import java.util.Arrays;
import java.util.List;

/**
 * @author Charlie
 */
public class Transaction extends BaseNulsData implements Cloneable {

    private int type;

    private byte[] coinData;

    private byte[] txData;

    private long time;

    private byte[] transactionSignature;

    private byte[] remark;

    private transient NulsHash hash;

    private transient long blockHeight = -1L;

    private transient TxStatusEnum status = TxStatusEnum.UNCONFIRM;

    private transient int size;

    private transient CoinData coinDataInstance;

    /**
     * 在区块中的顺序,存储在rocksDB中是无序的,保存区块时赋值,取出后根据此值排序
     */
    private int inBlockIndex;

    public Transaction() {

    }

    public Transaction(int type) {
        this.type = type;
    }

    @Override
    public int size() {
        int size = 0;
        //type
        size += SerializeUtils.sizeOfUint16();
        //time
        size += SerializeUtils.sizeOfUint32();
        size += SerializeUtils.sizeOfBytes(remark);
        size += SerializeUtils.sizeOfBytes(txData);
        size += SerializeUtils.sizeOfBytes(coinData);
        size += SerializeUtils.sizeOfBytes(transactionSignature);
        return size;
    }

    @Override
    public void serializeToStream(NulsOutputStreamBuffer stream) throws IOException {
        stream.writeUint16(type);
        stream.writeUint32(time);
        stream.writeBytesWithLength(remark);
        stream.writeBytesWithLength(txData);
        stream.writeBytesWithLength(coinData);
        stream.writeBytesWithLength(transactionSignature);
    }

    public byte[] serializeForHash() throws IOException {
        ByteArrayOutputStream bos = null;
        try {
            int size = size() - SerializeUtils.sizeOfBytes(transactionSignature);
            bos = new UnsafeByteArrayOutputStream(size);
            NulsOutputStreamBuffer buffer = new NulsOutputStreamBuffer(bos);
            if (size == 0) {
                bos.write(ToolsConstant.PLACE_HOLDER);
            } else {
                buffer.writeUint16(type);
                buffer.writeUint32(time);
                buffer.writeBytesWithLength(remark);
                buffer.writeBytesWithLength(txData);
                buffer.writeBytesWithLength(coinData);
            }
            return bos.toByteArray();
        } finally {
            if (bos != null) {
                bos.close();
            }
        }
    }

    @Override
    public void parse(NulsByteBuffer byteBuffer) throws NulsException {
        type = byteBuffer.readUint16();
        time = byteBuffer.readUint32();
        remark = byteBuffer.readByLengthByte();
        txData = byteBuffer.readByLengthByte();
        this.coinData = byteBuffer.readByLengthByte();
        transactionSignature = byteBuffer.readByLengthByte();
    }

    public byte[] getTxData() {
        return txData;
    }

    public long getTime() {
        return time;
    }

    public void setTime(long time) {
        this.time = time;
    }

    public void setType(int type) {
        this.type = type;
    }

    public int getType() {
        return type;
    }

    public byte[] getRemark() {
        return remark;
    }

    public void setRemark(byte[] remark) {
        this.remark = remark;
    }

    public NulsHash getHash() {
        if (hash == null) {
            try {
                hash = NulsHash.calcHash(serializeForHash());
            } catch (IOException e) {
                e.printStackTrace();
            }
        }
        return hash;
    }

    public void setHash(NulsHash hash) {
        this.hash = hash;
    }

    public byte[] getTransactionSignature() {
        return transactionSignature;
    }

    public void setTransactionSignature(byte[] transactionSignature) {
        this.transactionSignature = transactionSignature;
    }

    public void setTxData(byte[] txData) {
        this.txData = txData;
    }

    public long getBlockHeight() {
        return blockHeight;
    }

    public void setBlockHeight(long blockHeight) {
        this.blockHeight = blockHeight;
    }

    public TxStatusEnum getStatus() {
        return status;
    }

    public void setStatus(TxStatusEnum status) {
        this.status = status;
    }

    public byte[] getCoinData() {
        return coinData;
    }

    public int getInBlockIndex() {
        return inBlockIndex;
    }

    public void setInBlockIndex(int inBlockIndex) {
        this.inBlockIndex = inBlockIndex;
    }

    public CoinData getCoinDataInstance() throws NulsException {
        if (null == this.coinData || this.coinData.length == 0) {
            return null;
        }
        if (coinDataInstance == null) {
            coinDataInstance = new CoinData();
            coinDataInstance.parse(new NulsByteBuffer(this.coinData));
        }
        return coinDataInstance;
    }

    public void setCoinData(byte[] coinData) {
        this.coinData = coinData;
    }

    public int getSize() {
        if (size == 0) {
            size = size();
        }
        return size;
    }

    public void setSize(int size) {
        this.size = size;
    }

    public static Transaction getInstance(byte[] txBytes) throws NulsException {
        NulsByteBuffer nulsByteBuffer = new NulsByteBuffer(txBytes);
        Transaction transaction = new Transaction();
        transaction.parse(nulsByteBuffer);
        return transaction;
    }

    /**
     * 获取交易的手续费
     *
     * @return
     * @throws NulsException
     */
    public BigInteger getFee() throws NulsException {
        BigInteger fee = BigInteger.ZERO;
        if (null != coinData && type > 1) {
            CoinData cData = getCoinDataInstance();
            if (cData.getFrom().size() > 0) {
                BigInteger toAmount = BigInteger.ZERO;
                for (CoinTo coinTo : cData.getTo()) {
                    toAmount = toAmount.add(coinTo.getAmount());
                }
                BigInteger fromAmount = BigInteger.ZERO;
                for (CoinFrom coinFrom : cData.getFrom()) {
                    fromAmount = fromAmount.add(coinFrom.getAmount());
                }
                fee = fromAmount.subtract(toAmount);
            }
        }
        return fee;
    }

    /**
     * 判断交易是否为多签交易
     * Judging whether a transaction is a multi-signature transaction
     */
    public boolean isMultiSignTx() throws NulsException {
        if (null == coinData) {
            return false;
        }
        CoinData cData = getCoinDataInstance();
        List from = cData.getFrom();
        if (from == null || from.size() == 0) {
            return false;
        }
        CoinFrom coinFrom = from.get(0);
        return AddressTool.isMultiSignAddress(coinFrom.getAddress());
    }

    @Override
    public boolean equals(Object obj) {
        if (obj == null) {
            return false;
        }
        if (!(obj instanceof Transaction)) {
            return false;
        }
        return this.getHash().equals(((Transaction) obj).getHash());
    }

    @Override
    public int hashCode() {
        int result = type;
        result = 31 * result + Arrays.hashCode(coinData);
        result = 31 * result + Arrays.hashCode(txData);
        result = 31 * result + (int) (time ^ (time >>> 32));
        result = 31 * result + Arrays.hashCode(transactionSignature);
        result = 31 * result + Arrays.hashCode(remark);
        return result;
    }

    public String format() throws Exception {
        return format(null);
    }

    /**
     * 生成可视化交易字符串(格式化)
     *
     * @return String
     * @throws Exception
     */
    public String format(Class txDataClasz) throws NulsException {
       /* if (this.type == 1) {
            return;
        }*/
        StringBuilder builder = new StringBuilder();
        try {
            String lineSeparator = System.lineSeparator();
            builder.append(lineSeparator).append("*********************** Transaction information ***********************").append(lineSeparator);
            builder.append(String.format("type: %s", this.type)).append(lineSeparator);
            builder.append(String.format("txHash: %s", this.getHash().toHex())).append(lineSeparator);
            builder.append(String.format("time: %s", this.getTime())).append(lineSeparator);
            builder.append(String.format("time(format): %s", DateUtils.timeStamp2DateStr(this.getTime() * 1000L))).append(lineSeparator);
            builder.append(String.format("size: %s byte,", String.valueOf(this.getSize()))).append(lineSeparator);
            byte[] remark = this.getRemark();
            String remarkStr = remark == null ? null : new String(this.getRemark(), "UTF-8");
            builder.append(String.format("remark: %s", remarkStr)).append(lineSeparator);

            CoinData coinData = null;
            if (this.getCoinData() != null) {
                coinData = this.getCoinDataInstance();
                builder.append("coinData:").append(lineSeparator).append(coinData.toString());
               /*
                List coinFromList = coinData.getFrom();
                if (coinFromList == null) {
                    builder.append("\tcoinFrom: null").append(lineSeparator);
                } else if (coinFromList.size() == 0) {
                    builder.append("\tcoinFrom: size 0").append(lineSeparator);
                } else {
                    builder.append("\tcoinFrom: ").append(lineSeparator);
                    for (int i = 0; i < coinFromList.size(); i++) {
                        CoinFrom coinFrom = coinFromList.get(i);
                        builder.append(String.format("\t\tfroms-%s:", i)).append(lineSeparator);
                        builder.append(String.format("\t\t\taddress: %s", AddressTool.getStringAddressByBytes(coinFrom.getAddress()))).append(lineSeparator);
                        builder.append(String.format("\t\t\tamount: %s", coinFrom.getAmount())).append(lineSeparator);
                        builder.append(String.format("\t\t\tassetChainId: %s", coinFrom.getAssetsChainId())).append(lineSeparator);
                        builder.append(String.format("\t\t\tassetId: %s", coinFrom.getAssetsId())).append(lineSeparator);
                        builder.append(String.format("\t\t\tnonce: %s", HexUtil.encode(coinFrom.getNonce()))).append(lineSeparator);
                        builder.append(String.format("\t\t\tlocked: %s", coinFrom.getLocked())).append("\t(0普通交易,-1解锁金额交易(退出共识,退出委托))").append(lineSeparator);
                        builder.append(lineSeparator);
                    }
                }

                List coinToList = coinData.getTo();
                if (coinToList == null) {
                    builder.append("\tcoinTo: null").append(lineSeparator);
                } else if (coinToList.size() == 0) {
                    builder.append("\tcoinTo: size 0").append(lineSeparator);
                } else {
                    builder.append("\tcoinTo: ").append(lineSeparator);
                    for (int i = 0; i < coinToList.size(); i++) {
                        CoinTo coinTo = coinToList.get(i);
                        builder.append(String.format("\t\ttos-%s:", i)).append(lineSeparator);
                        builder.append(String.format("\t\t\taddress: %s", AddressTool.getStringAddressByBytes(coinTo.getAddress()))).append(lineSeparator);
                        builder.append(String.format("\t\t\tamount: %s", coinTo.getAmount())).append(lineSeparator);
                        builder.append(String.format("\t\t\tassetChainId: %s", coinTo.getAssetsChainId())).append(lineSeparator);
                        builder.append(String.format("\t\t\tassetId: %s", coinTo.getAssetsId())).append(lineSeparator);
                        builder.append(String.format("\t\t\tlocked: %s", coinTo.getLockTime())).append("\t(解锁高度或解锁时间,-1为永久锁定)").append(lineSeparator);
                        builder.append(lineSeparator);
                    }
                }
                */
            } else {
                builder.append("coinData: null").append(lineSeparator);
            }
            if (null != txDataClasz) {
                if (null != this.txData && this.txData.length != 0) {
                    try {
                        String txDataStr = getInstance(this.txData, txDataClasz).toString();
                        builder.append("txData:").append(lineSeparator).append(txDataStr).append(lineSeparator);
                    } catch (Exception e) {
                        Log.error("format txData error", e);
                    }
                } else {
                    builder.append("txData: null").append(lineSeparator);
                }
            } else {
                if (null != this.txData && this.txData.length != 0) {
                    builder.append("txData hex: ").append(HexUtil.encode(this.txData)).append(lineSeparator);
                } else {
                    builder.append("txData: null").append(lineSeparator);
                }
            }
            builder.append("***********************************************************************").append(lineSeparator);

        } catch (UnsupportedEncodingException e) {
            e.printStackTrace();
            throw new NulsException(e);
        }
        return builder.toString();
    }

    public static  T getInstance(byte[] bytes, Class clazz) throws Exception {
        if (null == bytes || bytes.length == 0) {
            throw new Exception(CommonCodeConstanst.DESERIALIZE_ERROR.getMsg());
        }
        try {
            BaseNulsData baseNulsData = clazz.getDeclaredConstructor().newInstance();
            baseNulsData.parse(new NulsByteBuffer(bytes));
            return (T) baseNulsData;
        } catch (Exception e) {
            throw new Exception(CommonCodeConstanst.DESERIALIZE_ERROR.getMsg());
        }
    }
}




© 2015 - 2024 Weber Informatics LLC | Privacy Policy