From a23597631172a6b7ca95fe9c80384a889f380e04 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Jos=C3=A9=20Juli=C3=A1n=20Espina?= Date: Thu, 9 Jan 2025 16:39:24 -0600 Subject: [PATCH] Improve implementation of example `JobQueue`s (#4111) * Improve implementation of example `JobQueue`s * Rename example files * Rewrite examples again --- Cargo.lock | 424 +++++++++++------- Cargo.toml | 4 +- examples/Cargo.toml | 3 +- examples/src/bin/futures.rs | 200 --------- ...{module_fetch.rs => module_fetch_async.rs} | 149 +++--- examples/src/bin/smol_event_loop.rs | 250 +++++++++++ examples/src/bin/tokio_event_loop.rs | 265 +++++++++++ tests/macros/tests/derive.rs | 3 +- 8 files changed, 863 insertions(+), 435 deletions(-) delete mode 100644 examples/src/bin/futures.rs rename examples/src/bin/{module_fetch.rs => module_fetch_async.rs} (66%) create mode 100644 examples/src/bin/smol_event_loop.rs create mode 100644 examples/src/bin/tokio_event_loop.rs diff --git a/Cargo.lock b/Cargo.lock index 6967b07f7ac..d91d84fa0a8 100644 --- a/Cargo.lock +++ b/Cargo.lock @@ -46,9 +46,9 @@ dependencies = [ [[package]] name = "allocator-api2" -version = "0.2.20" +version = "0.2.21" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "45862d1c77f2228b9e10bc609d5bc203d86ebc9b87ad8d5d5167a6c9abf739d9" +checksum = "683d7910e743518b0e34f1186f92494becacb047c7b6bf616c96772180fef923" [[package]] name = "android_system_properties" @@ -160,7 +160,7 @@ checksum = "30ca9a001c1e8ba5149f91a74362376cc6bc5b919d92d988668657bd570bdcec" dependencies = [ "async-task", "concurrent-queue", - "fastrand 2.2.0", + "fastrand 2.3.0", "futures-lite 2.5.0", "slab", ] @@ -437,10 +437,11 @@ dependencies = [ "boa_interner", "boa_parser", "boa_runtime", - "futures-util", + "futures-concurrency", "isahc", "smol", "time", + "tokio", ] [[package]] @@ -633,9 +634,9 @@ dependencies = [ [[package]] name = "bytemuck_derive" -version = "1.8.0" +version = "1.8.1" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "bcfcc3cd946cb52f0bbfdbbcfa2f4e24f75ebb6c0e1002f7c25904fada18b9ec" +checksum = "3fa76293b4f7bb636ab88fd78228235b5248b4d05cc589aed610f954af5d7c7a" dependencies = [ "proc-macro2", "quote", @@ -650,15 +651,15 @@ checksum = "1fd0f2584146f6f2ef48085050886acf353beff7305ebd1ae69500e27c67f64b" [[package]] name = "bytes" -version = "1.8.0" +version = "1.9.0" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "9ac0150caa2ae65ca5bd83f25c7de183dea78d4d366469f148435e2acfbad0da" +checksum = "325918d6fe32f23b19878fe4b34794ae41fc19ddbe53b10571a4874d44ffd39b" [[package]] name = "calendrical_calculations" -version = "0.1.1" +version = "0.1.2" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "cec493b209a1b81fa32312d7ceca1b547d341c7b5f16a3edbf32b1d8b455bbdf" +checksum = "f27ca2b6e2f7d75f43e001ded6f25e79b80bded5abbe764cbdf78c25a3051f4b" dependencies = [ "core_maths", "displaydoc", @@ -675,9 +676,9 @@ dependencies = [ [[package]] name = "cargo-platform" -version = "0.1.8" +version = "0.1.9" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "24b1f0365a6c6bb4020cd05806fd0d33c44d38046b8bd7f0e40814b9763cabfc" +checksum = "e35af189006b9c0f00a064685c727031e3ed2d8020f7ba284d78cc2671bd36ea" dependencies = [ "serde", ] @@ -710,9 +711,9 @@ checksum = "a2698f953def977c68f935bb0dfa959375ad4638570e969e2f1e9f433cbf1af6" [[package]] name = "cc" -version = "1.2.1" +version = "1.2.7" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "fd9de9f2205d5ef3fd67e685b0df337994ddd4495e2a28d185500d0e1edfea47" +checksum = "a012a0df96dd6d06ba9a1b29d6402d1a5d77c6befd2566afdc26e10603dc93d7" dependencies = [ "shlex", ] @@ -895,6 +896,16 @@ dependencies = [ "wasm-bindgen", ] +[[package]] +name = "cordyceps" +version = "0.3.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "ec10f0a762d93c4498d2e97a333805cb6250d60bead623f71d8034f9a4152ba3" +dependencies = [ + "loom", + "tracing", +] + [[package]] name = "core-foundation-sys" version = "0.8.7" @@ -963,18 +974,18 @@ checksum = "790eea4361631c5e7d22598ecd5723ff611904e3344ce8720784c93e3d83d40b" [[package]] name = "crossbeam-channel" -version = "0.5.13" +version = "0.5.14" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "33480d6946193aa8033910124896ca395333cae7e2d1113d1fef6c3272217df2" +checksum = "06ba6d68e24814cb8de6bb986db8222d3a027d15872cabc0d18817bc3c0e4471" dependencies = [ "crossbeam-utils", ] [[package]] name = "crossbeam-deque" -version = "0.8.5" +version = "0.8.6" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "613f8cc01fe9cf1a3eb3d7f488fd2fa8388403e97039e2f73692932e291a770d" +checksum = "9dd111b7b7f7d55b72c0a6ae361660ee5853c9af73f70c3c2ef6858b950e2e51" dependencies = [ "crossbeam-epoch", "crossbeam-utils", @@ -991,9 +1002,9 @@ dependencies = [ [[package]] name = "crossbeam-utils" -version = "0.8.20" +version = "0.8.21" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "22ec99545bb0ed0ea7bb9b8e1e9122ea386ff8a48c0922e43f36d45ab09e0e80" +checksum = "d0a5c400df2834b80a4c3327b3aad3a4c4cd4de0629063962b03235697506a28" [[package]] name = "crossterm" @@ -1093,9 +1104,9 @@ dependencies = [ [[package]] name = "deduplicating_array" -version = "0.1.6" +version = "0.1.7" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "1d579f0dbb23612a78e2d3a483dd1bff5edafa84592d15ba5eb5e3fd689b7d98" +checksum = "7165fb0d6692995d6b2874eab9b2c1c3ead785af3f7f6e698274adfed5ab10dc" dependencies = [ "serde", ] @@ -1136,6 +1147,12 @@ dependencies = [ "thousands", ] +[[package]] +name = "diatomic-waker" +version = "0.2.3" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "ab03c107fafeb3ee9f5925686dbb7a73bc76e3932abb0d2b365cb64b169cf04c" + [[package]] name = "displaydoc" version = "0.2.5" @@ -1206,12 +1223,12 @@ dependencies = [ [[package]] name = "errno" -version = "0.3.9" +version = "0.3.10" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "534c5cf6194dfab3db3242765c03bbe257cf92f22b38f6bc0c58d59108a820ba" +checksum = "33d852cb9b869c2a9b3df2f71a3074817f01e1844f839a144f5fcef059a4eb5d" dependencies = [ "libc", - "windows-sys 0.52.0", + "windows-sys 0.59.0", ] [[package]] @@ -1239,9 +1256,9 @@ dependencies = [ [[package]] name = "event-listener-strategy" -version = "0.5.2" +version = "0.5.3" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "0f214dc438f977e6d4e3500aaa277f5ad94ca83fbbd9b1a15713ce2344ccc5a1" +checksum = "3c3e4e0dd3673c1139bf041f3008816d9cf2946bbfac2945c09e523b8d7b05b2" dependencies = [ "event-listener 5.3.1", "pin-project-lite", @@ -1274,9 +1291,9 @@ dependencies = [ [[package]] name = "fastrand" -version = "2.2.0" +version = "2.3.0" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "486f806e73c5707928240ddc295403b1b93c96a02038563881c4a2fd84b81ac4" +checksum = "37909eebbb50d72f9059c3b6d82c0463f2ff062c9e95845c43a6c9c0355411be" [[package]] name = "fd-lock" @@ -1301,6 +1318,12 @@ dependencies = [ "writeable 0.5.5", ] +[[package]] +name = "fixedbitset" +version = "0.5.7" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "1d674e81391d1e1ab681a28d99df07927c6d4aa5b027d7da16ba32d1d21ecd99" + [[package]] name = "flate2" version = "1.0.35" @@ -1308,7 +1331,7 @@ source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "c936bfdafb507ebbf50b8074c54fa31c5be9a1e7e5f467dd659697041407d07c" dependencies = [ "crc32fast", - "miniz_oxide 0.8.0", + "miniz_oxide 0.8.2", ] [[package]] @@ -1328,9 +1351,9 @@ checksum = "3f9eec918d3f24069decb9af1554cad7c880e2da24a9afd88aca000531ab82c1" [[package]] name = "foldhash" -version = "0.1.3" +version = "0.1.4" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "f81ec6369c545a7d40e4589b5597581fa1c441fe1cce96dd1de43159910a36a2" +checksum = "a0d2fde1f7b3d48b8395d5f2de76c18a528bd6a9cdde438df747bfcba3e05d6f" [[package]] name = "form_urlencoded" @@ -1341,6 +1364,33 @@ dependencies = [ "percent-encoding", ] +[[package]] +name = "futures-buffered" +version = "0.2.9" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "34acda8ae8b63fbe0b2195c998b180cff89a8212fb2622a78b572a9f1c6f7684" +dependencies = [ + "cordyceps", + "diatomic-waker", + "futures-core", + "pin-project-lite", +] + +[[package]] +name = "futures-concurrency" +version = "7.6.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "d9b724496da7c26fcce66458526ce68fc2ecf4aaaa994281cf322ded5755520c" +dependencies = [ + "fixedbitset", + "futures-buffered", + "futures-core", + "futures-lite 1.13.0", + "pin-project", + "slab", + "smallvec", +] + [[package]] name = "futures-core" version = "0.3.31" @@ -1374,44 +1424,13 @@ version = "2.5.0" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "cef40d21ae2c515b51041df9ed313ed21e572df340ea58a922a0aefe7e8891a1" dependencies = [ - "fastrand 2.2.0", + "fastrand 2.3.0", "futures-core", "futures-io", "parking", "pin-project-lite", ] -[[package]] -name = "futures-macro" -version = "0.3.31" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "162ee34ebcb7c64a8abebc059ce0fee27c2262618d7b60ed8faf72fef13c3650" -dependencies = [ - "proc-macro2", - "quote", - "syn", -] - -[[package]] -name = "futures-task" -version = "0.3.31" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "f90f7dce0722e95104fcb095585910c0977252f286e354b5e3bd38902cd99988" - -[[package]] -name = "futures-util" -version = "0.3.31" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "9fa08315bb612088cc391249efdc3bc77536f16c91f6cf495e6fbe85b20a4a81" -dependencies = [ - "futures-core", - "futures-macro", - "futures-task", - "pin-project-lite", - "pin-utils", - "slab", -] - [[package]] name = "gen-icu4x-data" version = "0.20.0" @@ -1430,6 +1449,19 @@ dependencies = [ "simple_logger", ] +[[package]] +name = "generator" +version = "0.7.5" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "5cc16584ff22b460a382b7feec54b23d2908d858152e5739a120b949293bd74e" +dependencies = [ + "cc", + "libc", + "log", + "rustversion", + "windows", +] + [[package]] name = "getrandom" version = "0.2.15" @@ -1451,9 +1483,9 @@ checksum = "4271d37baee1b8c7e4b708028c57d816cf9d2434acb33a549475f78c181f6253" [[package]] name = "glob" -version = "0.3.1" +version = "0.3.2" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "d2fabcfbdc87f4758337ca535fb41a6d701b65693ce38287d856d1674551ec9b" +checksum = "a8d1add55171497b4705a648c6b583acafb01d58050a51727785f0b2c8e0a2b2" [[package]] name = "half" @@ -2279,9 +2311,9 @@ checksum = "91335e575850c5c4c673b9bd467b0e025f164ca59d0564f69d0c2ee0ffad4653" [[package]] name = "js-sys" -version = "0.3.74" +version = "0.3.76" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "a865e038f7f6ed956f788f0d7d60c541fff74c7bd74272c5d4cf15c63743e705" +checksum = "6717b6b5b077764fb5966237269cb3c64edddde4b14ce42647430a78ced9e7b7" dependencies = [ "once_cell", "wasm-bindgen", @@ -2295,9 +2327,9 @@ checksum = "bbd2bcb4c963f2ddae06a2efc7e9f3591312473c50c6685e1f298068316e66fe" [[package]] name = "libc" -version = "0.2.164" +version = "0.2.169" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "433bfe06b8c75da9b2e3fbea6e5329ff87748f0b144ef75306e674c3f6f7c13f" +checksum = "b5aba8db14291edd000dfcc4d620c7ebfb122c613afb886ca8803fa4e128a20a" [[package]] name = "libm" @@ -2335,9 +2367,9 @@ checksum = "78b3ae25bc7c8c38cec158d1f2757ee79e9b3740fbc7ccf0e59e4b08d793fa89" [[package]] name = "litemap" -version = "0.7.3" +version = "0.7.4" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "643cb0b8d4fcc284004d5fd0d67ccf61dfffadb7f75e1e71bc420f4688a3a704" +checksum = "4ee93343901ab17bd981295f2cf0026d4ad018c7c31ba84549a4ddbb47a45104" dependencies = [ "serde", ] @@ -2358,6 +2390,28 @@ version = "0.4.22" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "a7a70ba024b9dc04c27ea2f0c0548feb474ec5c54bba33a7f72f873a39d07b24" +[[package]] +name = "loom" +version = "0.5.6" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "ff50ecb28bb86013e935fb6683ab1f6d3a20016f123c76fd4c27470076ac30f5" +dependencies = [ + "cfg-if", + "generator", + "scoped-tls", + "tracing", + "tracing-subscriber", +] + +[[package]] +name = "matchers" +version = "0.1.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "8263075bb86c5a1b1427b5ae862e8889656f126e9f77c484496e8b47cf5c5558" +dependencies = [ + "regex-automata 0.1.10", +] + [[package]] name = "matrixmultiply" version = "0.3.9" @@ -2433,9 +2487,9 @@ dependencies = [ [[package]] name = "miniz_oxide" -version = "0.8.0" +version = "0.8.2" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "e2d80299ef12ff69b16a84bb182e3b9df68b5a91574d3d4fa6e41b65deec4df1" +checksum = "4ffbe83022cedc1d264172192511ae958937694cd57ce297164951b8b3568394" dependencies = [ "adler2", ] @@ -2471,6 +2525,16 @@ dependencies = [ "libc", ] +[[package]] +name = "nu-ansi-term" +version = "0.46.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "77a8165726e8236064dbb45459242600304b42a5ea24ee2948e18e023bf7ba84" +dependencies = [ + "overload", + "winapi", +] + [[package]] name = "num-bigint" version = "0.4.6" @@ -2610,6 +2674,12 @@ dependencies = [ "vcpkg", ] +[[package]] +name = "overload" +version = "0.1.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "b15813163c1d831bf4a13c3610c05c0d03b39feb07f7e09fa234dac9b15aaf39" + [[package]] name = "owo-colors" version = "3.5.0" @@ -2668,9 +2738,9 @@ dependencies = [ [[package]] name = "phf" -version = "0.11.2" +version = "0.11.3" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "ade2d8b8f33c7333b51bcf0428d37e217e9f32192ae4772156f65063b8ce03dc" +checksum = "1fd6780a80ae0c52cc120a26a1a42c1ae51b247a253e4e06113d23d2c2edd078" dependencies = [ "phf_macros", "phf_shared", @@ -2678,9 +2748,9 @@ dependencies = [ [[package]] name = "phf_generator" -version = "0.11.2" +version = "0.11.3" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "48e4cc64c2ad9ebe670cb8fd69dd50ae301650392e81c05f9bfcb2d5bdbc24b0" +checksum = "3c80231409c20246a13fddb31776fb942c38553c51e871f8cbd687a4cfb5843d" dependencies = [ "phf_shared", "rand", @@ -2688,9 +2758,9 @@ dependencies = [ [[package]] name = "phf_macros" -version = "0.11.2" +version = "0.11.3" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "3444646e286606587e49f3bcf1679b8cef1dc2c5ecc29ddacaffc305180d464b" +checksum = "f84ac04429c13a7ff43785d75ad27569f2951ce0ffd30a3321230db2fc727216" dependencies = [ "phf_generator", "phf_shared", @@ -2701,27 +2771,27 @@ dependencies = [ [[package]] name = "phf_shared" -version = "0.11.2" +version = "0.11.3" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "90fcb95eef784c2ac79119d1dd819e162b5da872ce6f3c3abe1e8ca1c082f72b" +checksum = "67eabc2ef2a60eb7faa00097bd1ffdb5bd28e62bf39990626a582201b7a754e5" dependencies = [ "siphasher", ] [[package]] name = "pin-project" -version = "1.1.7" +version = "1.1.8" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "be57f64e946e500c8ee36ef6331845d40a93055567ec57e8fae13efd33759b95" +checksum = "1e2ec53ad785f4d35dac0adea7f7dc6f1bb277ad84a680c7afefeae05d1f5916" dependencies = [ "pin-project-internal", ] [[package]] name = "pin-project-internal" -version = "1.1.7" +version = "1.1.8" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "3c0f5fad0874fc7abcd4d750e76917eaebbecaa2c20bde22e1dbeeba8beb758c" +checksum = "d56a66c0c55993aa927429d0f8a0abfd74f084e4d9c192cffed01e418d83eefb" dependencies = [ "proc-macro2", "quote", @@ -2730,15 +2800,9 @@ dependencies = [ [[package]] name = "pin-project-lite" -version = "0.2.15" +version = "0.2.16" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "915a1e146535de9163f3987b8944ed8cf49a18bb0056bcebcdcece385cece4ff" - -[[package]] -name = "pin-utils" -version = "0.1.0" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "8b870d8c151b6f2fb93e84a13146138f05d02ed11c7e7c54f8826aaaf7c9f184" +checksum = "3b3cff922bd51709b605d9ead9aa71031d81447142d828eb4a6eba76fe619f9b" [[package]] name = "piper" @@ -2747,7 +2811,7 @@ source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "96c8c490f422ef9a4efd2cb5b42b76c8613d7e7dfc1caf667b8a3350a5acc066" dependencies = [ "atomic-waker", - "fastrand 2.2.0", + "fastrand 2.3.0", "futures-io", ] @@ -2830,9 +2894,9 @@ checksum = "280dc24453071f1b63954171985a0b0d30058d287960968b9b2aca264c8d4ee6" [[package]] name = "postcard" -version = "1.0.10" +version = "1.1.1" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "5f7f0a8d620d71c457dd1d47df76bb18960378da56af4527aaa10f515eee732e" +checksum = "170a2601f67cc9dba8edd8c4870b15f71a6a2dc196daec8c83f72b59dff628a8" dependencies = [ "cobs", "embedded-io 0.4.0", @@ -2950,9 +3014,9 @@ dependencies = [ [[package]] name = "redox_syscall" -version = "0.5.7" +version = "0.5.8" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "9b6dfecf2c74bce2466cabf93f6664d6998a69eb21e39f4207930065b27b771f" +checksum = "03a862b389f93e68874fbf580b9de08dd02facb9a788ebadaf4a3fd33cf58834" dependencies = [ "bitflags 2.6.0", ] @@ -2969,6 +3033,15 @@ dependencies = [ "regex-syntax 0.8.5", ] +[[package]] +name = "regex-automata" +version = "0.1.10" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "6c230d73fb8d8c1b9c0b3135c5142a8acee3a0558fb8db5cf1cb65f8d7862132" +dependencies = [ + "regex-syntax 0.6.29", +] + [[package]] name = "regex-automata" version = "0.2.0" @@ -3047,22 +3120,22 @@ checksum = "c7fb8039b3032c191086b10f11f319a6e99e1e82889c5cc6046f515c9db1d497" [[package]] name = "rustix" -version = "0.38.41" +version = "0.38.42" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "d7f649912bc1495e167a6edee79151c84b1bad49748cb4f1f1167f459f6224f6" +checksum = "f93dc38ecbab2eb790ff964bb77fa94faf256fd3e73285fd7ba0903b76bedb85" dependencies = [ "bitflags 2.6.0", "errno", "libc", "linux-raw-sys", - "windows-sys 0.52.0", + "windows-sys 0.59.0", ] [[package]] name = "rustls" -version = "0.23.18" +version = "0.23.20" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "9c9cc1d47e243d655ace55ed38201c19ae02c148ae56412ab8750e8f0166ab7f" +checksum = "5065c3f250cbd332cd894be57c40fa52387247659b14a2d6041d121547903b1b" dependencies = [ "log", "once_cell", @@ -3075,9 +3148,9 @@ dependencies = [ [[package]] name = "rustls-pki-types" -version = "1.10.0" +version = "1.10.1" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "16f1201b3c9a7ee8039bcadc17b7e605e2945b27eee7631788c1bd2b0643674b" +checksum = "d2bf47e6ff922db3825eb750c4e2ff784c6ff8fb9e13046ef6a1d1c5401b0b37" [[package]] name = "rustls-webpki" @@ -3092,9 +3165,9 @@ dependencies = [ [[package]] name = "rustversion" -version = "1.0.18" +version = "1.0.19" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "0e819f2bc632f285be6d7cd36e25940d45b2391dd6d9b939e79de557f7014248" +checksum = "f7c45b9784283f1b2e7fb61b42047c2fd678ef0960d4f6f1eba131594cc369d4" [[package]] name = "rustyline" @@ -3181,9 +3254,9 @@ dependencies = [ [[package]] name = "semver" -version = "1.0.23" +version = "1.0.24" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "61697e0a1c7e512e84a621326239844a24d8207b4669b41bc18b32ea5cbf988b" +checksum = "3cb6eb87a131f756572d7fb904f6e7b68633f09cca868c5df1c4b8d1a694bbba" dependencies = [ "serde", ] @@ -3220,9 +3293,9 @@ dependencies = [ [[package]] name = "serde_json" -version = "1.0.134" +version = "1.0.135" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "d00f4175c42ee48b15416f6193a959ba3a0d67fc699a0db9ad12df9f83991c7d" +checksum = "2b0d7ba2887406110130a978386c4e1befb98c674b4fba677954e4db976630d9" dependencies = [ "itoa", "memchr", @@ -3301,9 +3374,9 @@ dependencies = [ [[package]] name = "siphasher" -version = "0.3.11" +version = "1.0.1" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "38b58827f4464d87d377d175e90bf58eb00fd8716ff0a62f80356b5e61555d0d" +checksum = "56199f7ddabf13fe5074ce809e7d3f42b42ae711800501b5b16ea82ad029c39d" [[package]] name = "slab" @@ -3359,9 +3432,9 @@ dependencies = [ [[package]] name = "socket2" -version = "0.5.7" +version = "0.5.8" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "ce305eb0b4296696835b71df73eb912e0f1ffd2556a501fcede6e0c50349191c" +checksum = "c970269d99b64e60ec3bd6ad27270092a5394c4e309314b18ae3fe575695fbe8" dependencies = [ "libc", "windows-sys 0.52.0", @@ -3644,6 +3717,28 @@ dependencies = [ "serde_json", ] +[[package]] +name = "tokio" +version = "1.42.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "5cec9b21b0450273377fc97bd4c33a8acffc8c996c987a7c5b319a0083707551" +dependencies = [ + "backtrace", + "pin-project-lite", + "tokio-macros", +] + +[[package]] +name = "tokio-macros" +version = "2.4.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "693d596312e88961bc67d7f1f97af8a70227d9f90c31bba5806eec004978d752" +dependencies = [ + "proc-macro2", + "quote", + "syn", +] + [[package]] name = "toml" version = "0.5.11" @@ -3689,9 +3784,9 @@ dependencies = [ [[package]] name = "tracing" -version = "0.1.40" +version = "0.1.41" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "c3523ab5a71916ccf420eebdf5521fcef02141234bbc0b8a49f2fdc4544364ef" +checksum = "784e0ac535deb450455cbfa28a6f0df145ea1bb7ae51b821cf5e7927fdcfbdd0" dependencies = [ "log", "pin-project-lite", @@ -3701,9 +3796,9 @@ dependencies = [ [[package]] name = "tracing-attributes" -version = "0.1.27" +version = "0.1.28" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "34704c8d6ebcbc939824180af020566b01a7c01f80641264eba0999f6c2b6be7" +checksum = "395ae124c09f9e6918a2310af6038fba074bcf474ac352496d5910dd59a2226d" dependencies = [ "proc-macro2", "quote", @@ -3712,9 +3807,9 @@ dependencies = [ [[package]] name = "tracing-core" -version = "0.1.32" +version = "0.1.33" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "c06d3da6113f116aaee68e4d601191614c9053067f9ab7f6edbcb161237daa54" +checksum = "e672c95779cf947c5311f83787af4fa8fffd12fb27e4993211a84bdfd9610f9c" dependencies = [ "once_cell", "valuable", @@ -3722,9 +3817,9 @@ dependencies = [ [[package]] name = "tracing-error" -version = "0.2.0" +version = "0.2.1" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "d686ec1c0f384b1277f097b2f279a2ecc11afe8c133c1aabf036a27cb4cd206e" +checksum = "8b1581020d7a273442f5b45074a6a57d5757ad0a47dac0e9f0bd57b81936f3db" dependencies = [ "tracing", "tracing-subscriber", @@ -3740,15 +3835,33 @@ dependencies = [ "tracing", ] +[[package]] +name = "tracing-log" +version = "0.2.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "ee855f1f400bd0e5c02d150ae5de3840039a3f54b025156404e34c23c03f47c3" +dependencies = [ + "log", + "once_cell", + "tracing-core", +] + [[package]] name = "tracing-subscriber" -version = "0.3.18" +version = "0.3.19" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "ad0f048c97dbd9faa9b7df56362b8ebcaa52adb06b498c050d2f4e32f90a7a8b" +checksum = "e8189decb5ac0fa7bc8b96b7cb9b2701d60d48805aca84a238004d665fcc4008" dependencies = [ + "matchers", + "nu-ansi-term", + "once_cell", + "regex", "sharded-slab", + "smallvec", "thread_local", + "tracing", "tracing-core", + "tracing-log", ] [[package]] @@ -3830,9 +3943,9 @@ checksum = "8ecb6da28b8a351d773b68d5825ac39017e680750f980f3a1a85cd8dd28a47c1" [[package]] name = "ureq" -version = "2.10.1" +version = "2.12.1" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "b74fc6b57825be3373f7054754755f03ac3a8f5d70015ccad699ba2029956f4a" +checksum = "02d1a66277ed75f640d608235660df48c8e3c19f3b4edb6a263315626cc3c01d" dependencies = [ "base64", "flate2", @@ -3915,9 +4028,9 @@ checksum = "9c8d87e72b64a3b4db28d11ce29237c246188f4f51057d65a7eab63b7987e423" [[package]] name = "wasm-bindgen" -version = "0.2.97" +version = "0.2.99" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "d15e63b4482863c109d70a7b8706c1e364eb6ea449b201a76c5b89cedcec2d5c" +checksum = "a474f6281d1d70c17ae7aa6a613c87fce69a127e2624002df63dcb39d6cf6396" dependencies = [ "cfg-if", "once_cell", @@ -3926,13 +4039,12 @@ dependencies = [ [[package]] name = "wasm-bindgen-backend" -version = "0.2.97" +version = "0.2.99" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "8d36ef12e3aaca16ddd3f67922bc63e48e953f126de60bd33ccc0101ef9998cd" +checksum = "5f89bb38646b4f81674e8f5c3fb81b562be1fd936d84320f3264486418519c79" dependencies = [ "bumpalo", "log", - "once_cell", "proc-macro2", "quote", "syn", @@ -3941,9 +4053,9 @@ dependencies = [ [[package]] name = "wasm-bindgen-futures" -version = "0.4.47" +version = "0.4.49" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "9dfaf8f50e5f293737ee323940c7d8b08a66a95a419223d9f41610ca08b0833d" +checksum = "38176d9b44ea84e9184eff0bc34cc167ed044f816accfe5922e54d84cf48eca2" dependencies = [ "cfg-if", "js-sys", @@ -3954,9 +4066,9 @@ dependencies = [ [[package]] name = "wasm-bindgen-macro" -version = "0.2.97" +version = "0.2.99" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "705440e08b42d3e4b36de7d66c944be628d579796b8090bfa3471478a2260051" +checksum = "2cc6181fd9a7492eef6fef1f33961e3695e4579b9872a6f7c83aee556666d4fe" dependencies = [ "quote", "wasm-bindgen-macro-support", @@ -3964,9 +4076,9 @@ dependencies = [ [[package]] name = "wasm-bindgen-macro-support" -version = "0.2.97" +version = "0.2.99" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "98c9ae5a76e46f4deecd0f0255cc223cfa18dc9b261213b8aa0c7b36f61b3f1d" +checksum = "30d7a95b763d3c45903ed6c81f156801839e5ee968bb07e534c44df0fcd330c2" dependencies = [ "proc-macro2", "quote", @@ -3977,19 +4089,18 @@ dependencies = [ [[package]] name = "wasm-bindgen-shared" -version = "0.2.97" +version = "0.2.99" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "6ee99da9c5ba11bd675621338ef6fa52296b76b83305e9b6e5c77d4c286d6d49" +checksum = "943aab3fdaaa029a6e0271b35ea10b72b943135afe9bffca82384098ad0e06a6" [[package]] name = "wasm-bindgen-test" -version = "0.3.47" +version = "0.3.49" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "3d919bb60ebcecb9160afee6c71b43a58a4f0517a2de0054cd050d02cec08201" +checksum = "c61d44563646eb934577f2772656c7ad5e9c90fac78aa8013d776fcdaf24625d" dependencies = [ "js-sys", "minicov", - "once_cell", "scoped-tls", "wasm-bindgen", "wasm-bindgen-futures", @@ -3998,9 +4109,9 @@ dependencies = [ [[package]] name = "wasm-bindgen-test-macro" -version = "0.3.47" +version = "0.3.49" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "222ebde6ea87fbfa6bdd2e9f1fd8a91d60aee5db68792632176c4e16a74fc7d8" +checksum = "54171416ce73aa0b9c377b51cc3cb542becee1cd678204812e8392e5b0e4a031" dependencies = [ "proc-macro2", "quote", @@ -4049,9 +4160,9 @@ dependencies = [ [[package]] name = "web-sys" -version = "0.3.74" +version = "0.3.76" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "a98bc3c33f0fe7e59ad7cd041b89034fa82a7c2d4365ca538dda6cdaf513863c" +checksum = "04dd7223427d52553d3702c004d3b2fe07c148165faa56313cb00211e31c12bc" dependencies = [ "js-sys", "wasm-bindgen", @@ -4107,6 +4218,15 @@ version = "0.4.0" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "712e227841d057c1ee1cd2fb22fa7e5a5461ae8e48fa2ca79ec42cfc1931183f" +[[package]] +name = "windows" +version = "0.48.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "e686886bc078bc1b0b600cac0147aadb815089b6e4da64016cbd754b6342700f" +dependencies = [ + "windows-targets 0.48.5", +] + [[package]] name = "windows-core" version = "0.52.0" @@ -4266,9 +4386,9 @@ checksum = "589f6da84c646204747d1270a2a5661ea66ed1cced2631d546fdfb155959f9ec" [[package]] name = "winnow" -version = "0.6.20" +version = "0.6.22" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "36c1fec1a2bb5866f07c25f68c26e565c4c200aebb96d7e55710c19d3e8ac49b" +checksum = "39281189af81c07ec09db316b302a3e67bf9bd7cbf6c820b50e35fee9c2fa980" dependencies = [ "memchr", ] @@ -4350,9 +4470,9 @@ dependencies = [ [[package]] name = "zerofrom-derive" -version = "0.1.4" +version = "0.1.5" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "0ea7b4a3637ea8669cedf0f1fd5c286a17f3de97b8dd5a70a6c167a1730e63a5" +checksum = "595eed982f7d355beb85837f651fa22e90b3c044842dc7f2c2842c086f295808" dependencies = [ "proc-macro2", "quote", diff --git a/Cargo.toml b/Cargo.toml index 8a4bf43b939..459aa6623de 100644 --- a/Cargo.toml +++ b/Cargo.toml @@ -86,7 +86,6 @@ getrandom = { version = "0.2.15", default-features = false } console_error_panic_hook = "0.1.7" wasm-bindgen-test = "0.3.47" smol = "2.0.2" -futures-util = "0.3.31" isahc = "1.7.2" rustyline = { version = "15.0.0", default-features = false } dhat = "0.3.3" @@ -120,6 +119,9 @@ futures-lite = "2.5.0" test-case = "3.3.1" winapi = { version = "0.3.9", default-features = false } url = "2.5.4" +tokio = { version = "1.42.0", default-features = false } +futures-concurrency = "7.6.2" + # ICU4X diff --git a/examples/Cargo.toml b/examples/Cargo.toml index e1c4e17b19f..f9a3990ce17 100644 --- a/examples/Cargo.toml +++ b/examples/Cargo.toml @@ -18,8 +18,9 @@ boa_parser.workspace = true boa_runtime.workspace = true time.workspace = true smol.workspace = true -futures-util.workspace = true +futures-concurrency.workspace = true isahc.workspace = true +tokio = { workspace = true, features = ["rt", "rt-multi-thread", "time", "macros"] } # use explicit lints for examples, since we don't need to lint for docs [lints.rust] diff --git a/examples/src/bin/futures.rs b/examples/src/bin/futures.rs deleted file mode 100644 index c90217a99fc..00000000000 --- a/examples/src/bin/futures.rs +++ /dev/null @@ -1,200 +0,0 @@ -use std::{ - cell::{Cell, RefCell}, - collections::VecDeque, - rc::Rc, - time::{Duration, Instant}, -}; - -use boa_engine::{ - context::ContextBuilder, - job::{FutureJob, JobQueue, NativeJob}, - js_string, - native_function::NativeFunction, - property::Attribute, - Context, JsArgs, JsResult, JsValue, Source, -}; -use boa_runtime::Console; -use futures_util::{stream::FuturesUnordered, Future}; -use smol::{future, stream::StreamExt, LocalExecutor}; - -/// An event queue that also drives futures to completion. -struct Queue<'a> { - executor: LocalExecutor<'a>, - futures: RefCell>, - jobs: RefCell>, -} - -impl<'a> Queue<'a> { - fn new(executor: LocalExecutor<'a>) -> Self { - Self { - executor, - futures: RefCell::default(), - jobs: RefCell::default(), - } - } -} - -impl JobQueue for Queue<'_> { - fn enqueue_promise_job(&self, job: NativeJob, _context: &mut Context) { - self.jobs.borrow_mut().push_back(job); - } - - fn enqueue_future_job(&self, future: FutureJob, _context: &mut Context) { - self.futures.borrow().push(future); - } - - fn run_jobs(&self, context: &mut Context) { - // Early return in case there were no jobs scheduled. - if self.jobs.borrow().is_empty() && self.futures.borrow().is_empty() { - return; - } - - let context = RefCell::new(context); - - future::block_on(self.executor.run(async move { - // Used to sync the finalization of both tasks - let finished = Cell::new(0b00u8); - - let fut_queue = async { - loop { - if self.futures.borrow().is_empty() { - finished.set(finished.get() | 0b01); - if finished.get() >= 0b11 { - // All possible futures and jobs were completed. Exit. - return; - } - // All possible jobs were completed, but `jqueue` could have - // pending jobs. Yield to the executor to try to progress on - // `jqueue` until we have more pending futures. - future::yield_now().await; - continue; - } - finished.set(finished.get() & 0b10); - - // Blocks on all the enqueued futures, driving them all to completion. - let futures = &mut std::mem::take(&mut *self.futures.borrow_mut()); - while let Some(job) = futures.next().await { - // Important to schedule the returned `job` into the job queue, since that's - // what allows updating the `Promise` seen by ECMAScript for when the future - // completes. - self.enqueue_promise_job(job, &mut context.borrow_mut()); - } - } - }; - - let job_queue = async { - loop { - if self.jobs.borrow().is_empty() { - finished.set(finished.get() | 0b10); - if finished.get() >= 0b11 { - // All possible futures and jobs were completed. Exit. - return; - } - // All possible jobs were completed, but `fqueue` could have - // pending futures. Yield to the executor to try to progress on - // `fqueue` until we have more pending jobs. - future::yield_now().await; - continue; - }; - finished.set(finished.get() & 0b01); - - let jobs = std::mem::take(&mut *self.jobs.borrow_mut()); - for job in jobs { - if let Err(e) = job.call(&mut context.borrow_mut()) { - eprintln!("Uncaught {e}"); - } - future::yield_now().await; - } - } - }; - - // Wait for both queues to complete - future::zip(fut_queue, job_queue).await; - })); - } -} - -// Example async code. Note that the returned future must be 'static. -fn delay( - _this: &JsValue, - args: &[JsValue], - context: &mut Context, -) -> impl Future> { - let millis = args.get_or_undefined(0).to_u32(context); - - async move { - let millis = millis?; - println!("Delaying for {millis} milliseconds ..."); - let now = Instant::now(); - smol::Timer::after(Duration::from_millis(u64::from(millis))).await; - let elapsed = now.elapsed().as_secs_f64(); - Ok(elapsed.into()) - } -} - -/// Adds the custom runtime to the context. -fn add_runtime(context: &mut Context) { - // First add the `console` object, to be able to call `console.log()`. - let console = Console::init(context); - context - .register_global_property(Console::NAME, console, Attribute::all()) - .expect("the console builtin shouldn't exist"); - - // Then, bind the defined async function to the ECMAScript function "delay". - context - .register_global_builtin_callable( - js_string!("delay"), - 1, - NativeFunction::from_async_fn(delay), - ) - .expect("the delay builtin shouldn't exist"); -} - -fn main() { - // Initialize the required executors and the context - let executor = LocalExecutor::new(); - let queue = Queue::new(executor); - let context = &mut ContextBuilder::new() - .job_queue(Rc::new(queue)) - .build() - .unwrap(); - - // Then, add a custom runtime. - add_runtime(context); - - // Multiple calls to multiple async timers. - let script = r" - function print(elapsed) { - console.log(`Finished. elapsed time: ${elapsed * 1000} ms`) - } - delay(1000).then(print); - delay(500).then(print); - delay(200).then(print); - delay(600).then(print); - delay(30).then(print); - "; - - let now = Instant::now(); - context.eval(Source::from_bytes(script)).unwrap(); - - // Important to run this after evaluating, since this is what triggers to run the enqueued jobs. - context.run_jobs(); - - println!("Total elapsed time: {:?}", now.elapsed()); - - // Example output: - - // Delaying for 1000 milliseconds ... - // Delaying for 500 milliseconds ... - // Delaying for 200 milliseconds ... - // Delaying for 600 milliseconds ... - // Delaying for 30 milliseconds ... - // Finished. elapsed time: 30.073821000000002 ms - // Finished. elapsed time: 200.079116 ms - // Finished. elapsed time: 500.10745099999997 ms - // Finished. elapsed time: 600.098433 ms - // Finished. elapsed time: 1000.118099 ms - // Total elapsed time: 1.002628715s - - // The queue concurrently drove several timers to completion! -} diff --git a/examples/src/bin/module_fetch.rs b/examples/src/bin/module_fetch_async.rs similarity index 66% rename from examples/src/bin/module_fetch.rs rename to examples/src/bin/module_fetch_async.rs index 991d58fd0ed..6f9b2a7b09d 100644 --- a/examples/src/bin/module_fetch.rs +++ b/examples/src/bin/module_fetch_async.rs @@ -1,8 +1,4 @@ -use std::{ - cell::{Cell, RefCell}, - collections::VecDeque, - rc::Rc, -}; +use std::{cell::RefCell, collections::VecDeque, future::Future, pin::Pin, rc::Rc}; use boa_engine::{ builtins::promise::PromiseState, @@ -12,15 +8,12 @@ use boa_engine::{ Context, JsNativeError, JsResult, JsString, JsValue, Module, }; use boa_parser::Source; -use futures_util::{stream::FuturesUnordered, StreamExt}; +use futures_concurrency::future::FutureGroup; use isahc::{ config::{Configurable, RedirectPolicy}, AsyncReadResponseExt, Request, RequestExt, }; -use smol::{future, LocalExecutor}; - -// Most of the boilerplate is taken from the `futures.rs` example. -// This file only explains what is exclusive of async module loading. +use smol::{future, stream::StreamExt}; #[derive(Debug, Default)] struct HttpModuleLoader; @@ -115,9 +108,8 @@ fn main() -> JsResult<()> { export default result; "#; - let queue = Rc::new(Queue::new(LocalExecutor::new())); let context = &mut Context::builder() - .job_queue(queue) + .job_queue(Rc::new(Queue::new())) // NEW: sets the context module loader to our custom loader .module_loader(Rc::new(HttpModuleLoader)) .build()?; @@ -175,94 +167,93 @@ fn main() -> JsResult<()> { Ok(()) } -// Taken from the `futures.rs` example. -struct Queue<'a> { - executor: LocalExecutor<'a>, - futures: RefCell>, +// Taken from the `smol_event_loop.rs` example. +/// An event queue using smol to drive futures to completion. +struct Queue { + futures: RefCell>, jobs: RefCell>, } -impl<'a> Queue<'a> { - fn new(executor: LocalExecutor<'a>) -> Self { +impl Queue { + fn new() -> Self { Self { - executor, futures: RefCell::default(), jobs: RefCell::default(), } } + + fn drain_jobs(&self, context: &mut Context) { + let jobs = std::mem::take(&mut *self.jobs.borrow_mut()); + for job in jobs { + if let Err(e) = job.call(context) { + eprintln!("Uncaught {e}"); + } + } + } } -impl JobQueue for Queue<'_> { +impl JobQueue for Queue { fn enqueue_promise_job(&self, job: NativeJob, _context: &mut Context) { self.jobs.borrow_mut().push_back(job); } fn enqueue_future_job(&self, future: FutureJob, _context: &mut Context) { - self.futures.borrow().push(future) + self.futures.borrow_mut().push(future); } + // While the sync flavor of `run_jobs` will block the current thread until all the jobs have finished... fn run_jobs(&self, context: &mut Context) { - // Early return in case there were no jobs scheduled. - if self.jobs.borrow().is_empty() && self.futures.borrow().is_empty() { - return; - } - - let context = RefCell::new(context); - - future::block_on(self.executor.run(async move { - // Used to sync the finalization of both tasks - let finished = Cell::new(0b00u8); - - let fqueue = async { - loop { - if self.futures.borrow().is_empty() { - finished.set(finished.get() | 0b01); - if finished.get() >= 0b11 { - // All possible futures and jobs were completed. Exit. - return; - } - // All possible jobs were completed, but `jqueue` could have - // pending jobs. Yield to the executor to try to progress on - // `jqueue` until we have more pending futures. - future::yield_now().await; - continue; - } - finished.set(finished.get() & 0b10); + smol::block_on(smol::LocalExecutor::new().run(self.run_jobs_async(context))); + } - let futures = &mut std::mem::take(&mut *self.futures.borrow_mut()); - while let Some(job) = futures.next().await { - self.enqueue_promise_job(job, &mut context.borrow_mut()); - } - } - }; - - let jqueue = async { - loop { - if self.jobs.borrow().is_empty() { - finished.set(finished.get() | 0b10); - if finished.get() >= 0b11 { - // All possible futures and jobs were completed. Exit. - return; - } - // All possible jobs were completed, but `fqueue` could have - // pending futures. Yield to the executor to try to progress on - // `fqueue` until we have more pending jobs. - future::yield_now().await; - continue; + // ...the async flavor won't, which allows concurrent execution with external async tasks. + fn run_jobs_async<'a, 'ctx, 'fut>( + &'a self, + context: &'ctx mut Context, + ) -> Pin + 'fut>> + where + 'a: 'fut, + 'ctx: 'fut, + { + Box::pin(async move { + // Early return in case there were no jobs scheduled. + if self.jobs.borrow().is_empty() && self.futures.borrow().is_empty() { + return; + } + let mut group = FutureGroup::new(); + loop { + group.extend(std::mem::take(&mut *self.futures.borrow_mut())); + + if self.jobs.borrow().is_empty() { + let Some(job) = group.next().await else { + // Both queues are empty. We can exit. + return; }; - finished.set(finished.get() & 0b01); - - let jobs = std::mem::take(&mut *self.jobs.borrow_mut()); - for job in jobs { - if let Err(e) = job.call(&mut context.borrow_mut()) { - eprintln!("Uncaught {e}"); - } - future::yield_now().await; - } + + // Important to schedule the returned `job` into the job queue, since that's + // what allows updating the `Promise` seen by ECMAScript for when the future + // completes. + self.enqueue_promise_job(job, context); + continue; } - }; - future::zip(fqueue, jqueue).await; - })) + // We have some jobs pending on the microtask queue. Try to poll the pending + // tasks once to see if any of them finished, and run the pending microtasks + // otherwise. + let Some(job) = future::poll_once(group.next()).await.flatten() else { + // No completed jobs. Run the microtask queue once. + self.drain_jobs(context); + continue; + }; + + // Important to schedule the returned `job` into the job queue, since that's + // what allows updating the `Promise` seen by ECMAScript for when the future + // completes. + self.enqueue_promise_job(job, context); + + // Only one macrotask can be executed before the next drain of the microtask queue. + self.drain_jobs(context); + } + }) } } diff --git a/examples/src/bin/smol_event_loop.rs b/examples/src/bin/smol_event_loop.rs new file mode 100644 index 00000000000..45b9ade8914 --- /dev/null +++ b/examples/src/bin/smol_event_loop.rs @@ -0,0 +1,250 @@ +use std::{ + cell::RefCell, + collections::VecDeque, + future::Future, + rc::Rc, + time::{Duration, Instant}, +}; + +use boa_engine::{ + context::ContextBuilder, + job::{FutureJob, JobQueue, NativeJob}, + js_string, + native_function::NativeFunction, + property::Attribute, + Context, JsArgs, JsResult, JsValue, Script, Source, +}; +use boa_runtime::Console; +use futures_concurrency::future::FutureGroup; +use smol::{future, stream::StreamExt}; + +// This example shows how to create an event loop using the smol runtime. +// The example contains two "flavors" of event loops: +fn main() { + // An internally async event loop. This event loop blocks the execution of the thread + // while executing tasks, but internally uses async to run its tasks. + internally_async_event_loop(); + + // An externally async event loop. This event loop can yield to the runtime to concurrently + // run tasks with it. + externally_async_event_loop(); +} + +/// An event queue using smol to drive futures to completion. +struct Queue { + futures: RefCell>, + jobs: RefCell>, +} + +impl Queue { + fn new() -> Self { + Self { + futures: RefCell::default(), + jobs: RefCell::default(), + } + } + + fn drain_jobs(&self, context: &mut Context) { + let jobs = std::mem::take(&mut *self.jobs.borrow_mut()); + for job in jobs { + if let Err(e) = job.call(context) { + eprintln!("Uncaught {e}"); + } + } + } +} + +impl JobQueue for Queue { + fn enqueue_promise_job(&self, job: NativeJob, _context: &mut Context) { + self.jobs.borrow_mut().push_back(job); + } + + fn enqueue_future_job(&self, future: FutureJob, _context: &mut Context) { + self.futures.borrow_mut().push(future); + } + + // While the sync flavor of `run_jobs` will block the current thread until all the jobs have finished... + fn run_jobs(&self, context: &mut Context) { + smol::block_on(smol::LocalExecutor::new().run(self.run_jobs_async(context))); + } + + // ...the async flavor won't, which allows concurrent execution with external async tasks. + fn run_jobs_async<'a, 'ctx, 'fut>( + &'a self, + context: &'ctx mut Context, + ) -> std::pin::Pin + 'fut>> + where + 'a: 'fut, + 'ctx: 'fut, + { + Box::pin(async move { + // Early return in case there were no jobs scheduled. + if self.jobs.borrow().is_empty() && self.futures.borrow().is_empty() { + return; + } + let mut group = FutureGroup::new(); + loop { + group.extend(std::mem::take(&mut *self.futures.borrow_mut())); + + if self.jobs.borrow().is_empty() { + let Some(job) = group.next().await else { + // Both queues are empty. We can exit. + return; + }; + + // Important to schedule the returned `job` into the job queue, since that's + // what allows updating the `Promise` seen by ECMAScript for when the future + // completes. + self.enqueue_promise_job(job, context); + continue; + } + + // We have some jobs pending on the microtask queue. Try to poll the pending + // tasks once to see if any of them finished, and run the pending microtasks + // otherwise. + let Some(job) = future::poll_once(group.next()).await.flatten() else { + // No completed jobs. Run the microtask queue once. + self.drain_jobs(context); + continue; + }; + + // Important to schedule the returned `job` into the job queue, since that's + // what allows updating the `Promise` seen by ECMAScript for when the future + // completes. + self.enqueue_promise_job(job, context); + + // Only one macrotask can be executed before the next drain of the microtask queue. + self.drain_jobs(context); + } + }) + } +} + +// Example async function. Note that the returned future must be 'static. +fn delay( + _this: &JsValue, + args: &[JsValue], + context: &mut Context, +) -> impl Future> { + let millis = args.get_or_undefined(0).to_u32(context); + + async move { + let millis = millis?; + println!("Delaying for {millis} milliseconds ..."); + let now = Instant::now(); + smol::Timer::after(Duration::from_millis(u64::from(millis))).await; + let elapsed = now.elapsed().as_secs_f64(); + Ok(elapsed.into()) + } +} + +/// Adds the custom runtime to the context. +fn add_runtime(context: &mut Context) { + // First add the `console` object, to be able to call `console.log()`. + let console = Console::init(context); + context + .register_global_property(Console::NAME, console, Attribute::all()) + .expect("the console builtin shouldn't exist"); + + // Then, bind the defined async function to the ECMAScript function "delay". + context + .register_global_builtin_callable( + js_string!("delay"), + 1, + NativeFunction::from_async_fn(delay), + ) + .expect("the delay builtin shouldn't exist"); +} + +// Script that does multiple calls to multiple async timers. +const SCRIPT: &str = r" + function print(elapsed) { + console.log(`Finished delay. Elapsed time: ${elapsed * 1000} ms`) + } + delay(1000).then(print); + delay(500).then(print); + delay(200).then(print); + delay(600).then(print); + delay(30).then(print); + + for(let i = 0; i <= 100000; i++) { + // Emulate a long-running evaluation of a script. + } +"; + +// This flavor is most recommended when you have an application that: +// - Needs to wait until the engine finishes executing; depends on the execution result to continue. +// - Delegates the execution of the application to the engine's event loop. +fn internally_async_event_loop() { + println!("====== Internally async event loop. ======"); + + // Initialize the queue and the context + let queue = Queue::new(); + let context = &mut ContextBuilder::new() + .job_queue(Rc::new(queue)) + .build() + .unwrap(); + + // Then, add the custom runtime. + add_runtime(context); + + let now = Instant::now(); + println!("Evaluating script..."); + context.eval(Source::from_bytes(SCRIPT)).unwrap(); + + // Important to run this after evaluating, since this is what triggers to run the enqueued jobs. + println!("Running jobs..."); + context.run_jobs(); + + println!("Total elapsed time: {:?}\n", now.elapsed()); +} + +// This flavor is most recommended when you have an application that: +// - Cannot afford to block until the engine finishes executing. +// - Needs to process IO requests between executions that will be consumed by the engine. +fn externally_async_event_loop() { + println!("====== Externally async event loop. ======"); + let executor = smol::Executor::new(); + + smol::block_on(executor.run(async { + // Initialize the queue and the context + let queue = Queue::new(); + let context = &mut ContextBuilder::new() + .job_queue(Rc::new(queue)) + .build() + .unwrap(); + + // Then, add the custom runtime. + add_runtime(context); + + let now = Instant::now(); + + // Example of an asynchronous workload that must be run alongside the engine. + let counter = executor.spawn(async { + let mut interval = smol::Timer::interval(Duration::from_millis(100)); + println!("Starting smol interval job..."); + for i in 0..10 { + interval.next().await; + println!("Executed interval tick {i}"); + } + println!("Finished smol interval job...") + }); + + let engine = async { + let script = Script::parse(Source::from_bytes(SCRIPT), None, context).unwrap(); + + // `Script::evaluate_async` will yield to the executor from time to time, Unlike `Context::run` + // or `Script::evaluate` which block the current thread until the execution finishes. + println!("Evaluating script..."); + script.evaluate_async(context).await.unwrap(); + + // Run the jobs asynchronously, which avoids blocking the main thread. + println!("Running jobs..."); + context.run_jobs_async().await; + }; + + future::zip(counter, engine).await; + + println!("Total elapsed time: {:?}\n", now.elapsed()); + })); +} diff --git a/examples/src/bin/tokio_event_loop.rs b/examples/src/bin/tokio_event_loop.rs new file mode 100644 index 00000000000..c6e0cd248d2 --- /dev/null +++ b/examples/src/bin/tokio_event_loop.rs @@ -0,0 +1,265 @@ +use std::{ + cell::RefCell, + collections::VecDeque, + future::Future, + pin::Pin, + rc::Rc, + time::{Duration, Instant}, +}; + +use boa_engine::{ + context::ContextBuilder, + job::{FutureJob, JobQueue, NativeJob}, + js_string, + native_function::NativeFunction, + property::Attribute, + Context, JsArgs, JsResult, JsValue, Script, Source, +}; +use boa_runtime::Console; +use tokio::{task, time}; + +// This example shows how to create an event loop using the tokio runtime. +// The example contains two "flavors" of event loops: +fn main() { + // An internally async event loop. This event loop blocks the execution of the thread + // while executing tasks, but internally uses async to run its tasks. + internally_async_event_loop(); + + // An externally async event loop. This event loop can yield to the runtime to concurrently + // run tasks with it. + externally_async_event_loop(); +} + +/// An event queue using tokio to drive futures to completion. +struct Queue { + futures: RefCell>, + jobs: RefCell>, +} + +impl Queue { + fn new() -> Self { + Self { + futures: RefCell::default(), + jobs: RefCell::default(), + } + } + + fn drain_jobs(&self, context: &mut Context) { + let jobs = std::mem::take(&mut *self.jobs.borrow_mut()); + for job in jobs { + if let Err(e) = job.call(context) { + eprintln!("Uncaught {e}"); + } + } + } +} + +impl JobQueue for Queue { + fn enqueue_promise_job(&self, job: NativeJob, _context: &mut Context) { + self.jobs.borrow_mut().push_back(job); + } + + fn enqueue_future_job(&self, future: FutureJob, _context: &mut Context) { + self.futures.borrow_mut().push(future); + } + + // While the sync flavor of `run_jobs` will block the current thread until all the jobs have finished... + fn run_jobs(&self, context: &mut Context) { + let runtime = tokio::runtime::Builder::new_current_thread() + .enable_time() + .build() + .unwrap(); + + task::LocalSet::default().block_on(&runtime, self.run_jobs_async(context)); + } + + // ...the async flavor won't, which allows concurrent execution with external async tasks. + fn run_jobs_async<'a, 'ctx, 'fut>( + &'a self, + context: &'ctx mut Context, + ) -> Pin + 'fut>> + where + 'a: 'fut, + 'ctx: 'fut, + { + Box::pin(async move { + // Early return in case there were no jobs scheduled. + if self.jobs.borrow().is_empty() && self.futures.borrow().is_empty() { + return; + } + let mut join_set = task::JoinSet::new(); + loop { + for future in std::mem::take(&mut *self.futures.borrow_mut()) { + join_set.spawn_local(future); + } + + if self.jobs.borrow().is_empty() { + let Some(job) = join_set.join_next().await else { + // Both queues are empty. We can exit. + return; + }; + + // Important to schedule the returned `job` into the job queue, since that's + // what allows updating the `Promise` seen by ECMAScript for when the future + // completes. + match job { + Ok(job) => self.enqueue_promise_job(job, context), + Err(e) => eprintln!("{e}"), + } + + continue; + } + + // We have some jobs pending on the microtask queue. Try to poll the pending + // tasks once to see if any of them finished, and run the pending microtasks + // otherwise. + let Some(job) = join_set.try_join_next() else { + // No completed jobs. Run the microtask queue once. + self.drain_jobs(context); + + task::yield_now().await; + continue; + }; + + // Important to schedule the returned `job` into the job queue, since that's + // what allows updating the `Promise` seen by ECMAScript for when the future + // completes. + match job { + Ok(job) => self.enqueue_promise_job(job, context), + Err(e) => eprintln!("{e}"), + } + + // Only one macrotask can be executed before the next drain of the microtask queue. + self.drain_jobs(context); + } + }) + } +} + +// Example async function. Note that the returned future must be 'static. +fn delay( + _this: &JsValue, + args: &[JsValue], + context: &mut Context, +) -> impl Future> { + let millis = args.get_or_undefined(0).to_u32(context); + + async move { + let millis = millis?; + println!("Delaying for {millis} milliseconds ..."); + let now = Instant::now(); + time::sleep(Duration::from_millis(u64::from(millis))).await; + let elapsed = now.elapsed().as_secs_f64(); + Ok(elapsed.into()) + } +} + +/// Adds the custom runtime to the context. +fn add_runtime(context: &mut Context) { + // First add the `console` object, to be able to call `console.log()`. + let console = Console::init(context); + context + .register_global_property(Console::NAME, console, Attribute::all()) + .expect("the console builtin shouldn't exist"); + + // Then, bind the defined async function to the ECMAScript function "delay". + context + .register_global_builtin_callable( + js_string!("delay"), + 1, + NativeFunction::from_async_fn(delay), + ) + .expect("the delay builtin shouldn't exist"); +} + +// Script that does multiple calls to multiple async timers. +const SCRIPT: &str = r" + function print(elapsed) { + console.log(`Finished delay. Elapsed time: ${elapsed * 1000} ms`) + } + delay(1000).then(print); + delay(500).then(print); + delay(200).then(print); + delay(600).then(print); + delay(30).then(print); + + for(let i = 0; i <= 100000; i++) { + // Emulate a long-running evaluation of a script. + } +"; + +// This flavor is most recommended when you have an application that: +// - Needs to wait until the engine finishes executing; depends on the execution result to continue. +// - Delegates the execution of the application to the engine's event loop. +fn internally_async_event_loop() { + println!("====== Internally async event loop. ======"); + + // Initialize the queue and the context + let queue = Queue::new(); + let context = &mut ContextBuilder::new() + .job_queue(Rc::new(queue)) + .build() + .unwrap(); + + // Then, add the custom runtime. + add_runtime(context); + + let now = Instant::now(); + println!("Evaluating script..."); + context.eval(Source::from_bytes(SCRIPT)).unwrap(); + + // Important to run this after evaluating, since this is what triggers to run the enqueued jobs. + println!("Running jobs..."); + context.run_jobs(); + + println!("Total elapsed time: {:?}\n", now.elapsed()); +} + +// This flavor is most recommended when you have an application that: +// - Cannot afford to block until the engine finishes executing. +// - Needs to process IO requests between executions that will be consumed by the engine. +#[tokio::main] +async fn externally_async_event_loop() { + println!("====== Externally async event loop. ======"); + // Initialize the queue and the context + let queue = Queue::new(); + let context = &mut ContextBuilder::new() + .job_queue(Rc::new(queue)) + .build() + .unwrap(); + + // Then, add the custom runtime. + add_runtime(context); + + let now = Instant::now(); + + // Example of an asynchronous workload that must be run alongside the engine. + let counter = tokio::spawn(async { + let mut interval = time::interval(Duration::from_millis(100)); + println!("Starting tokio interval job..."); + for i in 0..10 { + interval.tick().await; + println!("Executed interval tick {i}"); + } + println!("Finished tokio interval job...") + }); + + let local_set = &mut task::LocalSet::default(); + let engine = local_set.run_until(async { + let script = Script::parse(Source::from_bytes(SCRIPT), None, context).unwrap(); + + // `Script::evaluate_async` will yield to the executor from time to time, Unlike `Context::run` + // or `Script::evaluate` which block the current thread until the execution finishes. + println!("Evaluating script..."); + script.evaluate_async(context).await.unwrap(); + + // Run the jobs asynchronously, which avoids blocking the main thread. + println!("Running jobs..."); + context.run_jobs_async().await; + Ok(()) + }); + + tokio::try_join!(counter, engine).unwrap(); + + println!("Total elapsed time: {:?}\n", now.elapsed()); +} diff --git a/tests/macros/tests/derive.rs b/tests/macros/tests/derive.rs index fd786e7e1f9..1798df9ee86 100644 --- a/tests/macros/tests/derive.rs +++ b/tests/macros/tests/derive.rs @@ -5,6 +5,5 @@ #[test] fn try_from_js() { let t = trybuild::TestCases::new(); - t.pass("tests/derive/simple_struct.rs"); - t.pass("tests/derive/from_js_with.rs"); + t.pass("tests/derive/*.rs"); }