Extism runtime and rust-sdk
This repo contains the code for the Extism runtime and rust-sdk. It can be embedded in any Rust application to call Extism plug-ins.
Note: If you're unsure what Extism is or what an SDK is see our homepage: https://extism.org.
Installation
Cargo
To use the extism crate, you can add it to your Cargo file:
[]
= "1.4.1"
Environment variables
There are a few environment variables that can be used for debugging purposes:
EXTISM_ENABLE_WASI_OUTPUT=1: show WASI stdout/stderrEXTISM_MEMDUMP=extism.mem: dump Extism linear memory to a fileEXTISM_COREDUMP=extism.core: write coredump to a file when a WebAssembly function trapsEXTISM_DEBUG=1: generate debug informationEXTISM_PROFILE=perf|jitdump|vtune: enable Wasmtime profilingEXTISM_CACHE_CONFIG=path/to/config.toml: enable Wasmtime cache, details here
Note: The debug and coredump info will only be written if the plug-in has an error.
Getting Started
This guide should walk you through some of the concepts in Extism and the extism crate.
Creating A Plug-in
The primary concept in Extism is the plug-in. You can think of a plug-in as a code module stored in a .wasm file.
Since you may not have an Extism plug-in on hand to test, let's load a demo plug-in from the web:
use *;
Note: See the Manifest docs as it has a rich schema and a lot of options.
Calling A Plug-in's Exports
This plug-in was written in Rust and it does one thing, it counts vowels in a string. As such, it exposes one "export" function: count_vowels. We can call exports using Extism::Plugin::call:
let res = plugin..unwrap;
println!;
# =>
All exports have a simple interface of bytes-in and bytes-out. This plug-in happens to take a string and return a JSON encoded string with a report of results.
The call function uses extism-convert to determine which input/output types can be used. If we wanted to use a concrete type for
the count_vowels result, we could defined a struct:
Then we can use Json to get the JSON results decoded into VowelCount:
let Json = plugin..unwrap;
println!;
# => VowelCount
Plug-in State
Plug-ins may be stateful or stateless. Plug-ins can maintain state between calls by the use of variables. Our count vowels plug-in remembers the total number of vowels it's ever counted in the "total" key in the result. You can see this by making subsequent calls to the export:
let res = plugin..unwrap;
println!;
# =>
let res = plugin..unwrap;
println!;
# =>
These variables will persist until this plug-in is freed or you initialize a new one.
Configuration
Plug-ins may optionally take a configuration object. This is a static way to configure the plug-in. Our count-vowels plugin takes an optional configuration to change out which characters are considered vowels. Example:
let manifest = new;
let mut plugin = new;
let res = plugin..unwrap;
println!;
# =>
let manifest = new.with_config_key;
let mut plugin = new.unwrap;
let res = plugin..unwrap;
println!;
# =>
Host Functions
Let's extend our count-vowels example a little bit: Instead of storing the total in an ephemeral plug-in var, let's store it in a persistent key-value store!
Wasm can't use our KV store on it's own. This is where Host Functions come in.
Host functions allow us to grant new capabilities to our plug-ins from our application. They are simply some Rust functions you write which can be passed down and invoked from any language inside the plug-in.
Let's load the manifest like usual but load up this count_vowels_kvstore plug-in:
let url = url;
let manifest = new;
Note: The source code for this is here and is written in rust, but it could be written in any of our PDK languages.
Unlike our previous plug-in, this plug-in expects you to provide host functions that satisfy our its import interface for a KV store.
We want to expose two functions to our plugin, kv_write(key: String, value: Bytes) which writes a bytes value to a key and kv_read(key: String) -> Bytes which reads the bytes at the given key.
use *;
// pretend this is redis or something :)
type KVStore = BTreeMap;
// When a first argument separated with a semicolon is provided to `host_fn` it is used as the
// variable name and type for the `UserData` parameter
host_fn!;
host_fn!;
Note: In order to write host functions you should get familiar with the methods on the CurrentPlugin and UserData types.
Now we can invoke the event:
let res = plugin..unwrap;
println!;
# => Read from key=count-vowels"
# => Writing value=3 from key=count-vowels"
# =>
let res = plugin..unwrap;
println!;
# => Read from key=count-vowels"
# => Writing value=6 from key=count-vowels"
# =>
Logging
Plug-ins can't directly print anything to the console. They can however use Extism's built-in logging functionality, for example the log! macro in the rust-pdk or the logInfo function in the haskell-pdk.
Inside your host application, the rust-sdk emits these as tracing events. The simplest way to make the logged messages visible is by adding the tracing_subscriber dependency to your crate and then initializing a tracing subscriber at the top of your main function:
init;
Wasmtime Caching
To enable or disable caching for plugin compilation, you need to provide a configuration file that will be used by the wasmtime crate.
For more information and values that can be used for configuring caching, take a look at the docs.
Note: As of now extism uses wasmtime
version = ">= 27.0.0, < 31.0.0", but theenabledkey requirement was removed fromwasmtimeand its documentation, this could explain thefailed to parse config fileerror you might encounter without it.
An example configuration for caching would be:
[]
= true # This value is required
= "/some/path"
You can :
- Create a global
wasmtimeconfiguration file in$HOME/.config/wasmtime/config.toml. - Set the
EXTISM_CACHE_CONFIGenvironment variable - Set the configuration file path using
PluginBuilder
Using a configuration file
The wasmtime crate, by default, will look for a configuration file in your systems' default configuration directory (for example on UNIX systems: $HOME/.config/wasmtime/config.toml),
for more information on this behaviour.
Using an environment variable
You can set the EXTISM_CACHE_CONFIG=path/to/config.toml environment variable to set the path of the configuration file used by wasmtime.
Setting the variable to an empty string will disable caching (it won't load any configuration file).
Note: If the environment variable is not set,
wasmtimewill still try to read from a configuration file that may exist in your system's default configuration folder (e.g.$HOME/.config/wasmtime/config.toml).
The environment variable does not override the path you might have set using PluginBuilder. will only be checked for if you did not specify a cache configuration path in PluginBuilder.
Using PluginBuilder
If you use a PluginBuilder, you can set the wasmtime configuration path using the with_cache_config method.
This will override the EXTISM_CACHE_CONFIG environment variable if it's set, so you could have a "global" and per plugin configuration if needed.