WebAssembly System Interface (WASI)
[Stable: 1 - Experimental]
Stable: 1 Stability: 1 - Experimental
Модуль node:wasi в настоящее время не предоставляет всесторонние свойства безопасности файловой системы, предоставляемые некоторыми средами выполнения WASI. Полная поддержка безопасной песочницы файловой системы может быть или не быть реализована в будущем. Тем временем не полагайтесь на него для запуска ненадежного кода.
Исходный код: lib/wasi.js
API WASI предоставляет реализацию спецификации WebAssembly System Interface. WASI предоставляет приложениям WebAssembly доступ к базовой операционной системе через набор функций, подобных POSIX.
import { readFile } from 'node:fs/promises';
import { WASI } from 'node:wasi';
import { argv, env } from 'node:process';
const wasi = new WASI({
version: 'preview1',
args: argv,
env,
preopens: {
'/local': '/some/real/path/that/wasm/can/access',
},
});
const wasm = await WebAssembly.compile(
await readFile(new URL('./demo.wasm', import.meta.url)),
);
const instance = await WebAssembly.instantiate(wasm, wasi.getImportObject());
wasi.start(instance);'use strict';
const { readFile } = require('node:fs/promises');
const { WASI } = require('node:wasi');
const { argv, env } = require('node:process');
const { join } = require('node:path');
const wasi = new WASI({
version: 'preview1',
args: argv,
env,
preopens: {
'/local': '/some/real/path/that/wasm/can/access',
},
});
(async () => {
const wasm = await WebAssembly.compile(
await readFile(join(__dirname, 'demo.wasm')),
);
const instance = await WebAssembly.instantiate(wasm, wasi.getImportObject());
wasi.start(instance);
})();Чтобы запустить приведенный выше пример, создайте новый текстовый файл WebAssembly с именем demo.wat:
(module
;; Import the required fd_write WASI function which will write the given io vectors to stdout
;; The function signature for fd_write is:
;; (File Descriptor, *iovs, iovs_len, nwritten) -> Returns number of bytes written
(import "wasi_snapshot_preview1" "fd_write" (func $fd_write (param i32 i32 i32 i32) (result i32)))
(memory 1)
(export "memory" (memory 0))
;; Write 'hello world\n' to memory at an offset of 8 bytes
;; Note the trailing newline which is required for the text to appear
(data (i32.const 8) "hello world\n")
(func $main (export "_start")
;; Creating a new io vector within linear memory
(i32.store (i32.const 0) (i32.const 8)) ;; iov.iov_base - This is a pointer to the start of the 'hello world\n' string
(i32.store (i32.const 4) (i32.const 12)) ;; iov.iov_len - The length of the 'hello world\n' string
(call $fd_write
(i32.const 1) ;; file_descriptor - 1 for stdout
(i32.const 0) ;; *iovs - The pointer to the iov array, which is stored at memory location 0
(i32.const 1) ;; iovs_len - We're printing 1 string stored in an iov - so one.
(i32.const 20) ;; nwritten - A place in memory to store the number of bytes written
)
drop ;; Discard the number of bytes written from the top of the stack
)
)Используйте wabt для компиляции .wat в .wasm
wat2wasm demo.watБезопасность
[История]
| Версия | Изменения |
|---|---|
| v21.2.0, v20.11.0 | Уточнены свойства безопасности WASI. |
| v21.2.0, v20.11.0 | Добавлено в: v21.2.0, v20.11.0 |
WASI предоставляет модель на основе возможностей, посредством которой приложениям предоставляются их собственные пользовательские возможности env, preopens, stdin, stdout, stderr и exit.
Текущая модель угроз Node.js не обеспечивает безопасную песочницу, как это реализовано в некоторых средах выполнения WASI.
Хотя функции возможностей поддерживаются, они не формируют модель безопасности в Node.js. Например, песочница файловой системы может быть преодолена с помощью различных методов. Проект изучает, можно ли добавить эти гарантии безопасности в будущем.
Класс: WASI
Добавлено в: v13.3.0, v12.16.0
Класс WASI предоставляет API системных вызовов WASI и дополнительные удобные методы для работы с приложениями на основе WASI. Каждый экземпляр WASI представляет собой отдельную среду.
new WASI([options])
[История]
| Версия | Изменения |
|---|---|
| v20.1.0 | Значение по умолчанию для returnOnExit изменено на true. |
| v20.0.0 | Опция version теперь обязательна и не имеет значения по умолчанию. |
| v19.8.0 | Поле version добавлено в options. |
| v13.3.0, v12.16.0 | Добавлено в: v13.3.0, v12.16.0 |
options<Object>args<Array> Массив строк, который WebAssembly-приложение увидит как аргументы командной строки. Первый аргумент — это виртуальный путь к самой команде WASI. По умолчанию:[].env<Object> Объект, аналогичныйprocess.env, который WebAssembly-приложение увидит в качестве своей среды. По умолчанию:{}.preopens<Object> Этот объект представляет локальную структуру каталогов WebAssembly-приложения. Строковые ключиpreopensрассматриваются как каталоги в файловой системе. Соответствующие значения вpreopens— это реальные пути к этим каталогам на хост-машине.returnOnExit<boolean> По умолчанию, когда приложения WASI вызывают__wasi_proc_exit(),wasi.start()возвращает код выхода, а не завершает процесс. Установка этого параметра вfalseприведет к тому, что процесс Node.js завершится с указанным кодом выхода. По умолчанию:true.stdin<integer> Файловый дескриптор, используемый в качестве стандартного ввода в WebAssembly-приложении. По умолчанию:0.stdout<integer> Файловый дескриптор, используемый в качестве стандартного вывода в WebAssembly-приложении. По умолчанию:1.stderr<integer> Файловый дескриптор, используемый в качестве стандартной ошибки в WebAssembly-приложении. По умолчанию:2.version<string> Запрашиваемая версия WASI. В настоящее время поддерживаются только версииunstableиpreview1. Эта опция является обязательной.
wasi.getImportObject()
Добавлено в: v19.8.0
Возвращает объект импорта, который можно передать в WebAssembly.instantiate(), если кроме тех, что предоставляются WASI, другие импорты WASM не нужны.
Если в конструктор была передана версия unstable, он вернет:
{ wasi_unstable: wasi.wasiImport }Если в конструктор была передана версия preview1 или версия не была указана, он вернет:
{ wasi_snapshot_preview1: wasi.wasiImport }wasi.start(instance)
Добавлено в: v13.3.0, v12.16.0
instance<WebAssembly.Instance>
Пытается начать выполнение instance в качестве команды WASI, вызвав его экспорт _start(). Если instance не содержит экспорт _start(), или если instance содержит экспорт _initialize(), то выбрасывается исключение.
start() требует, чтобы instance экспортировал WebAssembly.Memory с именем memory. Если у instance нет экспорта memory, выбрасывается исключение.
Если start() вызывается более одного раза, выбрасывается исключение.
wasi.initialize(instance)
Добавлено в: v14.6.0, v12.19.0
instance<WebAssembly.Instance>
Пытается инициализировать instance в качестве реактора WASI, вызвав его экспорт _initialize(), если он присутствует. Если instance содержит экспорт _start(), то выбрасывается исключение.
initialize() требует, чтобы instance экспортировал WebAssembly.Memory с именем memory. Если у instance нет экспорта memory, выбрасывается исключение.
Если initialize() вызывается более одного раза, выбрасывается исключение.
wasi.wasiImport
Добавлено в: v13.3.0, v12.16.0
wasiImport — это объект, который реализует API системных вызовов WASI. Этот объект следует передавать в качестве импорта wasi_snapshot_preview1 во время инстанцирования WebAssembly.Instance.