jlkiri opened issue #4145:
If I understand correctly,
stdout
method onWasiCtxBuilder
can be used to set what the instance will treat as stdout. This something needs to implementWasiFile
trait and I discovered thatFile
fromcap-std
create implements it. It also allows creating cap-std-files from usual std files. I guessed that I can then created a usual file, and then through some conversions turn it into aWasiCtxBuilder
-usable stdout target. Since all these methods own the file, I obtain the raw fd which I then use to open the file again and read its contents. However it appears to be empty, although I do print text in the Rust program compiled to wasm. Note that the execution itself is successful - I just don't see any side effects. Here's the code that I compile to wasm:fn main() { println!("Hello"); }
and here's what I use to execute it in wasmtime:
fn wasm_execute( engine: &wasmtime::Engine, module_path: impl AsRef<Path>, ) -> anyhow::Result<String> { use cap_std::fs::File as CapStdFile; use wasmtime::*; use wasmtime_wasi::{add_to_linker, sync::file::File, WasiCtxBuilder}; let module = Module::from_file(&engine, module_path)?; let tmpfile = tempfile::tempfile()?; let tmpfile_fd = tmpfile.as_raw_fd(); let cap_std_file = CapStdFile::from_std(tmpfile); let stdout_file = File::from_cap_std(cap_std_file); let wasi = WasiCtxBuilder::new().stdout(Box::new(stdout_file)).build(); let store_limits = StoreLimits::default(); let mut store = Store::new(&engine, Data { wasi, store_limits }); store.limiter(|data| &mut data.store_limits); let mut linker: Linker<Data> = Linker::new(&engine); add_to_linker(&mut linker, |state| &mut state.wasi)?; linker.module(&mut store, "", &module)?; linker.instantiate(&mut store, &module)?; linker .get_default(&mut store, "")? .typed::<(), (), _>(&store)? .call(&mut store, ())?; let mut output = String::new(); let mut stdout_file_handle = unsafe { std::fs::File::from_raw_fd(tmpfile_fd) }; let bytes_read = stdout_file_handle.read_to_string(&mut output)?; tracing::info!(bytes_read); // 0 bytes read Ok(output) }
Could it be that raw_fd is no longer valid at that point?
jlkiri edited issue #4145:
If I understand correctly,
stdout
method onWasiCtxBuilder
can be used to set what the instance will treat as stdout. This something needs to implementWasiFile
trait and I discovered thatFile
fromcap-std
create implements it. It also allows creating cap-std-files from usual std files. I guessed that I can then created a usual file, and then through some conversions turn it into aWasiCtxBuilder
-usable stdout target. Since all these methods own the file, I obtain the raw fd which I then use to open the file again and read its contents. However it appears to be empty, although I do print text in the Rust program compiled to wasm. Note that the execution itself is successful - I just don't see any side effects. Here's the code that I compile to wasm:fn main() { println!("Hello"); }
and here's what I use to execute it in wasmtime:
fn wasm_execute( engine: &wasmtime::Engine, module_path: impl AsRef<Path>, ) -> anyhow::Result<String> { use cap_std::fs::File as CapStdFile; use wasmtime::*; use wasmtime_wasi::{add_to_linker, sync::file::File, WasiCtxBuilder}; let module = Module::from_file(&engine, module_path)?; let tmpfile = tempfile::tempfile()?; let tmpfile_fd = tmpfile.as_raw_fd(); let cap_std_file = CapStdFile::from_std(tmpfile); let stdout_file = File::from_cap_std(cap_std_file); let wasi = WasiCtxBuilder::new().stdout(Box::new(stdout_file)).build(); let store_limits = StoreLimits::default(); let mut store = Store::new(&engine, Data { wasi, store_limits }); store.limiter(|data| &mut data.store_limits); let mut linker: Linker<Data> = Linker::new(&engine); add_to_linker(&mut linker, |state| &mut state.wasi)?; linker.module(&mut store, "", &module)?; linker.instantiate(&mut store, &module)?; linker .get_default(&mut store, "")? .typed::<(), (), _>(&store)? .call(&mut store, ())?; let mut output = String::new(); let mut stdout_file_handle = unsafe { std::fs::File::from_raw_fd(tmpfile_fd) }; let bytes_read = stdout_file_handle.read_to_string(&mut output)?; tracing::info!(bytes_read); // 0 bytes read Ok(output) }
Could it be that raw_fd is no longer valid at that point?
jlkiri edited issue #4145:
If I understand correctly,
stdout
method onWasiCtxBuilder
can be used to set what the instance will treat as stdout. This something needs to implementWasiFile
trait and I discovered thatFile
fromcap-std
create implements it. It also allows creating cap-std-files from usual std files. I guessed that I can then created a usual file, and then through some conversions turn it into aWasiCtxBuilder
-usable stdout target. Since all these methods own the file, I obtain the raw fd which I then use to open the file again and read its contents. However it appears to be empty, although I do print text in the Rust program compiled to wasm. Note that the execution itself is successful - I just don't see any side effects. Here's the code that I compile to wasm:fn main() { println!("Hello"); }
and here's what I use to execute it in wasmtime:
fn wasm_execute( engine: &wasmtime::Engine, module_path: impl AsRef<Path>, ) -> anyhow::Result<String> { use cap_std::fs::File as CapStdFile; use wasmtime::*; use wasmtime_wasi::{add_to_linker, sync::file::File, WasiCtxBuilder}; let module = Module::from_file(&engine, module_path)?; let tmpfile = tempfile::tempfile()?; let tmpfile_fd = tmpfile.as_raw_fd(); let cap_std_file = CapStdFile::from_std(tmpfile); let stdout_file = File::from_cap_std(cap_std_file); let wasi = WasiCtxBuilder::new().stdout(Box::new(stdout_file)).build(); let store_limits = StoreLimits::default(); let mut store = Store::new(&engine, Data { wasi, store_limits }); store.limiter(|data| &mut data.store_limits); let mut linker: Linker<Data> = Linker::new(&engine); add_to_linker(&mut linker, |state| &mut state.wasi)?; linker.module(&mut store, "", &module)?; linker.instantiate(&mut store, &module)?; linker .get_default(&mut store, "")? .typed::<(), (), _>(&store)? .call(&mut store, ())?; let mut output = String::new(); let mut stdout_file_handle = unsafe { std::fs::File::from_raw_fd(tmpfile_fd) }; let bytes_read = stdout_file_handle.read_to_string(&mut output)?; tracing::info!(bytes_read); // 0 bytes read Ok(output) }
Could it be that raw_fd is no longer valid at that point?
jlkiri edited issue #4145:
If I understand correctly,
stdout
method onWasiCtxBuilder
can be used to set what the instance will treat as stdout. This something needs to implementWasiFile
trait and I discovered thatFile
fromcap-std
crate implements it. It also allows creating cap-std-files from usual std files. I guessed that I can then created a usual file, and then through some conversions turn it into aWasiCtxBuilder
-usable stdout target. Since all these methods own the file, I obtain the raw fd which I then use to open the file again and read its contents. However it appears to be empty, although I do print text in the Rust program compiled to wasm. Note that the execution itself is successful - I just don't see any side effects. Here's the code that I compile to wasm:fn main() { println!("Hello"); }
and here's what I use to execute it in wasmtime:
fn wasm_execute( engine: &wasmtime::Engine, module_path: impl AsRef<Path>, ) -> anyhow::Result<String> { use cap_std::fs::File as CapStdFile; use wasmtime::*; use wasmtime_wasi::{add_to_linker, sync::file::File, WasiCtxBuilder}; let module = Module::from_file(&engine, module_path)?; let tmpfile = tempfile::tempfile()?; let tmpfile_fd = tmpfile.as_raw_fd(); let cap_std_file = CapStdFile::from_std(tmpfile); let stdout_file = File::from_cap_std(cap_std_file); let wasi = WasiCtxBuilder::new().stdout(Box::new(stdout_file)).build(); let store_limits = StoreLimits::default(); let mut store = Store::new(&engine, Data { wasi, store_limits }); store.limiter(|data| &mut data.store_limits); let mut linker: Linker<Data> = Linker::new(&engine); add_to_linker(&mut linker, |state| &mut state.wasi)?; linker.module(&mut store, "", &module)?; linker.instantiate(&mut store, &module)?; linker .get_default(&mut store, "")? .typed::<(), (), _>(&store)? .call(&mut store, ())?; let mut output = String::new(); let mut stdout_file_handle = unsafe { std::fs::File::from_raw_fd(tmpfile_fd) }; let bytes_read = stdout_file_handle.read_to_string(&mut output)?; tracing::info!(bytes_read); // 0 bytes read Ok(output) }
Could it be that raw_fd is no longer valid at that point?
jlkiri edited issue #4145:
If I understand correctly,
stdout
method onWasiCtxBuilder
can be used to set what the instance will treat as stdout. This something needs to implementWasiFile
trait and I discovered thatFile
fromcap-std
crate implements it. It also allows creating cap-std-files from usual std files. I guessed that I can then created a usual file, and then through some conversions turn it into aWasiCtxBuilder
-usable stdout target. Since all these methods own the file, I obtain the raw fd which I then use to open the file again and read its contents. However it appears to be empty, although I do print text in the Rust program compiled to wasm. Note that the execution itself is successful - I just don't see any side effects. Here's the code that I compile to wasm:fn main() { println!("Hello"); }
and here's what I use to execute it in wasmtime:
fn wasm_execute( engine: &wasmtime::Engine, module_path: impl AsRef<Path>, ) -> anyhow::Result<String> { use cap_std::fs::File as CapStdFile; use wasmtime::*; use wasmtime_wasi::{add_to_linker, sync::file::File, WasiCtxBuilder}; let module = Module::from_file(&engine, module_path)?; let tmpfile = tempfile::tempfile()?; let tmpfile_fd = tmpfile.as_raw_fd(); let cap_std_file = CapStdFile::from_std(tmpfile); let stdout_file = File::from_cap_std(cap_std_file); let wasi = WasiCtxBuilder::new().stdout(Box::new(stdout_file)).build(); let store_limits = StoreLimits::default(); let mut store = Store::new(&engine, Data { wasi, store_limits }); store.limiter(|data| &mut data.store_limits); let mut linker: Linker<Data> = Linker::new(&engine); add_to_linker(&mut linker, |state| &mut state.wasi)?; linker.module(&mut store, "", &module)?; linker.instantiate(&mut store, &module)?; linker .get_default(&mut store, "")? .typed::<(), (), _>(&store)? .call(&mut store, ())?; let mut output = String::new(); let mut stdout_file_handle = unsafe { std::fs::File::from_raw_fd(tmpfile_fd) }; let bytes_read = stdout_file_handle.read_to_string(&mut output)?; tracing::info!(bytes_read); // 0 bytes read Ok(output) }
Could it be that raw_fd is no longer valid at that point?
jlkiri edited issue #4145:
If I understand correctly,
stdout
method onWasiCtxBuilder
can be used to set what the instance will treat as stdout. This something needs to implementWasiFile
trait and I discovered thatFile
fromcap-std
crate implements it. It also allows creating cap-std-files from usual std files. I guessed that I can then create a usual file, and then through some conversions turn it into aWasiCtxBuilder
-usable stdout target. Since all these methods own the file, I obtain the raw fd which I then use to open the file again and read its contents. However it appears to be empty, although I do print text in the Rust program compiled to wasm. Note that the execution itself is successful - I just don't see any side effects. Here's the code that I compile to wasm:fn main() { println!("Hello"); }
and here's what I use to execute it in wasmtime:
fn wasm_execute( engine: &wasmtime::Engine, module_path: impl AsRef<Path>, ) -> anyhow::Result<String> { use cap_std::fs::File as CapStdFile; use wasmtime::*; use wasmtime_wasi::{add_to_linker, sync::file::File, WasiCtxBuilder}; let module = Module::from_file(&engine, module_path)?; let tmpfile = tempfile::tempfile()?; let tmpfile_fd = tmpfile.as_raw_fd(); let cap_std_file = CapStdFile::from_std(tmpfile); let stdout_file = File::from_cap_std(cap_std_file); let wasi = WasiCtxBuilder::new().stdout(Box::new(stdout_file)).build(); let store_limits = StoreLimits::default(); let mut store = Store::new(&engine, Data { wasi, store_limits }); store.limiter(|data| &mut data.store_limits); let mut linker: Linker<Data> = Linker::new(&engine); add_to_linker(&mut linker, |state| &mut state.wasi)?; linker.module(&mut store, "", &module)?; linker.instantiate(&mut store, &module)?; linker .get_default(&mut store, "")? .typed::<(), (), _>(&store)? .call(&mut store, ())?; let mut output = String::new(); let mut stdout_file_handle = unsafe { std::fs::File::from_raw_fd(tmpfile_fd) }; let bytes_read = stdout_file_handle.read_to_string(&mut output)?; tracing::info!(bytes_read); // 0 bytes read Ok(output) }
Could it be that raw_fd is no longer valid at that point?
jlkiri edited issue #4145:
If I understand correctly,
stdout
method onWasiCtxBuilder
can be used to set what the instance will treat as stdout. This something needs to implementWasiFile
trait and I discovered thatFile
fromcap-std
crate implements it. It also allows creating cap-std-files from usual std files. I guessed that I can then create a usual file, and then through some conversions turn it into aWasiCtxBuilder
-usable stdout target. Since all these methods own the file, I obtain the raw fd which I then use to open the file again and read its contents. However it appears to be empty, although I do print text in the Rust program compiled to wasm. Note that the execution itself is successful - I just don't see any side effects. Also inheriting stdio also works properly. Here's the code that I compile to wasm:fn main() { println!("Hello"); }
and here's what I use to execute it in wasmtime:
fn wasm_execute( engine: &wasmtime::Engine, module_path: impl AsRef<Path>, ) -> anyhow::Result<String> { use cap_std::fs::File as CapStdFile; use wasmtime::*; use wasmtime_wasi::{add_to_linker, sync::file::File, WasiCtxBuilder}; let module = Module::from_file(&engine, module_path)?; let tmpfile = tempfile::tempfile()?; let tmpfile_fd = tmpfile.as_raw_fd(); let cap_std_file = CapStdFile::from_std(tmpfile); let stdout_file = File::from_cap_std(cap_std_file); let wasi = WasiCtxBuilder::new().stdout(Box::new(stdout_file)).build(); let store_limits = StoreLimits::default(); let mut store = Store::new(&engine, Data { wasi, store_limits }); store.limiter(|data| &mut data.store_limits); let mut linker: Linker<Data> = Linker::new(&engine); add_to_linker(&mut linker, |state| &mut state.wasi)?; linker.module(&mut store, "", &module)?; linker.instantiate(&mut store, &module)?; linker .get_default(&mut store, "")? .typed::<(), (), _>(&store)? .call(&mut store, ())?; let mut output = String::new(); let mut stdout_file_handle = unsafe { std::fs::File::from_raw_fd(tmpfile_fd) }; let bytes_read = stdout_file_handle.read_to_string(&mut output)?; tracing::info!(bytes_read); // 0 bytes read Ok(output) }
Could it be that raw_fd is no longer valid at that point?
jlkiri edited issue #4145:
If I understand correctly,
stdout
method onWasiCtxBuilder
can be used to set what the instance will treat as stdout. This something needs to implementWasiFile
trait and I discovered thatFile
fromcap-std
crate implements it. It also allows creating cap-std-files from usual std files. I guessed that I can then create a usual file, and then through some conversions turn it into aWasiCtxBuilder
-usable stdout target. Since all these methods own the file, I obtain the raw fd which I then use to open the file again and read its contents. However it appears to be empty, although I do print text in the Rust program compiled to wasm. Note that the execution itself is successful - I just don't see any side effects. Inheriting stdio also works properly. Here's the code that I compile to wasm:fn main() { println!("Hello"); }
and here's what I use to execute it in wasmtime:
fn wasm_execute( engine: &wasmtime::Engine, module_path: impl AsRef<Path>, ) -> anyhow::Result<String> { use cap_std::fs::File as CapStdFile; use wasmtime::*; use wasmtime_wasi::{add_to_linker, sync::file::File, WasiCtxBuilder}; let module = Module::from_file(&engine, module_path)?; let tmpfile = tempfile::tempfile()?; let tmpfile_fd = tmpfile.as_raw_fd(); let cap_std_file = CapStdFile::from_std(tmpfile); let stdout_file = File::from_cap_std(cap_std_file); let wasi = WasiCtxBuilder::new().stdout(Box::new(stdout_file)).build(); let store_limits = StoreLimits::default(); let mut store = Store::new(&engine, Data { wasi, store_limits }); store.limiter(|data| &mut data.store_limits); let mut linker: Linker<Data> = Linker::new(&engine); add_to_linker(&mut linker, |state| &mut state.wasi)?; linker.module(&mut store, "", &module)?; linker.instantiate(&mut store, &module)?; linker .get_default(&mut store, "")? .typed::<(), (), _>(&store)? .call(&mut store, ())?; let mut output = String::new(); let mut stdout_file_handle = unsafe { std::fs::File::from_raw_fd(tmpfile_fd) }; let bytes_read = stdout_file_handle.read_to_string(&mut output)?; tracing::info!(bytes_read); // 0 bytes read Ok(output) }
Could it be that raw_fd is no longer valid at that point?
alexcrichton commented on issue #4145:
I believe you're reading the same file descriptor that was used for writes, so I think you might need to
seek
to the beginning to reset the internal cursor?
bjorn3 commented on issue #4145:
I think you will need to seek to the start of the file. Also you should probably use
cap_std::fs::File
directly instead of trying to convert it into anstd::fs::File
. I'm pretty sure the code you have right now would close the fd twice.
jlkiri commented on issue #4145:
Seeking to start of the file solved the problem!
Also you should probably use cap_std::fs::File directly instead of trying to convert it into an std::fs::File
Are you talking about this line?File::from_cap_std(cap_std_file);
This File is
wasmtime::File
.
jlkiri edited a comment on issue #4145:
Seeking to start of the file solved the problem!
Also you should probably use cap_std::fs::File directly instead of trying to convert it into an std::fs::File
Are you talking about this line?
File::from_cap_std(cap_std_file);
This File is
wasmtime::File
.
jlkiri edited a comment on issue #4145:
Seeking to start of the file solved the problem! Thanks!
Also you should probably use cap_std::fs::File directly instead of trying to convert it into an std::fs::File
Are you talking about this line?
File::from_cap_std(cap_std_file);
This File is
wasmtime::File
.
jlkiri edited a comment on issue #4145:
Seeking to the start of the file solved the problem! Thanks!
Also you should probably use cap_std::fs::File directly instead of trying to convert it into an std::fs::File
Are you talking about this line?
File::from_cap_std(cap_std_file);
This File is
wasmtime::File
.
bjorn3 commented on issue #4145:
Are you talking about this line?
No, I'm talking about
std::fs::File::from_raw_fd(tmpfile_fd)
. I think you should usecap_std_file
there instead. You may need to changeFile::from_cap_std(cap_std_file)
toFile::from_cap_std(cap_std_file.clone())
for that to be possible.
jlkiri commented on issue #4145:
Oh I see! Thanks, I thought that the file cannot have multiple owners by cloning.
jlkiri closed issue #4145:
If I understand correctly,
stdout
method onWasiCtxBuilder
can be used to set what the instance will treat as stdout. This something needs to implementWasiFile
trait and I discovered thatFile
fromcap-std
crate implements it. It also allows creating cap-std-files from usual std files. I guessed that I can then create a usual file, and then through some conversions turn it into aWasiCtxBuilder
-usable stdout target. Since all these methods own the file, I obtain the raw fd which I then use to open the file again and read its contents. However it appears to be empty, although I do print text in the Rust program compiled to wasm. Note that the execution itself is successful - I just don't see any side effects. Inheriting stdio also works properly. Here's the code that I compile to wasm:fn main() { println!("Hello"); }
and here's what I use to execute it in wasmtime:
fn wasm_execute( engine: &wasmtime::Engine, module_path: impl AsRef<Path>, ) -> anyhow::Result<String> { use cap_std::fs::File as CapStdFile; use wasmtime::*; use wasmtime_wasi::{add_to_linker, sync::file::File, WasiCtxBuilder}; let module = Module::from_file(&engine, module_path)?; let tmpfile = tempfile::tempfile()?; let tmpfile_fd = tmpfile.as_raw_fd(); let cap_std_file = CapStdFile::from_std(tmpfile); let stdout_file = File::from_cap_std(cap_std_file); let wasi = WasiCtxBuilder::new().stdout(Box::new(stdout_file)).build(); let store_limits = StoreLimits::default(); let mut store = Store::new(&engine, Data { wasi, store_limits }); store.limiter(|data| &mut data.store_limits); let mut linker: Linker<Data> = Linker::new(&engine); add_to_linker(&mut linker, |state| &mut state.wasi)?; linker.module(&mut store, "", &module)?; linker.instantiate(&mut store, &module)?; linker .get_default(&mut store, "")? .typed::<(), (), _>(&store)? .call(&mut store, ())?; let mut output = String::new(); let mut stdout_file_handle = unsafe { std::fs::File::from_raw_fd(tmpfile_fd) }; let bytes_read = stdout_file_handle.read_to_string(&mut output)?; tracing::info!(bytes_read); // 0 bytes read Ok(output) }
Could it be that raw_fd is no longer valid at that point?
Last updated: Jan 24 2025 at 00:11 UTC