Skip to main content
Version: 1.x

wash Plugin Developer Guide

tip

wash plugins are available starting with version 0.28.

This guide assumes some familiarity with building Wasm components. If you're not familiar with Wasm components and WIT, please read our existing developer documentation first.

Creating a plugin

The easiest way to get started creating a plugin is to use wash new component and select one of the plugin examples from the list. This will get you started with a fully functional example plugin that demonstrates how to use all of the available functionality.

Plugin API

The Plugin API is defined via WIT and is a small wrapper around the standard wasi:cli/run interface. The WIT is defined in the wash-lib crate and can be found here. To use this wit, just add wasmcloud:wash/subcommand@0.1.0; to your component's world.wit file.

A plugin can be any component that exports wasi:cli/run and the wasmCloud-defined interface called wasmcloud:wash/subcommand. This is a very small interface that expects a single function called register that returns metadata about the plugin. The definition for this data can be found in the WIT, but is defined below for convenience:

wit
/// Information about an argument
record argument {
    /// The description of the argument. Used for documentation in the CLI
    description: string,
    /// Whether or not the argument is a path. If the argument is a path, wash will load this
    // path (with access to only the file if it is a file path and access to the directory if
    /// it is a directory path) and pass it as a preopened dir at the exact same path
    is-path: bool,
    /// Whether or not the argument is required. 
    required: bool,
}

/// The metadata for a plugin used for registration and setup
record metadata {
    /// The friendly name of the plugin
    name: string,
    /// The ID of the plugin. This must be unique across all plugins and is used as the name of
    /// the subcommand added to wash. This ID should contain no whitespace
    id: string,
    /// The version of the plugin
    version: string,
    /// The author of the plugin
    author: string,
    /// The description of the plugin. This will be used as the top level help text for the plugin
    description: string,
    /// The list of flags and their documentation that can be used with this plugin. The key is
    /// the name of the flag.
    %flags: list<tuple<string, argument>>,
    /// The list of positional arguments that can be used with this plugin. The key is the name
    /// of the argument.
    arguments: list<tuple<string, argument>>,
}

Most of these fields are fairly straightforward, but the %flags and arguments fields are a bit more complicated. These fields are used to define the flags and positional arguments that can be used with the plugin. These fields are defined as a list of tuples (as WIT doesn't have support for a map type yet) where the first element of the tuple is the name of the flag or argument and the second element is additional configuration for an argument. Every argument should have a description and can also indicate to wash whether the argument is required and if it is a path (which matters for accessing files. See run function section for more information). For arguments, the order matters as they are positional arguments. This data is used to hook in to the CLI parsing library, meaning that when the plugin is run, the CLI will automatically parse the flags and arguments to make sure they are valid. There is not currently any way to inform the CLI parsing library which arguments are required and which are optional, so it only provides a basic sanity check (such as if an unknown flag is passed) and to hook in help text to wash.

Because a plugin just uses wasi:cli/run, any component that already implements that interface can be used by composing it with another component that exports the wasmcloud:wash/subcommand interface. In the future we plan on adding an example of doing this.

Availability of other resources during registration

When wash first loads a plugin to call the register function, it will be loaded with a "deny all" policy for all system resources. This means that the plugin can't access any of the resources that are normally available to a plugin. This is specifically for purposes of security as it ensures that a plugin is only returning the necessary information for registering the plugin. Once a plugin is loaded and is called directly, it is given access to system resources as described below.

run function implementation details

When a plugin is run, the wash will call the run function exported by your plugin. The plugin will be passed all arguments that were passed after wash, including the plugin name. So if you had a plugin called foo and the user ran wash foo --bar baz, the run function would be called with the arguments ["foo", "--bar", "baz"]. The run function is responsible for parsing the arguments in whatever way you see fit. The run function doesn't support returning a specific return code, but wash will ensure that the return code is 0 if the run function returns normally and a non-zero return code if the run function returns with an error. A plugin is also allowed to use wasi:cli/exit to exit early as well, but this may cause some wash cleanup to not run.

A plugin has access to the following resources while it is running:

  • Stdin
  • Stdout
  • Stderr
  • A single directory with full read/write privileges at $WASH_DIR/plugins/scratch/<plugin_id>
  • Write access to any directories passed as an argument (if the argument is marked as a path)
  • Read access to any directory containing a file path passed as an argument as well as full read/write access to the files in that directory (if the argument is marked as a path)
  • The ability to do outbound HTTP requests to any address
  • Environment variable passthrough for all environment variables that start with WASH_PLUGIN_<plugin_id_uppercase>_ (e.g. WASH_PLUGIN_FOO_ for a plugin with the id of "foo")

Just like a normal CLI, the plugin can read the incoming stdin and should write any messages/logs/etc to stdout/stderr.

Accessing files in the plugin

When defining your arguments returned from the register function for the plugin, you are able to indicate that any given argument should be treated as a path. If a user passes that argument to the plugin, it will be configured to allow access for the plugin using a Wasi preopen directory. Please note that the directory (whether it is the directory containing a file or a directory itself) must exist before the plugin can actually load it.

The default plugin scratch directory is "mounted" into the plugin at /. Every other directory/file path goes through a canonicalization step before it is passed to the plugin. This canonicalization step ensures that the plugin is only given access to the directories/files that it is explicitly allowed to access. For example, if the following example plugin accepted two arguments, one a directory, and the other one a path, this is how things would be parsed:

shell
wash my-plugin ./bar ../other/baz.txt

For purposes of this example, we'll assume ./bar is at a fully qualified path of /foo/bar and ../other/baz.txt has a fully qualified path of /my-files/other/baz.txt. Before running your plugin, wash will normalize the paths to /foo/bar and /my-files/other/baz.txt respectively. It will then replace the path arguments passed to the plugin with the canonicalized paths. So with the example above, the arguments array given to the plugin would be ["my-plugin", "/foo/bar", "/my-files/other/baz.txt"].

Each of the paths passed as arguments will be "mounted" at the exact same paths in the plugin. In the case of a file path, it will mount the file's parent directory. So for the example above, you will be able to access /foo/bar and /my-files/other/ from within the plugin. The example plugin template has code that shows how you can access each of the files.

Windows Paths

Because Wasm is platform agnostic, some languages (such as Rust) will not be able to parse file paths properly that do not use the standard unix file separator (i.e. /). On Windows systems, in addition to fully canonicalizing the paths, wash will also convert all file paths to use the standard unix file separator (/). For example, if you pass a path like .\my\files to the plugin, it will be normalized, mounted in the plugin, and passed as an argument in the form //?//C://fully/qualified/path/to/my/files. The ? prefix is an artifact of the canonicalization and shouldn't effect your loading of the path

Configuring your plugin

All plugins are expected to document their configuration options (whether it be a file, environment variables, or command line flags) in a README or other well-known location. As noted in previous sections, all command line options will be added to the autogenerated help text for wash.

Outside of command line flags, there are generally two ways to configure a CLI: environment variables or configuration files (or both). Every plugin is given a list of environment variables that are prefixed with WASH_PLUGIN_<plugin_id_uppercase>_. So if you have a plugin with the id of "foo", any variable with the prefix WASH_PLUGIN_FOO_ will be available for the plugin to consume. This means that any configuration via environment variables must be done via this naming scheme.

For configuration files, every plugin is given a directory at $WASH_DIR/plugins/scratch/<plugin_id>. A plugin can use this directory to store any configuration files it needs (in addition to any specific application data it needs to persist). The configuration file format and parsing is left to the maintainer of a plugin, but it should also be documented. So if you had a config.toml file for your plugin, you should let users know that if they need to create or edit that file, it should be at $WASH_DIR/plugins/scratch/<plugin_id>/config.toml.

Future development

Currently we only support subcommand-style plugins for wash. In the future, we plan to add support for auth plugins, registry plugins, and resource plugins (plugins used to access custom enterprise resources that hook into wasmCloud such as ingress points, queues, DBs, etc.). Each of these will likely have a separate interface that is part of the wasmcloud:wash WIT package.

The subcommands world will also have support for the CTL interface added to it in the near future. This will allow plugins access to interact with wasmCloud lattices.