mix volt.build
Building Tailwind CSS...
app-1a2b3c4d.css 23.9 KB
Built Tailwind in 43ms
Building "assets/js/app.ts"...
app-5e6f7a8b.js 128.4 KB
manifest.json 2 entries
Built in 15msReads configuration from config :volt. CLI flags override config values.
What production builds do
Production builds run the same framework/plugin compilation pipeline as the dev server, then apply build-only graph and output steps:
- expands
import.meta.glob()and simple relative dynamic import variables - rewrites
new URL("./asset.ext", import.meta.url)through the asset pipeline - rewrites relative CSS
url(...)asset references through the asset pipeline - copies JavaScript- and CSS-referenced assets with content hashes
- tree-shakes, minifies, and optionally code-splits JavaScript
- writes a manifest that Phoenix can use for digested asset paths and chunk preload metadata
- optionally copies a Vite-style public directory to the static root without transforming files
Public files in Phoenix apps
For Phoenix projects, stable root files usually belong in priv/static and are served by Phoenix through Plug.Static. Examples include favicon.ico, robots.txt, web app manifests, and touch icons. Keep those files at the Phoenix level when possible.
Volt handles files that are part of the frontend module graph instead:
- JavaScript asset imports
new URL("./asset.ext", import.meta.url)references Those graph assets are copied with content hashes and rewritten in production builds. CSS files are parsed and bundled by LightningCSS through Vize, and relative CSSurl(...)references are rewritten through Vize's parser-backed CSS AST API. CSS-referenced emitted assets are listed on the CSS manifest entry.
Optional Vite-style public directory
public_dir is disabled by default. Enable it only when you intentionally want Vite-style public directory behavior, for example during migration from Vite:
config :volt,
public_dir: "public"When enabled, files are copied as-is to the static root. With the default output directory, JavaScript and CSS are written below priv/static/assets, while public files are copied to priv/static:
public/favicon.svg -> priv/static/favicon.svg
public/robots.txt -> priv/static/robots.txt
assets/js/app.ts -> priv/static/assets/js/app-a1b2c3d4.jsReference public files with root-absolute URLs such as /favicon.svg. They are not transformed, hashed, or included in the module graph.
CLI: mix volt.build --public-dir path/to/public.
Asset URL Prefix
Production JavaScript and CSS asset references use /assets by default, matching Phoenix's conventional priv/static/assets mount. Change only the public URL prefix with asset_url_prefix; this does not change the filesystem outdir or Phoenix endpoint/static URL configuration.
config :volt, asset_url_prefix: "/my-app/assets"CLI: mix volt.build --asset-url-prefix /my-app/assets.
Tree Shaking
JavaScript tree shaking is enabled by default for production builds. Disable it only when you need to preserve unused exports or debug bundling output:
config :volt, tree_shaking: falseCLI: mix volt.build --no-tree-shaking.
Source Maps
sourcemap: true— write.mapfiles and append//# sourceMappingURLcomment (default)sourcemap: :hidden— write.mapfiles without the URL comment (for Sentry, Datadog, etc.)sourcemap: false— no source maps
CLI: --sourcemap hidden or --sourcemap false.
External Modules
Exclude packages that the host page already provides:
config :volt, external: ~w(phoenix phoenix_html phoenix_live_view)Or per-build: mix volt.build --external phoenix --external phoenix_html
Module Preloading
For code-split builds, the production manifest records static imports, dynamic imports, chunk-local CSS, and emitted assets. Use Volt.Preload.tags/2 in your layout to preload the entry and its static chunk dependencies:
<%= Volt.Preload.tags("priv/static/assets/js/manifest.json", "/assets/js", entry: "app.js") %>Runtime dynamic imports are rewritten through Volt's preload helper when the async chunk has dependency chunks or CSS. The helper preloads those files before executing import(), avoiding extra round trips while keeping async chunks lazy.
Deploy Alias
The installer generates an assets.deploy alias:
"assets.deploy": ["volt.build --tailwind", "phx.digest"]This builds assets with content hashes, then generates the Phoenix digest manifest for CDN deployment.