Version: 23.0.0

This documentation is for Taquito version 23.0.0 and is no longer being updated. Some information may be outdated. View the latest version .

TZIP-16 Contract Metadata and Views

Written by Roxane Letourneau

The @taquito/tzip16 package allows retrieving metadata associated with a smart contract. These metadata can be stored on-chain (tezos-storage) or off-chain (HTTP(S) or IPFS). The package also provides a way to execute the MichelsonStorageView found in the metadata. More information about the TZIP-16 standard can be found here.

How to use the tzip16 package

The package can be used as an extension to the well known Taquito contract abstraction.

  1. We first need to create an instance of Tzip16Module and add it as an extension to our TezosToolkit

The constructor of the Tzip16Module takes an optional MetadataProvider as a parameter. When none is passed, the default MetadataProvider of Taquito is instantiated and the default handlers (HttpHandler, IpfsHandler, and TezosStorageHandler) are used.

import { TezosToolkit } from '@taquito/taquito';
import { Tzip16Module, tzip16 } from '@taquito/tzip16';

const Tezos = new TezosToolkit('rpcUrl');
Tezos.addExtension(new Tzip16Module());

In some cases, we may want to use a customized metadata provider. The constructor of the Tzip16Module class takes an optional metadata provider as a parameter. This allows to inject a custom metadata provider with custom protocol handlers if desired. For example, if we want to use a different IPFS gateway than the default one, which is ipfs.io, or if we want to use a different HTTP handler to support authentication or custom headers. Here is an example:

import { Handler, IpfsHttpHandler, TezosStorageHandler, MetadataProvider, Tzip16Module, tzip16 } from '@taquito/tzip16';

const Tezos = new TezosToolkit('rpcUrl');

// The constructor of the `MetadataProvider` class takes a `Map<string, Handler>` as a parameter.
const customHandler = new Map<string, Handler>([
  ['ipfs', new IpfsHttpHandler('dweb.link')], // Constructor of IpfsHttpHandler takes an optional gateway
  ['http', 'customHttpHandler'], // Custom HTTP(S) handler
  ['https', 'customHttpHandler'],
  ['tezos-storage', new TezosStorageHandler()],
]);

const customMetadataProvider = new MetadataProvider(customHandler);
Tezos.addExtension(new Tzip16Module(customMetadataProvider));

A list of public gateways is accessible here.

  1. Use the tzip16 function to extend a contract abstraction
const contract = await Tezos.contract.at('contractAddress', tzip16);
  1. Call the methods of the Tzip16ContractAbstraction class

The namespace tzip16() need to be specified when calling a method of the Tzip16ContractAbstraction class:

const metadata = await contract.tzip16().getMetadata();
const views = await contract.tzip16().metadataViews();

All other methods of the ContractAbstraction class can be called as usual on the contract object.

Get the metadata

The getMetadata method returns an object which contains the URI, the metadata in JSON format, an optional SHA256 hash of the metadata and an optional integrity check result.

A sequence diagram can be found here.

Tezos-storage example

// import { TezosToolkit } from '@taquito/taquito';
// import { Tzip16Module, tzip16 } from "@taquito/tzip16";
// const Tezos = new TezosToolkit('rpc_url');

Tezos.addExtension(new Tzip16Module());

const contractAddress = 'KT1V9mi4SiN85aUKjkJGRDDxELSbMSMdBMcy';

try {
  const contract = await Tezos.contract.at(contractAddress, tzip16);
  console.log(`Fetching the metadata for ${contractAddress}...`);
  const metadata = await contract.tzip16().getMetadata();
  console.log(JSON.stringify(metadata, null, 2));
} catch (error) {
  console.log(`Error: ${JSON.stringify(error, null, 2)}`);
}
// import { TezosToolkit } from '@taquito/taquito';
// import { Tzip16Module, tzip16 } from "@taquito/tzip16";
// const Tezos = new TezosToolkit('rpc_url');

Tezos.addExtension(new Tzip16Module());

const contractAddress = 'KT1V9mi4SiN85aUKjkJGRDDxELSbMSMdBMcy';

try {
  const wallet = await Tezos.wallet.at(contractAddress, tzip16);
  console.log(`Fetching the metadata for ${contractAddress}...`);
  const metadata = await wallet.tzip16().getMetadata();
  console.log(JSON.stringify(metadata, null, 2));
} catch (error) {
  console.log(`Error: ${JSON.stringify(error, null, 2)}`);
}

HTTPS example

// import { TezosToolkit } from '@taquito/taquito';
// import { Tzip16Module, tzip16 } from "@taquito/tzip16";
// const Tezos = new TezosToolkit('rpc_url');

Tezos.addExtension(new Tzip16Module());

const contractAddress = 'KT1MJ7wAZ9LBB797zhGJrXByaaUwvLGfe3qz';

try {
  const contract = await Tezos.contract.at(contractAddress, tzip16);
  console.log(`Fetching the metadata for ${contractAddress}...`);
  const metadata = await contract.tzip16().getMetadata();
  console.log(JSON.stringify(metadata, null, 2));
} catch (error) {
  console.log(`Error: ${JSON.stringify(error, null, 2)}`);
}
// import { TezosToolkit } from '@taquito/taquito';
// import { Tzip16Module, tzip16 } from "@taquito/tzip16";
// const Tezos = new TezosToolkit('rpc_url');

Tezos.addExtension(new Tzip16Module());

const contractAddress = 'KT1MJ7wAZ9LBB797zhGJrXByaaUwvLGfe3qz';

try {
  const wallet = await Tezos.wallet.at(contractAddress, tzip16);
  console.log(`Fetching the metadata for ${contractAddress}...`);
  const metadata = await wallet.tzip16().getMetadata();
  console.log(JSON.stringify(metadata, null, 2));
} catch (error) {
  console.log(`Error: ${JSON.stringify(error, null, 2)}`);
}

Example having a SHA256 hash:

// import { TezosToolkit } from '@taquito/taquito';
// import { Tzip16Module, tzip16 } from "@taquito/tzip16";
// const Tezos = new TezosToolkit('rpc_url');

Tezos.addExtension(new Tzip16Module());

const contractAddress = 'KT1JbEzvHn2Y2DjVQ7kgK8H8pxrspG893JsX';

try {
  const contract = await Tezos.contract.at(contractAddress, tzip16);
  console.log(`Fetching the metadata for ${contractAddress}...`);
  const metadata = await contract.tzip16().getMetadata();
  console.log(JSON.stringify(metadata, null, 2));
} catch (error) {
  console.log(`Error: ${JSON.stringify(error, null, 2)}`);
}
// import { TezosToolkit } from '@taquito/taquito';
// import { Tzip16Module, tzip16 } from "@taquito/tzip16";
// const Tezos = new TezosToolkit('rpc_url');

Tezos.addExtension(new Tzip16Module());

const contractAddress = 'KT1JbEzvHn2Y2DjVQ7kgK8H8pxrspG893JsX';

try {
  const wallet = await Tezos.wallet.at(contractAddress, tzip16);
  console.log(`Fetching the metadata for ${contractAddress}...`);
  const metadata = await wallet.tzip16().getMetadata();
  console.log(JSON.stringify(metadata, null, 2));
} catch (error) {
  console.log(`Error: ${JSON.stringify(error, null, 2)}`);
}

IPFS example

// import { TezosToolkit } from '@taquito/taquito';
// import { Tzip16Module, tzip16 } from "@taquito/tzip16";
// const Tezos = new TezosToolkit('rpc_url');

Tezos.addExtension(new Tzip16Module());

const contractAddress = 'KT1SDtsB4DHdh1QwFNgvsavxDwQJBdimgrcL';

try {
  const contract = await Tezos.contract.at(contractAddress, tzip16);
  console.log(`Fetching the metadata for ${contractAddress}...`);
  const metadata = await contract.tzip16().getMetadata();
  console.log(JSON.stringify(metadata, null, 2));
} catch (error) {
  console.log(`Error: ${JSON.stringify(error, null, 2)}`);
}
// import { TezosToolkit } from '@taquito/taquito';
// import { Tzip16Module, tzip16 } from "@taquito/tzip16";
// const Tezos = new TezosToolkit('rpc_url');

Tezos.addExtension(new Tzip16Module());

const contractAddress = 'KT1SDtsB4DHdh1QwFNgvsavxDwQJBdimgrcL';

try {
  const wallet = await Tezos.wallet.at(contractAddress, tzip16);
  console.log(`Fetching the metadata for ${contractAddress}...`);
  const metadata = await wallet.tzip16().getMetadata();
  console.log(JSON.stringify(metadata, null, 2));
} catch (error) {
  console.log(`Error: ${JSON.stringify(error, null, 2)}`);
}

Execute off-chain views

A sequence diagram can be found here.

In the next example, we will run a view named someJson that can be found in the metadata of the contract KT1Vms3NQK8rCQJ6JkimLFtAC9NhpAq9vLqE. When we inspect those metadata, we can see that this view takes no parameter, has a returnType of bytes and has the following code:

"code":
[
  {
    "prim": "DROP",
    "args": [],
    "annots": []
  },
  {
    "prim": "PUSH",
    "args": [
      {
        "prim": "bytes",
        "args": [],
        "annots": []
      },
      {
        "bytes": "7b2268656c6c6f223a22776f726c64222c226d6f7265223a7b226c6f72656d223a34322c22697073756d223a5b22222c226f6e65222c2232225d7d7d"
      }
    ],
    "annots": []
  }
]

Try to run the view:

// import { TezosToolkit } from '@taquito/taquito';
// import { Tzip16Module, tzip16, bytesToString } from "@taquito/tzip16";
// const Tezos = new TezosToolkit('rpc_url');

Tezos.addExtension(new Tzip16Module());

const contractAddress = 'KT1XdXkU9piczYpTU8ToAAGJunzFiGCWRvVK';

try {
  const contract = await Tezos.contract.at(contractAddress, tzip16);
  console.log(`Initialising the views for ${contractAddress}...`);
  const views = await contract.tzip16().metadataViews();
  console.log(`The following view names were found in the metadata: ${Object.keys(views)}`);
  const result = await views.someJson().executeView();
  console.log(`Result of the view someJson: ${result}`);
  console.log(`Transform result to char: ${bytesToString(result)}`);
} catch (error) {
  console.log(`Error: ${JSON.stringify(error, null, 2)}`);
}
// import { TezosToolkit } from '@taquito/taquito';
// import { Tzip16Module, tzip16, bytesToString } from "@taquito/tzip16";
// const Tezos = new TezosToolkit('rpc_url');

Tezos.addExtension(new Tzip16Module());

const contractAddress = 'KT1XdXkU9piczYpTU8ToAAGJunzFiGCWRvVK';

try {
  const wallet = await Tezos.wallet.at(contractAddress, tzip16);
  console.log(`Initialising the views for ${contractAddress}...`);
  const views = await wallet.tzip16().metadataViews();
  console.log(`The following view names were found in the metadata: ${Object.keys(views)}`);
  const result = await views.someJson().executeView();
  console.log(`Result of the view someJson: ${result}`);
  console.log(`Transform result to char: ${bytesToString(result)}`);
} catch (error) {
  console.log(`Error: ${JSON.stringify(error, null, 2)}`);
}

In the next example, we will run a view named multiply-the-nat-in-storage that can be found in the metadata of the contract KT19rDkTYg1355Wp1XM5Q23CxuLgRnA3SiGq. When we inspect those metadata, we can see that this view takes a nat has a parameter, has a returnType of nat and has the following instructions: DUP, CDR, CAR, SWAP, CAR, MUL.

Try to run the view:

// import { TezosToolkit } from '@taquito/taquito';
// import { Tzip16Module, tzip16 } from "@taquito/tzip16";
// const Tezos = new TezosToolkit('rpc_url');

Tezos.addExtension(new Tzip16Module());

const contractAddress = 'KT1CVoo3PxuvH3BuBpNTYRDafAQ7aRTfj8bd';

try {
  const contract = await Tezos.contract.at(contractAddress, tzip16);
  const storage = await contract.storage();
  console.log(`The nat in the storage of the contract is: ${storage[0]}`);
  console.log(`Initialising the views for ${contractAddress}...`);
  const views = await contract.tzip16().metadataViews();
  console.log(`The following view names were found in the metadata: ${Object.keys(views)}`);
  const result = await views['multiply-the-nat-in-storage']().executeView(10);
  console.log(`Result of the view 'multiply-the-nat-in-storage': ${result}`);
} catch (error) {
  console.log(`Error: ${JSON.stringify(error, null, 2)}`);
}
// import { TezosToolkit } from '@taquito/taquito';
// import { Tzip16Module, tzip16 } from "@taquito/tzip16";
// const Tezos = new TezosToolkit('rpc_url');

Tezos.addExtension(new Tzip16Module());

const contractAddress = 'KT1CVoo3PxuvH3BuBpNTYRDafAQ7aRTfj8bd';

try {
  const wallet = await Tezos.wallet.at(contractAddress, tzip16);
  const storage = await wallet.storage();
  console.log(`The nat in the storage of the contract is: ${storage[0]}`);
  console.log(`Initialising the views for ${contractAddress}...`);
  const views = await wallet.tzip16().metadataViews();
  console.log(`The following view names were found in the metadata: ${Object.keys(views)}`);
  const result = await views['multiply-the-nat-in-storage']().executeView(10);
  console.log(`Result of the view 'multiply-the-nat-in-storage': ${result}`);
} catch (error) {
  console.log(`Error: ${JSON.stringify(error, null, 2)}`);
}

Execute a custom view

In the next example we execute the view multiply-the-nat-in-storage in a custom way:

// import { TezosToolkit, RpcReadAdapter } from '@taquito/taquito';
// import { MichelsonStorageView } from "@taquito/tzip16";
// const Tezos = new TezosToolkit('rpc_url');

const contractAddress = 'KT1CVoo3PxuvH3BuBpNTYRDafAQ7aRTfj8bd';

try {
  const contract = await Tezos.contract.at(contractAddress);
  const view = new MichelsonStorageView(
    'test', // view name
    contract, // contract abstraction
    Tezos.rpc, // rpc
    new RpcReadAdapter(Tezos.rpc), // readProvider
    { prim: 'nat' }, // returnType
    [
      { prim: 'DUP' },
      { prim: 'CDR' },
      { prim: 'CAR' },
      { prim: 'SWAP' },
      { prim: 'CAR' },
      { prim: 'MUL' },
    ], // code of the view
    { prim: 'nat' } // parameter type
  );

  const result = await view.executeView(2);
  console.log(`Result of the custom view: ${result}`);
} catch (error) {
  console.log(error);
}
// import { TezosToolkit, RpcReadAdapter } from '@taquito/taquito';
// import { MichelsonStorageView } from "@taquito/tzip16";
// const Tezos = new TezosToolkit('rpc_url');

const contractAddress = 'KT1CVoo3PxuvH3BuBpNTYRDafAQ7aRTfj8bd';

try {
  const wallet = await Tezos.wallet.at(contractAddress);
  const view = new MichelsonStorageView(
    'test', // view name
    wallet, // contract abstraction
    Tezos.rpc, // rpc,
    new RpcReadAdapter(Tezos.rpc), // readProvider
    { prim: 'nat' }, // returnType
    [
      { prim: 'DUP' },
      { prim: 'CDR' },
      { prim: 'CAR' },
      { prim: 'SWAP' },
      { prim: 'CAR' },
      { prim: 'MUL' },
    ], // code of the view
    { prim: 'nat' } // parameter type
  );

  const result = await view.executeView(2);
  console.log(`Result of the custom view: ${result}`);
} catch (error) {
  console.log(`Error: ${JSON.stringify(error, null, 2)}`);
}