Skip to content

Latest commit

 

History

History
721 lines (677 loc) · 16.4 KB

psp-34.md

File metadata and controls

721 lines (677 loc) · 16.4 KB

Non-Fungible Token Standard for Substrate's contracts pallet

Summary

A standard for a Non-Fungible Token interface for WebAssembly smart contracts which run on Substrate's contracts pallet.

This proposal aims to define the standard Non-Fungible Token interface for WebAssembly smart contracts, just like EIP-721 for the Ethereum ecosystem.

Motivation

Without a standard interface for Non-Fungible Token every contract will have different signature and types. Hence, no interoperability is possible. This proposal aims to resolve that by defining one interface that shares the same ABI of permissionless methods between all implementations.

The goal is to have a standard contract interface that allows tokens deployed on Substrate's contracts pallet to be re-used by other applications: from wallets to decentralized exchanges.

Implementations

Examples of implementations:

Motivation for having a standard separate from ERC-721

Due to the different nature of WebAssembly smart contracts and the difference between EVM and the contracts pallet in Substrate, this standard proposal has specific rules and methods, therefore PSP-34 differs from ERC-721 in its implementation. Also the proposal contains new methods that should improve the usability of Non-Fungible Token.

This standard is at ABI level

Substrate's contracts pallet can execute any WebAssembly contract that implements its defined API; we do not want to restrain this standard to only Rust and the ink! language, but make it possible to be implemented by any language/framework that compiles to WebAssembly.

Specification

  1. Interfaces
  2. Extension
  3. Events
  4. Types
  5. Errors

Interfaces

PSP-34 Interface

This section defines the required interface for this standard.

collection_id() ➔ Id

Selector: 0xffa27a5f - first 4 bytes of blake2b_256("PSP34::collection_id")

{
  "args": [],
  "docs": [
    "Returns the collection `Id` of the NFT token.",
    "",
    "This can represents the relationship between tokens/contracts/pallets."
  ],
  "mutates": false,
  "label": "PSP34::collection_id",
  "payable": false,
  "returnType": {
    "displayName": [
      "Id"
    ],
    "type": "Id"
  },
  "selector": "0xffa27a5f"
}
balance_of(owner: AccountId) ➔ u32

Selector: 0xcde7e55f - first 4 bytes of blake2b_256("PSP34::balance_of")

{
  "args": [
    {
      "label": "owner",
      "type": {
        "displayName": [
          "AccountId"
        ],
        "type": "AccountId"
      }
    }
  ],
  "docs": [
    "Returns the balance of the owner.",
    "",
    "This represents the amount of unique tokens the owner has."
  ],
  "mutates": false,
  "label": "PSP34::balance_of",
  "payable": false,
  "returnType": {
    "displayName": [
      "u32"
    ],
    "type": 1
  },
  "selector": "0xcde7e55f"
}
owner_of(id: Id) ➔ Option

Selector: 0x1168624d - first 4 bytes of blake2b_256("PSP34::owner_of")

{
  "args": [
    {
      "label": "id",
      "type": {
        "displayName": [
           "Id"
        ],
        "type": "Id"
      }
    }
  ],
  "docs": [
    "Returns the owner of the token if any."
  ],
  "mutates": false,
  "label": "PSP34::owner_of",
  "payable": false,
  "returnType": {
    "displayName": [
      "Option"
    ],
    "type": "Option<AccountId>"
  },
  "selector": "0x1168624d"
}
allowance(owner: AccountId, operator: AccountId, id: Option) ➔ bool

Selector: 0x4790f55a - first 4 bytes of blake2b_256("PSP34::allowance")

{
  "args": [
    {
      "label": "owner",
      "type": {
        "displayName": [
          "AccountId"
        ],
        "type": "AccountId"
      }
    },
    {
      "label": "operator",
      "type": {
        "displayName": [
          "AccountId"
        ],
        "type": "AccountId"
      }
    },
    {
      "label": "id",
      "type": {
        "displayName": [
          "Option"
        ],
        "type": "Option<Id>"
      }
    }
  ],
  "docs": [
    "Returns `true` if the operator is approved by the owner to withdraw `id` token.",
    "If `id` is `None`, returns `true` if the operator is approved to withdraw all owner's tokens."
  ],
  "mutates": false,
  "label": "PSP34::allowance",
  "payable": false,
  "returnType": {
    "displayName": [
      "bool"
    ],
    "type": "bool"
  },
  "selector": "0x4790f55a"
}
approve(operator: AccountId, id: Option, approved: bool) ➔ Result<(), PSP34Error>

Selector: 0x1932a8b0 - first 4 bytes of blake2b_256("PSP34::approve")

{
  "args": [
    {
      "label": "operator",
      "type": {
        "displayName": [
          "AccountId"
        ],
        "type": "AccountId"
      }
    },
    {
      "label": "id",
      "type": {
        "displayName": [
          "Option"
        ],
        "type": "Option<Id>"
      }
    },
    {
      "name": "approved",
      "type": {
        "displayName": [
          "bool"
        ],
        "type": "bool"
      }
    }
  ],
  "docs": [
    "Approves `operator` to withdraw  the `id` token from the caller's account.",
    "If `id` is `None` approves or disapproves the operator for all tokens of the caller.",
    "",
    "An `Approval` event is emitted.",
    "",
    "# Errors",
    "",
    "Returns `SelfApprove` error if it is self approve.",
    "",
    "Returns `NotApproved` error if caller is not owner of `id`."
  ],
  "mutates": true,
  "label": "PSP34::approve",
  "payable": false,
  "returnType": {
    "displayName": [
      "Result"
    ],
    "type": 1
  },
  "selector": "0x1932a8b0"
}
transfer(to: AccountId, id: Id, data: [u8]) ➔ Result<(), PSP34Error>

Selector: 0x3128d61b - first 4 bytes of blake2b_256("PSP34::transfer")

{
  "args": [
    {
      "label": "to",
      "type": {
        "displayName": [
          "AccountId"
        ],
        "type": "AccountId"
      }
    },
    {
      "label": "id",
      "type": {
        "displayName": [
           "Id"
        ],
        "type": "Id"
      }
    },
    {
      "label": "data",
      "type": {
        "displayName": [
          "[u8]"
        ],
        "type": "[u8]"
      }
    }
  ],
  "docs": [
    "Transfer approved or owned token from caller.",
    "",
    "On success a `Transfer` event is emitted.",
    "",
    "# Errors",
    "",
    "Returns `TokenNotExists` error if `id` does not exist.",
    "",
    "Returns `NotApproved` error if `from` doesn't have allowance for transferring.",
    "",
    "Returns `SafeTransferCheckFailed` error if `to` doesn't accept transfer."
  ],
  "mutates": true,
  "label": "PSP34::transfer",
  "payable": false,
  "returnType": {
    "displayName": [
      "Result"
    ],
    "type": 1
  },
  "selector": "0x3128d61b"
}
total_supply() ➔ Balance

Selector: 0x628413fe - first 4 bytes of blake2b_256("PSP34::total_supply")

{
  "args": [],
  "docs": [
    "Returns the current total supply of the NFT."
  ],
  "mutates": false,
  "label": "PSP34::total_supply",
  "returnType": {
    "displayName": [
      "Balance"
    ],
    "type": "Balance"
  },
  "selector": "0x628413fe"
}

Extension

PSP34Metadata

PSP34Metadata is a recommended extension for this Non-Fungible Token standard because it is main feature of the NFT.

get_attribute(id: Id, key: [u8]) ➔ Option<[u8]>

Selector: 0xf19d48d1 - first 4 bytes of blake2b_256("PSP34Metadata::get_attribute")

{
  "args": [
    {
      "label": "id",
      "type": {
        "displayName": [
           "Id"
        ],
        "type": "Id"
      }
    },
    {
      "label": "key",
      "type": {
        "displayName": [
          "[u8]"
        ],
        "type": "[u8]"
      }
    }
  ],
  "docs": [
    "Returns the attribute of `id` for the given `key`.",
    "",
    "If `id` is a collection id of the token, it returns attributes for collection."
  ],
  "mutates": false,
  "label": "PSP34Metadata::get_attribute",
  "payable": false,
  "returnType": {
    "displayName": [
      "Option"
    ],
    "type": "Option<[u8]>"
  },
  "selector": "0xf19d48d1"
}

Attributes are more flexible than single tokenURI function like in Erc721. The list of required attributes for NFT should be defined in a separate proposal based on the scope of the usage.

PSP34Enumerable

PSP34Enumerable is an optional extension for this Multi Token standard to support enumeration of tokens.

owners_token_by_index(owner: AccountId, index: u128) ➔ Option

Selector: 0x3bcfb511 - first 4 bytes of blake2b_256("PSP34Enumerable::owners_token_by_index")

{
  "args": [
    {
      "label": "owner",
      "type": {
        "displayName": [
          "psp34enumerable_external",
          "OwnersTokenByIndexInput1"
        ],
        "type": 8
      }
    },
    {
      "label": "index",
      "type": {
        "displayName": [
          "psp34enumerable_external",
          "OwnersTokenByIndexInput2"
        ],
        "type": 6
      }
    }
  ],
  "docs": [
    " Returns a token `Id` owned by `owner` at a given `index` of its token list.",
    " Use along with `balance_of` to enumerate all of ``owner``'s tokens."
  ],
  "label": "PSP34Enumerable::owners_token_by_index",
  "mutates": false,
  "payable": false,
  "returnType": {
    "displayName": [
      "psp34enumerable_external",
      "OwnersTokenByIndexOutput"
    ],
    "type": 26
  },
  "selector": "0x3bcfb511"
}
token_by_index(index: u128) -> Option

Selector: 0xcd0340d0 - first 4 bytes of blake2b_256("PSP37Enumerable::owners_token_by_index")

{
  "args": [
    {
      "label": "index",
      "type": {
        "displayName": [
          "psp34enumerable_external",
          "TokenByIndexInput1"
        ],
        "type": 6
      }
    }
  ],
  "docs": [
    " Returns a token `Id` at a given `index` of all the tokens stored by the contract.",
    " Use along with `total_supply` to enumerate all tokens."
  ],
  "label": "PSP34Enumerable::token_by_index",
  "mutates": false,
  "payable": false,
  "returnType": {
    "displayName": [
      "psp34enumerable_external",
      "TokenByIndexOutput"
    ],
    "type": 26
  },
  "selector": "0xcd0340d0"
}

Events

Transfer

When a contract creates (mints) new tokens, from will be None. When a contract deletes (burns) tokens, to will be None.

{
  "args": [
    {
      "docs": [],
      "indexed": true,
      "label": "from",
      "type": {
        "displayName": [
          "Option"
        ],
        "type": "Option<AccountId>"
      }
    },
    {
      "docs": [],
      "indexed": true,
      "label": "to",
      "type": {
        "displayName": [
          "Option"
        ],
        "type": "Option<AccountId>"
      }
    },
    {
      "docs": [],
      "indexed": true,
      "label": "id",
      "type": {
        "displayName": [
           "Id"
        ],
        "type": "Id"
      }
    }
  ],
  "docs": [
    "Event emitted when a token transfer occurs."
  ],
  "label": "Transfer"
}

Approval

{
  "args": [
    {
      "docs": [],
      "indexed": true,
      "label": "owner",
      "type": {
        "displayName": [
          "AccountId"
        ],
        "type": "AccountId"
      }
    },
    {
      "docs": [],
      "indexed": true,
      "label": "operator",
      "type": {
        "displayName": [
          "AccountId"
        ],
        "type": "AccountId"
      }
    },
    {
      "docs": [],
      "indexed": true,
      "label": "id",
      "type": {
        "displayName": [
          "Option<Id>"
        ],
        "type": "Option<Id>"
      }
    },
    {
      "docs": [],
      "indexed": false,
      "label": "approved",
      "type": {
        "displayName": [
          "bool"
        ],
        "type": "bool"
      }
    }
  ],
  "docs": [
    "Event emitted when a token approve occurs."
  ],
  "label": "Approval"
}

AttributeSet

{
  "args": [
    {
      "docs": [],
      "indexed": false,
      "label": "id",
      "type": {
        "displayName": [
           "Id"
        ],
        "type": "Id"
      }
    },
    {
      "docs": [],
      "indexed": false,
      "label": "key",
      "type": {
        "displayName": [
          "[u8]"
        ],
        "type": "[u8]"
      }
    },
    {
      "docs": [],
      "indexed": false,
      "label": "data",
      "type": {
        "displayName": [
          "[u8]"
        ],
        "type": "[u8]"
      }
    }
  ],
  "docs": [
    "Event emitted when an attribute is set for a token.",
  ],
  "label": "AttributeSet"
}

Types

// Id is an Enum and its variant are types
enum Id {
    U8(u8),
    U16(u16),
    U32(u32),
    U64(u64),
    U128(u128),
    Bytes(Vec<u8>),
}

// AccountId is a 32 bytes Array, like in Substrate-based blockchains.
type AccountId = [u8; 32];

Return types

{
  "types": {
    "1": {
      "def": {
        "variant": {
          "variants": [
            {
              "fields": [
                {
                  "type": {
                    "def": {
                      "tuple": []
                    }
                  }
                }
              ],
              "name": "Ok"
            },
            {
              "fields": [
                {
                  "type": {
                    "def": {
                      "variant": {
                        "variants": [
                          {
                            "fields": [
                              {
                                "type": "string"
                              }
                            ],
                            "name": "Custom"
                          },
                          {
                            "name": "SelfApprove"
                          },
                          {
                            "name": "NotApproved"
                          },
                          {
                            "name": "TokenExists"
                          },
                          {
                            "name": "TokenNotExists"
                          },
                          {
                            "fields": [
                              {
                                "type": "string"
                              }
                            ],
                            "name": "SafeTransferCheckFailed"
                          }
                        ]
                      }
                    },
                    "path": [
                      "PSP34Error"
                    ]
                  }
                }
              ],
              "name": "Err"
            }
          ]
        }
      }
    }
  }
}

Errors

The suggested methods revert the transaction and return a SCALE-encoded Result type with one of the following Error enum variants:

enum PSP34Error {
    /// Custom error type for cases if writer of traits added own restrictions
    Custom(String),
    /// Returned if owner approves self
    SelfApprove,
    /// Returned if the caller doesn't have allowance for transferring.
    NotApproved,
    /// Returned if the owner already own the token.
    TokenExists,
    /// Returned if the token doesn't exist
    TokenNotExists,
    /// Returned if safe transfer check fails
    SafeTransferCheckFailed(String),
}

Copyright

This PSP is placed in the public domain.