Skip to content

Commit

Permalink
Support undeclared static imports
Browse files Browse the repository at this point in the history
  • Loading branch information
daxpedda committed Dec 5, 2024
1 parent 94b2dc6 commit 92a7912
Show file tree
Hide file tree
Showing 10 changed files with 173 additions and 29 deletions.
7 changes: 5 additions & 2 deletions CHANGELOG.md
Original file line number Diff line number Diff line change
Expand Up @@ -8,11 +8,14 @@
* Add support for multi-threading in Node.js.
[#4318](https://github.com/rustwasm/wasm-bindgen/pull/4318)

### Changed

* Add clear error message to communicate new feature resolver version requirements.
[#4312](https://github.com/rustwasm/wasm-bindgen/pull/4312)

### Changed

* `static FOO: Option<T>` now returns `None` if undeclared in JS instead of throwing an error in JS.
[#4319](https://github.com/rustwasm/wasm-bindgen/pull/4319)

### Fixed

* Fix macro-hygiene for calls to `std::thread_local!`.
Expand Down
17 changes: 14 additions & 3 deletions crates/cli-support/src/js/mod.rs
Original file line number Diff line number Diff line change
Expand Up @@ -2730,7 +2730,7 @@ __wbg_set_wasm(wasm);"
| AuxImport::Value(AuxValue::Setter(js, ..))
| AuxImport::ValueWithThis(js, ..)
| AuxImport::Instanceof(js)
| AuxImport::Static(js)
| AuxImport::Static { js, .. }
| AuxImport::StructuralClassGetter(js, ..)
| AuxImport::StructuralClassSetter(js, ..)
| AuxImport::IndexingGetterOfClass(js)
Expand Down Expand Up @@ -3265,11 +3265,22 @@ __wbg_set_wasm(wasm);"
Ok("result".to_owned())
}

AuxImport::Static(js) => {
AuxImport::Static { js, optional } => {
assert!(kind == AdapterJsImportKind::Normal);
assert!(!variadic);
assert_eq!(args.len(), 0);
self.import_name(js)
let js = self.import_name(js)?;

if *optional {
writeln!(
prelude,
"const result = typeof {js} === 'undefined' ? null : {js};"
)
.unwrap();
Ok("result".to_owned())
} else {
Ok(js)
}
}

AuxImport::String(string) => {
Expand Down
7 changes: 5 additions & 2 deletions crates/cli-support/src/wit/mod.rs
Original file line number Diff line number Diff line change
Expand Up @@ -788,6 +788,7 @@ impl<'a> Context<'a> {
None => return Ok(()),
Some(d) => d,
};
let optional = matches!(descriptor, Descriptor::Option(_));

// Register the signature of this imported shim
let id = self.import_adapter(
Expand All @@ -803,8 +804,10 @@ impl<'a> Context<'a> {

// And then save off that this function is is an instanceof shim for an
// imported item.
let import = self.determine_import(import, static_.name)?;
self.aux.import_map.insert(id, AuxImport::Static(import));
let js = self.determine_import(import, static_.name)?;
self.aux
.import_map
.insert(id, AuxImport::Static { js, optional });
Ok(())
}

Expand Down
2 changes: 1 addition & 1 deletion crates/cli-support/src/wit/nonstandard.rs
Original file line number Diff line number Diff line change
Expand Up @@ -233,7 +233,7 @@ pub enum AuxImport {

/// This import is expected to be a shim that returns the JS value named by
/// `JsImport`.
Static(JsImport),
Static { js: JsImport, optional: bool },

/// This import is expected to be a shim that returns an exported `JsString`.
String(String),
Expand Down
3 changes: 3 additions & 0 deletions crates/cli/tests/reference/static.d.ts
Original file line number Diff line number Diff line change
@@ -0,0 +1,3 @@
/* tslint:disable */
/* eslint-disable */
export function exported(): void;
77 changes: 77 additions & 0 deletions crates/cli/tests/reference/static.js
Original file line number Diff line number Diff line change
@@ -0,0 +1,77 @@
let wasm;
export function __wbg_set_wasm(val) {
wasm = val;
}


function isLikeNone(x) {
return x === undefined || x === null;
}

function addToExternrefTable0(obj) {
const idx = wasm.__externref_table_alloc();
wasm.__wbindgen_export_1.set(idx, obj);
return idx;
}

const lTextDecoder = typeof TextDecoder === 'undefined' ? (0, module.require)('util').TextDecoder : TextDecoder;

let cachedTextDecoder = new lTextDecoder('utf-8', { ignoreBOM: true, fatal: true });

cachedTextDecoder.decode();

let cachedUint8ArrayMemory0 = null;

function getUint8ArrayMemory0() {
if (cachedUint8ArrayMemory0 === null || cachedUint8ArrayMemory0.byteLength === 0) {
cachedUint8ArrayMemory0 = new Uint8Array(wasm.memory.buffer);
}
return cachedUint8ArrayMemory0;
}

function getStringFromWasm0(ptr, len) {
ptr = ptr >>> 0;
return cachedTextDecoder.decode(getUint8ArrayMemory0().subarray(ptr, ptr + len));
}

export function exported() {
wasm.exported();
}

export function __wbg_static_accessor_NAMESPACE_OPTIONAL_c9a4344c544120f4() {
const result = typeof test.NAMESPACE_OPTIONAL === 'undefined' ? null : test.NAMESPACE_OPTIONAL;
const ret = result;
return isLikeNone(ret) ? 0 : addToExternrefTable0(ret);
};

export function __wbg_static_accessor_NAMESPACE_PLAIN_784c8d7f5bbac62a() {
const ret = test.NAMESPACE_PLAIN;
return ret;
};

export function __wbg_static_accessor_OPTIONAL_ade71b6402851d0c() {
const result = typeof OPTIONAL === 'undefined' ? null : OPTIONAL;
const ret = result;
return isLikeNone(ret) ? 0 : addToExternrefTable0(ret);
};

export function __wbg_static_accessor_PLAIN_c0f08eb2f0db194c() {
const ret = PLAIN;
return ret;
};

export function __wbindgen_init_externref_table() {
const table = wasm.__wbindgen_export_1;
const offset = table.grow(4);
table.set(0, undefined);
table.set(offset + 0, undefined);
table.set(offset + 1, null);
table.set(offset + 2, true);
table.set(offset + 3, false);
;
};

export function __wbindgen_throw(arg0, arg1) {
throw new Error(getStringFromWasm0(arg0, arg1));
};

24 changes: 24 additions & 0 deletions crates/cli/tests/reference/static.rs
Original file line number Diff line number Diff line change
@@ -0,0 +1,24 @@
// DEPENDENCY: js-sys = { path = '{root}/crates/js-sys' }

use wasm_bindgen::prelude::*;
use js_sys::Number;

#[wasm_bindgen]
extern "C" {
#[wasm_bindgen(thread_local_v2)]
static PLAIN: JsValue;
#[wasm_bindgen(thread_local_v2)]
static OPTIONAL: Option<Number>;
#[wasm_bindgen(thread_local_v2, js_namespace = test)]
static NAMESPACE_PLAIN: JsValue;
#[wasm_bindgen(thread_local_v2, js_namespace = test)]
static NAMESPACE_OPTIONAL: Option<Number>;
}

#[wasm_bindgen]
pub fn exported() {
let _ = PLAIN.with(JsValue::clone);
let _ = OPTIONAL.with(Option::clone);
let _ = NAMESPACE_PLAIN.with(JsValue::clone);
let _ = NAMESPACE_OPTIONAL.with(Option::clone);
}
16 changes: 16 additions & 0 deletions crates/cli/tests/reference/static.wat
Original file line number Diff line number Diff line change
@@ -0,0 +1,16 @@
(module $reference_test.wasm
(type (;0;) (func))
(type (;1;) (func (result i32)))
(import "./reference_test_bg.js" "__wbindgen_init_externref_table" (func (;0;) (type 0)))
(func $__externref_table_alloc (;1;) (type 1) (result i32))
(func $exported (;2;) (type 0))
(table (;0;) 128 externref)
(memory (;0;) 17)
(export "memory" (memory 0))
(export "exported" (func $exported))
(export "__externref_table_alloc" (func $__externref_table_alloc))
(export "__wbindgen_export_1" (table 0))
(export "__wbindgen_start" (func 0))
(@custom "target_features" (after code) "\04+\0amultivalue+\0fmutable-globals+\0freference-types+\08sign-ext")
)

35 changes: 14 additions & 21 deletions crates/js-sys/src/lib.rs
Original file line number Diff line number Diff line change
Expand Up @@ -6055,14 +6055,6 @@ pub fn global() -> Object {
}

fn get_global_object() -> Object {
// This is a bit wonky, but we're basically using `#[wasm_bindgen]`
// attributes to synthesize imports so we can access properties of the
// form:
//
// * `globalThis.globalThis`
// * `self.self`
// * ... (etc)
//
// Accessing the global object is not an easy thing to do, and what we
// basically want is `globalThis` but we can't rely on that existing
// everywhere. In the meantime we've got the fallbacks mentioned in:
Expand All @@ -6076,26 +6068,27 @@ pub fn global() -> Object {
extern "C" {
type Global;

#[wasm_bindgen(getter, catch, static_method_of = Global, js_class = globalThis, js_name = globalThis)]
fn get_global_this() -> Result<Object, JsValue>;
#[wasm_bindgen(thread_local_v2, js_name = globalThis)]
static GLOBAL_THIS: Option<Object>;

#[wasm_bindgen(getter, catch, static_method_of = Global, js_class = self, js_name = self)]
fn get_self() -> Result<Object, JsValue>;
#[wasm_bindgen(thread_local_v2, js_name = self)]
static SELF: Option<Object>;

#[wasm_bindgen(getter, catch, static_method_of = Global, js_class = window, js_name = window)]
fn get_window() -> Result<Object, JsValue>;
#[wasm_bindgen(thread_local_v2, js_name = window)]
static WINDOW: Option<Object>;

#[wasm_bindgen(getter, catch, static_method_of = Global, js_class = global, js_name = global)]
fn get_global() -> Result<Object, JsValue>;
#[wasm_bindgen(thread_local_v2, js_name = global)]
static GLOBAL: Option<Object>;
}

// The order is important: in Firefox Extension Content Scripts `globalThis`
// is a Sandbox (not Window), so `globalThis` must be checked after `window`.
let static_object = Global::get_self()
.or_else(|_| Global::get_window())
.or_else(|_| Global::get_global_this())
.or_else(|_| Global::get_global());
if let Ok(obj) = static_object {
let static_object = SELF
.with(Option::clone)
.or_else(|| WINDOW.with(Option::clone))
.or_else(|| GLOBAL_THIS.with(Option::clone))
.or_else(|| GLOBAL.with(Option::clone));
if let Some(obj) = static_object {
if !obj.is_undefined() {
return obj;
}
Expand Down
14 changes: 14 additions & 0 deletions guide/src/reference/static-js-objects.md
Original file line number Diff line number Diff line change
Expand Up @@ -65,6 +65,20 @@ extern "C" {
}
```

## Undeclared

When accessing an in JS undeclared value, it will throw in JS. This can be accounted for by using `Option`.

```rust
extern "C" {
type Crypto;
#[wasm_bindgen(thread_local_v2, js_name = crypto)]
static CRYPTO: Option<Crypto>;
}
```

If `crypto` is undeclared in JS, it will simply return `None` in Rust.

## Static strings

Strings can be imported to avoid going through `TextDecoder/Encoder` when requiring just a `JsString`. This can be useful when dealing with environments where `TextDecoder/Encoder` is not available, like in audio worklets.
Expand Down

0 comments on commit 92a7912

Please sign in to comment.