mirror of
https://github.com/facebook/react.git
synced 2026-02-23 20:23:02 +00:00
Stacked on #26557 Supporting Float methods such as ReactDOM.preload() are challenging for flight because it does not have an easy means to convey direct executions in other environments. Because the flight wire format is a JSON-like serialization that is expected to be rendered it currently only describes renderable elements. We need a way to convey a function invocation that gets run in the context of the client environment whether that is Fizz or Fiber. Fiber is somewhat straightforward because the HostDispatcher is always active and we can just have the FlightClient dispatch the serialized directive. Fizz is much more challenging becaue the dispatcher is always scoped but the specific request the dispatch belongs to is not readily available. Environments that support AsyncLocalStorage (or in the future AsyncContext) we will use this to be able to resolve directives in Fizz to the appropriate Request. For other environments directives will be elided. Right now this is pragmatic and non-breaking because all directives are opportunistic and non-critical. If this changes in the future we will need to reconsider how widespread support for async context tracking is. For Flight, if AsyncLocalStorage is available Float methods can be called before and after await points and be expected to work. If AsyncLocalStorage is not available float methods called in the sync phase of a component render will be captured but anything after an await point will be a noop. If a float call is dropped in this manner a DEV warning should help you realize your code may need to be modified. This PR also introduces a way for resources (Fizz) and hints (Flight) to flush even if there is not active task being worked on. This will help when Float methods are called in between async points within a function execution but the task is blocked on the entire function finishing. This PR also introduces deduping of Hints in Flight using the same resource keys used in Fizz. This will help shrink payload sizes when the same hint is attempted to emit over and over again
237 lines
6.0 KiB
JavaScript
237 lines
6.0 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 type {HintModel} from 'react-server/src/ReactFlightServerConfig';
|
|
import type {RowEncoding, JSONValue} from './ReactFlightDOMRelayProtocol';
|
|
|
|
import type {
|
|
Request,
|
|
ReactClientValue,
|
|
} from 'react-server/src/ReactFlightServer';
|
|
|
|
import type {JSResourceReference} from 'JSResourceReference';
|
|
import JSResourceReferenceImpl from 'JSResourceReferenceImpl';
|
|
|
|
import hasOwnProperty from 'shared/hasOwnProperty';
|
|
import isArray from 'shared/isArray';
|
|
|
|
export type ClientReference<T> = JSResourceReference<T>;
|
|
export type ServerReference<T> = T;
|
|
export type ServerReferenceId = {};
|
|
|
|
import type {
|
|
Destination,
|
|
BundlerConfig as ClientManifest,
|
|
ClientReferenceMetadata,
|
|
} from 'ReactFlightDOMRelayServerIntegration';
|
|
|
|
import {resolveModelToJSON} from 'react-server/src/ReactFlightServer';
|
|
|
|
import {
|
|
emitRow,
|
|
resolveClientReferenceMetadata as resolveClientReferenceMetadataImpl,
|
|
close,
|
|
} from 'ReactFlightDOMRelayServerIntegration';
|
|
|
|
export type {
|
|
Destination,
|
|
BundlerConfig as ClientManifest,
|
|
ClientReferenceMetadata,
|
|
} from 'ReactFlightDOMRelayServerIntegration';
|
|
|
|
export function isClientReference(reference: Object): boolean {
|
|
return reference instanceof JSResourceReferenceImpl;
|
|
}
|
|
|
|
export function isServerReference(reference: Object): boolean {
|
|
return false;
|
|
}
|
|
|
|
export type ClientReferenceKey = ClientReference<any>;
|
|
|
|
export function getClientReferenceKey(
|
|
reference: ClientReference<any>,
|
|
): ClientReferenceKey {
|
|
// We use the reference object itself as the key because we assume the
|
|
// object will be cached by the bundler runtime.
|
|
return reference;
|
|
}
|
|
|
|
export function resolveClientReferenceMetadata<T>(
|
|
config: ClientManifest,
|
|
resource: ClientReference<T>,
|
|
): ClientReferenceMetadata {
|
|
return resolveClientReferenceMetadataImpl(config, resource);
|
|
}
|
|
|
|
export function getServerReferenceId<T>(
|
|
config: ClientManifest,
|
|
resource: ServerReference<T>,
|
|
): ServerReferenceId {
|
|
throw new Error('Not implemented.');
|
|
}
|
|
|
|
export function getServerReferenceBoundArguments<T>(
|
|
config: ClientManifest,
|
|
resource: ServerReference<T>,
|
|
): Array<ReactClientValue> {
|
|
throw new Error('Not implemented.');
|
|
}
|
|
|
|
export type Chunk = RowEncoding;
|
|
|
|
export function processErrorChunkProd(
|
|
request: Request,
|
|
id: number,
|
|
digest: string,
|
|
): Chunk {
|
|
if (__DEV__) {
|
|
// These errors should never make it into a build so we don't need to encode them in codes.json
|
|
// eslint-disable-next-line react-internal/prod-error-codes
|
|
throw new Error(
|
|
'processErrorChunkProd should never be called while in development mode. Use processErrorChunkDev instead. This is a bug in React.',
|
|
);
|
|
}
|
|
|
|
return [
|
|
'E',
|
|
id,
|
|
{
|
|
digest,
|
|
},
|
|
];
|
|
}
|
|
|
|
export function processErrorChunkDev(
|
|
request: Request,
|
|
id: number,
|
|
digest: string,
|
|
message: string,
|
|
stack: string,
|
|
): Chunk {
|
|
if (!__DEV__) {
|
|
// These errors should never make it into a build so we don't need to encode them in codes.json
|
|
// eslint-disable-next-line react-internal/prod-error-codes
|
|
throw new Error(
|
|
'processErrorChunkDev should never be called while in production mode. Use processErrorChunkProd instead. This is a bug in React.',
|
|
);
|
|
}
|
|
|
|
return [
|
|
'E',
|
|
id,
|
|
{
|
|
digest,
|
|
message,
|
|
stack,
|
|
},
|
|
];
|
|
}
|
|
|
|
function convertModelToJSON(
|
|
request: Request,
|
|
parent: {+[key: string]: ReactClientValue} | $ReadOnlyArray<ReactClientValue>,
|
|
key: string,
|
|
model: ReactClientValue,
|
|
): JSONValue {
|
|
const json = resolveModelToJSON(request, parent, key, model);
|
|
if (typeof json === 'object' && json !== null) {
|
|
if (isArray(json)) {
|
|
const jsonArray: Array<JSONValue> = [];
|
|
for (let i = 0; i < json.length; i++) {
|
|
jsonArray[i] = convertModelToJSON(request, json, '' + i, json[i]);
|
|
}
|
|
return jsonArray;
|
|
} else {
|
|
const jsonObj: {[key: string]: JSONValue} = {};
|
|
for (const nextKey in json) {
|
|
if (hasOwnProperty.call(json, nextKey)) {
|
|
jsonObj[nextKey] = convertModelToJSON(
|
|
request,
|
|
json,
|
|
nextKey,
|
|
json[nextKey],
|
|
);
|
|
}
|
|
}
|
|
return jsonObj;
|
|
}
|
|
}
|
|
return json;
|
|
}
|
|
|
|
export function processModelChunk(
|
|
request: Request,
|
|
id: number,
|
|
model: ReactClientValue,
|
|
): Chunk {
|
|
const json = convertModelToJSON(request, {}, '', model);
|
|
return ['O', id, json];
|
|
}
|
|
|
|
export function processReferenceChunk(
|
|
request: Request,
|
|
id: number,
|
|
reference: string,
|
|
): Chunk {
|
|
return ['O', id, reference];
|
|
}
|
|
|
|
export function processImportChunk(
|
|
request: Request,
|
|
id: number,
|
|
clientReferenceMetadata: ClientReferenceMetadata,
|
|
): Chunk {
|
|
// The clientReferenceMetadata is already a JSON serializable value.
|
|
return ['I', id, clientReferenceMetadata];
|
|
}
|
|
|
|
export function processHintChunk(
|
|
request: Request,
|
|
id: number,
|
|
code: string,
|
|
model: HintModel,
|
|
): Chunk {
|
|
// The hint is already a JSON serializable value.
|
|
return ['H', code, model];
|
|
}
|
|
|
|
export function scheduleWork(callback: () => void) {
|
|
callback();
|
|
}
|
|
|
|
export function flushBuffered(destination: Destination) {}
|
|
|
|
export const supportsRequestStorage = false;
|
|
export const requestStorage: AsyncLocalStorage<Request> = (null: any);
|
|
|
|
export function beginWriting(destination: Destination) {}
|
|
|
|
export function writeChunk(destination: Destination, chunk: Chunk): void {
|
|
// $FlowFixMe[incompatible-call] `Chunk` doesn't flow into `JSONValue` because of the `E` row type.
|
|
emitRow(destination, chunk);
|
|
}
|
|
|
|
export function writeChunkAndReturn(
|
|
destination: Destination,
|
|
chunk: Chunk,
|
|
): boolean {
|
|
// $FlowFixMe[incompatible-call] `Chunk` doesn't flow into `JSONValue` because of the `E` row type.
|
|
emitRow(destination, chunk);
|
|
return true;
|
|
}
|
|
|
|
export function completeWriting(destination: Destination) {}
|
|
|
|
export {close};
|
|
|
|
export function closeWithError(destination: Destination, error: mixed): void {
|
|
close(destination);
|
|
}
|