This was developed for the Untitled NFT Hackaton while working on a safe and hopefully in the future standard way to create Interactive NFTs.
This project is the open source Sandbox for executing and viewing Interactive NFTs.
If you're just looking to easily embed NFTs, you might be looking for the BeyondNFT/embeddable project.
The Sandbox has no idea about the existence of the Blockchain (it works with JSON already loaded), when BeyondNFT/embeddable makes direct calls to the smart contracts to get all the data needed.
This Sandbox project is more an "in deep" presentation of what are Interactive NFTs and how they work.
This is the good place to see the end schema of Interactive NFT's JSON, which can guide platforms into creating their own Sandbox if they do not trust this one. (you could for example run the NFTs in an iframe that you host on a subdomain of your website if srcdoc iframes are not your thing)
Creator: the entity (Artist, Developer, Platform, ...) who created the NFT. Owner: the current NFT owner. Viewer: the person viewing the NFT (might be any user, including Creator or Owner).
What follows in mainly for developers to know what the Sandbox expects for properties to work and to participate to the development of this (maybe?) new standard.
As a Creator or an Owner, you should probably never have to edit any of those values by hand. The tools provided by the platform you used to create Interactive NFTs should be enough.
If like me, you prefer reading code with comments better than long walls of text, just jump to Usage
Interactive NFT is a project that aims to:
- Allow NFTs to be dynamic and/or interactives to the Viewer (procedural art with js, html, external data call, music player, video player...).
- Allow a Creator to declare some values "configurable/variables" and an Owner to configure those values, making the NFT evolve. (a bit like Async, but because the NFT is code running, it can go much deeper)
This way, an Artist could for example create a procedural piece of art, and allow the future Owners to set some key values used during the art rendering, thus making the Art evolutive. The Owner could for example edit colors, animation durations, texts, textures or anything that the Creator declared as editable.
Another example would be a Card on which the Viewer could click; the card would then flip and present its attributes (that are stored in the NFT's JSON or even retrieven with an ajax call). All this directly in a Gallery / Website / Marketplace.
This Sandbox aims to display the NFT code in a safe way for the Viewer.
For Security reasons, the NFT Code is sandboxed in an iframe, using srcdoc.
By default, only "allow-script", "allow-pointer-lock", "allow-downloads" and "allow-popups" are enabled. So no access to parent context or same origin stuffs (cookies, localStorage & co).
eval
and alert
are also disabled.
Idea (but only idea, this is risky and would need to be handled with a lot of care): Create a system of Permissions, allowing Creators to request some permissions that the Viewer would have to accept or decline.
The idea is to have a property interactive_nft
in the NFT's JSON that declares the NFT as an Interactive NFT.
This propery references where to find the code, the dependencies, the default configuration (if any) etc...
Under interactive_nft.version
must be declared the version of the Interactive NFT used. This in order to help Sandboxed in the futur to know how to render the NFT, if the "standard" comes to evolve.
The code of the NFT. It is expected to be VALID HTML that will be inserted at the end of the iframe's body
tag.
There are 2 ways to declare code in the interactive_nft
.
- (recommended) as an URI under
interactive_nft.code_uri
(1) . The Sandbox will detect this property, fetch the URI as a text file and use the content as the NFT code. - (not recommended) As a string under
interactive_nft.code
. The Sandbox will also use this code as the NFT code. However, because the code is HTML and can contain special characters that are not playing well with JSON, it is recommended to not save the code directly into the NFT JSON, and to use thecode_uri
property instead.interactive_nft.code
is mainly here to be used when creating the NFTs, because the code won't be hosted already.
This gives a huge flexibility to Creators. They can then add HTML, JavaScript and CSS to the NFT.
<style>.foo { font-size: 3rem; }</style>
<div class="foo"></div>
<script type="text/javascript">
document.querySelector('.foo').innerText = "Hello world!";
</script>
is a perfectly fine NFT code.
(1) preferably hosted somewhere on a decentralized host (IPFS, Arweave or the like).
Recommended: some platforms have requested that the code_uri
(or code
if used) was signed (using the Wallet, i.e web3.eth.sign(message, account)
) by the Creator.
This, allowing Viewer to be able to define an "allow-list" of creators they want to execute the content automatically. Else, the platform will use an "Execute this interactive NFT" button, to protect user from possible non expected rendering.
Therefore, if you can, before saving the JSON, ask creators to sign code_uri
(and save the result signature in code_uri_signature
) or code
(and save in code_signature
)
** This is not used internally by the Sandbox, it's mostly to have a proof of who created the NFT, allowing more control for the Viewer **
Dependencies are declared under interactive_nft.dependencies
.
It is an array of object of the form : { url: String, type: String }
When loading the NFT, the Sandbox will add the dependencies into script
and style
tags in the iframe before the NFT code.
JSON Schema
{
"title": "Interactive NFT Dependencies",
"type": "array",
"items": {
"type": "object",
"properties": {
"required": ["type", "url"],
"type": {
"type": "string",
"enum": ["script", "style"],
"description": "Type of the dependency (script or style tag)."
},
"url": {
"type": "string",
"description": "URL of the dependency."
}
}
}
}
Example:
{
"name": "Interactive NFT #1",
"description": "The first of its kind.",
"image": "http://gateway.ipfs.io/Qxn...",
"interactive_nft": {
"code_uri": "http://gateway.ipfs.io/Qxn...",
"code_uri_signature": "0x0123456789abcdef...",
"dependencies": [{
"type": "script",
"url": "https://cdnjs.cloudflare.com/ajax/libs/p5.js/1.1.9/p5.min.js"
}, {
"type": "style",
"url": "https://stackpath.bootstrapcdn.com/bootstrap/4.5.2/css/bootstrap.min.css"
}]
}
}
Contract implementing configurable Interactive NFTs are expected to provide a public method interactiveConfURI(_tokenId, _owner) public returns (string)
which works for both ERC721 and ERC1155, and returns the conf URI for a tokenId (if set by the Owner).
It should also be paired with a setter method setInteractiveConfURI(_tokenId, _uri)
for Owners to be able to set the URI see ERC721Configurable
If the Creator declared some configurable properties, they MUST have a default value: this to be able to reset configuration if anything is wrongly configured.
Configurable properties are declared under interactive_nft.properties
.
It is an Array of Objects of the form { name: String, type: String, value: String|Number|Array|Object }
JSON Schema
{
"title": "Interactive NFT Configurable Properties",
"type": "array",
"items": {
"type": "object",
"properties": {
"required": ["name", "type", "value"],
"name": {
"type": "string",
"description": "Name of the property.",
},
"type": {
"type": "string",
"enum": ["string", "number", "array", "object"],
"description": "Type of the property.",
},
"value": {
"type": ["string", "number", "array", "object"],
"description": "This is the default value. May be a string, number, object or array.",
},
"description": {
"type": "string",
"description": "Description of what the property is used for. Might be needed if name not explicit enough.",
}
}
}
}
Example:
{
"name": "Interactive NFT #1",
"description": "The first of its kind.",
"image": "http://gateway.ipfs.io/Qxn...",
"interactive_nft": {
"code_uri": "http://gateway.ipfs.io/Qxn...",
"code_uri_signature": "0x0123456789abcdef...",
"properties": [{
"name": "duration",
"type": "number",
"value": 500,
"description": "Animation duration"
},
{
"name": "name",
"type": "string",
"value": "John Doe",
"description": "Your cat name",
},
{
"name": "fruits",
"type": "array",
"value": ["orange", "banane"],
"description": "Your favorite fruits",
},
{
"name": "fullname",
"type": "object",
"value": {
"name": "Doe",
"surname": "John"
},
"description": "Your cat fullname",
}]
}
}
(What follows is automatically done when using BeyondNFT/embeddable to show Interactive NFTs to Viewers.)
When loading the NFT Metadata, the platform showing the NFT to the Viewers should check the existence of this interactive_nft.properties
, and, if defined, should call the NFT contract interactiveConfURI(_tokenId, _owner)
to see if the current NFT owner has a configuration file for this NFT.
If a configuration file exists, its content as a JavaScript object will be expected to be passed to the sandbox under owner_properties
(see usage)
When detecting interactive_nft.properties
, the Sandbox will automatically search for owner_properties
.
It will then override interactive_nft.properties
with owner_properties
and write the whole properties object to window.context.properties
. It will then be accessible in the js code using const propValue = window.context.properties[propertyName];
There is nothing that forces the configuration file to be on IPFS or any decentralized network.
Platform could just mint the NFT with an URI to an empty configuration JSON that they host on their server (e.g http://mynftplatform.com/tokens/0xcontract/{id}.config.json
), or Owners could just set the configuration file to any raw gist.
This way, when the Owner wants to edit some values, they can do it without having to do a new transaction because they already have control over the configuration file (either through gist or through the platform editing the NFTs).
It is sure less decentralized, but it saves the cost of transactions and the Owner can always set the URI stored in the contract to another URI, having full control over the file.
As of now, accepted types are 'string', 'number', 'object' and 'array' (see Usage)
This Sandbox is to be used as follow:
new SandBox({ target, props })
Construction parameter is an Object with two required properties :
target
: an HTML element where to render the Sandbox
props
: An object of properties read by the Sandbox
Props Schema :
{
"title": "Sandbox Props definition",
"type": "object",
"properties": {
"required": ["data"],
"data": {
"type": ["object", "string"],
"description": "NFT's JSON URI or Object (result of JSON.parse of the tokenURI -erc721- or uri -erc1155-)."
},
"owner": {
"type": "string",
"description": "Address of the current token owner. Default to solidity's address(0)."
},
"owner_properties": {
"type": ["object", "string"],
"description": "Owner configuration JSON URI or Object (result of JSON.parse of the interactiveConfURI contract method).""
}
}
}
Full example of usage, because code is ten times better than words (You can also see ./public/index.html to see another one)
import Sandbox from 'https://cdn.jsdelivr.net/npm/@beyondnft/sandbox@0.0.8/dist/nftsandbox.es.min.js';
const sandbox = new SandBox({
target: document.querySelector('#viewer'),
props: {
// data: required | Mixed
// This is either the tokenURI (string) or the content (object - JSON.parsed) of tokenURI|uri
//
// The Sandbox will look for the `interactive_nft` property in thos object
// Else it will show the `image` property
//
// This whole JSON will also be available in the iframe as a JavaScript object
// under `window.context.nft_json` so the code can read data from it
// (for example to get attributes or the NFT name)
data: {
name: 'Interactive NFT #1',
description: "The first of its kind.",,
image: "http://gateway.ipfs.io/Qxn...",
// interactive_nft: required (for Interactive NFT)
// this is the property the Sandbox will look for in order to render the NFT
interactive_nft: {
// code_uri (optional) - URI where to find the code to execute
code_uri: 'ipfs://Qx....',
// the code_uri_signature contains the signature of the URI by the Creator, proving that they
// are the one that created this code
code_uri_signature: "0x0123456789abcdef...",
// code (optional) (non recommended, mostly used when creating the NFT in a codepen-like env)
code: '<script>console.log("here");</script>',
// code_signature (optional)
// the code_signature is used when using `code` and not `code_uri`
// it contains the signature of the `code` by the Creator, proving that they
// are the one that created this code
code_signature: "0x0123456789abcdef...",
// version: required
// version of the Sandbox used to create this Interactive NFT, might be important at some point
// if the way the sandbox works changes a lot, we will need to know what Sandbox to use to load
// the NFT
version: '0.0.8',
// dependencies: optional
// Array of dependencies that the Sandbox should load before executing the NFT code.
//
// @dev When a creator makes an NFT with dependencies, please show a reminder that if the
// dependencies break (404, cdn stops working, ...) the NFT will not work anymore
// Maybe offer to host those on IPFS?!
dependencies: [
{
// type: required (script|style)
// type of the dependency, will define how it is handled by the Sandbox
type: 'script',
// url: required
// url where to find the dependency
url: 'https://cdn.jsdelivr.net/npm/p5@1.1.9/lib/p5.js',
},
{
type: 'style',
url: 'https://fonts.googleapis.com/css2?family=Roboto:wght@100&display=swap',
},
// ...
],
// properties: optional
// an array of configurable properties
// those properties are here to configure the NFT rendering
// because those are the one the Owner of the NFT can modify and store somewhere
// for them to be loaded when users are viewing their NFT
// These values will be set in `window.context.properties` and be accessed in the NFT Javascript as follow
// `const propertyValue = window.context.properties[propertyName]`;
properties: [{
// name: required
// name of the property
name: "duration",
// type: required
// type of the property, can be number, string, array or object
// mainly for configuration editors to know what they deal with
type: "number",
// value: required
// this is the default value of the property.
// @dev if is very important to have a value here, because if an owner "breaks" their NFT by saving a boggus configuration file, this value can be used to recreate a default configuration file that won't break the NFT. even empty values ('', 0, [], {}) should be enough
value: 500,
// description: optional
// small text describing what this value is used for.
// might be needed if the name is not explicit enough
description: "Duration of the animation"
},
{
name: "name",
type: "string",
value: "John Doe"
},
{
name: "fruits",
type: "array",
value: ["orange", "banane"]
},
{
name: "fullname",
type: "object",
value: {
name: "Doe",
surname: "John"
}
}
// ...
]
}
},
// owner: optional (but should be set when possible - defaults to 0x0000000000000000000000000000000000000000)
// Address of the current owner/holder. Accessible in the code under `window.context.owner`
owner: '0xAb5801a7D398351b8bE11C439e05C5B3259aeC9B',
// owner_properties: optional | Mixed
// This is either the uri where to find the configuration file (string) or the content of this file (object - JSON.parsed)
// if it's an URI the sandbox will first retrieve the data and JSON.parse it
//
// object containing all the properties the Owner configured
// this should be the content of the NFT's stored configuration file (if set by the owner and NFT is configurable)
// the Sandbox will override the default props with the values in here, making the NFT
// configurable by its owner
owner_properties: {
duration: 3000,
name: 'Jane Doe',
fruits: ['kiwi', 'strawberries'],
fullname: {
name: "Doe",
surname: "Jane"
}
}
}
});
The Sandbox internal processus is as follow :
- Sandbox checks if data is set
- if it's a string, it will try to fetch it and JSON.parse
- else it should be a JavaScript Object, content of the NFT JSON
- Sandbox checks if
owner_properties
is set- if it's a string, it will try to fetch it and JSON.parse the result
- else it should be a JavaScript Object, content of the configuration JSON
- Sandbox checks if code is provided in
data.interactive_nft.code
(not recommended - only use in dev mode)- If not Sandbox will try to fetch code from
data.interactive_nft.code_uri
property - else it won't run and trhow an error
- If not Sandbox will try to fetch code from
- Sandbox will load
data.interactive_nft.dependencies
in respective script and style tags - Sandbox will assign
owner_properties
todata.interactive_nft.properties
- the result will be assigned to
window.context.properties
making it accessible to the javascript.
- the result will be assigned to
- Sandbox will set
window.context.owner
to the provided owner property (or default to address(0)) -
- Sandbox will assign to
window.context.nft_json
the value of the current json file
- Sandbox will assign to
- Sandbox will execute the code
This way in your code's JavaScript you can access data as follow:
window.context.nft_json
contains all the NFT data (name, description, attributes, image, ...)window.context.owner
contains the NFT Owner addresswindow.context.properties
contains the configurable properties
The Sandbox dispatch events to let you know what happens inside.
You can listen to those events using sandboxInstance.$on(eventName, fn)
For the moment, events are:
loaded
: when the Sandbox loaded all dependencies, created the configuration object and added the code to the iframe (and it didn't throw an error)
error
: When any unhandledrejection
happens in the iframe. The iframe content will blur and stop any interaction if that happens (@TODO: reflect whether to make this configurable directly when instantiating the sandbox?!)
warning
: is emitted when something odd happens, but is not blocking the rendering (for example if the owner_properties is a string that doesn't resolve to a valid JSON file when fetched)
Executing "user submitted" code is always a bit tricky.
Using a Sandboxed iframe should already help to stop a lot of problem that can happen, but still, users should be warned to not do anything that seems suspicious when the NFT runs.
Therefore, if you use this Sandbox on your website, before the first rendering it would be nice to tell users that they will be shown an Interactive NFT and that they must be carefull because you do not have control over the code itself.
@TODO: Should we set that directly in the Sandbox? @TODO: Research if there is a way to create an AST of all the javascript code in the NFT code, and add things to it, to prevent for example infitinie loop or other things
Feel free to help develop or correct bug as this is highly POC for the moment.
The Sandbox is created using Svelte (because... what else?).
npm run build
to build the Sandbox code
npm run dev
to edit with a watch (autoreload and so on).
npm run srcdoc
need to be run after modifying the iframe html in dev mode (automatically done before a build)
npm run start
will serve the files under public, which let you play a bit with the Sandbox if you edit ./public/index.html
The Sandbox will dispatch event in case of an error occuring in the iframe. It will also show a message to the Viewer and blur the Sandbox.
TODO: Make this configurable with a prop showerror
?
Here are the Configurable contracts for ERC721 and ERC1155 tokens
ERC721
// SPDX-License-Identifier: MIT
pragma solidity ^0.6.0;
import "@openzeppelin/contracts/token/ERC721/ERC721.sol";
abstract contract ERC721Configurable is ERC721 {
// map of tokenId => interactiveConfURI.
mapping(uint256 => string) private _interactiveConfURIs;
function _setInteractiveConfURI(
uint256 tokenId,
string calldata _interactiveConfURI
) internal virtual {
require(
_exists(tokenId),
"ERC721Configurable: Configuration URI for unknown token"
);
_interactiveConfURIs[tokenId] = _interactiveConfURI;
}
/**
* Configuration uri for tokenId
*/
function interactiveConfURI(uint256 tokenId)
public
virtual
view
returns (string memory)
{
require(
_exists(tokenId),
"ERC721Configurable: Configuration URI query for unknown token"
);
return _interactiveConfURIs[tokenId];
}
}
ERC1155
// SPDX-License-Identifier: MIT
pragma solidity ^0.6.0;
import "@openzeppelin/contracts/token/ERC1155/ERC1155.sol";
abstract contract ERC1155Configurable is ERC1155 {
// map of tokenId => interactiveConfURI.
mapping(uint256 => mapping(address => string)) private _interactiveConfURIs;
function _setInteractiveConfURI(
uint256 _tokenId,
address _owner,
string calldata _interactiveConfURI
) internal virtual {
_interactiveConfURIs[_tokenId][_owner] = _interactiveConfURI;
}
/**
* Configuration uri for tokenId
*/
function interactiveConfURI(uint256 _tokenId, address _owner)
public
virtual
view
returns (string memory)
{
return _interactiveConfURIs[_tokenId][_owner];
}
}