A Model Context Protocol (MCP) Server that exposes Wasm Components as Tools.
This is currently an early-stage non-production prototype.
(auth and observability are on the roadmap)
Prerequisite: a current rust toolchain
Clone the toolbelt project if you have not already.
Then from within the toolbelt
directory:
cargo install --path .
That will build the binary with the release
profile and add
it to your cargo bin directory which should be on your PATH.
Provide the path to one or more .wasm
files as command line arguments:
toolbelt hello.wasm calculator.wasm
Or you can specify OCI URIs for published Wasm Components, such as these:
toolbelt oci://ghcr.io/modulewise/demo/hello:0.1.0 \
oci://ghcr.io/modulewise/demo/calculator:0.1.0
Tip
If you'd like to build the Wasm Components locally, clone the modulewise/demos project and follow the build instructions in components/README.md
By default, components operate in a least-privilege capability mode.
If your component requires features from the host runtime, you can
specify those features in a .toml
file. The enables
property
indicates the scope within which they will be available:
[wasip2]
uri = "wasmtime:wasip2"
enables = "any"
[http]
uri = "wasmtime:http"
enables = "any"
And then define the "exposed" tool component that "expects" those features:
[flights]
uri = "file:///path/to/flight-search.wasm"
expects = ["wasip2", "http"]
exposed = true
Pass the definition file to the server instead of direct .wasm
files:
toolbelt flights.toml
Wasm Components can also be defined to enable other components, and they may have their own dependencies. Notice this time the runtime features are not directly available to exposed "tool" components, but only to the internal components that are enabling exposed components:
runtime-features.toml
[wasip2]
uri = "wasmtime:wasip2"
enables = "unexposed"
[inherit-network]
uri = "wasmtime:inherit-network"
enables = "unexposed"
Those runtime features are then expected by an enabling component:
keyvalue.toml
[keyvalue]
uri = "../demos/components/lib/valkey-client.wasm"
expects = ["wasip2", "inherit-network"]
enables = "exposed"
Finally, that component can be composed into exposed "tool" components. In this case, the exposed component will also be composed with config:
incrementor.toml
[incrementor]
uri = "../demos/components/lib/incrementor.wasm"
expects = ["keyvalue"]
exposed = true
[incrementor.config]
bucket = "increments"
Now these files can all be passed to the server:
toolbelt runtime-features.toml keyvalue.toml incrementor.toml
This allows for various combinations of runtime features, enabling components, and components that will be exposed as tools. It also promotes responsibility-driven separation of concerns between supporting infrastructure and exposed functionality.
-
Run the server as described above.
-
Start the MCP Inspector.
-
Ensure the
Streamable HTTP
Transport Type is selected. -
Ensure the specified URL is
http://127.0.0.1:3001/mcp
(replace host or port if not using defaults). -
Click
Connect
and thenList Tools
. -
Select a Tool, provide parameter values, and click
Run Tool
.
Copyright (c) 2025 Modulewise Inc and the Modulewise Toolbelt contributors.
Apache License v2.0: see LICENSE for details.