mirror of
https://github.com/facebook/react.git
synced 2025-11-01 09:12:30 +00:00
b775564d35
Moves writing queues to renderState. We shouldn't need the resource tracking's value. We just need to know if that resource has already been emitted. We can use a Set for this. To ensure that set is directly serializable we can just use a dictionary-like object with no value. See individual commits for special cases.
104 lines
2.5 KiB
JavaScript
104 lines
2.5 KiB
JavaScript
/**
|
|
* Copyright (c) Meta Platforms, Inc. and affiliates.
|
|
*
|
|
* This source code is licensed under the MIT license found in the
|
|
* LICENSE file in the root directory of this source tree.
|
|
*
|
|
* @flow
|
|
*/
|
|
|
|
import ReactVersion from 'shared/ReactVersion';
|
|
|
|
import type {ReactNodeList} from 'shared/ReactTypes';
|
|
|
|
import {
|
|
createRequest,
|
|
startWork,
|
|
startFlowing,
|
|
abort,
|
|
} from 'react-server/src/ReactFizzServer';
|
|
|
|
import {
|
|
createResumableState,
|
|
createRenderState,
|
|
createRootFormatContext,
|
|
} from 'react-dom-bindings/src/server/ReactFizzConfigDOMLegacy';
|
|
|
|
type ServerOptions = {
|
|
identifierPrefix?: string,
|
|
};
|
|
|
|
function onError() {
|
|
// Non-fatal errors are ignored.
|
|
}
|
|
|
|
function renderToStringImpl(
|
|
children: ReactNodeList,
|
|
options: void | ServerOptions,
|
|
generateStaticMarkup: boolean,
|
|
abortReason: string,
|
|
): string {
|
|
let didFatal = false;
|
|
let fatalError = null;
|
|
let result = '';
|
|
const destination = {
|
|
// $FlowFixMe[missing-local-annot]
|
|
push(chunk) {
|
|
if (chunk !== null) {
|
|
result += chunk;
|
|
}
|
|
return true;
|
|
},
|
|
// $FlowFixMe[missing-local-annot]
|
|
destroy(error) {
|
|
didFatal = true;
|
|
fatalError = error;
|
|
},
|
|
};
|
|
|
|
let readyToStream = false;
|
|
function onShellReady() {
|
|
readyToStream = true;
|
|
}
|
|
const resumableState = createResumableState(
|
|
options ? options.identifierPrefix : undefined,
|
|
undefined,
|
|
);
|
|
const request = createRequest(
|
|
children,
|
|
resumableState,
|
|
createRenderState(resumableState, generateStaticMarkup),
|
|
createRootFormatContext(),
|
|
Infinity,
|
|
onError,
|
|
undefined,
|
|
onShellReady,
|
|
undefined,
|
|
undefined,
|
|
undefined,
|
|
);
|
|
startWork(request);
|
|
// If anything suspended and is still pending, we'll abort it before writing.
|
|
// That way we write only client-rendered boundaries from the start.
|
|
abort(request, abortReason);
|
|
startFlowing(request, destination);
|
|
if (didFatal && fatalError !== abortReason) {
|
|
throw fatalError;
|
|
}
|
|
|
|
if (!readyToStream) {
|
|
// Note: This error message is the one we use on the client. It doesn't
|
|
// really make sense here. But this is the legacy server renderer, anyway.
|
|
// We're going to delete it soon.
|
|
throw new Error(
|
|
'A component suspended while responding to synchronous input. This ' +
|
|
'will cause the UI to be replaced with a loading indicator. To fix, ' +
|
|
'updates that suspend should be wrapped with startTransition.',
|
|
);
|
|
}
|
|
|
|
return result;
|
|
}
|
|
|
|
export {renderToStringImpl, ReactVersion as version};
|