diff --git a/src/masternodes/consensus/loans.cpp b/src/masternodes/consensus/loans.cpp index cabaf73481..5f8c44f7ea 100644 --- a/src/masternodes/consensus/loans.cpp +++ b/src/masternodes/consensus/loans.cpp @@ -579,7 +579,7 @@ Res CLoansConsensus::operator()(const CLoanPaybackLoanV2Message& obj) const { balances.Add(CTokenAmount{loanTokenId, subAmount}); balances.Add(CTokenAmount{paybackTokenId, penalty}); - attributes->attributes[liveKey] = balances; + attributes->SetValue(liveKey, std::move(balances)); LogPrint(BCLog::LOAN, "CLoanPaybackLoanMessage(): Burning interest and loan in %s directly - total loan %lld (%lld %s), height - %d\n", paybackToken->symbol, subLoan + subInterest, subInToken, paybackToken->symbol, height); @@ -592,7 +592,7 @@ Res CLoansConsensus::operator()(const CLoanPaybackLoanV2Message& obj) const { balances.tokensPayback.Add(CTokenAmount{loanTokenId, subAmount}); balances.tokensFee.Add(CTokenAmount{paybackTokenId, penalty}); - attributes->attributes[liveKey] = balances; + attributes->SetValue(liveKey, balances); LogPrint(BCLog::LOAN, "CLoanPaybackLoanMessage(): Swapping %s to DFI and burning it - total loan %lld (%lld %s), height - %d\n", paybackToken->symbol, subLoan + subInterest, subInToken, paybackToken->symbol, height); diff --git a/src/masternodes/consensus/smartcontracts.cpp b/src/masternodes/consensus/smartcontracts.cpp index f9ba26b025..efe1f13907 100644 --- a/src/masternodes/consensus/smartcontracts.cpp +++ b/src/masternodes/consensus/smartcontracts.cpp @@ -178,7 +178,7 @@ Res CSmartContractsConsensus::operator()(const CFutureSwapMessage& obj) const { balances.Add(obj.source); } - attributes->attributes[liveKey] = balances; + attributes->SetValue(liveKey, std::move(balances)); return mnview.SetVariable(*attributes); } diff --git a/src/masternodes/govvariables/attributes.cpp b/src/masternodes/govvariables/attributes.cpp index 8f09b1ad53..04d1dae1e6 100644 --- a/src/masternodes/govvariables/attributes.cpp +++ b/src/masternodes/govvariables/attributes.cpp @@ -170,6 +170,7 @@ const std::map>& ATTRIBUTES::displayKeys {EconomyKeys::DFIP2203Current, "dfip2203_current"}, {EconomyKeys::DFIP2203Burned, "dfip2203_burned"}, {EconomyKeys::DFIP2203Minted, "dfip2203_minted"}, + {EconomyKeys::DexTokens, "dex"}, } }, }; @@ -491,7 +492,7 @@ Res ATTRIBUTES::RefundFuturesContracts(CCustomCSView &mnview, const uint32_t hei } } - attributes[liveKey] = balances; + SetValue(liveKey, std::move(balances)); return Res::Ok(); } @@ -519,11 +520,11 @@ Res ATTRIBUTES::Import(const UniValue & val) { } else { newAttr.key = TokenKeys::PaybackDFIFeePCT; } - attributes[newAttr] = attrValue; + SetValue(newAttr, attrValue); return Res::Ok(); } } - attributes[attribute] = attrValue; + SetValue(attribute, attrValue); return Res::Ok(); } ); @@ -581,6 +582,18 @@ UniValue ATTRIBUTES::Export() const { result.pushKV("paybackfees", AmountsToJSON(paybacks->tokensFee.balances)); result.pushKV("paybacktokens", AmountsToJSON(paybacks->tokensPayback.balances)); ret.pushKV(key, result); + } else if (auto balances = std::get_if(&attribute.second)) { + for (const auto& pool : *balances) { + auto& dexTokenA = pool.second.totalTokenA; + auto& dexTokenB = pool.second.totalTokenB; + auto poolkey = KeyBuilder(key, pool.first.v); + ret.pushKV(KeyBuilder(poolkey, "total_commission_a"), ValueFromUint(dexTokenA.commissions)); + ret.pushKV(KeyBuilder(poolkey, "total_commission_b"), ValueFromUint(dexTokenB.commissions)); + ret.pushKV(KeyBuilder(poolkey, "fee_burn_a"), ValueFromUint(dexTokenA.feeburn)); + ret.pushKV(KeyBuilder(poolkey, "fee_burn_b"), ValueFromUint(dexTokenB.feeburn)); + ret.pushKV(KeyBuilder(poolkey, "total_swap_a"), ValueFromUint(dexTokenA.swaps)); + ret.pushKV(KeyBuilder(poolkey, "total_swap_b"), ValueFromUint(dexTokenB.swaps)); + } } } catch (const std::out_of_range&) { // Should not get here, that's mean maps are mismatched diff --git a/src/masternodes/govvariables/attributes.h b/src/masternodes/govvariables/attributes.h index 22ef9e0084..94f227b8fb 100644 --- a/src/masternodes/govvariables/attributes.h +++ b/src/masternodes/govvariables/attributes.h @@ -37,6 +37,7 @@ enum EconomyKeys : uint8_t { DFIP2203Current = 'c', DFIP2203Burned = 'd', DFIP2203Minted = 'e', + DexTokens = 'f', }; enum DFIPKeys : uint8_t { @@ -111,8 +112,38 @@ struct CTokenPayback { } }; +struct CDexTokenInfo { + + struct CTokenInfo { + uint64_t swaps; + uint64_t feeburn; + uint64_t commissions; + + ADD_SERIALIZE_METHODS; + + template + inline void SerializationOp(Stream& s, Operation ser_action) { + READWRITE(swaps); + READWRITE(feeburn); + READWRITE(commissions); + } + }; + + CTokenInfo totalTokenA; + CTokenInfo totalTokenB; + + ADD_SERIALIZE_METHODS; + + template + inline void SerializationOp(Stream& s, Operation ser_action) { + READWRITE(totalTokenA); + READWRITE(totalTokenB); + } +}; + +using CDexBalances = std::map; using CAttributeType = std::variant; -using CAttributeValue = std::variant; +using CAttributeValue = std::variant; class ATTRIBUTES : public GovVariable, public AutoRegistrator { @@ -150,6 +181,21 @@ class ATTRIBUTES : public GovVariable, public AutoRegistrator + void SetValue(const K& key, T&& value) { + static_assert(std::is_convertible_v); + static_assert(std::is_convertible_v); + changed.insert(key); + attributes[key] = std::forward(value); + } + + template + void EraseKey(const K& key) { + static_assert(std::is_convertible_v); + changed.insert(key); + attributes.erase(key); + } + template [[nodiscard]] bool CheckKey(const K& key) const { static_assert(std::is_convertible_v); @@ -177,16 +223,17 @@ class ATTRIBUTES : public GovVariable, public AutoRegistrator attributes; uint32_t time{0}; - // For formatting in export static const std::map& displayVersions(); static const std::map& displayTypes(); static const std::map& displayParamsIDs(); static const std::map>& displayKeys(); private: + friend class CGovView; bool futureBlockUpdated{}; + std::set changed; + std::map attributes; // Defined allowed arguments static const std::map& allowedVersions(); diff --git a/src/masternodes/gv.cpp b/src/masternodes/gv.cpp index 345d0df54a..4b992a6e32 100644 --- a/src/masternodes/gv.cpp +++ b/src/masternodes/gv.cpp @@ -15,7 +15,29 @@ Res CGovView::SetVariable(GovVariable const & var) { - return WriteBy(var.GetName(), var) ? Res::Ok() : Res::Err("can't write to DB"); + auto WriteVar = [this](GovVariable const & var) { + return WriteBy(var.GetName(), var) ? Res::Ok() : Res::Err("can't write to DB"); + }; + if (var.GetName() != "ATTRIBUTES") { + return WriteVar(var); + } + auto attributes = GetAttributes(); + if (!attributes) { + return WriteVar(var); + } + auto& current = dynamic_cast(var); + if (current.changed.empty()) { + return Res::Ok(); + } + for (auto& key : current.changed) { + auto it = current.attributes.find(key); + if (it == current.attributes.end()) { + attributes->attributes.erase(key); + } else { + attributes->attributes[key] = it->second; + } + } + return WriteVar(*attributes); } std::shared_ptr CGovView::GetVariable(std::string const & name) const diff --git a/src/masternodes/mn_checks.cpp b/src/masternodes/mn_checks.cpp index c9e8059f94..d8a4744701 100644 --- a/src/masternodes/mn_checks.cpp +++ b/src/masternodes/mn_checks.cpp @@ -14,6 +14,7 @@ #include #include #include +#include #include #include #include @@ -704,6 +705,14 @@ Res CPoolSwap::ExecuteSwap(CCustomCSView& view, std::vector poolIDs, boo mnview.Flush(); } + auto attributes = view.GetAttributes(); + if (!attributes) { + attributes = std::make_shared(); + } + + CDataStructureV0 dexKey{AttributeTypes::Live, ParamIDs::Economy, EconomyKeys::DexTokens}; + auto dexBalances = attributes->GetValue(dexKey, CDexBalances{}); + // Set amount to be swapped in pool CTokenAmount swapAmountResult{obj.idTokenFrom, obj.amountFrom}; @@ -745,6 +754,18 @@ Res CPoolSwap::ExecuteSwap(CCustomCSView& view, std::vector poolIDs, boo auto dexfeeInPct = view.GetDexFeeInPct(currentID, swapAmount.nTokenId); + auto& balances = dexBalances[currentID]; + auto forward = swapAmount.nTokenId == pool->idTokenA; + + auto& totalTokenA = forward ? balances.totalTokenA : balances.totalTokenB; + auto& totalTokenB = forward ? balances.totalTokenB : balances.totalTokenA; + + const auto& reserveAmount = forward ? pool->reserveA : pool->reserveB; + const auto& blockCommission = forward ? pool->blockCommissionA : pool->blockCommissionB; + + const auto initReserveAmount = reserveAmount; + const auto initBlockCommission = blockCommission; + // Perform swap poolResult = pool->Swap(swapAmount, dexfeeInPct, poolPrice, [&] (const CTokenAmount& dexfeeInAmount, const CTokenAmount& tokenAmount) { // Save swap amount for next loop @@ -792,6 +813,7 @@ Res CPoolSwap::ExecuteSwap(CCustomCSView& view, std::vector poolIDs, boo if (!res) { return res; } + totalTokenA.feeburn += dexfeeInAmount.nValue; } // burn the dex out amount @@ -800,6 +822,14 @@ Res CPoolSwap::ExecuteSwap(CCustomCSView& view, std::vector poolIDs, boo if (!res) { return res; } + totalTokenB.feeburn += dexfeeOutAmount.nValue; + } + + totalTokenA.swaps += (reserveAmount - initReserveAmount); + totalTokenA.commissions += (blockCommission - initBlockCommission); + + if (lastSwap && obj.to == Params().GetConsensus().burnAddress) { + totalTokenB.feeburn += swapAmountResult.nValue; } return res; @@ -820,6 +850,10 @@ Res CPoolSwap::ExecuteSwap(CCustomCSView& view, std::vector poolIDs, boo } } + if (!testOnly) { + attributes->SetValue(dexKey, std::move(dexBalances)); + view.SetVariable(*attributes); + } // Assign to result for loop testing best pool swap result result = swapAmountResult.nValue; diff --git a/src/validation.cpp b/src/validation.cpp index 52b39e2486..f8b7979c1c 100644 --- a/src/validation.cpp +++ b/src/validation.cpp @@ -3403,11 +3403,11 @@ void CChainState::ProcessFutures(const CBlockIndex* pindex, CCustomCSView& cache cache.EraseFuturesUserValues(key); } - attributes->attributes[burnKey] = burned; - attributes->attributes[mintedKey] = minted; + attributes->SetValue(burnKey, std::move(burned)); + attributes->SetValue(mintedKey, std::move(minted)); if (!unpaidContracts.empty()) { - attributes->attributes[liveKey] = balances; + attributes->SetValue(liveKey, std::move(balances)); } cache.SetVariable(*attributes); diff --git a/test/functional/feature_loan_vault.py b/test/functional/feature_loan_vault.py index c7fb31047a..3a0e436a55 100755 --- a/test/functional/feature_loan_vault.py +++ b/test/functional/feature_loan_vault.py @@ -744,7 +744,9 @@ def loan_and_collateral_token_to_govvar(self): # Invalidate fork block self.nodes[0].invalidateblock(self.nodes[0].getblockhash(self.nodes[0].getblockcount())) - assert_equal(len(self.nodes[0].getgov('ATTRIBUTES')['ATTRIBUTES']), 0) + result = self.nodes[0].getgov('ATTRIBUTES')['ATTRIBUTES'] + for key in result.keys(): + assert key.startswith('v0/live') # Move to hard fork again self.move_to_gw_fork() diff --git a/test/functional/feature_poolswap.py b/test/functional/feature_poolswap.py index 7566a205ff..dbe1cb3197 100755 --- a/test/functional/feature_poolswap.py +++ b/test/functional/feature_poolswap.py @@ -196,6 +196,11 @@ def run_test(self): # 7 Sync self.sync_blocks([self.nodes[0], self.nodes[2]]) + attributes = self.nodes[0].getgov('ATTRIBUTES')['ATTRIBUTES'] + # silver is tokenB + assert_equal(attributes['v0/live/economy/dex/%s/total_swap_b'%(idGS)], Decimal('9.0')) + assert_equal(attributes['v0/live/economy/dex/%s/total_commission_b'%(idGS)], Decimal('1.0')) + # 8 Checking that poolswap is correct goldCheckN0 = self.nodes[2].getaccount(accountGN0, {}, True)[idGold] silverCheckN0 = self.nodes[2].getaccount(accountGN0, {}, True)[idSilver] @@ -253,6 +258,10 @@ def run_test(self): ) self.nodes[0].generate(1) + attributes = self.nodes[0].getgov('ATTRIBUTES')['ATTRIBUTES'] + assert_equal(attributes['v0/live/economy/dex/%s/total_swap_b'%(idGS)], Decimal('189.0')) + assert_equal(attributes['v0/live/economy/dex/%s/total_commission_b'%(idGS)], Decimal('21.0')) + maxPrice = self.nodes[0].listpoolpairs()['1']['reserveB/reserveA'] # exchange tokens each other should work self.nodes[0].poolswap({ @@ -274,6 +283,12 @@ def run_test(self): }) self.nodes[0].generate(1) + attributes = self.nodes[0].getgov('ATTRIBUTES')['ATTRIBUTES'] + assert_equal(attributes['v0/live/economy/dex/%s/total_swap_a'%(idGS)], Decimal('180.0')) + assert_equal(attributes['v0/live/economy/dex/%s/total_commission_a'%(idGS)], Decimal('20.0')) + assert_equal(attributes['v0/live/economy/dex/%s/total_swap_b'%(idGS)], Decimal('369.0')) + assert_equal(attributes['v0/live/economy/dex/%s/total_commission_b'%(idGS)], Decimal('41.0')) + # Test fort canning max price change disconnect_nodes(self.nodes[0], 1) disconnect_nodes(self.nodes[0], 2) @@ -390,6 +405,12 @@ def run_test(self): }) self.nodes[0].generate(1) + attributes = self.nodes[0].getgov('ATTRIBUTES')['ATTRIBUTES'] + assert_equal(attributes['v0/live/economy/dex/%s/total_swap_a'%(idBL)], Decimal('0.0')) + assert_equal(attributes['v0/live/economy/dex/%s/total_commission_a'%(idBL)], Decimal('0.0')) + assert_equal(attributes['v0/live/economy/dex/%s/total_swap_b'%(idBL)], Decimal('0.00000189')) + assert_equal(attributes['v0/live/economy/dex/%s/total_commission_b'%(idBL)], Decimal('1E-8')) + assert_equal(self.nodes[0].getaccount(new_dest, {}, True)[idBTC], Decimal('0.00000002')) # Reset swap and move to Fort Canning Park Height and try swap again @@ -428,7 +449,9 @@ def run_test(self): self.nodes[0].setgov({"ATTRIBUTES":{'v0/poolpairs/%s/token_a_fee_pct'%(idGS): '0.05', 'v0/poolpairs/%s/token_b_fee_pct'%(idGS): '0.08'}}) self.nodes[0].generate(1) - assert_equal(self.nodes[0].getgov('ATTRIBUTES')['ATTRIBUTES'], {'v0/poolpairs/%s/token_a_fee_pct'%(idGS): '0.05', 'v0/poolpairs/%s/token_b_fee_pct'%(idGS): '0.08'}) + attributes = self.nodes[0].getgov('ATTRIBUTES')['ATTRIBUTES'] + assert_equal(attributes['v0/poolpairs/%s/token_a_fee_pct'%(idGS)], '0.05') + assert_equal(attributes['v0/poolpairs/%s/token_b_fee_pct'%(idGS)], '0.08') result = self.nodes[0].getpoolpair(idGS) assert_equal(result[idGS]['dexFeePctTokenA'], Decimal('0.05')) @@ -461,11 +484,17 @@ def run_test(self): assert_equal(self.nodes[0].getburninfo()['dexfeetokens'].sort(), ['%.8f'%(dexinfee)+symbolGOLD, '%.8f'%(dexoutfee)+symbolSILVER].sort()) + attributes = self.nodes[0].getgov('ATTRIBUTES')['ATTRIBUTES'] + assert_equal(attributes['v0/live/economy/dex/%s/fee_burn_a'%(idGS)], dexinfee) + assert_equal(attributes['v0/live/economy/dex/%s/fee_burn_b'%(idGS)], dexoutfee) + # set 1% token dex fee and commission self.nodes[0].setgov({"ATTRIBUTES":{'v0/poolpairs/%s/token_a_fee_pct'%(idGS): '0.01', 'v0/poolpairs/%s/token_b_fee_pct'%(idGS): '0.01'}}) self.nodes[0].generate(1) - assert_equal(self.nodes[0].getgov('ATTRIBUTES')['ATTRIBUTES'], {'v0/poolpairs/%s/token_a_fee_pct'%(idGS): '0.01', 'v0/poolpairs/%s/token_b_fee_pct'%(idGS): '0.01'}) + attributes = self.nodes[0].getgov('ATTRIBUTES')['ATTRIBUTES'] + assert_equal(attributes['v0/poolpairs/%s/token_a_fee_pct'%(idGS)], '0.01') + assert_equal(attributes['v0/poolpairs/%s/token_b_fee_pct'%(idGS)], '0.01') self.nodes[0].updatepoolpair({"pool": "GS", "commission": 0.01}) self.nodes[0].generate(1) @@ -489,7 +518,11 @@ def run_test(self): self.nodes[0].setgov({"ATTRIBUTES":{'v0/poolpairs/%s/token_a_fee_pct'%(idBL): '0.01', 'v0/poolpairs/%s/token_b_fee_pct'%(idBL): '0.01'}}) self.nodes[0].generate(1) - assert_equal(self.nodes[0].getgov('ATTRIBUTES')['ATTRIBUTES'], {'v0/poolpairs/%s/token_a_fee_pct'%(idGS): '0.01', 'v0/poolpairs/%s/token_b_fee_pct'%(idGS): '0.01', 'v0/poolpairs/%s/token_a_fee_pct'%(idBL): '0.01', 'v0/poolpairs/%s/token_b_fee_pct'%(idBL): '0.01'}) + attributes = self.nodes[0].getgov('ATTRIBUTES')['ATTRIBUTES'] + assert_equal(attributes['v0/poolpairs/%s/token_a_fee_pct'%(idGS)], '0.01') + assert_equal(attributes['v0/poolpairs/%s/token_b_fee_pct'%(idGS)], '0.01') + assert_equal(attributes['v0/poolpairs/%s/token_a_fee_pct'%(idBL)], '0.01') + assert_equal(attributes['v0/poolpairs/%s/token_b_fee_pct'%(idBL)], '0.01') self.nodes[0].invalidateblock(self.nodes[0].getblockhash(self.nodes[0].getblockcount())) self.nodes[0].clearmempool() @@ -528,6 +561,10 @@ def run_test(self): dexoutfee = round(trunc(amountA * Decimal(0.05) * coin) / coin, 8) assert_equal(round(amountA - Decimal(dexoutfee), 8), round(swapped, 8)) + attributes = self.nodes[0].getgov('ATTRIBUTES')['ATTRIBUTES'] + assert_equal(attributes['v0/live/economy/dex/%s/fee_burn_b'%(idBL)], round(dexinfee, 8)) + assert_equal(attributes['v0/live/economy/dex/%s/fee_burn_a'%(idBL)], Decimal(str(round(dexoutfee, 8)))) + # REVERTING: #======================== print ("Reverting...") diff --git a/test/lint/lint-circular-dependencies.sh b/test/lint/lint-circular-dependencies.sh index bbbf59f23c..ab5c4e9040 100755 --- a/test/lint/lint-circular-dependencies.sh +++ b/test/lint/lint-circular-dependencies.sh @@ -51,6 +51,7 @@ EXPECTED_CIRCULAR_DEPENDENCIES=( "masternodes/govvariables/attributes -> masternodes/gv -> masternodes/govvariables/attributes" "masternodes/govvariables/attributes -> masternodes/masternodes -> masternodes/govvariables/attributes" "masternodes/govvariables/attributes -> masternodes/masternodes -> validation -> masternodes/govvariables/attributes" + "masternodes/govvariables/attributes -> masternodes/mn_checks -> masternodes/govvariables/attributes" "masternodes/govvariables/icx_takerfee_per_btc -> masternodes/gv -> masternodes/govvariables/icx_takerfee_per_btc" "masternodes/govvariables/loan_daily_reward -> masternodes/gv -> masternodes/govvariables/loan_daily_reward" "masternodes/govvariables/loan_daily_reward -> masternodes/masternodes -> validation -> masternodes/govvariables/loan_daily_reward"