mirror of
https://github.com/nushell/nushell
synced 2025-01-13 21:55:07 +00:00
Replace std::time::Instant
with web_time::Instant
(#14668)
# Description The `std::time::Instant` type panics in the WASM context. To prevent this, I replaced all uses of `std::time::Instant` in WASM-relevant crates with `web_time::Instant`. This ensures commands using `Instant` work in WASM without issues. For non-WASM targets, `web-time` simply reexports `std::time`, so this change doesn’t affect regular builds ([docs](https://docs.rs/web-time/latest/web_time/)). To ensure future code doesn't reintroduce `std::time::Instant` in WASM contexts, I added a `clippy wasm` command to the toolkit. This runs `cargo clippy` with a `clippy.toml` configured to disallow `std::time::Instant`. Since `web-time` aliases `std::time` by default, the `clippy.toml` is stored in `clippy/wasm` and is only loaded when targeting WASM. I also added a new CI job that tests this too. # User-Facing Changes None.
This commit is contained in:
parent
c29bcc94e7
commit
4b1f4e63c3
12 changed files with 118 additions and 58 deletions
56
.github/workflows/ci.yml
vendored
56
.github/workflows/ci.yml
vendored
|
@ -163,7 +163,23 @@ jobs:
|
||||||
echo "no changes in working directory";
|
echo "no changes in working directory";
|
||||||
fi
|
fi
|
||||||
|
|
||||||
build-wasm:
|
wasm:
|
||||||
|
env:
|
||||||
|
WASM_OPTIONS: --no-default-features --target wasm32-unknown-unknown
|
||||||
|
CLIPPY_CONF_DIR: ${{ github.workspace }}/clippy/wasm/
|
||||||
|
|
||||||
|
strategy:
|
||||||
|
matrix:
|
||||||
|
job:
|
||||||
|
- name: Build WASM
|
||||||
|
command: cargo build
|
||||||
|
args:
|
||||||
|
- name: Clippy WASM
|
||||||
|
command: cargo clippy
|
||||||
|
args: -- $CLIPPY_OPTIONS
|
||||||
|
|
||||||
|
name: ${{ matrix.job.name }}
|
||||||
|
|
||||||
runs-on: ubuntu-latest
|
runs-on: ubuntu-latest
|
||||||
steps:
|
steps:
|
||||||
- uses: actions/checkout@v4.1.7
|
- uses: actions/checkout@v4.1.7
|
||||||
|
@ -174,22 +190,22 @@ jobs:
|
||||||
- name: Add wasm32-unknown-unknown target
|
- name: Add wasm32-unknown-unknown target
|
||||||
run: rustup target add wasm32-unknown-unknown
|
run: rustup target add wasm32-unknown-unknown
|
||||||
|
|
||||||
- run: cargo build -p nu-cmd-base --no-default-features --target wasm32-unknown-unknown
|
- run: ${{ matrix.job.command }} -p nu-cmd-base $WASM_OPTIONS ${{ matrix.job.args }}
|
||||||
- run: cargo build -p nu-cmd-extra --no-default-features --target wasm32-unknown-unknown
|
- run: ${{ matrix.job.command }} -p nu-cmd-extra $WASM_OPTIONS ${{ matrix.job.args }}
|
||||||
- run: cargo build -p nu-cmd-lang --no-default-features --target wasm32-unknown-unknown
|
- run: ${{ matrix.job.command }} -p nu-cmd-lang $WASM_OPTIONS ${{ matrix.job.args }}
|
||||||
- run: cargo build -p nu-color-config --no-default-features --target wasm32-unknown-unknown
|
- run: ${{ matrix.job.command }} -p nu-color-config $WASM_OPTIONS ${{ matrix.job.args }}
|
||||||
- run: cargo build -p nu-command --no-default-features --target wasm32-unknown-unknown
|
- run: ${{ matrix.job.command }} -p nu-command $WASM_OPTIONS ${{ matrix.job.args }}
|
||||||
- run: cargo build -p nu-derive-value --no-default-features --target wasm32-unknown-unknown
|
- run: ${{ matrix.job.command }} -p nu-derive-value $WASM_OPTIONS ${{ matrix.job.args }}
|
||||||
- run: cargo build -p nu-engine --no-default-features --target wasm32-unknown-unknown
|
- run: ${{ matrix.job.command }} -p nu-engine $WASM_OPTIONS ${{ matrix.job.args }}
|
||||||
- run: cargo build -p nu-glob --no-default-features --target wasm32-unknown-unknown
|
- run: ${{ matrix.job.command }} -p nu-glob $WASM_OPTIONS ${{ matrix.job.args }}
|
||||||
- run: cargo build -p nu-json --no-default-features --target wasm32-unknown-unknown
|
- run: ${{ matrix.job.command }} -p nu-json $WASM_OPTIONS ${{ matrix.job.args }}
|
||||||
- run: cargo build -p nu-parser --no-default-features --target wasm32-unknown-unknown
|
- run: ${{ matrix.job.command }} -p nu-parser $WASM_OPTIONS ${{ matrix.job.args }}
|
||||||
- run: cargo build -p nu-path --no-default-features --target wasm32-unknown-unknown
|
- run: ${{ matrix.job.command }} -p nu-path $WASM_OPTIONS ${{ matrix.job.args }}
|
||||||
- run: cargo build -p nu-pretty-hex --no-default-features --target wasm32-unknown-unknown
|
- run: ${{ matrix.job.command }} -p nu-pretty-hex $WASM_OPTIONS ${{ matrix.job.args }}
|
||||||
- run: cargo build -p nu-protocol --no-default-features --target wasm32-unknown-unknown
|
- run: ${{ matrix.job.command }} -p nu-protocol $WASM_OPTIONS ${{ matrix.job.args }}
|
||||||
- run: cargo build -p nu-std --no-default-features --target wasm32-unknown-unknown
|
- run: ${{ matrix.job.command }} -p nu-std $WASM_OPTIONS ${{ matrix.job.args }}
|
||||||
- run: cargo build -p nu-system --no-default-features --target wasm32-unknown-unknown
|
- run: ${{ matrix.job.command }} -p nu-system $WASM_OPTIONS ${{ matrix.job.args }}
|
||||||
- run: cargo build -p nu-table --no-default-features --target wasm32-unknown-unknown
|
- run: ${{ matrix.job.command }} -p nu-table $WASM_OPTIONS ${{ matrix.job.args }}
|
||||||
- run: cargo build -p nu-term-grid --no-default-features --target wasm32-unknown-unknown
|
- run: ${{ matrix.job.command }} -p nu-term-grid $WASM_OPTIONS ${{ matrix.job.args }}
|
||||||
- run: cargo build -p nu-utils --no-default-features --target wasm32-unknown-unknown
|
- run: ${{ matrix.job.command }} -p nu-utils $WASM_OPTIONS ${{ matrix.job.args }}
|
||||||
- run: cargo build -p nuon --no-default-features --target wasm32-unknown-unknown
|
- run: ${{ matrix.job.command }} -p nuon $WASM_OPTIONS ${{ matrix.job.args }}
|
||||||
|
|
3
Cargo.lock
generated
3
Cargo.lock
generated
|
@ -3420,6 +3420,7 @@ dependencies = [
|
||||||
"uuid",
|
"uuid",
|
||||||
"v_htmlescape",
|
"v_htmlescape",
|
||||||
"wax",
|
"wax",
|
||||||
|
"web-time",
|
||||||
"which",
|
"which",
|
||||||
"windows 0.56.0",
|
"windows 0.56.0",
|
||||||
"winreg",
|
"winreg",
|
||||||
|
@ -3659,6 +3660,7 @@ dependencies = [
|
||||||
"tempfile",
|
"tempfile",
|
||||||
"thiserror 2.0.6",
|
"thiserror 2.0.6",
|
||||||
"typetag",
|
"typetag",
|
||||||
|
"web-time",
|
||||||
"windows-sys 0.48.0",
|
"windows-sys 0.48.0",
|
||||||
]
|
]
|
||||||
|
|
||||||
|
@ -3687,6 +3689,7 @@ dependencies = [
|
||||||
"ntapi",
|
"ntapi",
|
||||||
"procfs",
|
"procfs",
|
||||||
"sysinfo 0.32.1",
|
"sysinfo 0.32.1",
|
||||||
|
"web-time",
|
||||||
"windows 0.56.0",
|
"windows 0.56.0",
|
||||||
]
|
]
|
||||||
|
|
||||||
|
|
|
@ -178,6 +178,7 @@ uucore = "0.0.28"
|
||||||
uuid = "1.11.0"
|
uuid = "1.11.0"
|
||||||
v_htmlescape = "0.15.0"
|
v_htmlescape = "0.15.0"
|
||||||
wax = "0.6"
|
wax = "0.6"
|
||||||
|
web-time = "1.1.0"
|
||||||
which = "7.0.0"
|
which = "7.0.0"
|
||||||
windows = "0.56"
|
windows = "0.56"
|
||||||
windows-sys = "0.48"
|
windows-sys = "0.48"
|
||||||
|
|
3
clippy/wasm/clippy.toml
Normal file
3
clippy/wasm/clippy.toml
Normal file
|
@ -0,0 +1,3 @@
|
||||||
|
[[disallowed-types]]
|
||||||
|
path = "std::time::Instant"
|
||||||
|
reason = "WASM panics if used, use `web_time::Instant` instead"
|
|
@ -106,6 +106,7 @@ wax = { workspace = true }
|
||||||
which = { workspace = true, optional = true }
|
which = { workspace = true, optional = true }
|
||||||
unicode-width = { workspace = true }
|
unicode-width = { workspace = true }
|
||||||
data-encoding = { version = "2.6.0", features = ["alloc"] }
|
data-encoding = { version = "2.6.0", features = ["alloc"] }
|
||||||
|
web-time = { workspace = true }
|
||||||
|
|
||||||
[target.'cfg(windows)'.dependencies]
|
[target.'cfg(windows)'.dependencies]
|
||||||
winreg = { workspace = true }
|
winreg = { workspace = true }
|
||||||
|
|
|
@ -1,6 +1,6 @@
|
||||||
use nu_engine::{command_prelude::*, ClosureEvalOnce};
|
use nu_engine::{command_prelude::*, ClosureEvalOnce};
|
||||||
use nu_protocol::engine::Closure;
|
use nu_protocol::engine::Closure;
|
||||||
use std::time::Instant;
|
use web_time::Instant;
|
||||||
|
|
||||||
#[derive(Clone)]
|
#[derive(Clone)]
|
||||||
pub struct TimeIt;
|
pub struct TimeIt;
|
||||||
|
|
|
@ -20,9 +20,9 @@ use std::{
|
||||||
io::{IsTerminal, Read},
|
io::{IsTerminal, Read},
|
||||||
path::PathBuf,
|
path::PathBuf,
|
||||||
str::FromStr,
|
str::FromStr,
|
||||||
time::Instant,
|
|
||||||
};
|
};
|
||||||
use url::Url;
|
use url::Url;
|
||||||
|
use web_time::Instant;
|
||||||
|
|
||||||
const STREAM_PAGE_SIZE: usize = 1000;
|
const STREAM_PAGE_SIZE: usize = 1000;
|
||||||
|
|
||||||
|
|
|
@ -40,6 +40,7 @@ thiserror = "2.0"
|
||||||
typetag = "0.2"
|
typetag = "0.2"
|
||||||
os_pipe = { workspace = true, optional = true, features = ["io_safety"] }
|
os_pipe = { workspace = true, optional = true, features = ["io_safety"] }
|
||||||
log = { workspace = true }
|
log = { workspace = true }
|
||||||
|
web-time = { workspace = true }
|
||||||
|
|
||||||
[target.'cfg(unix)'.dependencies]
|
[target.'cfg(unix)'.dependencies]
|
||||||
nix = { workspace = true, default-features = false, features = ["signal"] }
|
nix = { workspace = true, default-features = false, features = ["signal"] }
|
||||||
|
|
|
@ -10,8 +10,8 @@ use crate::{
|
||||||
ir::IrBlock,
|
ir::IrBlock,
|
||||||
record, PipelineData, ShellError, Span, Value,
|
record, PipelineData, ShellError, Span, Value,
|
||||||
};
|
};
|
||||||
use std::time::Instant;
|
|
||||||
use std::{borrow::Borrow, io::BufRead};
|
use std::{borrow::Borrow, io::BufRead};
|
||||||
|
use web_time::Instant;
|
||||||
|
|
||||||
#[derive(Debug, Clone, Copy)]
|
#[derive(Debug, Clone, Copy)]
|
||||||
struct ElementId(usize);
|
struct ElementId(usize);
|
||||||
|
|
|
@ -20,6 +20,7 @@ libc = { workspace = true }
|
||||||
log = { workspace = true }
|
log = { workspace = true }
|
||||||
sysinfo = { workspace = true }
|
sysinfo = { workspace = true }
|
||||||
itertools = { workspace = true }
|
itertools = { workspace = true }
|
||||||
|
web-time = { workspace = true }
|
||||||
|
|
||||||
[target.'cfg(target_family = "unix")'.dependencies]
|
[target.'cfg(target_family = "unix")'.dependencies]
|
||||||
nix = { workspace = true, default-features = false, features = ["fs", "term", "process", "signal"] }
|
nix = { workspace = true, default-features = false, features = ["fs", "term", "process", "signal"] }
|
||||||
|
|
|
@ -18,7 +18,8 @@ use std::ptr;
|
||||||
use std::ptr::null_mut;
|
use std::ptr::null_mut;
|
||||||
use std::sync::LazyLock;
|
use std::sync::LazyLock;
|
||||||
use std::thread;
|
use std::thread;
|
||||||
use std::time::{Duration, Instant};
|
use std::time::Duration;
|
||||||
|
use web_time::Instant;
|
||||||
|
|
||||||
use windows::core::{PCWSTR, PWSTR};
|
use windows::core::{PCWSTR, PWSTR};
|
||||||
|
|
||||||
|
|
101
toolkit.nu
101
toolkit.nu
|
@ -6,6 +6,8 @@
|
||||||
# (**2**) catch classical flaws in the new changes with *clippy* and (**3**)
|
# (**2**) catch classical flaws in the new changes with *clippy* and (**3**)
|
||||||
# make sure all the tests pass.
|
# make sure all the tests pass.
|
||||||
|
|
||||||
|
const toolkit_dir = path self .
|
||||||
|
|
||||||
# check standard code formatting and apply the changes
|
# check standard code formatting and apply the changes
|
||||||
export def fmt [
|
export def fmt [
|
||||||
--check # do not apply the format changes, only check the syntax
|
--check # do not apply the format changes, only check the syntax
|
||||||
|
@ -363,40 +365,6 @@ export def build [
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
# build crates for wasm
|
|
||||||
export def "build wasm" [] {
|
|
||||||
^rustup target add wasm32-unknown-unknown
|
|
||||||
|
|
||||||
# these crates should compile for wasm
|
|
||||||
let compatible_crates = [
|
|
||||||
"nu-cmd-base",
|
|
||||||
"nu-cmd-extra",
|
|
||||||
"nu-cmd-lang",
|
|
||||||
"nu-color-config",
|
|
||||||
"nu-command",
|
|
||||||
"nu-derive-value",
|
|
||||||
"nu-engine",
|
|
||||||
"nu-glob",
|
|
||||||
"nu-json",
|
|
||||||
"nu-parser",
|
|
||||||
"nu-path",
|
|
||||||
"nu-pretty-hex",
|
|
||||||
"nu-protocol",
|
|
||||||
"nu-std",
|
|
||||||
"nu-system",
|
|
||||||
"nu-table",
|
|
||||||
"nu-term-grid",
|
|
||||||
"nu-utils",
|
|
||||||
"nuon"
|
|
||||||
]
|
|
||||||
|
|
||||||
for crate in $compatible_crates {
|
|
||||||
print $'(char nl)Building ($crate) for wasm'
|
|
||||||
print '----------------------------'
|
|
||||||
^cargo build -p $crate --target wasm32-unknown-unknown --no-default-features
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
def "nu-complete list features" [] {
|
def "nu-complete list features" [] {
|
||||||
open Cargo.toml | get features | transpose feature dependencies | get feature
|
open Cargo.toml | get features | transpose feature dependencies | get feature
|
||||||
}
|
}
|
||||||
|
@ -625,4 +593,69 @@ export def 'release-pkg windows' [
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
# these crates should compile for wasm
|
||||||
|
const wasm_compatible_crates = [
|
||||||
|
"nu-cmd-base",
|
||||||
|
"nu-cmd-extra",
|
||||||
|
"nu-cmd-lang",
|
||||||
|
"nu-color-config",
|
||||||
|
"nu-command",
|
||||||
|
"nu-derive-value",
|
||||||
|
"nu-engine",
|
||||||
|
"nu-glob",
|
||||||
|
"nu-json",
|
||||||
|
"nu-parser",
|
||||||
|
"nu-path",
|
||||||
|
"nu-pretty-hex",
|
||||||
|
"nu-protocol",
|
||||||
|
"nu-std",
|
||||||
|
"nu-system",
|
||||||
|
"nu-table",
|
||||||
|
"nu-term-grid",
|
||||||
|
"nu-utils",
|
||||||
|
"nuon"
|
||||||
|
]
|
||||||
|
|
||||||
|
def "prep wasm" [] {
|
||||||
|
^rustup target add wasm32-unknown-unknown
|
||||||
|
}
|
||||||
|
|
||||||
|
# build crates for wasm
|
||||||
|
export def "build wasm" [] {
|
||||||
|
prep wasm
|
||||||
|
|
||||||
|
for crate in $wasm_compatible_crates {
|
||||||
|
print $'(char nl)Building ($crate) for wasm'
|
||||||
|
print '----------------------------'
|
||||||
|
(
|
||||||
|
^cargo build
|
||||||
|
-p $crate
|
||||||
|
--target wasm32-unknown-unknown
|
||||||
|
--no-default-features
|
||||||
|
)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
# make sure no api is used that doesn't work with wasm
|
||||||
|
export def "clippy wasm" [] {
|
||||||
|
prep wasm
|
||||||
|
|
||||||
|
$env.CLIPPY_CONF_DIR = $toolkit_dir | path join clippy wasm
|
||||||
|
|
||||||
|
for crate in $wasm_compatible_crates {
|
||||||
|
print $'(char nl)Checking ($crate) for wasm'
|
||||||
|
print '----------------------------'
|
||||||
|
(
|
||||||
|
^cargo clippy
|
||||||
|
-p $crate
|
||||||
|
--target wasm32-unknown-unknown
|
||||||
|
--no-default-features
|
||||||
|
--
|
||||||
|
-D warnings
|
||||||
|
-D clippy::unwrap_used
|
||||||
|
-D clippy::unchecked_duration_subtraction
|
||||||
|
)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
export def main [] { help toolkit }
|
export def main [] { help toolkit }
|
||||||
|
|
Loading…
Reference in a new issue