Note: users of v0.1.0-v0.1.3 should upgrade to v0.1.4 if they're using Rust 2021, as an upstream bug occurs if you attempt to compile your app in release mode. See here for further details.
This is a very simple plugin for Perseus that applies size optimizations automatically, which decrease the size of your final Wasm bundle significantly, meaning faster loads for users because a smaller amount of data needs to be transferred to their browsers. Because Perseus renders a page almost immediately through static generation, and then the Wasm bundle is needed to make that page interactive, applying this plugin will decrease your app's time-to-interactive and its total blocking time (a Lighthouse metric that normal Perseus apps don't do so well in on mobile).
If you're new to Perseus, check it out on it's website and on GitHub! Basically though, it's a really fast and fully-featured web framework for Rust!
In your src/lib.rs
, call the following function on PerseusApp
:
PerseusApp::new()
[...]
.plugins(Plugins::new()
.plugin(
perseus_size_opt,
SizeOpts::default()
))
I'm using `define_app!`
If you're still using define_app!
from v0.3.3, you should upgrade to using PerseusApp
soon, but you can still use this plugin by adding the following to the bottom of the define_app!
call:
plugins: Plugins::new().plugin(perseus_size_opt, SizeOpts::default())
If you have any other plugins defined, add the .plugin()
call where appropriate. You'll also need to add the following imports:
use perseus_size_opt::{perseus_size_opt, SizeOpts};
Once that's done, run perseus tinker
to apply the optimizations (this needs a separate command because they involve modifying the .perseus/
directory), and then you can work with your app as normal!
If you ever want to uninstall the plugin, just remove the relevant .plugin()
call and re-run perseus tinker
, and it'll be completely removed.
This plugin currently performs the following optimizations:
wee_alloc
-- an alternative allocator designed for Wasm that reduces binary size at the expense of slightly slower allocationslto
-- reduces binary size when set totrue
by enabling link-time optimizationsopt-level
-- optimizes aggressively for binary size when set toz
codegen-units
-- makes faster and smaller code when set to lower values (but makes compile times slower in release mode)
Note that all optimizations will only apply to release builds. except for the use of wee_alloc
, which will also affect development builds.
There are a few defaults available for setting size optimization levels, or you can build your own custom settings easily by constructing
SizeOpts
manually.
::default()
-- enables all optimizations::default_no_lto()
-- enables all optimizations exceptlto = true
, because that can break compilation of execution on some hosting providers, like Netlify.::only_wee_alloc()
-- only useswee_alloc
, applying no other optimizations::no_wee_alloc()
-- applies all optimizations other thanwee_alloc
This is due to an upstream issue in Rust 2021 that leads to size optimizations of the fluent-bundle
package causing an overload of LLVM (see this issue). As of v0.1.4, this plugin accounts for this and does not attempt to optimize fluent-bundle
with the default()
options. However, this will increase bundle size, so it's recommended that, until this upstream issue is fixed, users of this plugin remain on Rust 2018 for now and use the default_2018()
function instead. This will optimize fluent-bundle
appropriately while ensuring that this bug does not occur. If moving to Rust 2018 is infeasible, you'll have to put up with slightly larger bundles for now until the upstream issue is fixed (this seems to depend on the LLVM team now).
This plugin is considered quite stable due to how basic its optimizations are (the whole thing is one file), and so its stability is mostly dependent on that of Perseus. If you're happy to use Perseus, you shouldn't need to worry about using this plugin as well (in fact, it's recommended that all Perseus apps use this plugin).
See LICENSE
.