Stream: general

Topic: composing components


view this post on Zulip Peter Huene (Nov 17 2022 at 00:16):

For those of you that are experimenting with building wasm components, there's the wasm-tools compose command for stitching components together into a new component based off what's on your local file system. I've also put up https://wasmbuilder.app as a simple GUI for wiring up component instances explicitly using a graph representation (see the linked repo for instructions) . Hopefully they'll prove useful for someone

view this post on Zulip Peter Huene (Nov 17 2022 at 00:20):

The cool thing about the GUI is that itself is implemented using a wasm component (via wit-bindgen's js host bindings); :turtle: s all the way down

view this post on Zulip Christoph Brewing (Apr 06 2023 at 20:05):

@Peter Huene Do you expect any difficulties when generating Python bindings of a composed component?
I ask because when creating Python bindings for a simple (not composed) component, I do not face any problems. When I compose two components to one and do

python -m wasmtime.bindgen composed.wasm --out-dir python

I get an

[..]
 wasm trap: wasm `unreachable` instruction executed
[..]

I tried to break it down to a minimal example but still I am not completely sure if I have a mistake in the way I use wasm-tools compose (or sth. else)

view this post on Zulip Peter Huene (Apr 06 2023 at 20:29):

i'm not familiar with the wasmtime python host bindings, so i can't say where the problem might lie from just this information

view this post on Zulip Peter Huene (Apr 06 2023 at 20:32):

i'd test first on a single component that exports a single simple function, maybe returning an easily verifiable string value; then I would compose that component (as the import) with a component that imports an interface with the same simple function and exports the same function, implementing the export in terms of calling the import; to the host, it should return the same expected value

view this post on Zulip Peter Huene (Apr 06 2023 at 20:33):

the bindings should actually be the same in both cases since the world would just contain the same exported function and no imports

view this post on Zulip Peter Huene (Apr 06 2023 at 20:33):

so if it doesn't work, i'd expect a bug in wasmtime

view this post on Zulip Peter Huene (Apr 06 2023 at 20:35):

if that simple case doesn't reproduce, then it's something specific to your component and we'd need to figure out what that trap represents

view this post on Zulip Christoph Brewing (Apr 06 2023 at 20:36):

Agreed, I will do so.

view this post on Zulip Christoph Brewing (Apr 07 2023 at 21:56):

I tried to follow your description. The result I get is the same as in my 'real world' project. I assembled this minimal example in this respository. Can you recommend any next steps? Should I open an issue at wasmtime?
Before I do so: in case you or anyone recognizes any mistakes in the example, please do not hesitate to leave me a note.

Contribute to Finfalter/wasm-compositions development by creating an account on GitHub.

view this post on Zulip Peter Huene (Apr 07 2023 at 22:50):

The issue probably belongs on the wasmtime-py repo; I can take a look on Monday to see what's going on

view this post on Zulip Peter Huene (Apr 10 2023 at 18:39):

@Christoph Brewing so the trap is coming from this line, which indicates that wasmtime-py doesn't yet support handing a composed component that exchanges strings. @Alex Crichton will likely have more context than I regarding implementation status of wasmtime-py

Python WebAssembly runtime powered by Wasmtime. Contribute to bytecodealliance/wasmtime-py development by creating an account on GitHub.

view this post on Zulip Peter Huene (Apr 10 2023 at 18:40):

Probably the best bet is to file an issue on the wasmtime-py repo linking to your repro repo

view this post on Zulip Alex Crichton (Apr 10 2023 at 18:40):

oh wow I didn't expect that unimplemented! to get hit for a much longer time!

That shouldn't be the hardest thing in the world to implement but also not trivial. An issue would be much appreciated!

view this post on Zulip Alex Crichton (Apr 10 2023 at 18:41):

also if you're only getting unreachable and no actual error message printed out then that's also a bug I need to fix

view this post on Zulip Peter Huene (Apr 10 2023 at 18:42):

debug backtrace:

wasmtime._trap.Trap: error while executing at wasm backtrace:
    0: 0x713c7d - panic_abort::__rust_start_panic::abort::hce48e00841fe5fb9
                    at /rustc/8460ca823e8367a30dda430efda790588b8c84d3/library/panic_abort/src/lib.rs:83:17              - __rust_start_panic
                    at /rustc/8460ca823e8367a30dda430efda790588b8c84d3/library/panic_abort/src/lib.rs:37:5
    1: 0x7139bb - rust_panic
                    at /rustc/8460ca823e8367a30dda430efda790588b8c84d3/library/std/src/panicking.rs:740:9
    2: 0x713982 - std::panicking::rust_panic_with_hook::h582baf536187e93d
                    at /rustc/8460ca823e8367a30dda430efda790588b8c84d3/library/std/src/panicking.rs:710:5
    3: 0x7129fa - std::panicking::begin_panic_handler::{{closure}}::h10886906d9416d04
                    at /rustc/8460ca823e8367a30dda430efda790588b8c84d3/library/std/src/panicking.rs:577:13
    4: 0x71295c - std::sys_common::backtrace::__rust_end_short_backtrace::hd12587e8e9c19331
                    at /rustc/8460ca823e8367a30dda430efda790588b8c84d3/library/std/src/sys_common/backtrace.rs:137:18
    5: 0x712ff0 - rust_begin_unwind
                    at /rustc/8460ca823e8367a30dda430efda790588b8c84d3/library/std/src/panicking.rs:575:5
    6: 0x71939c - core::panicking::panic_fmt::hb0157e5eb5d13797
                    at /rustc/8460ca823e8367a30dda430efda790588b8c84d3/library/core/src/panicking.rs:64:14
    7: 0x719905 - core::panicking::panic::hd3974465c50a40c4
                    at /rustc/8460ca823e8367a30dda430efda790588b8c84d3/library/core/src/panicking.rs:114:5
    8: 0xbfa99 - bindgen::bindgen::Instantiator::global_initializer::ha042365cb24e1e89
                    at /Users/peterhuene/src/bytecodealliance/wasmtime-py/rust/bindgen/src/bindgen.rs:452:49
    9: 0xbd94d - bindgen::bindgen::WasmtimePy::instantiate::h9843581d0d938b8c
                    at /Users/peterhuene/src/bytecodealliance/wasmtime-py/rust/bindgen/src/bindgen.rs:232:13
   10: 0xbbdc2 - bindgen::bindgen::WasmtimePy::generate::h18c37b18ad175b4f
                    at /Users/peterhuene/src/bytecodealliance/wasmtime-py/rust/bindgen/src/bindgen.rs:150:9
   11: 0x5bcc5 - <bindgen::bindings::PythonBindings as bindgen::bindings::WasmtimePy>::generate::h741c41ef729d8851
                    at /Users/peterhuene/src/bytecodealliance/wasmtime-py/rust/bindgen/src/lib.rs:48:13
   12: 0x6673b - bindgen::bindings::call_generate::h1e265052a1f06de3
                    at /Users/peterhuene/src/bytecodealliance/wasmtime-py/rust/bindgen/src/lib.rs:38:5
   13: 0x5bf94 - generate
                    at /Users/peterhuene/src/bytecodealliance/wasmtime-py/rust/bindgen/src/lib.rs:38:5
   14: 0x7261ec - <unknown>!generate.command_export

Caused by:
    wasm trap: wasm `unreachable` instruction executed

view this post on Zulip Alex Crichton (Apr 10 2023 at 18:43):

in theory the standard Rust unimplemented!() panic message should be printed somewhere, but I may have forgotten to wire that up

view this post on Zulip Peter Huene (Apr 10 2023 at 18:43):

it does: thread '<unnamed>' panicked at 'not implemented', bindgen/src/bindgen.rs:452:49

view this post on Zulip Alex Crichton (Apr 10 2023 at 18:48):

ah ok phew just making sure!

view this post on Zulip Alex Crichton (Apr 10 2023 at 18:48):

not that that's an "official" error mesage of course, just wanted to make sure that was printed

view this post on Zulip Christoph Brewing (Apr 11 2023 at 13:18):

:heart: Thank you very much for your investigations! I created issue no. 143.

As discussed in Zulip > general > composing components, wasmtime-py doesn't yet support handling a composed component that exchanges strings. Since I really would like to use this feature in one of...

view this post on Zulip Alex Crichton (Apr 11 2023 at 14:30):

Thanks! I'll write up some more notes on the issue of what to do

view this post on Zulip Christoph Brewing (Apr 14 2023 at 10:51):

I am about to compose an application comprising two components but the dependencies are not found.

The lower layer, called exporter, provides functionality in terms of multiple exported interfaces. The upper layer, called importer, is supposed to import lower layer's interfaces, do sth. with them and export another interface. I think of it as a facade like pattern. However, I do not arrive to compose the importer and exporter components:

[exporter component]

$ wasm-tools component wit exporter_component.wasm
interface ione {
  greet: func(s: string) -> string
}

interface itwo {
  greet: func(s: string) -> string
}

default world exporter {
  export first: self.ione
  export second: self.itwo
}

[importer component]

$ wasm-tools component wit importer_component.wasm
interface ione {
  greet: func(s: string) -> string
}

interface itwo {
  greet: func(s: string) -> string
}

default world importer {
  import first: self.ione
  import second: self.itwo
  export something_else
}

When I try to compose

$ wasm-tools compose -o composed.wasm importer_component.wasm -p .
instance `first` will be imported because a dependency named `first` could not be found
instance `second` will be imported because a dependency named `second` could not be found
Error: no dependencies of component `importer_component.wasm` were found

In the past, I only had a single interface to be imported. I got around the no dependencies .. where found error by naming the exporter according to the exported artefact, e.g. first.

What do I have to do that the command wasm-tools compose recognizes multiple exports?

view this post on Zulip Peter Huene (Apr 14 2023 at 17:58):

By default, wasm-compose looks for components with a filename matching the import name to satisfy an individual import; with this more complicated linking, you'll need to author a config file to feed the tool, which explicitly wires up the first and second exports from exporter_component to the instantiation arguments of importer_component. The config file syntax is documented here

Low level tooling for WebAssembly in Rust. Contribute to bytecodealliance/wasm-tools development by creating an account on GitHub.

view this post on Zulip Peter Huene (Apr 14 2023 at 17:59):

you can also use https://wasmbuilder.app to quickly wire these things up manually in a composition

view this post on Zulip Christoph Brewing (Apr 14 2023 at 20:03):

Excellent, that works! The following config file seems to do the trick:

dependencies:
  a: importer_component.wasm
  first:
    path: exporter_component.wasm
  second:
    path: exporter_component.wasm

Right, I have tried your super convenient service .. was just thinking about an integration pipeline and so I am pretty glad to have checked this path as well.

view this post on Zulip Peter Huene (Apr 14 2023 at 21:16):

I believe that config will create two different instances of the exporter component, though. there's a way to create a single instance of it and explicitly tell the composer to use an export from that instance as an instantiation argument for the root instance

view this post on Zulip Peter Huene (Apr 14 2023 at 21:18):

Something like:

dependencies:
  exporter:
    path: exporter_component.wasm

instantiations:
  $input:
    arguments:
      first:
        instance: exporter
        export: first
      second:
        instance: exporter
        export: second

view this post on Zulip Peter Huene (Apr 14 2023 at 21:19):

i believe that will create a single instance of the exporter component, if that's desired


Last updated: Jan 24 2025 at 00:11 UTC