X-Git-Url: https://git.novaco.in/?a=blobdiff_plain;f=Novacoin%2FCTransaction.cs;h=11d99c99f44e80f42d0283042da2bfa7bcb9ab37;hb=04ac3ad40812b5ff686f16e182a10a78b9816bf4;hp=b0d4f08799612d6f8722764277e0300a1688fd23;hpb=183708642533185adadd50c36470927d8dd4b914;p=NovacoinLibrary.git diff --git a/Novacoin/CTransaction.cs b/Novacoin/CTransaction.cs index b0d4f08..11d99c9 100644 --- a/Novacoin/CTransaction.cs +++ b/Novacoin/CTransaction.cs @@ -57,11 +57,16 @@ namespace Novacoin /// One coin = 1000000 satoshis. /// public const ulong nCoin = 1000000; + /// /// Sanity checking threshold. /// public const ulong nMaxMoney = 2000000000 * nCoin; + public const ulong nMinTxFee = nCent / 10; + public const ulong nMinRelayTxFee = nCent / 50; + public const ulong nMinTxoutAmount = nCent / 100; + /// /// Maximum transaction size is 250Kb /// @@ -151,12 +156,12 @@ namespace Novacoin 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)) + TxOutItem txOutCursor; + if (!CBlockStore.Instance.GetTxOutCursor(outpoint, out txOutCursor)) return false; if (!ScriptCode.VerifyScript(vin[i].scriptSig, txOutCursor.scriptPubKey, this, i, (int)scriptflag.SCRIPT_VERIFY_P2SH, 0)) @@ -334,11 +339,11 @@ namespace Novacoin /// /// Serialized size /// - public int Size + public uint Size { get { - int nSize = 12; // nVersion, nTime, nLockLime + uint nSize = 12; // nVersion, nTime, nLockLime nSize += VarInt.GetEncodedSize(vin.Length); nSize += VarInt.GetEncodedSize(vout.Length); @@ -552,14 +557,131 @@ namespace Novacoin return new CTxOut(outItem.nValue, outItem.scriptPubKey); } - internal bool GetCoinAge(ref Dictionary inputs, out ulong nCoinAge) + /// + /// Calculate coin*age. + /// + /// Note, only those coins meeting minimum age requirement counts. + /// + /// Inputs set. + /// Coin age calculation result. + /// Result + public bool GetCoinAge(ref Dictionary inputs, out ulong nCoinAge) { - throw new NotImplementedException(); + BigInteger bnCentSecond = 0; // coin age in the unit of cent-seconds + nCoinAge = 0; + + if (IsCoinBase) + { + // Nothing spent by coinbase, coinage is always zero. + return true; + } + + for( var i = 0; i nTime) + { + continue; // only count coins meeting min age requirement + } + + ulong nValueIn = input.nValue; + bnCentSecond += new BigInteger(nValueIn) * (nTime - merkleItem.nTime) / nCent; + } + + BigInteger bnCoinDay = bnCentSecond * nCent / nCoin / (24 * 60 * 60); + nCoinAge = (ulong)bnCoinDay; + + return true; } - internal static ulong GetMinFee(int v1, bool v2, MinFeeMode gMF_BLOCK, int nTxSize) + public ulong GetMinFee(uint nBlockSize, bool fAllowFree, MinFeeMode mode) { - throw new NotImplementedException(); + ulong nMinTxFee = CTransaction.nMinTxFee, nMinRelayTxFee = CTransaction.nMinRelayTxFee; + uint nBytes = Size; + + if (IsCoinStake) + { + // Enforce 0.01 as minimum fee for old approach or coinstake + nMinTxFee = nCent; + nMinRelayTxFee = nCent; + + if (nTime < NetInfo.nStakeValidationSwitchTime) + { + // Enforce zero size for compatibility with old blocks. + nBytes = 0; + } + } + + // Base fee is either nMinTxFee or nMinRelayTxFee + ulong nBaseFee = (mode == MinFeeMode.GMF_RELAY) ? nMinRelayTxFee : nMinTxFee; + + uint nNewBlockSize = nBlockSize + nBytes; + ulong nMinFee = (1 + (ulong)nBytes / 1000) * nBaseFee; + + if (fAllowFree) + { + if (nBlockSize == 1) + { + // Transactions under 1K are free + if (nBytes < 1000) + nMinFee = 0; + } + else + { + // Free transaction area + if (nNewBlockSize < 27000) + nMinFee = 0; + } + } + + // To limit dust spam, require additional MIN_TX_FEE/MIN_RELAY_TX_FEE for + // each non empty output which is less than 0.01 + // + // It's safe to ignore empty outputs here, because these inputs are allowed + // only for coinbase and coinstake transactions. + foreach (var txout in vout) + { + if (txout.nValue < nCent && !txout.IsEmpty) + { + nMinFee += nBaseFee; + } + } + + var nMaxBlockSizeGen = CBlock.nMaxBlockSize / 2; + + // Raise the price as the block approaches full + if (nBlockSize != 1 && nNewBlockSize >= nMaxBlockSizeGen / 2) + { + if (nNewBlockSize >= nMaxBlockSizeGen) + { + return nMaxMoney; + } + + nMinFee *= nMaxBlockSizeGen / (nMaxBlockSizeGen - nNewBlockSize); + } + + if (!MoneyRange(nMinFee)) + { + nMinFee = nMaxMoney; + } + + return nMinFee; } } }