SIGN IN SIGN UP
vitejs / vite UNCLAIMED

Next generation frontend tooling. It's fast!

79383 0 1 TypeScript
2021-01-01 16:40:16 -05:00
# Backend Integration
:::tip Note
If you want to serve the HTML using a traditional backend (e.g. Rails, Laravel) but use Vite for serving assets, check for existing integrations listed in [Awesome Vite](https://github.com/vitejs/awesome-vite#integrations-with-backends).
If you need a custom integration, you can follow the steps in this guide to configure it manually.
:::
2021-01-01 16:40:16 -05:00
1. In your Vite config, configure the entry and enable build manifest:
```js twoslash [vite.config.js]
import { defineConfig } from 'vite'
// ---cut---
2021-07-27 17:00:51 +02:00
export default defineConfig({
server: {
cors: {
// the origin you will be accessing via browser
origin: 'http://my-backend.example.com',
},
},
build: {
// generate .vite/manifest.json in outDir
manifest: true,
rollupOptions: {
// overwrite default .html entry
input: '/path/to/main.js',
},
},
2021-07-27 17:00:51 +02:00
})
```
2021-01-01 16:40:16 -05:00
2022-06-19 18:12:59 +09:00
If you haven't disabled the [module preload polyfill](/config/build-options.md#build-polyfillmodulepreload), you also need to import the polyfill in your entry
2021-07-29 07:17:13 +02:00
```js
// add the beginning of your app entry
import 'vite/modulepreload-polyfill'
```
2. For development, inject the following in your server's HTML template (substitute `http://localhost:5173` with the local URL Vite is running at):
2021-01-01 16:40:16 -05:00
```html
<!-- if development -->
<script type="module" src="http://localhost:5173/@vite/client"></script>
<script type="module" src="http://localhost:5173/main.js"></script>
```
2021-01-01 16:40:16 -05:00
In order to properly serve assets, you have two options:
- Make sure the server is configured to proxy static assets requests to the Vite server
2022-06-19 18:12:59 +09:00
- Set [`server.origin`](/config/server-options.md#server-origin) so that generated asset URLs will be resolved using the back-end server URL instead of a relative path
This is needed for assets such as images to load properly.
2021-01-01 16:40:16 -05:00
Note if you are using React with `@vitejs/plugin-react`, you'll also need to add this before the above scripts, since the plugin is not able to modify the HTML you are serving (substitute `http://localhost:5173` with the local URL Vite is running at):
```html
<script type="module">
import RefreshRuntime from 'http://localhost:5173/@react-refresh'
2021-03-31 22:06:16 +02:00
RefreshRuntime.injectIntoGlobalHook(window)
window.$RefreshReg$ = () => {}
window.$RefreshSig$ = () => (type) => type
window.__vite_plugin_react_preamble_installed__ = true
</script>
```
3. For production, after running `vite build`, a `.vite/manifest.json` file will be generated alongside other asset files. An example manifest file looks like this:
```json [.vite/manifest.json] style:max-height:400px
{
2024-05-07 19:16:36 +09:00
"_shared-B7PI925R.js": {
"file": "assets/shared-B7PI925R.js",
"name": "shared",
"css": ["assets/shared-ChJ_j-JJ.css"]
},
"_shared-ChJ_j-JJ.css": {
"file": "assets/shared-ChJ_j-JJ.css",
"src": "_shared-ChJ_j-JJ.css"
},
"logo.svg": {
"file": "assets/logo-BuPIv-2h.svg",
"src": "logo.svg"
},
2024-05-07 19:16:36 +09:00
"baz.js": {
"file": "assets/baz-B2H3sXNv.js",
"name": "baz",
"src": "baz.js",
"isDynamicEntry": true
},
"views/bar.js": {
"file": "assets/bar-gkvgaI9m.js",
"name": "bar",
"src": "views/bar.js",
"isEntry": true,
2024-05-07 19:16:36 +09:00
"imports": ["_shared-B7PI925R.js"],
"dynamicImports": ["baz.js"]
},
"views/foo.js": {
2024-05-07 19:16:36 +09:00
"file": "assets/foo-BRBmoGS9.js",
"name": "foo",
"src": "views/foo.js",
2024-05-07 19:16:36 +09:00
"isEntry": true,
"imports": ["_shared-B7PI925R.js"],
"css": ["assets/foo-5UjPuW-k.css"]
}
}
```
2026-03-04 21:09:26 +09:00
The manifest maps source files to their build outputs and dependencies:
```dot
digraph manifest {
rankdir=TB
node [shape=box style="rounded,filled" fontname="Arial" fontsize=10 margin="0.2,0.1" fontcolor="${#3c3c43|#ffffff}" color="${#c2c2c4|#3c3f44}"]
edge [color="${#67676c|#98989f}" fontname="Arial" fontsize=9 fontcolor="${#67676c|#98989f}"]
bgcolor="transparent"
foo [label="views/foo.js\n(entry)" fillcolor="${#e9eaff|#222541}"]
bar [label="views/bar.js\n(entry)" fillcolor="${#e9eaff|#222541}"]
shared [label="_shared-B7PI925R.js\n(common chunk)" fillcolor="${#f2ecfc|#2c273e}"]
baz [label="baz.js\n(dynamic import)" fillcolor="${#fcf4dc|#38301a}"]
foocss [label="foo.css" shape=ellipse fillcolor="${#fde4e8|#3a1d27}"]
sharedcss [label="shared.css" shape=ellipse fillcolor="${#fde4e8|#3a1d27}"]
logo [label="logo.svg\n(asset)" shape=ellipse fillcolor="${#def5ed|#15312d}"]
foo -> shared [label="imports"]
bar -> shared [label="imports"]
bar -> baz [label="dynamicImports" style=dashed]
foo -> foocss [label="css"]
shared -> sharedcss [label="css"]
}
```
The manifest has a `Record<name, chunk>` structure where each chunk follows the `ManifestChunk` interface:
```ts style:max-height:400px
interface ManifestChunk {
/**
* The input file name of this chunk / asset if known
*/
src?: string
/**
* The output file name of this chunk / asset
*/
file: string
/**
* The list of CSS files imported by this chunk
*/
css?: string[]
/**
* The list of asset files imported by this chunk, excluding CSS files
*/
assets?: string[]
/**
* Whether this chunk or asset is an entry point
*/
isEntry?: boolean
/**
* The name of this chunk / asset if known
*/
name?: string
/**
* Whether this chunk is a dynamic entry point
*
* This field is only present in JS chunks.
*/
isDynamicEntry?: boolean
/**
* The list of statically imported chunks by this chunk
*
* The values are the keys of the manifest. This field is only present in JS chunks.
*/
imports?: string[]
/**
* The list of dynamically imported chunks by this chunk
*
* The values are the keys of the manifest. This field is only present in JS chunks.
*/
dynamicImports?: string[]
}
```
Each entry in the manifest represents one of the following:
- **Entry chunks**: Generated from files specified in [`build.rollupOptions.input`](https://rollupjs.org/configuration-options/#input). These chunks have `isEntry: true` and their key is the relative src path from project root.
- **Dynamic entry chunks**: Generated from dynamic imports. These chunks have `isDynamicEntry: true` and their key is the relative src path from project root.
- **Non-entry chunks**: Their key is the base name of the generated file prefixed with `_`.
- **Asset chunks**: Generated from imported assets like images, fonts. Their key is the relative src path from project root.
- **CSS files**: When [`build.cssCodeSplit`](/config/build-options.md#build-csscodesplit) is `false`, a single CSS file is generated with the key `style.css`. When `build.cssCodeSplit` is not `false`, the key is generated similar to JS chunks (i.e. entry chunks will not have `_` prefix and non-entry chunks will have `_` prefix).
JS chunks (chunks other than assets or CSS) will contain information on their static and dynamic imports (both are keys that map to the corresponding chunk in the manifest). Chunks also list their corresponding CSS and asset files if they have any.
4. You can use this file to render links or preload directives with hashed filenames.
Here is an example HTML template to render the proper links. The syntax here is for
explanation only, substitute with your server templating language. The `importedChunks`
function is for illustration and isn't provided by Vite.
2021-01-01 16:40:16 -05:00
```html
<!-- if production -->
<!-- for cssFile of manifest[name].css -->
<link rel="stylesheet" href="/{{ cssFile }}" />
<!-- for chunk of importedChunks(manifest, name) -->
<!-- for cssFile of chunk.css -->
<link rel="stylesheet" href="/{{ cssFile }}" />
<script type="module" src="/{{ manifest[name].file }}"></script>
<!-- for chunk of importedChunks(manifest, name) -->
<link rel="modulepreload" href="/{{ chunk.file }}" />
```
Specifically, a backend generating HTML should include the following tags given a manifest
file and an entry point. Note that following this order is recommended for optimal performance:
1. A `<link rel="stylesheet">` tag for each file in the entry point chunk's `css` list (if it exists)
2. Recursively follow all chunks in the entry point's `imports` list and include a
`<link rel="stylesheet">` tag for each CSS file of each imported chunk's `css` list (if it exists).
3. A tag for the `file` key of the entry point chunk. This can be `<script type="module">` for JavaScript, `<link rel="stylesheet">` for CSS.
4. Optionally, `<link rel="modulepreload">` tag for the `file` of each imported JavaScript
chunk, again recursively following the imports starting from the entry point chunk.
2024-05-07 19:16:36 +09:00
Following the above example manifest, for the entry point `views/foo.js` the following tags should be included in production:
```html
2024-05-07 19:16:36 +09:00
<link rel="stylesheet" href="assets/foo-5UjPuW-k.css" />
<link rel="stylesheet" href="assets/shared-ChJ_j-JJ.css" />
<script type="module" src="assets/foo-BRBmoGS9.js"></script>
<!-- optional -->
2024-05-07 19:16:36 +09:00
<link rel="modulepreload" href="assets/shared-B7PI925R.js" />
```
2024-05-07 19:16:36 +09:00
While the following should be included for the entry point `views/bar.js`:
```html
2024-05-07 19:16:36 +09:00
<link rel="stylesheet" href="assets/shared-ChJ_j-JJ.css" />
<script type="module" src="assets/bar-gkvgaI9m.js"></script>
<!-- optional -->
2024-05-07 19:16:36 +09:00
<link rel="modulepreload" href="assets/shared-B7PI925R.js" />
```
::: details Pseudo implementation of `importedChunks`
An example pseudo implementation of `importedChunks` in TypeScript (This will
need to be adapted for your programming language and templating language):
```ts
import type { Manifest, ManifestChunk } from 'vite'
export default function importedChunks(
manifest: Manifest,
name: string,
): ManifestChunk[] {
const seen = new Set<string>()
function getImportedChunks(chunk: ManifestChunk): ManifestChunk[] {
const chunks: ManifestChunk[] = []
for (const file of chunk.imports ?? []) {
const importee = manifest[file]
if (seen.has(file)) {
continue
}
seen.add(file)
chunks.push(...getImportedChunks(importee))
chunks.push(importee)
}
return chunks
}
return getImportedChunks(manifest[name])
}
```
:::