@modern-js/plugin-data-loader 2.5.0-alpha.0 → 2.6.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
@@ -1,142 +0,0 @@
1
- import {
2
- UNSAFE_DeferredData as DeferredData,
3
- AbortedDeferredError
4
- } from "@modern-js/utils/remix-router";
5
- const DEFERRED_VALUE_PLACEHOLDER_PREFIX = "__deferred_promise:";
6
- async function parseDeferredReadableStream(stream) {
7
- if (!stream) {
8
- throw new Error("parseDeferredReadableStream requires stream argument");
9
- }
10
- let deferredData;
11
- const deferredResolvers = {};
12
- try {
13
- const sectionReader = readStreamSections(stream);
14
- const initialSectionResult = await sectionReader.next();
15
- const initialSection = initialSectionResult.value;
16
- if (!initialSection) {
17
- throw new Error("no critical data");
18
- }
19
- const criticalData = JSON.parse(initialSection);
20
- if (typeof criticalData === "object" && criticalData !== null) {
21
- for (const [eventKey, value] of Object.entries(criticalData)) {
22
- if (typeof value !== "string" || !value.startsWith(DEFERRED_VALUE_PLACEHOLDER_PREFIX)) {
23
- continue;
24
- }
25
- deferredData = deferredData || {};
26
- deferredData[eventKey] = new Promise((resolve, reject) => {
27
- deferredResolvers[eventKey] = {
28
- resolve: (value2) => {
29
- resolve(value2);
30
- delete deferredResolvers[eventKey];
31
- },
32
- reject: (error) => {
33
- reject(error);
34
- delete deferredResolvers[eventKey];
35
- }
36
- };
37
- });
38
- }
39
- }
40
- (async () => {
41
- try {
42
- for await (const section of sectionReader) {
43
- const [event, ...sectionDataStrings] = section.split(":");
44
- const sectionDataString = sectionDataStrings.join(":");
45
- const data = JSON.parse(sectionDataString);
46
- if (event === "data") {
47
- for (const [key, value] of Object.entries(data)) {
48
- if (deferredResolvers[key]) {
49
- deferredResolvers[key].resolve(value);
50
- }
51
- }
52
- } else if (event === "error") {
53
- for (const [key, value] of Object.entries(data)) {
54
- const err = new Error(value.message);
55
- err.stack = value.stack;
56
- if (deferredResolvers[key]) {
57
- deferredResolvers[key].reject(err);
58
- }
59
- }
60
- }
61
- }
62
- for (const [key, resolver] of Object.entries(deferredResolvers)) {
63
- resolver.reject(
64
- new AbortedDeferredError(`Deferred ${key} will never resolved`)
65
- );
66
- }
67
- } catch (error) {
68
- for (const resolver of Object.values(deferredResolvers)) {
69
- resolver.reject(error);
70
- }
71
- }
72
- })();
73
- return new DeferredData({ ...criticalData, ...deferredData });
74
- } catch (error) {
75
- for (const resolver of Object.values(deferredResolvers)) {
76
- resolver.reject(error);
77
- }
78
- throw error;
79
- }
80
- }
81
- async function* readStreamSections(stream) {
82
- const reader = stream.getReader();
83
- let buffer = [];
84
- let sections = [];
85
- let closed = false;
86
- const encoder = new TextEncoder();
87
- const decoder = new TextDecoder();
88
- const readStreamSection = async () => {
89
- if (sections.length > 0) {
90
- return sections.shift();
91
- }
92
- while (!closed && sections.length === 0) {
93
- const chunk = await reader.read();
94
- if (chunk.done) {
95
- closed = true;
96
- break;
97
- }
98
- buffer.push(chunk.value);
99
- try {
100
- const bufferedString = decoder.decode(mergeArrays(...buffer));
101
- const splitSections = bufferedString.split("\n\n");
102
- if (splitSections.length >= 2) {
103
- sections.push(...splitSections.slice(0, -1));
104
- buffer = [encoder.encode(splitSections.slice(-1).join("\n\n"))];
105
- }
106
- if (sections.length > 0) {
107
- break;
108
- }
109
- } catch {
110
- continue;
111
- }
112
- }
113
- if (sections.length > 0) {
114
- return sections.shift();
115
- }
116
- if (buffer.length > 0) {
117
- const bufferedString = decoder.decode(mergeArrays(...buffer));
118
- sections = bufferedString.split("\n\n").filter((s) => s);
119
- buffer = [];
120
- }
121
- return sections.shift();
122
- };
123
- let section = await readStreamSection();
124
- while (section) {
125
- yield section;
126
- section = await readStreamSection();
127
- }
128
- }
129
- function mergeArrays(...arrays) {
130
- const out = new Uint8Array(
131
- arrays.reduce((total, arr) => total + arr.length, 0)
132
- );
133
- let offset = 0;
134
- for (const arr of arrays) {
135
- out.set(arr, offset);
136
- offset += arr.length;
137
- }
138
- return out;
139
- }
140
- export {
141
- parseDeferredReadableStream
142
- };
@@ -1,143 +0,0 @@
1
- import {
2
- installGlobals,
3
- writeReadableStreamToWritable,
4
- Response as NodeResponse
5
- } from "@remix-run/node";
6
- import {
7
- transformNestedRoutes
8
- } from "@modern-js/utils";
9
- import {
10
- createStaticHandler,
11
- ErrorResponse,
12
- UNSAFE_DEFERRED_SYMBOL as DEFERRED_SYMBOL
13
- } from "@modern-js/utils/remix-router";
14
- import { LOADER_ID_PARAM } from "../common/constants";
15
- import { createDeferredReadableStream } from "./response";
16
- installGlobals();
17
- const redirectStatusCodes = /* @__PURE__ */ new Set([301, 302, 303, 307, 308]);
18
- function isRedirectResponse(status) {
19
- return redirectStatusCodes.has(status);
20
- }
21
- function isResponse(value) {
22
- return value != null && typeof value.status === "number" && typeof value.statusText === "string" && typeof value.headers === "object" && typeof value.body !== "undefined";
23
- }
24
- function convertModernRedirectResponse(headers) {
25
- const newHeaders = new Headers(headers);
26
- newHeaders.set("X-Modernjs-Redirect", headers.get("Location"));
27
- newHeaders.delete("Location");
28
- return new NodeResponse(null, {
29
- status: 204,
30
- headers: newHeaders
31
- });
32
- }
33
- const createLoaderHeaders = (requestHeaders) => {
34
- const headers = new Headers();
35
- for (const [key, values] of Object.entries(requestHeaders)) {
36
- if (values) {
37
- if (Array.isArray(values)) {
38
- for (const value of values) {
39
- headers.append(key, value);
40
- }
41
- } else {
42
- headers.set(key, values);
43
- }
44
- }
45
- }
46
- return headers;
47
- };
48
- const createLoaderRequest = (context) => {
49
- const origin = `${context.protocol}://${context.host}`;
50
- const url = new URL(context.url, origin);
51
- const controller = new AbortController();
52
- const init = {
53
- method: context.method,
54
- headers: createLoaderHeaders(context.headers),
55
- signal: controller.signal
56
- };
57
- return new Request(url.href, init);
58
- };
59
- const sendLoaderResponse = async (res, nodeResponse) => {
60
- res.statusMessage = nodeResponse.statusText;
61
- res.statusCode = nodeResponse.status;
62
- for (const [key, value] of nodeResponse.headers.entries()) {
63
- res.setHeader(key, value);
64
- }
65
- if (nodeResponse.body) {
66
- await writeReadableStreamToWritable(nodeResponse.body, res);
67
- } else {
68
- res.end();
69
- }
70
- };
71
- const matchEntry = (pathname, entries) => {
72
- return entries.find((entry) => pathname.startsWith(entry.urlPath));
73
- };
74
- const handleRequest = async ({
75
- context,
76
- serverRoutes,
77
- routes
78
- }) => {
79
- const { method, query } = context;
80
- const routeId = query[LOADER_ID_PARAM];
81
- if (!routeId) {
82
- throw new Error(`Missing parameter: ${LOADER_ID_PARAM}`);
83
- }
84
- if (method.toLowerCase() !== "get") {
85
- throw new Error("Only support httpp GET method");
86
- }
87
- const entry = matchEntry(context.path, serverRoutes);
88
- if (!entry) {
89
- throw new Error("Route is not matched");
90
- }
91
- const dataRoutes = transformNestedRoutes(routes);
92
- const staticHandler = createStaticHandler(dataRoutes, {
93
- basename: entry.urlPath
94
- });
95
- const { res } = context;
96
- const request = createLoaderRequest(context);
97
- let response;
98
- try {
99
- response = await staticHandler.queryRoute(request, {
100
- routeId,
101
- requestContext: context
102
- });
103
- if (isResponse(response) && isRedirectResponse(response.status)) {
104
- response = convertModernRedirectResponse(response.headers);
105
- } else if (DEFERRED_SYMBOL in response) {
106
- const deferredData = response[DEFERRED_SYMBOL];
107
- const body = createDeferredReadableStream(deferredData, request.signal);
108
- const init = deferredData.init || {};
109
- if (init.status && isRedirectResponse(init.status)) {
110
- if (!init.headers) {
111
- throw new Error("redirect response includes no headers");
112
- }
113
- response = convertModernRedirectResponse(new Headers(init.headers));
114
- } else {
115
- const headers = new Headers(init.headers);
116
- headers.set("Content-Type", "text/modernjs-deferred");
117
- init.headers = headers;
118
- response = new NodeResponse(body, init);
119
- }
120
- } else {
121
- response = isResponse(response) ? response : new NodeResponse(JSON.stringify(response), {
122
- headers: {
123
- "Content-Type": "application/json; charset=utf-8"
124
- }
125
- });
126
- }
127
- } catch (error) {
128
- const message = error instanceof ErrorResponse ? error.data : String(error);
129
- response = new NodeResponse(message, {
130
- status: 500,
131
- headers: {
132
- "Content-Type": "text/plain"
133
- }
134
- });
135
- }
136
- sendLoaderResponse(res, response);
137
- };
138
- export {
139
- handleRequest,
140
- isRedirectResponse,
141
- isResponse,
142
- matchEntry
143
- };
@@ -1,79 +0,0 @@
1
- import { TextEncoder } from "util";
2
- import { serializeJson } from "@modern-js/utils";
3
- function isTrackedPromise(value) {
4
- return value != null && typeof value.then === "function" && value._tracked === true;
5
- }
6
- const DEFERRED_VALUE_PLACEHOLDER_PREFIX = "__deferred_promise:";
7
- function createDeferredReadableStream(deferredData, signal) {
8
- const encoder = new TextEncoder();
9
- const stream = new ReadableStream({
10
- async start(controller) {
11
- const criticalData = {};
12
- const preresolvedKeys = [];
13
- for (const [key, value] of Object.entries(deferredData.data)) {
14
- if (isTrackedPromise(value)) {
15
- criticalData[key] = `${DEFERRED_VALUE_PLACEHOLDER_PREFIX}${key}`;
16
- if (typeof value._data !== "undefined" || typeof value._error !== "undefined") {
17
- preresolvedKeys.push(key);
18
- }
19
- } else {
20
- criticalData[key] = value;
21
- }
22
- }
23
- controller.enqueue(encoder.encode(`${JSON.stringify(criticalData)}
24
-
25
- `));
26
- for (const preresolvedKey of preresolvedKeys) {
27
- enqueueTrackedPromise(
28
- controller,
29
- encoder,
30
- preresolvedKey,
31
- deferredData.data[preresolvedKey]
32
- );
33
- }
34
- const unsubscribe = deferredData.subscribe((aborted, settledKey) => {
35
- if (settledKey) {
36
- enqueueTrackedPromise(
37
- controller,
38
- encoder,
39
- settledKey,
40
- deferredData.data[settledKey]
41
- );
42
- }
43
- });
44
- await deferredData.resolveData(signal);
45
- unsubscribe();
46
- controller.close();
47
- }
48
- });
49
- return stream;
50
- }
51
- function enqueueTrackedPromise(controller, encoder, settledKey, promise) {
52
- var _a;
53
- if ("_error" in promise) {
54
- const { _error } = promise;
55
- controller.enqueue(
56
- encoder.encode(
57
- `error:${serializeJson({
58
- [settledKey]: {
59
- message: _error.message,
60
- stack: _error.stack
61
- }
62
- })}
63
-
64
- `
65
- )
66
- );
67
- } else {
68
- controller.enqueue(
69
- encoder.encode(
70
- `data:${JSON.stringify({ [settledKey]: (_a = promise._data) != null ? _a : null })}
71
-
72
- `
73
- )
74
- );
75
- }
76
- }
77
- export {
78
- createDeferredReadableStream
79
- };
@@ -1,6 +0,0 @@
1
- /**
2
- * modified from https://github.com/remix-run/remix/blob/main/packages/remix-react/data.ts
3
- * license at https://github.com/remix-run/remix/blob/main/LICENSE.md
4
- */
5
- import { UNSAFE_DeferredData as DeferredData } from '@modern-js/utils/remix-router';
6
- export declare function parseDeferredReadableStream(stream: ReadableStream<Uint8Array>): Promise<DeferredData>;
@@ -1,15 +0,0 @@
1
- import type { ModernServerContext, NestedRoute, ServerRoute } from '@modern-js/types';
2
- import { Response as NodeResponse } from '@remix-run/node';
3
- export type ServerContext = Pick<ModernServerContext, 'req' | 'res' | 'params' | 'headers' | 'method' | 'url' | 'host' | 'protocol' | 'origin' | 'href' | 'path' | 'query'>;
4
- export declare function isRedirectResponse(status: number): boolean;
5
- export declare function isResponse(value: any): value is NodeResponse;
6
- export declare const matchEntry: (pathname: string, entries: ServerRoute[]) => ServerRoute | undefined;
7
- export declare const handleRequest: ({
8
- context,
9
- serverRoutes,
10
- routes
11
- }: {
12
- context: ServerContext;
13
- serverRoutes: ServerRoute[];
14
- routes: NestedRoute[];
15
- }) => Promise<void>;
@@ -1,2 +0,0 @@
1
- import { type UNSAFE_DeferredData as DeferredData } from '@modern-js/utils/remix-router';
2
- export declare function createDeferredReadableStream(deferredData: DeferredData, signal: AbortSignal): any;