백엔드 프레임워크와 함께 사용하기
참고
기존의 백엔드(Rails, Laravel 등)를 사용해 HTML을 제공하기를 원하지만, 에셋은 Vite를 이용하고자 한다면, Awesome Vite의 방법을 확인해 보세요.
만약 이를 직접 구성하고자 한다면, 아래의 가이드를 따라 진행할 수 있습니다.
Vite 설정 파일에서, 진입점을 설정하고 매니페스트를 활성화합니다:
jsexport default
defineConfig({build: { // outDir 위치에 .vite/manifest.json 파일을 생성manifest: true,rollupOptions: { // 기본 .html 진입점을 변경input: '/path/to/main.js' } } })만약 module preload polyfill을 비활성화하지 않았다면, 진입점에 폴리필을 가져와야 합니다.
js// 앱 진입점에 추가 import 'vite/modulepreload-polyfill'
개발 단계에서는 서버의 HTML 템플릿에 다음을 추가합니다(
http://localhost:5173
을 Vite가 실행 중인 로컬 URL로 대체):html<!-- 개발 시 --> <script type="module" src="http://localhost:5173/@vite/client"></script> <script type="module" src="http://localhost:5173/main.js"></script>
이후 Vite의 에셋에 접근할 수 있도록 아래 두 가지 옵션 중 하나를 적용해 주세요:
- 백엔드 서버가 Vite 서버에 대한 에셋 요청을 프락시 하도록 설정
- 에셋의 URL이 상대 경로 대신 백엔드 서버 URL을 사용해 가져와질 수 있도록
server.origin
옵션값을 설정
위 옵션은 이미지와 같은 에셋이 정상적으로 로드되기 위해 필요합니다.
만약 React를
@vitejs/plugin-react
와 함께 사용한다면, 플러그인이 HTML을 수정할 수 없기 때문에, 위의 스크립트 이전에 다음을 추가해야 합니다(http://localhost:5173
을 Vite가 실행 중인 로컬 URL로 대체):html<script type="module"> import RefreshRuntime from "http://localhost:5173/@react-refresh" RefreshRuntime.injectIntoGlobalHook(window) window.$RefreshReg$ = () => {} window.$RefreshSig$ = () => (type) => type window.__vite_plugin_react_preamble_installed__ = true </script>
프로덕션 빌드 단계:
vite build
를 실행하게 되면.vite/manifest.json
파일이 에셋 파일과 함께 생성됩니다. 매니페스트 파일은 다음과 같은 구조를 가집니다:json{ "_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" }, "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, "imports": ["_shared-B7PI925R.js"], "dynamicImports": ["baz.js"] }, "views/foo.js": { "file": "assets/foo-BRBmoGS9.js", "name": "foo", "src": "views/foo.js", "isEntry": true, "imports": ["_shared-B7PI925R.js"], "css": ["assets/foo-5UjPuW-k.css"] } }
- 매니페스트는
Record<name, chunk>
구조를 가집니다. - 진입점 또는 동적 진입점의 청크는 프로젝트 루트로부터의 상대적인 src 경로를 가집니다.
- 진입점이 아닌 청크의 경우, 키는
_
로 접두사가 붙은 파일명이 됩니다. - 청크에는 정적 및 동적 임포트에 대한 정보(둘 다 매니페스트에서 해당 청크를 매핑하는 키)와 CSS 및 에셋 파일(있는 경우)이 포함됩니다.
- build.cssCodeSplit이
false
일 때 생성되는 CSS 파일 키는style.css
입니다.
- 매니페스트는
해시된 파일 이름으로 링크를 렌더링하거나 지시문을 미리 로드하기 위해 이 파일을 사용할 수 있습니다.
아래는 올바른 링크를 렌더링하는 HTML 템플릿 예시입니다. 여기서 사용된 구문은 설명을 위한 것이므로, 실제 사용 시에는 서버 템플릿 언어로 대체해 주세요.
importedChunks
함수 역시 예시를 위한 것이며, Vite에서 제공하지 않습니다.html<!-- 프로덕션 (아래 `for ... of` 는 구문입니다 - 옮긴이) --> <!-- 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 }}" />
구체적으로, HTML을 생성하는 백엔드는 매니페스트 파일과 진입점이 주어졌을 때 다음 태그를 포함해야 합니다:
- 진입점 청크의
css
목록에 있는 각 파일에 대한<link rel="stylesheet">
태그 - 진입점의
imports
목록에 있는 모든 청크를 재귀적으로 따라가며, 임포트된 청크 내 CSS 파일에 대한<link rel="stylesheet">
태그 - 진입점 청크의
file
키에 대한 태그 (JavaScript는<script type="module">
, CSS는<link rel="stylesheet">
) - 선택 사항으로, 진입점의
imports
목록에 있는 JavaScript 청크를 재귀적으로 따라가며, 임포트된 청크의file
에 대한<link rel="modulepreload">
태그
위 예시 매니페스트를 예로 들자면, 진입점
views/foo.js
에 대해 다음 태그가 포함되어야 합니다.html<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> <!-- 선택 사항 --> <link rel="modulepreload" href="assets/shared-B7PI925R.js" />
진입점
views/foo.js
에 대해서는 아래 코드가 포함되어야 합니다.html<link rel="stylesheet" href="assets/shared-ChJ_j-JJ.css" /> <script type="module" src="assets/bar-gkvgaI9m.js"></script> <!-- 선택 사항 --> <link rel="modulepreload" href="assets/shared-B7PI925R.js" />
importedChunks
의사 구현체아래 예시 코드는
importedChunks
의사 구현체입니다. TypeScript로 작성되어 있으며, 실제 적용 시에는 사용하는 프로그래밍 언어와 템플릿 언어에 맞게 수정이 필요합니다:tsimport 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]) }
- 진입점 청크의