-
Notifications
You must be signed in to change notification settings - Fork 84
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
SIP-033 The Generalized Standard for non-custodial Marketplace traits #120
Open
314159265359879
wants to merge
1
commit into
stacksgov:main
Choose a base branch
from
no314:SIP022-branch
base: main
Could not load branches
Branch not found: {{ refName }}
Loading
Could not load tags
Nothing to show
Loading
Are you sure you want to change the base?
Some commits from the old base branch may be removed from the timeline,
and old review comments may become outdated.
Open
Changes from all commits
Commits
File filter
Filter by extension
Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
There are no files selected for viewing
310 changes: 310 additions & 0 deletions
310
sips/sip-022/sip-022_generalized_marketplace_standard.md
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,310 @@ | ||
# Preamble | ||
|
||
SIP Number: 022 | ||
|
||
Title: The Generalized Standard for non-custodial Marketplace traits | ||
|
||
Authors: werner.btc (werner at stx.fan), Friedger, Mike, Jason and Jamil. | ||
|
||
Consideration: Technical | ||
|
||
Type: Standard | ||
|
||
Status: draft | ||
|
||
Created: 10 January 2023 | ||
|
||
License: CC0-1.0 | ||
|
||
Sign-off: Brice Dobry <[email protected]>, Technical Consideration Advisory Boards (CAB) Chair | ||
Sign-off: Jude Nelson <[email protected]>, Technical Steering Committee Chair | ||
|
||
# Abstract | ||
|
||
This standard provides additional functions to digital assets (i.e. tokens) so that they can be made availble to buy from the owners public address. This in effect enables an open, decentralized marketplace for digital assets. | ||
|
||
# License and Copyright | ||
|
||
This SIP is made available under the terms of the Creative Commons CC0 1.0 Universal license, available at https://creativecommons.org/publicdomain/zero/1.0/ | ||
This SIP’s copyright is held by the Stacks Open Internet Foundation. | ||
|
||
# Introduction | ||
|
||
Tokens are digital assets registered on a blockchain through a smart contract. A non-fungible token (NFT) is a token that is globally unique and can be identified through its unique identifier. | ||
|
||
In blockchains with smart contracts, including the Stacks blockchain, developers and users can use smart contracts to register and interact with (non-)fungible tokens. | ||
|
||
To extend the functions of tokens this SIP defines two traits with smart contract functions that enable an open, decentralized marketplace for digital assets. These assets must implement the `marketplace trait` and `commission trait`. Buyers should be able to buy the assets from the owner with STX or another SIP010 compliant token. | ||
|
||
The `marketplace trait` can be implemented as part of the asset contract (usually not recommended) or as an independent contract. | ||
|
||
The marketplace functions have been used since the first Megapont contract was deployed to mainnet in October 2022 and have since become the pseudo standard for non-custodial listing of assets on the Stacks Blockchain. Having an this official standard ratified can help further adoption of the standard. [(1)](https://) [(2)](https://github.com/stacksgov/sips/issues/92#issuecomment-1360144565) | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. Link 1 is broken - should remove or replace with appropriate reference |
||
|
||
# Specification | ||
|
||
Every SIP-011 compliant smart contract on the Stacks blockchain must implement these traits; | ||
1. `marketplace-functions`, defined in the [Marketplace-Trait-and-Commision-Trait](#Marketplace-Trait-and-Commision-Trait) section, | ||
2. `commission-functions`, defined in the [Marketplace-Trait-and-Commision-Trait](#Marketplace-Trait-and-Commision-Trait) section; | ||
|
||
Optionally seperate out royalties from commission, as suggested by Jamil? | ||
3. royalties? | ||
|
||
Note: Not sure about "transferrable trait" instead of token-trait-buy? | ||
|
||
\/\/\/\/\/\/\/\/\/\/\/\/\/\/\/\/\/\ to be updated later \/\/\/\/\/\/\/\/\/\/\/\/\/\/\/\/ | ||
### Marketplace-Trait function 1: List in token | ||
`(list-in-token (<transferable-trait> uint uint <commission-trait>) (response bool uint))` | ||
|
||
This function announces the listing to a global marketplace. This must return `(ok true)` on success, never `(ok false)` and it must send a list event. | ||
|
||
The Function takes a transferable-trait, a token id or token amount to be listed and the third argument is the price it should be listed at, lastly the function takes the commission trait. | ||
|
||
### Marketplace-Trait function 2: Unlist in token | ||
`(unlist-in-token (<transferable-trait> uint) (response bool uint))` | ||
|
||
This function announces the unlisting to a global marketplace. This must return `(ok true)` on success, never `(ok false)` and it must send a delist event. | ||
|
||
The Function takes a transferable-trait, a token id or token amount to be unlisted. | ||
|
||
### Marketplace-Trait function 3: Buy in tokens and annonce delisting | ||
`(buy-in-token (<transferable-trait> uint <commission-trait>) (response bool uint))` | ||
|
||
This function buys the listed token for the selling price and delists it. | ||
|
||
This function takes transferable-trait and an amount of the token (selling price). Additionally the function takes a commission-trait as defined below it will dictate actions to happen after the sale (owner, artist, marketplace, etc. getting their share of the sale). | ||
|
||
This function must be defined with define-public, as it alters state, and must be externally callable. The function must return `(ok true)` on success. | ||
|
||
Werner: Should this be explained here, I do not understand it: https://github.com/stacksgov/sips/issues/60#issuecomment-1050075826 . | ||
|
||
### Marketplace-Trait function 4: Get asset | ||
`(get-asset () (response {fq-contract: string, asset-class: string} uint))` | ||
|
||
The is a read only function defining the asset. | ||
|
||
|
||
### Commision-Trait function: | ||
`(pay (uint uint) (response bool uint))` | ||
Werner: (this should take an optional token trait if generalized to any token?... should there be an allowlist for token traits in that case or would this always be uSTX?) | ||
|
||
An additional action after the marketplace sale happened. Usually a token transfer (fee) to the marketplace. | ||
|
||
The function takes | ||
|
||
It is recommended to use error codes from standardized list of codes and implement the function for converting the error codes to messages function that are defined in a separate SIP. | ||
|
||
/\/\/\/\/\/\/\/\/\/\/\/\/\/\/\/\/\/\/\/\/\/\/\/\/\/\/\/\/\/\/\/\/\/\/\/\/\/\/\/\/\/\/\/\ | ||
|
||
## Marketplace-Trait-and-Commision-Trait | ||
Adapted based on most recent discussion here by Friedger and Jamil dec 2022: https://github.com/stacksgov/sips/issues/92#issuecomment-1359529405 | ||
|
||
``` | ||
(use-trait commission-trait .commissions.trait) | ||
(define-trait marketplace | ||
( | ||
;; announce listing to global marketplace | ||
;; must return `(ok true)` on success, never `(ok false)` | ||
;; must send a list event | ||
;; @param; token-trait-sell (of token(s) to list) | ||
;; @param id (to list); identifier of NFT or SFT (ignored if <token-trait-sell> = FT) | ||
;; @param amount (to list); amount of FT or SFT (ignored if <token-trait-sell> = NFT) | ||
;; @param; token-trait-buy (to buy with) | ||
;; @param token-id; identifier of NFT or SFT to list for (ignored if <token-trait-buy> = FT) | ||
;; @param price(amount): of sale in smallest denomination of the token defined by <token-trait-buy> (ignored if <token-trait-buy> = NFT) | ||
;; @param commission: action to happen after sale | ||
(list-in-token (<token-trait-sell> uint uint <token-trait-buy> uint uint <commission-trait>) (response bool uint)) | ||
|
||
;; announce delisting to global marketplace | ||
;; must return `(ok true)` on success, never `(ok false)` | ||
;; must send a delist event | ||
;; @param id; identifier of NFT or SFT (ignored if <token-trait-sell> = FT) | ||
;; @param amount; amount of FT or SFT (ignored if <token-trait-sell> = NFT) (note: you can also list FT's on a global marketplace) | ||
(unlist-in-token (uint uint) (response bool uint)) | ||
|
||
;; buy and announce delisting to global marketplace | ||
;; must return `(ok true)` on success, never `(ok false)` | ||
;; commission must match the one set during listing | ||
;; must send a delist event | ||
;; @param; token-trait-buy (token to buy with) | ||
;; @param id; identifier of NFT or SFT (ignored if FT) | ||
;; @param amount; amount of FT or SFT (ignored if NFT) | ||
;; @param commission: action to happen after sale | ||
(buy-in-token (<token-trait-buy> uint uint '<commission-trait>') (response bool uint)) | ||
|
||
;; read-only function defining the asset | ||
(get-asset () (response {fq-contract: string, asset-class: string} uint)) | ||
) | ||
) | ||
|
||
(define-trait commission | ||
( | ||
;; additional action after a sale happened, usually a fee transfer for marketplaces | ||
;; must return `(ok true)` on success, never `(ok false)` | ||
;; @param id; identifier of NFT or SFT (ignored if FT) | ||
;; @param amount; amount of FT or SFT (ignored if NFT) | ||
;; @param; token-trait-buy | ||
;; @param token-id; identifier of NFT or SFT sold for (ignored if <token-trait-buy> = FT) | ||
;; @param price(amount): of sale in smallest denomination of the token defined by <token-trait-buy> (ignored if <token-trait-buy> = NFT) | ||
(pay (uint uint <token-trait-buy> uint uint) (response bool uint)) (Note: when buying with SFT's or an NFT, how to fractonalise for commissions if at all? standard rounding down to whole tokens? commissions in STX or no commissions, subscriptions instead?) | ||
) | ||
) | ||
``` | ||
|
||
## Security | ||
As `commission-traits` can call any functions in the name of the tx-sender, it is important that a web app only offers commission contracts that are well understood. In particular, appropriate post-conditions have to be created. | ||
|
||
If asset contracts want to control trades they have to restrict which operators are approved. Note, that royalties to an artist of an NFT can be part of the commission if agreed with the marketplace. They can also be implemented in the NFT directly. | ||
|
||
## Use of native asset functions | ||
Werner: not sure what to do here (section just a copy from sip009) | ||
|
||
Although it is not possible to mandate in a Clarity trait, contract implementers must define at least one built-in native non-fungible [asset class](https://app.sigle.io/friedger.id/FDwT_3yuMrHDQm-Ai1OVS) that are provided as Clarity primitives. This allows clients to use Post Conditions (explained below), and takes advantages of other benefits, like native support for these asset balances and transfers through `stacks-blockchain-api`. The reference implementations included in this SIP use the native asset primitives, and provide a good boilerplate for their usage. | ||
|
||
The native asset functions include: | ||
|
||
- `define-non-fungible-token` | ||
- `nft-burn?` | ||
- `nft-get-owner?` | ||
- `nft-mint?` | ||
- `nft-transfer?` | ||
|
||
The following requirements for using native asset functions are defined: | ||
### Transfer | ||
|
||
If the `transfer` function is called from a client without a [post-condition](https://docs.blockstack.org/understand-stacks/transactions#post-conditions) in deny mode or without any NFT condition about a changed owner, then the function call must fail with `abort_by_post_condition`. | ||
|
||
### Recommendation for using STX as SIP010 token | ||
|
||
For the generalized interface and hence the `_in-token_`-functions (`list-in-token`, `buy-in-token` etc.) as a single function to use any token, including STX. It is recommended that the Stacks Foundation deploy the following contract as a means to allow the use of STX as a SIP010 token without the need for hold wrapped STX. | ||
|
||
From lNow 2022: https://github.com/stacksgov/sips/issues/60#issue-1148066438 | ||
|
||
A global wrapped STX (SIP-010 compliant) FT that is not issuing a new token, but uses plain STX: | ||
|
||
`(impl-trait 'SP3FBR2AGK5H9QBDH3EEN6DF8EK8JY7RX8QJ5SVTE.sip-010-trait-ft-standard.sip-010-trait) | ||
|
||
(define-read-only (get-balance (owner principal)) | ||
(ok (stx-get-balance owner)) | ||
) | ||
|
||
(define-read-only (get-decimals) | ||
(ok u6) | ||
) | ||
|
||
(define-read-only (get-name) | ||
(ok "Global Wrapped STX") | ||
) | ||
|
||
(define-read-only (get-symbol) | ||
(ok "GWSTX") | ||
) | ||
|
||
(define-read-only (get-token-uri) | ||
(ok (some u"https://www.stacks.co")) | ||
) | ||
|
||
(define-read-only (get-total-supply) | ||
(ok stx-liquid-supply) | ||
) | ||
|
||
(define-public (transfer (amount uint) (sender principal) (recipient principal) (memo (optional (buff 34)))) | ||
(begin | ||
(try! (stx-transfer? amount sender recipient)) | ||
(match memo to-print (print to-print) 0x) | ||
(ok true) | ||
) | ||
)` | ||
|
||
Werner: WSTX/Wrapped STX is already used by Arkadiko and is a different contract. GWSTX does not have a naming collision. "Global" could also be replaced by "Foundation" if the contract is indeed deployed by the Stacks Foundation. Using just "wrapped STX" may help users from the ETH ecosystem understand what it is. I am not sure how important it is to prevent naming collisions here although the prior discussion did stear to avoid it. | ||
|
||
### Recommendations for displaying commissions | ||
|
||
It is advised to show the user what the listing price will be on the platform used and how much of that will go to the interested parties when sold such as the seller, the artist and a marketplace. | ||
|
||
It is recommended to marketplaces to do the following if a user lists something at 1 STX, don't send 1000000 to list-in-ustx, but rather 1000000/(1+comm) as the price argument. | ||
|
||
Some examples of how the user would be informed as displayed in December 2022 on Gamma.io and Tradeport.xyz: | ||
|
||
**Example 1 tradeport.xyz** | ||
![image](https://user-images.githubusercontent.com/33360391/208997197-61111244-b597-46fd-9c08-559550a127aa.png) | ||
|
||
**Example 2 gamma.io** | ||
![image](https://user-images.githubusercontent.com/33360391/208997558-6dcbdef7-692f-4b41-b7eb-df429c3cfead.png) | ||
|
||
### Recommendations for error codes | ||
|
||
From Friedger 2022: https://github.com/stacksgov/sips/issues/60#issuecomment-1048832083 | ||
|
||
unsupported token: u500 | ||
insufficient balance: u103 | ||
not authorized to mint: u403 | ||
|
||
# Using NFTs in applications | ||
|
||
Developers who wish to use a non-fungible token contract in an application should first be provided, or keep track of, various different non-fungible token implementations. When validating a non-fungible token contract, they should fetch the interface and/or source code for that contract. If the contract implements the trait, then the application can use this standard's contract interface for making transfers and getting other details defined in this standard. | ||
|
||
All of the functions in this trait return the `response` type, which is a requirement of trait definitions in Clarity. However, some of these functions should be "fail-proof", in the sense that they should never return an error. These "fail-proof" functions are those that have been recommended as read-only. If a contract that implements this trait returns an error for these functions, it may be an indication of a non-compliant contract, and consumers of those contracts should proceed with caution. | ||
|
||
## Use of Post-Conditions | ||
|
||
The Stacks blockchain includes a feature known as "Post-Conditions" or "Constraints". By defining post-conditions, users can create transactions that include pre-defined guarantees about what might happen in that contract. | ||
|
||
For example, when applications call the `transfer` function, they should _always_ use post conditions to specify that the new owner of the NFT is the recipient principal in the `transfer` function call. | ||
|
||
|
||
From: https://github.com/stacksgov/sips/issues/60#issuecomment-1050196178 | ||
friedger commented on Feb 24 | ||
|
||
How do you build post conditions for these txs if they can transfer either STXs or FTs? | ||
@LNow | ||
Author | ||
LNow commented on Feb 24 | ||
|
||
The same way Alex/Arkadiko/Stackswaps are doing it. Based on token address user picked on the UI. With one additional if statement. | ||
If user choose wrapped stx -> STX post-condition, otherwise -> FT post-condition. | ||
|
||
# Related Work | ||
|
||
Loopbom | ||
https://github.com/radicleart/clarity-market | ||
|
||
Megapont Ape Club | ||
https://explorer.stacks.co/txid/SP3D6PV2ACBPEKYJTCMH7HEN02KP87QSP8KTEH335.megapont-ape-club-nft?chain=mainnet | ||
|
||
# Backwards Compatibility | ||
|
||
Not applicable | ||
|
||
# Activation | ||
|
||
This SIP is activated after if there are no objections within the community before Bitcoin tip #780,000. There are already 20 or more contracts that have been deployed to mainnet that have implemented this standard. | ||
|
||
A trait that follows this specification is available on mainnet as: `to add later` | ||
|
||
# Alternative considered | ||
Instead of unified trait to allow the use of SIP010, SIP009 and SIP013 tokens for purchase have two traits one for SIP009 and SIP010 and the other for SIP013. This would eleviate the need for a uint that is ignored when using SIP009 and SIP010 but is required for using SIP013 tokens as one would have to define both a token identifier and an amount. | ||
|
||
# Reference Implementations list-in-token | ||
|
||
To be developed | ||
|
||
## Source code | ||
|
||
# Sources | ||
Marketplace function | ||
https://github.com/stacksgov/sips/issues/51 | ||
Generalized marketplace function (list-in-token vs. list-in-ustx) | ||
https://github.com/stacksgov/sips/issues/51#issuecomment-1151670018 | ||
implementation 1: https://github.com/radicleart/clarity-market/blob/main/projects/risidio/indige/contracts/indige.clar | ||
implementation 2: ? | ||
Optional send-many trait (from SIP013 semi-fungible token) | ||
https://github.com/stacksgov/sips/pull/42/files | ||
|
||
Examples of commission contracts | ||
SIP010 commision: https://github.com/radicleart/clarity-market/blob/main/projects/risidio/indige/contracts/commission-sip10-nop.clar | ||
Simple fixed fee: `?` | ||
Auction contract: `?` | ||
Marketplace commission and royalties split: gamma.io example | ||
|
||
Example Wrapped stx contract (defines STX as SIP010 token) | ||
https://github.com/radicleart/clarity-market/blob/main/projects/risidio/indige/contracts/wrapped-stx.clar |
Add this suggestion to a batch that can be applied as a single commit.
This suggestion is invalid because no changes were made to the code.
Suggestions cannot be applied while the pull request is closed.
Suggestions cannot be applied while viewing a subset of changes.
Only one suggestion per line can be applied in a batch.
Add this suggestion to a batch that can be applied as a single commit.
Applying suggestions on deleted lines is not supported.
You must change the existing code in this line in order to create a valid suggestion.
Outdated suggestions cannot be applied.
This suggestion has been applied or marked resolved.
Suggestions cannot be applied from pending reviews.
Suggestions cannot be applied on multi-line comments.
Suggestions cannot be applied while the pull request is queued to merge.
Suggestion cannot be applied right now. Please check back later.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Should add Traits field.
Traits:
marketplace-functions
andcommission-functions