A standard interface for creating and interacting with vanilla options.
I went through the EIP as well as the accompanying implementation. Great stuff! I am also interested in bringing different kinds of financial instruments to the blockchain, and vanilla options are a great product to start with.
I have a few comments:
(1) Creating a new contract for each option issuance seems a bit sub-optimal. Options are ephemeral by nature, so once the option has expired, the contract just stays on the blockchain and doesnāt do anything. I would propose a change where the parameters of the option are not defined in the contractās constructor, but instead are given on the create() call, and the create() would return a uint256 id.
The id value would then be passed as a parameter to the buy(), exercise(), retrieveExpiredTokens(), and cancel() function calls to signal which option issuance is the targeted action. Internally, inside the implementing contract, the option issuance data could be saved to a struct, and a mapping (and an incrementing counter) could be used for storing.
In this way, a single contract could hold a large number of option issuances, and in my opinion, it would be a better way to organize the option issuances and reduce fragmentation.
(2) When the counterparty pays the premium and takes the other side to buy a call or put, there should be tokens that should be minted for the buyer representing the options. The amount of tokens should be the same as the amount of the underlying asset.
As an example, in the āConcrete Example - Call Optionā part, the amount of underlying is 8000000000000000000 LINKs. When the counterparty pays the premium, they should get 8000000000000000000 tokens that represent the amount of LINK tokens that can be bought at the strike price.
Again, due to the ephemeral nature of the options, I donāt think the tokens should be represented by an ERC-20 contract that is created on-the-fly; instead, I would go for ERC-1155 to hold all option tokens in a single contract. Also, when options are represented as tokens, they could be traded independently to other parties.
(3) One potential limitation of the current interface design is that on the exercise() method, the buyer must exercise all options. I would suggest that there should be a parameter for the exercise() function to define how many option tokens the buyer wants to exercise. This would make sense, e.g., when the buyer might want to exercise only a portion of the option tokens to hedge and take gains of the currently favorable underlying price while still retaining the other portion to see if the market moves even more in the buyerās favor (given that buyer still has exercise time left).
When the buyer exercises they need to of course have the amount of option tokens and they need to pay strikePrice * amount of strikeToken to the seller. Anyone who holds option tokens would be able to exercise them during exercise window.
(4) For the create method, I would add an array of addresses that are the allowed counterparties to (initially) take the other side. This is the improvement idea mentioned at the end of the ERC, and I think it would make sense to add it. If the array is empty, then it is free-for-all. Now, if suggestions (2) and (3) are taken into account, then there should be no restrictions on who can exercise the underlying as long as they own the tokens that represent the options. If we want to limit this, then there could be an extra boolean parameter for the create() method called ārenounceableā that defines if the counterparty can buy/sell the option tokens to other parties as they wish or if those tokens are locked to the counterpartyās address, and only they can exercise them.
(5) I would remove from the state variables the type of the option (āeuropeanā / āamericanā). Also I would remove expiration state variable. Instead of these I would define two state variables, exerciseWindowStart and exerciseWindowEnd, which represent when user can start to exercise and when the exercising ends. After exerciseWindowEnd time has passed, seller can call retrieveExpiredTokens() to get the underlying tokens back that have not been exercised.
Hey, thanks so much for your interest in this EIP!
Hereās my answer for each suggestion:
(1) Yes thatās a really good idea! This would be a change a bit like Uniswap V4 = making a singleton contract. And this would be simpler for users to create a new contract (instead of creating, then calling create() to āenableā the option).
However, Iām not sure this would be a standard (ERC), but rather a āprivateā protocol on itself (because there would be only one contract on the blockchain). But if this standard is created by every dApp, we should think about how to transfer options between smart contracts.
Maybe another solution to improve gas: make the contract reusable. This mean that after the option is āinactiveā, someone (or the same writer), can change all the parameters to create a new option, instead of creating a new smart contract.
(2) So your idea is to create a kind of coupon represented as a token, so that the buyer can āshareā his right with multiple people to exercise according to the amount he gave to each of them? Itās really interesting and would brought even more possibilities. But do you think it is more gas efficient to use a system of token (with an ERC-1155), or to simply implement the functionality without token (e.g. a function shareExercise(uint256 amount, address recipient) that would store each amount for each address in a map, and when exercising, the contract would verify if msg.sender can exercise)?
(3) Great idea! It would offer more possibilities.
(4) Another great idea, this would (has you said) be the solution to the suggestion written in the EIP
(5) Very smart, it would be simpler and probably more gas efficient.
Hi,
my (current) proposal for the interface is at the fork that is available on GitHub - mlalma/ERC-7390: Fork of draft implementation of ERC-7390 spec for hatching out changes to the standard.
I think the key here would be the interface on IVanillaOption.sol
. It basically is IOption.sol
with all my prposed changes.
I have updated the README.md
and documented the interface there. If you could take a look at it, try out the fork and see that the reference implementation (and the related tests) work as expected and then provide some comments back that would be amazing.
For the summary, some of the key changes I would propose are:
VanillaOptionData
struct contains the essential features of a vanilla option in a single struct that is passed to thecreate()
function- None of the functions return bool stating success / failure. If the function returns, then the action is successful and for any failure an exception is thrown
- There is now ID parameter that needs to be passed around so that single contract can handle multiple option issuances due to the ephemeral nature of the instrument
- Option buyer does not need to buy the whole lot, they can define the amount of options they want to buy (and later on exercise)
- I have added
updatePremium()
function to enable option seller to change the premium to keep it more in synch
There are probably also other smaller changes, but I think those are the main ones.
-mla
Hey, sorry for the very late answer.
I looked at your changes and this looks very nice to me!
My only concerns are:
- in which case
data
fromVanillaOptionData
is useful? Do you have any idea or example? - in the
updatePremium
function, shouldnāt we verify thatblock.timestamp <= issuance[id].data.exerciseWindowEnd
?
Thanks a lot for your involvement in this EIP.
Hi
Great comments, please find my answers below:
-
My original thinking was that it would be possible to add e.g. access control lists for the addresses that are allowed to buy a particular option issuance or additional parameters for supporting exotic options such as knock-in / knock-out barrier levels.Thinking more of this, maybe any additional data should not be in an extra field on a structure named VanillaOptionData, but any derived contract implementations can handle them as needed. I have removed the
data
variable from the structure, it didnāt have any effect to the code itself. -
True - will add the check.
I have done the changes and will open a new pull request for you.
Hey, thanks for your reply. Sounds good to me, just merged the changes
Hi,
few proposals for the next things that could be worked on:
(1) (This is your idea ), We could create a separate interface IERC7390Metadata
for querying contract variables such as underlyingToken(uint256 id)
, premium(uint256 id)
, exerciseWindowStart(uint256 id)
and so forth for providing a defined interface getting the data out of the available option issuances.
(2) I propose adding to the VanillaOptionData
structure three additional fields: uint256 underlyingTokenId
, uint256 strikeTokenId
and uint256 premiumTokenId
. Reasoning is that right now (only) ERC-20 tokens can be used for underlying, strike and premium. If we added explicit token ids for all these three fields then tokens also from ERC-721 and ERC-1155 standards could be used. If the token is an ERC-20 token then the token id field is simply ignored. This would enable creating e.g. an option where underlying is a (single) ERC-721 token, compound options / options-on-options where underlying token is ERC-1155 from another option issuance or an issuance where option premiums are paid using other options. Lots of possibilities Making this change would of course require bunch of changes to reference implementation as it would need to check what is the interface that the given address supports using e.g. ERC-165.
(3) ā¦and that would also lead to my third change proposal, where I would suggest to implement the supportsInterface()
ERC-165 call on the reference implementation to let other contracts verify that IERC7390
is implemented.
Any thoughts / comments?
Hey,
These are really great ideas, especially the second one.
Nothing else to say tbh, except thanks for these improvement ideas.
Letās go building!
Hey, just to let you know that the previous PR was closed because they moved ERC-only proposals to another repository. Hereās the new PR: Add ERC-7390: Vanilla Options by Xeway Ā· Pull Request #53 Ā· ethereum/ERCs Ā· GitHub
Hey,
Just made small improvements:
- Instead of adding a brand new IERC7390Metadata interface, I thought it was actually useless to do it since we only want to implement one function (that I called
issuance(uint256 id)
). Moreover, by looking at OpenZeppelinās ERC20 contracts, I found thatbalanceOf
(which is similar toissuance
in the sense that it returns an element of a map with the key that correspond to the parameter) is not in the metadata extension interface but in the mainIERC20
interface. So I guess the best is to copy the rules of OpenZeppelinās contracts, so we know itās more formal. - Speaking of which, still by looking at OZ code, I found that functions are declared
external
in the interface but are declared aspublic
in the implementation. Iām not sure, but I think this is a tip to be able to call the function externally and internally (public
ās property) while being more gas-efficient (external
ās property). Iām waiting for OZ developers to confirm itās right. But I implemented it anyway in ERC7390 code implementation. - Made attributes private and only accessible with function defined in the interface.
issuanceCounter
is private but doesnāt have a function to access to, because this is not really an important data for an external entity.
Sorry this isnāt a lot of work, kinda busy these times.
Hope yāall have a great day
Hi @xeway
You may be interested in using EIP-6909 as a replacement for ERC1155 in the reference implementation.
Hey, so sorry for the late answer (as always ).
Didnāt know about ERC6909, it would way simpler for the current usecase of EIP-7390.
My concern is that ERC6909 is still a Draft, not very famousā¦ so is it worth the risk to implement it?
I would say it is worth looking into, the spec is quite close to being finalised and already has multiple implementations and drafts of extensions (like an ERC4626-like interface).