Skip to content

Commit

Permalink
[Fizz] Start initial work immediately
Browse files Browse the repository at this point in the history
In a recent update we make Flight start working immediately rather than waitin for a new task. This commit updates fizz to have similar mechanics. We start the render in the currently running task but we do so in a microtask to avoid reentrancy. This aligns Fizz with Flight.
  • Loading branch information
gnoff committed Sep 26, 2024
1 parent e495392 commit 0ea35c5
Show file tree
Hide file tree
Showing 2 changed files with 36 additions and 40 deletions.
4 changes: 2 additions & 2 deletions packages/react-dom/src/__tests__/ReactDOMFizzServer-test.js
Original file line number Diff line number Diff line change
Expand Up @@ -8119,7 +8119,7 @@ describe('ReactDOMFizzServer', () => {

prerendering = false;

const resumed = await ReactDOMFizzServer.resumeToPipeableStream(
const resumed = ReactDOMFizzServer.resumeToPipeableStream(
<App />,
JSON.parse(JSON.stringify(prerendered.postponed)),
{
Expand Down Expand Up @@ -8187,7 +8187,7 @@ describe('ReactDOMFizzServer', () => {
function onPostpone(reason) {
postpones.push(reason);
}
const result = await renderToPipeableStream(<App />, {
const result = renderToPipeableStream(<App />, {
onError,
onShellError,
onPostpone,
Expand Down
72 changes: 34 additions & 38 deletions packages/react-server/src/ReactFizzServer.js
Original file line number Diff line number Diff line change
Expand Up @@ -316,10 +316,11 @@ type Segment = {
textEmbedded: boolean,
};

const OPEN = 0;
const ABORTING = 1;
const CLOSING = 2;
const CLOSED = 3;
const OPENING = 10;
const OPEN = 11;
const ABORTING = 12;
const CLOSING = 13;
const CLOSED = 14;

export opaque type Request = {
destination: null | Destination,
Expand All @@ -328,7 +329,7 @@ export opaque type Request = {
+renderState: RenderState,
+rootFormatContext: FormatContext,
+progressiveChunkSize: number,
status: 0 | 1 | 2 | 3,
status: 10 | 11 | 12 | 13 | 14,
fatalError: mixed,
nextSegmentId: number,
allPendingTasks: number, // when it reaches zero, we can close the connection.
Expand Down Expand Up @@ -424,7 +425,7 @@ function RequestInstance(
progressiveChunkSize === undefined
? DEFAULT_PROGRESSIVE_CHUNK_SIZE
: progressiveChunkSize;
this.status = OPEN;
this.status = OPENING;
this.fatalError = null;
this.nextSegmentId = 0;
this.allPendingTasks = 0;
Expand Down Expand Up @@ -688,7 +689,7 @@ function pingTask(request: Request, task: Task): void {
pingedTasks.push(task);
if (request.pingedTasks.length === 1) {
request.flushScheduled = request.destination !== null;
if (request.trackedPostpones !== null) {
if (request.trackedPostpones !== null || request.status === OPENING) {
scheduleMicrotask(() => performWork(request));
} else {
scheduleWork(() => performWork(request));
Expand Down Expand Up @@ -4977,43 +4978,38 @@ function flushCompletedQueues(

export function startWork(request: Request): void {
request.flushScheduled = request.destination !== null;
if (request.trackedPostpones !== null) {
// When prerendering we use microtasks for pinging work
if (supportsRequestStorage) {
scheduleMicrotask(() =>
requestStorage.run(request, performWork, request),
);
} else {
scheduleMicrotask(() => performWork(request));
}
// When prerendering we use microtasks for pinging work
if (supportsRequestStorage) {
scheduleMicrotask(() => requestStorage.run(request, performWork, request));
} else {
// When rendering/resuming we use regular tasks and we also emit early preloads
if (supportsRequestStorage) {
scheduleWork(() => requestStorage.run(request, performWork, request));
} else {
scheduleWork(() => performWork(request));
scheduleMicrotask(() => performWork(request));
}
scheduleWork(() => {
if (request.status === OPENING) {
request.status = OPEN;
}
// this is either a regular render or a resume. For regular render we want
// to call emitEarlyPreloads after the first performWork because we want
// are responding to a live request and need to balance sending something early
// (i.e. don't want for the shell to finish) but we need something to send.
// The only implementation of this is for DOM at the moment and during resumes nothing
// actually emits but the code paths here are the same.
// During a prerender we don't want to be too aggressive in emitting early preloads
// because we aren't responding to a live request and we can wait for the prerender to
// postpone before we emit anything.
if (supportsRequestStorage) {
scheduleWork(() =>

if (request.trackedPostpones === null) {
// this is either a regular render or a resume. For regular render we want
// to call emitEarlyPreloads after the first performWork because we want
// are responding to a live request and need to balance sending something early
// (i.e. don't want for the shell to finish) but we need something to send.
// The only implementation of this is for DOM at the moment and during resumes nothing
// actually emits but the code paths here are the same.
// During a prerender we don't want to be too aggressive in emitting early preloads
// because we aren't responding to a live request and we can wait for the prerender to
// postpone before we emit anything.
if (supportsRequestStorage) {
requestStorage.run(
request,
enqueueEarlyPreloadsAfterInitialWork,
request,
),
);
} else {
scheduleWork(() => enqueueEarlyPreloadsAfterInitialWork(request));
);
} else {
enqueueEarlyPreloadsAfterInitialWork(request);
}
}
}
});
}

function enqueueEarlyPreloadsAfterInitialWork(request: Request) {
Expand Down Expand Up @@ -5095,7 +5091,7 @@ export function stopFlowing(request: Request): void {

// This is called to early terminate a request. It puts all pending boundaries in client rendered state.
export function abort(request: Request, reason: mixed): void {
if (request.status === OPEN) {
if (request.status === OPEN || request.status === OPENING) {
request.status = ABORTING;
}

Expand Down

0 comments on commit 0ea35c5

Please sign in to comment.