juntyr opened PR #8792 from juntyr:wasi-adapter-provider
to bytecodealliance:main
:
Fixes #8759
This PR adds a new
wasi-preview1-component-adapter-provider
crate that embeds the binary contents of the three wasi adapters. As per @alexcrichton's suggestion, this crate contains the adapter files as artefacts, which are checked in CI to match the freshly compiled versions of thewasi-preview1-component-adapter
crate. The new crate is intended to be published on crates.io and allow Rust code which programmatically builds WASM components to use the adapter without needing more than a normal crate dependency (instead of having to manually download the file).
juntyr requested alexcrichton for a review on PR #8792.
juntyr requested wasmtime-core-reviewers for a review on PR #8792.
juntyr requested wasmtime-default-reviewers for a review on PR #8792.
juntyr submitted PR review.
juntyr submitted PR review.
juntyr created PR review comment:
@alexcrichton I didn't find as much information on the proxy online and in code. Does it implement a pre-defined HTTP proxy world?
juntyr updated PR #8792.
@alexcrichton Well at least the CI check works :)
I tried rebuilding the adapters locally with 1.76.0 stable and the pinned wasm-tools version and got the local CI script to pass, but the GitHub CI check still failed. Is there something else I need to do to make the build reproducible and push a matching artefact?
alexcrichton submitted PR review:
Thanks for this! For the CI failure you say you're using Rust 1.76 but CI is using Rust 1.78, so that may be the culprit?
In addition to that there's a few more things here I think:
- By default the version of Rust used to compile the adapter will change whenever we bump the MSRV. This seems a little unfortunate and I think it'd be better to do this as a discrete step for the adapter to reduce the annoyance of updating our MSRV. Could the rust version be explicitly specified for the CI job that builds the adapter as exactly
1.78.0
? This might require splitting out the CI job so it's only build the adapter and doing nothing else.- Currently there's not an easy way to rebuild the adapter locally and then check in the results. Could the
cmp
commands in the build script be conditional somehow? For example they should only run on CI but locally you can skip them if you want to regenerate all the adapters.
alexcrichton submitted PR review:
Thanks for this! For the CI failure you say you're using Rust 1.76 but CI is using Rust 1.78, so that may be the culprit?
In addition to that there's a few more things here I think:
- By default the version of Rust used to compile the adapter will change whenever we bump the MSRV. This seems a little unfortunate and I think it'd be better to do this as a discrete step for the adapter to reduce the annoyance of updating our MSRV. Could the rust version be explicitly specified for the CI job that builds the adapter as exactly
1.78.0
? This might require splitting out the CI job so it's only build the adapter and doing nothing else.- Currently there's not an easy way to rebuild the adapter locally and then check in the results. Could the
cmp
commands in the build script be conditional somehow? For example they should only run on CI but locally you can skip them if you want to regenerate all the adapters.
alexcrichton created PR review comment:
It corresponds to this
world
, notablywasi:http/proxy
which doesn't have filesystem/env vars/cli arguments/etc
juntyr submitted PR review.
juntyr created PR review comment:
Thanks! I think what confused me is that the adapter defines the world inline here:
which includes most imports but not the incoming HTTP request handler import and not the outgoing HTTP request export.
juntyr updated PR #8792.
juntyr updated PR #8792.
Thanks for this! For the CI failure you say you're using Rust 1.76 but CI is using Rust 1.78, so that may be the culprit?
In addition to that there's a few more things here I think:
- By default the version of Rust used to compile the adapter will change whenever we bump the MSRV. This seems a little unfortunate and I think it'd be better to do this as a discrete step for the adapter to reduce the annoyance of updating our MSRV. Could the rust version be explicitly specified for the CI job that builds the adapter as exactly
1.78.0
? This might require splitting out the CI job so it's only build the adapter and doing nothing else.- Currently there's not an easy way to rebuild the adapter locally and then check in the results. Could the
cmp
commands in the build script be conditional somehow? For example they should only run on CI but locally you can skip them if you want to regenerate all the adapters.I'm a bit stumped. I check in CI and it now seems to build with 1.79 but even when I build locally with 1.79 it still produces different artefacts.
I absolutely agree we should explicitly lock down the adapter build settings to ensure that local build and CI builds remain the same unless someone explicitly changes that setting (including the Rust version). Once I have a successful build, I'll implement that.
What we could do perhaps is to offer two (mutually exclusive) options for the CI script:
- no options just build the adapters to the target folder, as before
--replace
also copies the newly built adapters into the provider crate--check
checks the newly built adapters against the provider crate (used in CI)I'll add that once CI works.
I restarted my dev container, cleared my target folder, followed all toolchain setup steps from the job output log, and reran the build script locally, which produced no change in the adapter binaries. So locally the build is reproducible, but something must still be different in CI.
alexcrichton commented on PR #8792:
Ah yesterday it was Rust 1.78 but we upgraded to 1.79 yesterday too, explaining that discrepancy. What os are you on? I've seen that macOS produces different binaries than Linux, for example, and that might be happening here. I can try to dig in today as well and see what's going on.
I was running on a Linux GitHub codespace container and tried both 1.78 and 1.79 today. I also used the same version of wasm tools as the one in CI and checked the Rust version hash from the CI job log. If you do have a bit of time to spare to investigate, that would be brilliant :)
alexcrichton updated PR #8792.
alexcrichton commented on PR #8792:
Hm interestingly I checked out this branch, ran the script, and it didn't fail. Locally I've got rustc 1.78 and wasm-tools 1.201, as opposed to CI which for the last build used rustc 1.79 and wasm-tools 1.0.27.
I added some debugging in the hopes of seeing what's happening in CI
alexcrichton updated PR #8792.
alexcrichton commented on PR #8792:
Ok now I think it's more clear what's happening:
Reference copy of adapter is not the same as the generated copy of the adapter reference copy: crates/wasi-preview1-component-adapter/provider/artefacts/wasi_snapshot_preview1.command.wasm built copy: target/wasm32-unknown-unknown/release/wasi_snapshot_preview1.command.wasm To automatically update the reference copy set `BLESS=1` in the environment --- /dev/fd/63 2024-06-14 18:50:51.726335825 +0000 +++ /dev/fd/62 2024-06-14 18:50:51.726335825 +0000 @@ -1,4 +1,4 @@ -(module $wasi_preview1_component_adapter.command.adapter: +(module $wasi_preview1_component_adapter.command.adapter:9a152d0ad902adeb995c2ab83556f62119151301 (type (;0;) (func)) (type (;1;) (func (param i32))) (type (;2;) (func (result i64)))
CI only sets the
VERSION
environment variable to the current git commit. That I think will no longer be viable with this approach so we'll need to remove that from CI and the script here.
Ok now I think it's more clear what's happening:
Reference copy of adapter is not the same as the generated copy of the adapter reference copy: crates/wasi-preview1-component-adapter/provider/artefacts/wasi_snapshot_preview1.command.wasm built copy: target/wasm32-unknown-unknown/release/wasi_snapshot_preview1.command.wasm To automatically update the reference copy set `BLESS=1` in the environment --- /dev/fd/63 2024-06-14 18:50:51.726335825 +0000 +++ /dev/fd/62 2024-06-14 18:50:51.726335825 +0000 @@ -1,4 +1,4 @@ -(module $wasi_preview1_component_adapter.command.adapter: +(module $wasi_preview1_component_adapter.command.adapter:9a152d0ad902adeb995c2ab83556f62119151301 (type (;0;) (func)) (type (;1;) (func (param i32))) (type (;2;) (func (result i64)))
CI only sets the
VERSION
environment variable to the current git commit. That I think will no longer be viable with this approach so we'll need to remove that from CI and the script here.Oooh, that makes sense! I never set the version variable locally and of course we can only know it once a commit is made.
I do like the idea that the version points to the last commit when the adapter was changed. Could git be used to check the file tree, excluding the artefacts folder, for the latest commit hash, and then use that?
alexcrichton commented on PR #8792:
Yeah I think that'd also be reasonable to do!
juntyr updated PR #8792.
juntyr updated PR #8792.
juntyr updated PR #8792.
juntyr updated PR #8792.
juntyr updated PR #8792.
Yeah I think that'd also be reasonable to do!
I tried, I tried. It unfortunately doesn't work out as any commit that would change the adapter would need to refer to itself to provide the updated artefacts (especially since in CI we run on merge-into-main commits which don't exist yet). So I've reverted to the most basic solution of just using the crate version :/
If we do want more info in the version, it could also include a hash of the source files (but this would only help to distinguish git versions, though without relation to their commit hash, not published versions)
alexcrichton submitted PR review:
I think it's reasonable to remove the hash yeah. One difficulty of using the crate version though is that these binaries would need to be updated whenever we bump the version of the repo which is currently an automated process. It's possible to update that automation, but I think it's probably just easiest to remove the version info entirely for now.
Otherwise can you also move the CI checks for the adapter to their own build job? That way it can pin rustc to a fixed version which isn't updated when we update the rest of CI in the future (done by updating our MSRV right now).
As some minor follow-up work:
- Can this update
scripts/publish.rs
as well? I think that'll end up getting caught when full CI runs on merge.- Mind updating
crates/test-programs/artifacts/build.rs to depend on this crate with a
[build-dependencies]`? That otherwise builds the adapters right now and it shouldn't need to any longer after this.
juntyr updated PR #8792.
I think it's reasonable to remove the hash yeah. One difficulty of using the crate version though is that these binaries would need to be updated whenever we bump the version of the repo which is currently an automated process. It's possible to update that automation, but I think it's probably just easiest to remove the version info entirely for now.
What if the adapter crate got its own explicit version, that could be bumped independently of the other crates? I've prototyped this for now, but I can also revert the change and use the workspace version and not add the version to the metadata.
Otherwise can you also move the CI checks for the adapter to their own build job? That way it can pin rustc to a fixed version which isn't updated when we update the rest of CI in the future (done by updating our MSRV right now).
Done - the adapter crate now lists an explicit MSRV which the build script uses.
As some minor follow-up work:
* Can this update `scripts/publish.rs` as well? I think that'll end up getting caught when full CI runs on merge.
Done!
* Mind updating `crates/test-programs/artifacts/build.rs to depend on this crate with a `[build-dependencies]`? That otherwise builds the adapters right now and it shouldn't need to any longer after this.
Done!
alexcrichton submitted PR review.
alexcrichton submitted PR review.
alexcrichton created PR review comment:
While having an explicit version here improves things my worry is that this will basically never get updated, even when it needs to. Currently all other version management in this repository is automated so I would expect that this would never change as a result.
I can think of two other primary alternatives for continuing to embed version information in the adapter:
- Update the bump-version script/CI configuration to additionally rebuild the adapters. I find this not super appealing though because it adds binary file churn to this repository which feels unfortunate.
- Add a
build.rs
to the provider crate which injects version information. To be maximally useful this would probably want to have zero build-dependencies which makes it a bit difficult to do, though.Overall I'd probably still be most in favor of just dropping the version information in the binary. AFAIK we haven't really ended up using it all that much in practice.
alexcrichton created PR review comment:
Could this be deferred to CI configuration rather than built-in to this script? It might make sense to double-check that the rustc version activated matches the
rust-version
on the crate itself but otherwise installation of the toolchain can get unwieldy to work with when running this script locally.
juntyr updated PR #8792.
juntyr updated PR #8792.
juntyr submitted PR review.
juntyr created PR review comment:
Ok, let's just remove the version info (for now) then
juntyr submitted PR review.
juntyr created PR review comment:
Done!
@alexcrichton I hope this is now good to go - thank you for all of your rounds of feedback!
alexcrichton submitted PR review:
Thank you as well for pushing on this!
alexcrichton merged PR #8792.
Last updated: Jan 24 2025 at 00:11 UTC