prerender
prerender, Web Stream பயன்படுத்தி React tree ஒன்றை static HTML string ஆக render செய்கிறது.
const {prelude, postponed} = await prerender(reactNode, options?)Reference
prerender(reactNode, options?)
உங்கள் app-ஐ static HTML ஆக render செய்ய prerender call செய்யவும்.
import { prerender } from 'react-dom/static';
async function handler(request, response) {
const {prelude} = await prerender(<App />, {
bootstrapScripts: ['/main.js']
});
return new Response(prelude, {
headers: { 'content-type': 'text/html' },
});
}Client-இல், server-generated HTML interactive ஆக hydrateRoot call செய்யவும்.
கீழே மேலும் examples பார்க்கவும்.
Parameters
-
reactNode: HTML-க்கு render செய்ய விரும்பும் React node. உதாரணமாக,<App />போன்ற JSX node. இது முழு document-ஐ represent செய்யும் என்று எதிர்பார்க்கப்படுகிறது; எனவே App component<html>tag-ஐ render செய்ய வேண்டும். -
optional
options: Static generation options கொண்ட object.- optional
bootstrapScriptContent: குறிப்பிடப்பட்டால், இந்த string inline<script>tag-இல் வைக்கப்படும். - optional
bootstrapScripts: Page-இல் emit செய்ய வேண்டிய<script>tags-க்கான string URLs array.hydrateRootcall செய்யும்<script>-ஐ include செய்ய இதைப் பயன்படுத்தவும். Client-இல் React run செய்ய வேண்டாம் என்றால் இதை omit செய்யவும். - optional
bootstrapModules:bootstrapScriptsபோலவே, ஆனால்<script type="module">emit செய்கிறது. - optional
identifierPrefix:useIdgenerate செய்யும் IDs-க்கு React பயன்படுத்தும் string prefix. ஒரே page-இல் multiple roots பயன்படுத்தும்போது conflicts தவிர்க்க பயனுள்ளது.hydrateRoot-க்கு pass செய்யப்பட்ட அதே prefix ஆக இருக்க வேண்டும். - optional
namespaceURI: Stream-க்கான root namespace URI கொண்ட string. Default regular HTML. SVG-க்கு'http://www.w3.org/2000/svg'அல்லது MathML-க்கு'http://www.w3.org/1998/Math/MathML'pass செய்யவும். - optional
onError: recoverable ஆனதோ அல்லாததோ, server error ஏற்படும் ஒவ்வொரு முறையும் fire ஆகும் callback. Default ஆக, இதுconsole.errorமட்டுமே call செய்கிறது. Crash reports log செய்ய override செய்தால், இன்னும்console.errorcall செய்வதை உறுதிசெய்யவும். Shell emit ஆகும்முன் status code adjust செய்யவும் இதைப் பயன்படுத்தலாம். - optional
progressiveChunkSize: ஒரு chunk-இல் உள்ள bytes எண்ணிக்கை. Default heuristic பற்றி மேலும் படிக்கவும். - optional
signal: Abort signal; இது prerendering abort செய்யவும் மீதியை client-இல் render செய்யவும் அனுமதிக்கிறது.
- optional
Returns
prerender ஒரு Promise return செய்கிறது:
- Rendering successful என்றால், Promise பின்வருவன கொண்ட object-க்கு resolve ஆகும்:
prelude: HTML-ன் Web Stream. இந்த stream-ஐ chunks-ஆக response அனுப்ப பயன்படுத்தலாம், அல்லது முழு stream-ஐ string ஆக read செய்யலாம்.postponed:prerenderமுடிவடையவில்லை என்றால்resume-க்கு pass செய்யக்கூடிய JSON-serializeable, opaque object. இல்லையெனில்null; அதாவதுpreludeஎல்லா content-யையும் கொண்டுள்ளது, resume தேவையில்லை.
- Rendering fail ஆனால், Promise reject செய்யப்படும். Fallback shell output செய்ய இதைப் பயன்படுத்தவும்.
Caveats
Prerendering செய்யும்போது nonce option கிடைக்காது. Nonces ஒவ்வொரு request-க்கும் unique ஆக இருக்க வேண்டும்; உங்கள் application-ஐ CSP மூலம் secure செய்ய nonces பயன்படுத்தினால், nonce value-ஐ prerender-இலேயே include செய்வது inappropriate மற்றும் insecure ஆகும்.
Usage
React tree-ஐ static HTML stream ஆக render செய்தல்
உங்கள் React tree-ஐ static HTML ஆக Readable Web Stream-க்குள் render செய்ய prerender call செய்யவும்:
import { prerender } from 'react-dom/static';
async function handler(request) {
const {prelude} = await prerender(<App />, {
bootstrapScripts: ['/main.js']
});
return new Response(prelude, {
headers: { 'content-type': 'text/html' },
});
}Root component-உடன், bootstrap <script> paths பட்டியலை வழங்க வேண்டும். உங்கள் root component root <html> tag உட்பட முழு document-ஐ return செய்ய வேண்டும்.
உதாரணமாக, அது இவ்வாறு இருக்கலாம்:
export default function App() {
return (
<html>
<head>
<meta charSet="utf-8" />
<meta name="viewport" content="width=device-width, initial-scale=1" />
<link rel="stylesheet" href="/styles.css"></link>
<title>என் app</title>
</head>
<body>
<Router />
</body>
</html>
);
}Resulting HTML stream-இல் React doctype மற்றும் உங்கள் bootstrap <script> tags-ஐ inject செய்யும்:
<!DOCTYPE html>
<html>
<!-- ... உங்கள் components-இலிருந்து HTML ... -->
</html>
<script src="/main.js" async=""></script>Client-இல், உங்கள் bootstrap script hydrateRoot call மூலம் முழு document-ஐ hydrate செய்ய வேண்டும்:
import { hydrateRoot } from 'react-dom/client';
import App from './App.js';
hydrateRoot(document, <App />);இது static server-generated HTML-க்கு event listeners attach செய்து interactive ஆக்கும்.
Deep Dive
Final asset URLs (JavaScript மற்றும் CSS files போன்றவை) build-க்கு பிறகு பெரும்பாலும் hashed ஆக இருக்கும். உதாரணமாக, styles.css-க்கு பதிலாக styles.123456.css கிடைக்கலாம். Static asset filenames hash செய்வது, அதே asset-ன் ஒவ்வொரு distinct build-க்கும் வேறு filename இருக்கும் என்பதை guarantee செய்கிறது. இது பயனுள்ளதாகும், ஏனெனில் static assets-க்கு long-term caching-ஐ safe ஆக enable செய்ய அனுமதிக்கிறது: குறிப்பிட்ட பெயர் கொண்ட file content எப்போதும் மாறாது.
ஆனால் build-க்கு பிறகே asset URLs தெரிந்தால், அவற்றை source code-இல் வைக்க வழியில்லை. உதாரணமாக, முன்புபோல் JSX-இல் "/styles.css" hardcode செய்வது வேலை செய்யாது. அவற்றை உங்கள் source code-இல் இருந்து வெளியே வைத்திருக்க, prop ஆக pass செய்யப்படும் map-இலிருந்து real filenames-ஐ உங்கள் root component படிக்கலாம்:
export default function App({ assetMap }) {
return (
<html>
<head>
<title>என் app</title>
<link rel="stylesheet" href={assetMap['styles.css']}></link>
</head>
...
</html>
);
}Server-இல், <App assetMap={assetMap} /> render செய்து asset URLs கொண்ட உங்கள் assetMap pass செய்யவும்:
// You'd need to get this JSON from your build tooling, e.g. read it from the build output.
const assetMap = {
'styles.css': '/styles.123456.css',
'main.js': '/main.123456.js'
};
async function handler(request) {
const {prelude} = await prerender(<App assetMap={assetMap} />, {
bootstrapScripts: [assetMap['/main.js']]
});
return new Response(prelude, {
headers: { 'content-type': 'text/html' },
});
}உங்கள் server இப்போது <App assetMap={assetMap} /> render செய்வதால், hydration errors தவிர்க்க client-இலும் assetMap உடன் render செய்ய வேண்டும். assetMap-ஐ serialize செய்து client-க்கு இவ்வாறு pass செய்யலாம்:
// You'd need to get this JSON from your build tooling.
const assetMap = {
'styles.css': '/styles.123456.css',
'main.js': '/main.123456.js'
};
async function handler(request) {
const {prelude} = await prerender(<App assetMap={assetMap} />, {
// Careful: It's safe to stringify() this because this data isn't user-generated.
bootstrapScriptContent: `window.assetMap = ${JSON.stringify(assetMap)};`,
bootstrapScripts: [assetMap['/main.js']],
});
return new Response(prelude, {
headers: { 'content-type': 'text/html' },
});
}மேலுள்ள example-இல், bootstrapScriptContent option client-இல் global window.assetMap variable set செய்யும் கூடுதல் inline <script> tag சேர்க்கிறது. இதனால் client code அதே assetMap-ஐ read செய்ய முடியும்:
import { hydrateRoot } from 'react-dom/client';
import App from './App.js';
hydrateRoot(document, <App assetMap={window.assetMap} />);Client மற்றும் server இரண்டும் அதே assetMap prop உடன் App render செய்வதால் hydration errors இல்லை.
React tree-ஐ static HTML string ஆக render செய்தல்
உங்கள் app-ஐ static HTML string ஆக render செய்ய prerender call செய்யவும்:
import { prerender } from 'react-dom/static';
async function renderToString() {
const {prelude} = await prerender(<App />, {
bootstrapScripts: ['/main.js']
});
const reader = prelude.getReader();
let content = '';
while (true) {
const {done, value} = await reader.read();
if (done) {
return content;
}
content += Buffer.from(value).toString('utf8');
}
}இது உங்கள் React components-ன் initial non-interactive HTML output உருவாக்கும். Client-இல், அந்த server-generated HTML-ஐ hydrate செய்து interactive ஆக்க hydrateRoot call செய்ய வேண்டும்.
எல்லா data load ஆக காத்திருத்தல்
Static HTML generation முடிந்து resolve ஆகும்முன், prerender எல்லா data load ஆக காத்திருக்கும். உதாரணமாக, cover, friends மற்றும் photos கொண்ட sidebar, posts பட்டியல் ஆகியவற்றைக் காட்டும் profile page ஒன்றைக் கருதுங்கள்:
function ProfilePage() {
return (
<ProfileLayout>
<ProfileCover />
<Sidebar>
<Friends />
<Photos />
</Sidebar>
<Suspense fallback={<PostsGlimmer />}>
<Posts />
</Suspense>
</ProfileLayout>
);
}<Posts /> சில data load செய்ய வேண்டியுள்ளது; அதற்கு சிறிது நேரம் எடுக்கும் என்று கற்பனை செய்யுங்கள். Ideally, posts முடியும் வரை காத்திருந்து, அது HTML-இல் include ஆக வேண்டும். இதை செய்ய, data மீது suspend செய்ய Suspense பயன்படுத்தலாம்; static HTML-க்கு resolve ஆகும்முன் suspended content முடியும் வரை prerender காத்திருக்கும்.
Prerendering abort செய்தல்
Timeout-க்கு பிறகு prerender-ஐ “give up” செய்ய force செய்யலாம்:
async function renderToString() {
const controller = new AbortController();
setTimeout(() => {
controller.abort()
}, 10000);
try {
// the prelude will contain all the HTML that was prerendered
// before the controller aborted.
const {prelude} = await prerender(<App />, {
signal: controller.signal,
});
//...Incomplete children கொண்ட எந்த Suspense boundaries-யும் fallback state-இல் prelude-இல் சேர்க்கப்படும்.
இதை resume அல்லது resumeAndPrerender உடன் சேர்த்து partial prerendering-க்கு பயன்படுத்தலாம்.
Troubleshooting
முழு app render ஆகும்வரை என் stream தொடங்கவில்லை
prerender response, resolve ஆகும்முன் அனைத்து Suspense boundaries resolve ஆக காத்திருப்பதையும் சேர்த்து முழு app rendering முடியும் வரை காத்திருக்கும். இது முன்கூட்டியே static site generation (SSG)-க்காக வடிவமைக்கப்பட்டுள்ளது; content load ஆகும் போது மேலும் content stream செய்வதை support செய்யாது.
Content load ஆகும் போதே stream செய்ய, renderToReadableStream போன்ற streaming server render API பயன்படுத்தவும்.