-
Notifications
You must be signed in to change notification settings - Fork 6
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
The logic of Market::getBuyPrice
and Market::getSellPrice
is flawed and encourages unhealthy behaviours that will make users lose their money
#270
Comments
minhquanym marked the issue as duplicate of #467 |
I acknowledge the validity of those concerns. However, this is rather a design choice than a design flaw or a bug of the protocol per se, therefore QA seems appropriate. |
MarioPoneder marked the issue as not a duplicate |
MarioPoneder changed the severity to QA (Quality Assurance) |
MarioPoneder marked the issue as grade-b |
@MarioPoneder, first of all, I would like to thank you a lot for reviewing this submission and sharing your point of view. However, I would still like to argue that it is a design flaw rather than design choice. The points I have raised are:
If it is a design choice made by the Sponsors, then it has the following implications:
I think that these implications are quite severe and it's not what Sponsors intended (hence I don't think it's a design choice). I think that the intended implications are as follows:
However, these implications will only hold when the issue that I reported is fixed, because otherwise, the first set of implications unfortunately apply as I have shown in my report and highlighted above. Last, but not least, the description for Medium severity that can be found in the docs state:
I believe that the leak of value is both for users (when dishonest user who bought a lot of tokens at the beginning dumps them) and for the protocol (as users may be reluctant to use the protocol, because of such unhealthy behaviour). Thank you once again and I would be very grateful, if you could review my arguments on this issue. |
Thank you for your comment! Again, I acknowledge the validity of your concerns, see also #294. The sponsor might want to provide further info about this @OpenCoreCH |
The points that are brought up here apply to every protocol with a bonding curve (friend.tech, song.tech) and you can definitely discuss the pros and cons of such a design, which is an interesting discussion to have (although this is probably the wrong place). I do not agree with all of the implications pointed out here:
So on the one hand, bots will always buy in the beginning, but on the other hand, no more users will buy? Why would the bots then continue to buy in the beginning, this is a risk for them because they need to pay fees. But if no bots buy, wouldn't it then be attractive again for users to buy following this logic? In protocols with such a bonding curve, it is of course true that "early" buyers are awarded. However, what actually is early? You do not know that when buying, if there is only one buyer for a market, even the first buyer is not early (and will make a loss), if there are 20000, buyer number 1,000 is still early and will make a profit. |
Lines of code
https://github.com/code-423n4/2023-11-canto/blob/335930cd53cf9a137504a57f1215be52c6d67cb3/1155tech-contracts/src/Market.sol#L132-L145
Vulnerability details
tl;dr
Since tokens will be more expensive to buy when the total supply increases, some malicious users can buy a lot of tokens right after protocol deployment just in order to dump them later on for a higher price when new users arrive. These new users will then be unable to sell their tokens for an acceptable price.
Detailed description
The current mechanism of calculating token prices in
Market::buy
andMarket::sell
makes token price to increase a lot when token supply increases - for example minting1000
th token will be roughly1000
times more expensive than buying the first one. it can be verified by analysing theLinearBondingCurve::getPriceAndFee
:It encourages users to buy a lot of tokens right after protocol deployment. This way, when new users enter, they (early users who bought a lot of tokens at the beginning) can dump their tokens and receive profit. But because of their actions, new users will pay significantly more for tokens than they would normally do.
In order to see this, consider the scenario:
1000
tokens.1001 * priceIncrease
for the first NFT instead of1 * priceIncrease
, which is over1000
more.1000 * priceIncrease
each time.Impact
Current fee calculation mechanism encourages unhealthy behaviour among users - users are incentivised to speculate and buy tokens early on only in order to sell them later, without really using the protocol (the additional incentive is that users with high token balance will get significant amount of protocol fees). This behaviour also harms innocent users discouraging them from using the protocol at all. In the scenario that I have described in the previous section, no "normal" user will be able to buy tokens for prices lower than
1000 * priceIncrease
- only attacker will do so.Note that attacker doesn't risk a lot here, as he will be able to sell his tokens at any time, in the worst case, for the price he initially bought them - he will only pay the fees, but they aren't high enough in order to discourage such attacks, especially that he will get a lot of fees paid by other users since he has a lot of tokens.
Note: This issue is different than the one about possible sandwitch attack that I submitted earlier. First of all, because the fix for that issue doesn't fix the problem described here, and secondly, this issue is about the design flaw that encourages unhealthy behaviours among users, that makes other users to be harmed.
Proof of Concept
First, please add the following function to the
MockERC20
inMarket.t.sol
:Then, please put the following test to
Market.t.sol
and run it:Tools Used
VS Code
Recommended Mitigation Steps
It's not so easy to fix this problem as incentives for the early users should still exist.
The best fix that comes to my mind is to limit the amount of tokens that users can buy for some time after each share has been created (say
1
week) to1
and after that time allow anybody to buy as many tokens as they want. It will still be possible to buy tokens from many accounts at the beginning, but it will be more costly:1000
tokens, he would have to create1000
accounts, send some tokens for each of these accounts and then make1000
buy
transactions - doing so many transactions will be costly and annoying, so it will deter the attackers. Moreover, if the attacker even managed to get1000
tokens (one per each account), then he will not benefit from protocol fees (as gas fees for1000
calls toclaimHolderFee
will consume the entire fees earned).Assessed type
Other
The text was updated successfully, but these errors were encountered: