@metamask/snaps-cli
TypeScript icon, indicating that this package has built-in type declarations

7.0.0 • Public • Published

@metamask/snaps-cli

A CLI for developing MetaMask Snaps.

Installation

Use Node.js 18.16.0 or later. We recommend nvm for managing Node.js versions.

Install a dev dependency in your snap project using yarn or npm:

  • npm install --save-dev @metamask/snaps-cli
  • yarn add -D @metamask/snaps-cli

Usage

Usage: mm-snap <command> [options]

Commands:
  mm-snap build     Build snap from source                          [aliases: b]
  mm-snap eval      Attempt to evaluate snap bundle in SES          [aliases: e]
  mm-snap manifest  Validate the snap.manifest.json file            [aliases: m]
  mm-snap serve     Locally serve Snap file(s) for testing          [aliases: s]
  mm-snap watch     Build Snap on change                            [aliases: w]

Options:
      --version  Show version number                                   [boolean]
  -c, --config   Path to config file                                    [string]
  -h, --help     Show help                                             [boolean]

Examples:
  mm-snap build                             Build './src/index.js' as './dist/bu
                                            ndle.js'
  mm-snap build --config ./snap.config.bui  Build './src/index.js' as './dist/bu
  ld.ts                                     ndle.js' using the config in './snap
                                            .config.build.ts'
  mm-snap manifest --fix                    Check the snap manifest, and fix any
                                             errors
  mm-snap watch --port 8000                 The snap input file for changes, and
                                             serve it on port 8000
  mm-snap serve --port 8000                 Serve the snap bundle on port 8000

MetaMask Snaps

MetaMask Snaps enables trustlessly extending the functionality of MetaMask at runtime. A snap consist of two things: a JSON manifest and a JavaScript bundle. At present, snaps can be published as npm packages on the public npm registry, or hosted locally during development. In the future, it will be possible to publish snaps on many different platforms, including arbitrary npm registries and IPFS.

We recommend building your snap using this tool. You can bundle your snap using your own tools, but it must run in SES and only use the global APIs that MetaMask exposes at runtime. Although snaps currently execute in the browser, some browser APIs are not available to snaps, and snaps do not have DOM access.

The Snap Manifest

Your manifest must be named snap.manifest.json and located in the root directory of your npm package. Here's an example manifest:

{
  "version": "0.7.0",
  "proposedName": "Dialog Example",
  "description": "A MetaMask Snap template.",
  "repository": {
    "type": "git",
    "url": "https://github.com/MetaMask/template-snap.git"
  },
  "source": {
    "shasum": "w3FltkDjKQZiPwM+AThnmypt0OFF7hj4ycg/kxxv+nU=",
    "location": {
      "npm": {
        "filePath": "dist/bundle.js",
        "iconPath": "images/icon.svg",
        "packageName": "@metamask/template-snap",
        "registry": "https://registry.npmjs.org/"
      }
    }
  },
  "initialPermissions": {
    "snap_dialog": {},
    "endowment:rpc": {
      "dapps": true,
      "snaps": false
    }
  },
  "manifestVersion": "0.1"
}

Refer to the Snaps publishing specification for more information about the manifest.

Note: If your snap is not compatible with the publishing specification, your snap may not work properly or install at all.

Assumed Project Structure

This tool has default arguments assuming the following project structure:

snap-project/
├─ package.json
├─ src/
│  ├─ index.js
├─ snap.manifest.json
├─ dist/
│  ├─ bundle.js
├─ ... (all other project files and folders)

Source files other than index.js are located through its imports. The defaults can be overwritten using the snap.config.js or (snap.config.ts) config file.

Configuration

The MetaMask Snaps CLI supports a configuration file. It can be either:

  • snap.config.js - a CommonJS module.
  • snap.config.ts - a TypeScript module.

The file should be placed in the project root directory.

Example

JavaScript

module.exports = {
  input: 'src/index.js',
  output: {
    path: 'dist',
  },
  server: {
    port: 9000,
  },
};

TypeScript

The CLI has full support for TypeScript out of the box. @metamask/snaps-cli exports a SnapConfig type that can be used to type the configuration object.

import type { SnapConfig } from '@metamask/snaps-cli';

const config: SnapConfig = {
  input: 'src/index.js',
  output: {
    path: 'dist',
  },
  server: {
    port: 9000,
  },
};

export default config;

The configuration file should not be published.

Options

input

  • Type: string
  • Default: "src/index.js"

The entry point of your snap. This is the file that will be bundled.

output

  • Type: object

The output configuration.

output.path
  • Type: string
  • Default: "dist"

The output directory.

output.filename
  • Type: string
  • Default: "bundle.js"

The output filename.

output.clean
  • Type: boolean
  • Default: false

Whether to clean the output directory before building.

output.minimize
  • Type: boolean
  • Default: true

Whether to minimize the bundle. This will remove comments and whitespace from the bundle, mangle variable names, and perform other optimizations.

sourceMap

  • Type: boolean | "inline"
  • Default: true

Whether to generate a source map. If "inline", the source map will be inlined in the bundle. Otherwise, it will be written to a separate file.

evaluate

  • Type: boolean

Whether to evaluate the bundle. This will check the bundle for compatibility issues with the MetaMask Snaps runtime. If there are any issues, the CLI will exit with an error.

manifest

  • Type: object

The snap manifest configuration.

manifest.path
  • Type: string
  • Default: "snap.manifest.json"

The path to the snap manifest, i.e., snap.manifest.json.

manifest.update
  • Type: boolean
  • Default: true

Whether to update the manifest with the bundle shasum, and any other possible updates. If false, the manifest will not be updated, and an error will be thrown if the manifest is not up-to-date.

server

  • Type: object

The development server configuration. The development server is used to test your snap during development, using the watch and serve commands.

server.enabled
  • Type: boolean
  • Default: true

Whether to enable the development server. If false, the development server will not be started when running the watch command. This option has no effect on the serve command.

server.root
  • Type: string
  • Default: process.cwd()

The root directory of the development server. This is the directory that will be served by the development server.

server.port
  • Type: number
  • Default: 8081

The port to run the development server on. If set to 0, a random port will be used.

environment

  • Type: Record<string, unknown>

The environment configuration. This is used to set environment variables for the snap, which can be accessed using process.env.

In addition to the environment variables set by the user, the following environment variables are set by the CLI:

  • NODE_ENV - "production".
  • NODE_DEBUG - false.
  • DEBUG - false.

stats

  • Type: object

The stats configuration, which controls the log output of the CLI.

stats.verbose
  • Type: boolean
  • Default: false

Whether to enable verbose logging. If true, the CLI will log more information.

stats.builtIns
  • Type: false | object
  • Default: { ignore: [] }

Whether to check for missing built-in modules. The MetaMask Snaps CLI does not support Node.js built-ins out of the box, and any used built-ins must be provided through the customizeWebpackConfig option. When enabled, the CLI shows a warning if a built-in module is used, but not provided.

stats.builtIns.ignore
  • Type: string[]

A list of built-in modules to ignore. This is useful if the built-in module is not actually used in the snap, but is added by a dependency.

stats.buffer
  • Type: boolean
  • Default: true

Whether to show a warning if Buffer is used, but not provided. The Buffer global is not available in the MetaMask Snaps runtime by default, and must be provided through the customizeWebpackConfig option.

customizeWebpackConfig

  • Type: (config: webpack.Configuration) => webpack.Configuration

A function that can be used to customize the Webpack configuration. This is useful if you need to add a Webpack plugin, provide a polyfill, add a loader, and so on.

The function receives the Webpack configuration object, and should return the modified configuration object. For convenience, @metamask/snaps-cli exports a merge function (re-exported from webpack-merge) that can be used to merge the configuration object with the default configuration.

The default configuration can be found in src/webpack/config.ts.

import type { SnapConfig } from '@metamask/snaps-cli';
import { merge } from '@metamask/snaps-cli';

const config: SnapConfig = {
  customizeWebpackConfig: (config) =>
    merge(config, {
      plugins: [
        // Add a plugin.
      ],
      module: {
        rules: [
          // Add a loader.
        ],
      },
    }),
};

export default config;

experimental

  • Type: object

Experimental features. These features are not stable, and may change in the future.

experimental.wasm
  • Type: boolean

Whether to enable WebAssembly support. When this is enabled, WebAssembly files can be imported in the snap, for example:

import program from './program.wasm';

// Program is initialised synchronously.
// ...

Readme

Keywords

Package Sidebar

Install

npm i @metamask/snaps-cli

Weekly Downloads

1,388

Version

7.0.0

License

ISC

Unpacked Size

712 kB

Total Files

373

Last publish

Collaborators

  • danfinlay
  • kumavis
  • mcmire
  • rekmarks
  • metamaskbot
  • gudahtt
  • nicholasellul
  • sethkfman
  • naugtur