X-Git-Url: https://git.novaco.in/?a=blobdiff_plain;f=Novacoin%2FCTransaction.cs;h=b0d4f08799612d6f8722764277e0300a1688fd23;hb=183708642533185adadd50c36470927d8dd4b914;hp=13ce0c773c144acd4d573c8e9151c63e6696c07b;hpb=cb0d54ef8359d1c9deabe23b25b145bc3dce38cf;p=NovacoinLibrary.git diff --git a/Novacoin/CTransaction.cs b/Novacoin/CTransaction.cs index 13ce0c7..b0d4f08 100644 --- a/Novacoin/CTransaction.cs +++ b/Novacoin/CTransaction.cs @@ -19,23 +19,70 @@ using System; using System.Text; using System.Collections.Generic; +using System.IO; +using System.Diagnostics.Contracts; +using System.Numerics; namespace Novacoin { + [Serializable] + public class TransactionConstructorException : Exception + { + public TransactionConstructorException() + { + } + + public TransactionConstructorException(string message) + : base(message) + { + } + + public TransactionConstructorException(string message, Exception inner) + : base(message, inner) + { + } + } + /// - /// Represents the transaction. Any transaction must provide one input and one output at least. + /// Represents the transaction. /// public class CTransaction { /// + /// One cent = 10000 satoshis. + /// + public const ulong nCent = 10000; + + /// + /// One coin = 1000000 satoshis. + /// + public const ulong nCoin = 1000000; + /// + /// Sanity checking threshold. + /// + public const ulong nMaxMoney = 2000000000 * nCoin; + + /// + /// Maximum transaction size is 250Kb + /// + public const uint nMaxTxSize = 250000; + + public enum MinFeeMode + { + GMF_BLOCK, + GMF_RELAY, + GMF_SEND, + } + + /// /// Version of transaction schema. /// - public uint nVersion = 1; + public uint nVersion; /// /// Transaction timestamp. /// - public uint nTime = 0; + public uint nTime; /// /// Array of transaction inputs @@ -50,7 +97,7 @@ namespace Novacoin /// /// Block height or timestamp when transaction is final /// - public uint nLockTime = 0; + public uint nLockTime; /// /// Initialize an empty instance @@ -60,8 +107,11 @@ namespace Novacoin // Initialize empty input and output arrays. Please note that such // configuration is not valid for real transaction, you have to supply // at least one input and one output. + nVersion = 1; + nTime = 0; vin = new CTxIn[0]; vout = new CTxOut[0]; + nLockTime = 0; } /// @@ -90,48 +140,221 @@ namespace Novacoin nLockTime = tx.nLockTime; } + /// + /// Attempts to execute all transaction scripts and validate the results. + /// + /// Checking result. + public bool VerifyScripts() + { + if (IsCoinBase) + { + return true; + } + + TxOutItem txOutCursor = null; + for (int i = 0; i < vin.Length; i++) + { + var outpoint = vin[i].prevout; + + if (!CBlockStore.Instance.GetTxOutCursor(outpoint, ref txOutCursor)) + return false; + + if (!ScriptCode.VerifyScript(vin[i].scriptSig, txOutCursor.scriptPubKey, this, i, (int)scriptflag.SCRIPT_VERIFY_P2SH, 0)) + return false; + } + + return true; + } /// - /// Parse byte sequence and initialize new instance of CTransaction + /// Calculate amount of signature operations without trying to properly evaluate P2SH scripts. /// - /// Byte sequence - public CTransaction(byte[] txBytes) + public uint LegacySigOpCount { - var wBytes = new ByteQueue(txBytes); + get + { + uint nSigOps = 0; + foreach (var txin in vin) + { + nSigOps += txin.scriptSig.GetSigOpCount(false); + } + foreach (var txout in vout) + { + nSigOps += txout.scriptPubKey.GetSigOpCount(false); + } - nVersion = BitConverter.ToUInt32(wBytes.Get(4), 0); - nTime = BitConverter.ToUInt32(wBytes.Get(4), 0); + return nSigOps; + } + } - int nInputs = (int)(int)wBytes.GetVarInt(); - vin = new CTxIn[nInputs]; + /// + /// Basic sanity checkings + /// + /// Checking result + public bool CheckTransaction() + { + if (Size > nMaxTxSize || vin.Length == 0 || vout.Length == 0) + { + return false; + } - for (int nCurrentInput = 0; nCurrentInput < nInputs; nCurrentInput++) + // Check for empty or overflow output values + ulong nValueOut = 0; + for (int i = 0; i < vout.Length; i++) { - // Fill inputs array - vin[nCurrentInput] = new CTxIn(); - - vin[nCurrentInput].prevout = new COutPoint(wBytes.Get(36)); + CTxOut txout = vout[i]; + if (txout.IsEmpty && !IsCoinBase && !IsCoinStake) + { + // Empty outputs aren't allowed for user transactions. + return false; + } - int nScriptSigLen = (int)wBytes.GetVarInt(); - vin[nCurrentInput].scriptSig = new CScript(wBytes.Get(nScriptSigLen)); + nValueOut += txout.nValue; + if (!MoneyRange(nValueOut)) + { + return false; + } + } + + // Check for duplicate inputs + var InOutPoints = new List(); + foreach (var txin in vin) + { + if (InOutPoints.IndexOf(txin.prevout) != -1) + { + // Duplicate input. + return false; + } + InOutPoints.Add(txin.prevout); + } - vin[nCurrentInput].nSequence = BitConverter.ToUInt32(wBytes.Get(4), 0); + if (IsCoinBase) + { + if (vin[0].scriptSig.Size < 2 || vin[0].scriptSig.Size > 100) + { + // Script size is invalid + return false; + } } + else + { + foreach (var txin in vin) + { + if (txin.prevout.IsNull) + { + // Null input in non-coinbase transaction. + return false; + } + } + } + + return true; + } - int nOutputs = (int)wBytes.GetVarInt(); - vout = new CTxOut[nOutputs]; + public bool IsFinal(uint nBlockHeight = 0, uint nBlockTime = 0) + { + // Time based nLockTime + if (nLockTime == 0) + { + return true; + } + if (nBlockHeight == 0) + { + nBlockHeight = uint.MaxValue; // TODO: stupid stub here, should be best height instead. + } + if (nBlockTime == 0) + { + nBlockTime = NetInfo.GetAdjustedTime(); + } + if (nLockTime < (nLockTime < NetInfo.nLockTimeThreshold ? nBlockHeight : nBlockTime)) + { + return true; + } + foreach (var txin in vin) + { + if (!txin.IsFinal) + { + return false; + } + } + return true; + } - for (int nCurrentOutput = 0; nCurrentOutput < nOutputs; nCurrentOutput++) + /// + /// Parse byte sequence and initialize new instance of CTransaction + /// + /// Byte sequence + public CTransaction(byte[] txBytes) + { + try { - // Fill outputs array - vout[nCurrentOutput] = new CTxOut(); - vout[nCurrentOutput].nValue = BitConverter.ToInt64(wBytes.Get(8), 0); + var stream = new MemoryStream(txBytes); + var reader = new BinaryReader(stream); + + nVersion = reader.ReadUInt32(); + nTime = reader.ReadUInt32(); + + int nInputs = (int)VarInt.ReadVarInt(ref reader); + vin = new CTxIn[nInputs]; + + for (int nCurrentInput = 0; nCurrentInput < nInputs; nCurrentInput++) + { + // Fill inputs array + vin[nCurrentInput] = new CTxIn(); - int nScriptPKLen = (int)wBytes.GetVarInt(); - vout[nCurrentOutput].scriptPubKey = new CScript(wBytes.Get(nScriptPKLen)); + vin[nCurrentInput].prevout = new COutPoint(reader.ReadBytes(36)); + + int nScriptSigLen = (int)VarInt.ReadVarInt(ref reader); + vin[nCurrentInput].scriptSig = new CScript(reader.ReadBytes(nScriptSigLen)); + + vin[nCurrentInput].nSequence = reader.ReadUInt32(); + } + + int nOutputs = (int)VarInt.ReadVarInt(ref reader); + vout = new CTxOut[nOutputs]; + + for (int nCurrentOutput = 0; nCurrentOutput < nOutputs; nCurrentOutput++) + { + // Fill outputs array + vout[nCurrentOutput] = new CTxOut(); + vout[nCurrentOutput].nValue = reader.ReadUInt64(); + + int nScriptPKLen = (int)VarInt.ReadVarInt(ref reader); + vout[nCurrentOutput].scriptPubKey = new CScript(reader.ReadBytes(nScriptPKLen)); + } + + nLockTime = reader.ReadUInt32(); + } + catch (Exception e) + { + throw new TransactionConstructorException("Deserialization failed", e); } + } - nLockTime = BitConverter.ToUInt32(wBytes.Get(4), 0); + /// + /// Serialized size + /// + public int Size + { + get + { + int nSize = 12; // nVersion, nTime, nLockLime + + nSize += VarInt.GetEncodedSize(vin.Length); + nSize += VarInt.GetEncodedSize(vout.Length); + + foreach (var input in vin) + { + nSize += input.Size; + } + + foreach (var output in vout) + { + nSize += output.Size; + } + + return nSize; + } } /// @@ -139,30 +362,38 @@ namespace Novacoin /// /// Bytes sequence /// Transactions array - public static CTransaction[] ReadTransactionsList(ref ByteQueue wTxBytes) + internal static CTransaction[] ReadTransactionsList(ref BinaryReader reader) { - // Read amount of transactions - int nTransactions = (int)wTxBytes.GetVarInt(); - var tx = new CTransaction[nTransactions]; - - for (int nTx = 0; nTx < nTransactions; nTx++) + try { - // Fill the transactions array - tx[nTx] = new CTransaction(); + // Read amount of transactions + int nTransactions = (int)VarInt.ReadVarInt(ref reader); + var tx = new CTransaction[nTransactions]; - tx[nTx].nVersion = BitConverter.ToUInt32(wTxBytes.Get(4), 0); - tx[nTx].nTime = BitConverter.ToUInt32(wTxBytes.Get(4), 0); + for (int nTx = 0; nTx < nTransactions; nTx++) + { + // Fill the transactions array + tx[nTx] = new CTransaction(); - // Inputs array - tx[nTx].vin = CTxIn.ReadTxInList(ref wTxBytes); + tx[nTx].nVersion = reader.ReadUInt32(); + tx[nTx].nTime = reader.ReadUInt32(); - // outputs array - tx[nTx].vout = CTxOut.ReadTxOutList(ref wTxBytes); + // Inputs array + tx[nTx].vin = CTxIn.ReadTxInList(ref reader); - tx[nTx].nLockTime = BitConverter.ToUInt32(wTxBytes.Get(4), 0); - } + // outputs array + tx[nTx].vout = CTxOut.ReadTxOutList(ref reader); - return tx; + tx[nTx].nLockTime = reader.ReadUInt32(); + } + + return tx; + + } + catch (Exception e) + { + throw new TransactionConstructorException("Deserialization failed", e); + } } public bool IsCoinBase @@ -181,40 +412,57 @@ namespace Novacoin /// /// Transaction hash /// - public Hash256 Hash + public uint256 Hash { - get { return Hash256.Compute256(Bytes); } + get { return CryptoUtils.ComputeHash256(this); } } /// - /// A sequence of bytes, which corresponds to the current state of CTransaction. + /// Amount of novacoins spent by this transaction. /// - public byte[] Bytes + public ulong nValueOut { get { - var resultBytes = new List(); - - resultBytes.AddRange(BitConverter.GetBytes(nVersion)); - resultBytes.AddRange(BitConverter.GetBytes(nTime)); - resultBytes.AddRange(VarInt.EncodeVarInt(vin.LongLength)); - - foreach (var input in vin) + ulong nValueOut = 0; + foreach (var txout in vout) { - resultBytes.AddRange(input.Bytes); + nValueOut += txout.nValue; + Contract.Assert(MoneyRange(txout.nValue) && MoneyRange(nValueOut)); } + return nValueOut; + } + } - resultBytes.AddRange(VarInt.EncodeVarInt(vout.LongLength)); + /// + /// A sequence of bytes, which corresponds to the current state of CTransaction. + /// + public static implicit operator byte[] (CTransaction tx) + { + var stream = new MemoryStream(); + var writer = new BinaryWriter(stream); - foreach (var output in vout) - { - resultBytes.AddRange(output.Bytes); - } + writer.Write(tx.nVersion); + writer.Write(tx.nTime); + writer.Write(VarInt.EncodeVarInt(tx.vin.LongLength)); + + foreach (var input in tx.vin) + { + writer.Write(input); + } - resultBytes.AddRange(BitConverter.GetBytes(nLockTime)); + writer.Write(VarInt.EncodeVarInt(tx.vout.LongLength)); - return resultBytes.ToArray(); + foreach (var output in tx.vout) + { + writer.Write(output); } + + writer.Write(tx.nLockTime); + var resultBytes = stream.ToArray(); + writer.Close(); + + return resultBytes; } public override string ToString() @@ -225,17 +473,93 @@ namespace Novacoin foreach (var txin in vin) { - sb.AppendFormat(" {0},\n", txin.ToString()); + sb.AppendFormat(" {0},\n", txin); } foreach (var txout in vout) { - sb.AppendFormat(" {0},\n", txout.ToString()); + sb.AppendFormat(" {0},\n", txout); } sb.AppendFormat("\nnLockTime={0}\n)", nLockTime); return sb.ToString(); } - } + + public static bool MoneyRange(ulong nValue) { return (nValue <= nMaxMoney); } + + /// + /// Get total sigops. + /// + /// Inputs map. + /// Amount of sigops. + public uint GetP2SHSigOpCount(ref Dictionary inputs) + { + if (IsCoinBase) + { + return 0; + } + + uint nSigOps = 0; + for (var i = 0; i < vin.Length; i++) + { + var prevout = GetOutputFor(vin[i], ref inputs); + if (prevout.scriptPubKey.IsPayToScriptHash) + { + nSigOps += prevout.scriptPubKey.GetSigOpCount(vin[i].scriptSig); + } + } + + return nSigOps; + } + + /// + /// Get sum of inputs spent by this transaction. + /// + /// Reference to innputs map. + /// Sum of inputs. + public ulong GetValueIn(ref Dictionary inputs) + { + if (IsCoinBase) + { + return 0; + } + + ulong nResult = 0; + for (int i = 0; i < vin.Length; i++) + { + nResult += GetOutputFor(vin[i], ref inputs).nValue; + } + + return nResult; + } + + /// + /// Helper method to find output in the map. + /// + /// Transaction input. + /// eference to inuts map. + /// Parent output. + private CTxOut GetOutputFor(CTxIn input, ref Dictionary inputs) + { + if (!inputs.ContainsKey(input.prevout)) + { + throw new Exception("No such input"); + } + + var outItem = inputs[input.prevout]; + + return new CTxOut(outItem.nValue, outItem.scriptPubKey); + } + + internal bool GetCoinAge(ref Dictionary inputs, out ulong nCoinAge) + { + throw new NotImplementedException(); + } + + internal static ulong GetMinFee(int v1, bool v2, MinFeeMode gMF_BLOCK, int nTxSize) + { + throw new NotImplementedException(); + } + } }