@modern-js/plugin-data-loader 2.6.1-alpha.0 → 2.8.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.
Files changed (48) hide show
  1. package/CHANGELOG.md +11 -0
  2. package/README.md +2 -2
  3. package/dist/cjs/cli/{create-request.js → createRequest.js} +17 -6
  4. package/dist/cjs/cli/data.js +162 -0
  5. package/dist/cjs/cli/{generate-client.js → generateClient.js} +4 -4
  6. package/dist/cjs/cli/loader.js +2 -2
  7. package/dist/cjs/common/constants.js +3 -0
  8. package/dist/cjs/runtime/index.js +180 -0
  9. package/dist/cjs/runtime/response.js +102 -0
  10. package/dist/cjs/server/index.js +22 -196
  11. package/dist/esm/cli/{create-request.js → createRequest.js} +42 -3
  12. package/dist/esm/cli/data.js +820 -0
  13. package/dist/esm/cli/{generate-client.js → generateClient.js} +1 -1
  14. package/dist/esm/cli/loader.js +1 -1
  15. package/dist/esm/common/constants.js +2 -1
  16. package/dist/{js/treeshaking/server → esm/runtime}/index.js +84 -220
  17. package/dist/esm/runtime/response.js +284 -0
  18. package/dist/esm/server/index.js +38 -447
  19. package/dist/esm-node/cli/{create-request.js → createRequest.js} +19 -4
  20. package/dist/esm-node/cli/data.js +142 -0
  21. package/dist/esm-node/cli/{generate-client.js → generateClient.js} +1 -1
  22. package/dist/esm-node/cli/loader.js +1 -1
  23. package/dist/esm-node/common/constants.js +2 -0
  24. package/dist/esm-node/runtime/index.js +162 -0
  25. package/dist/esm-node/runtime/response.js +79 -0
  26. package/dist/esm-node/server/index.js +21 -196
  27. package/dist/types/cli/{create-request.d.ts → createRequest.d.ts} +2 -1
  28. package/dist/types/cli/data.d.ts +6 -0
  29. package/dist/types/common/constants.d.ts +2 -1
  30. package/dist/types/runtime/index.d.ts +15 -0
  31. package/dist/types/runtime/response.d.ts +2 -0
  32. package/dist/types/server/index.d.ts +0 -15
  33. package/package.json +22 -12
  34. package/dist/js/modern/cli/create-request.js +0 -66
  35. package/dist/js/modern/cli/generate-client.js +0 -42
  36. package/dist/js/modern/cli/loader.js +0 -40
  37. package/dist/js/modern/common/constants.js +0 -6
  38. package/dist/js/modern/server/index.js +0 -237
  39. package/dist/js/node/cli/create-request.js +0 -90
  40. package/dist/js/node/cli/generate-client.js +0 -71
  41. package/dist/js/node/cli/loader.js +0 -61
  42. package/dist/js/node/common/constants.js +0 -30
  43. package/dist/js/node/server/index.js +0 -261
  44. package/dist/js/treeshaking/cli/create-request.js +0 -186
  45. package/dist/js/treeshaking/cli/generate-client.js +0 -45
  46. package/dist/js/treeshaking/cli/loader.js +0 -154
  47. package/dist/js/treeshaking/common/constants.js +0 -3
  48. /package/dist/types/cli/{generate-client.d.ts → generateClient.d.ts} +0 -0
package/CHANGELOG.md ADDED
@@ -0,0 +1,11 @@
1
+ # @modern-js/plugin-data-loader
2
+
3
+ ## 2.8.0
4
+
5
+ ### Patch Changes
6
+
7
+ - 0cf8540ffe: fix: should match the most similar entry
8
+ fix: 应该匹配最相似的 entry
9
+ - Updated dependencies [1104a9f18b]
10
+ - Updated dependencies [1f6ca2c7fb]
11
+ - @modern-js/utils@2.8.0
package/README.md CHANGED
@@ -19,8 +19,8 @@ Please follow [Quick Start](https://modernjs.dev/en/guides/get-started/quick-sta
19
19
 
20
20
  ## Contributing
21
21
 
22
- Please read the [Contributing Guide](https://github.com/modern-js-dev/modern.js/blob/main/CONTRIBUTING.md).
22
+ Please read the [Contributing Guide](https://github.com/web-infra-dev/modern.js/blob/main/CONTRIBUTING.md).
23
23
 
24
24
  ## License
25
25
 
26
- Modern.js is [MIT licensed](https://github.com/modern-js-dev/modern.js/blob/main/LICENSE).
26
+ Modern.js is [MIT licensed](https://github.com/web-infra-dev/modern.js/blob/main/LICENSE).
@@ -15,15 +15,16 @@ var __copyProps = (to, from, except, desc) => {
15
15
  return to;
16
16
  };
17
17
  var __toCommonJS = (mod) => __copyProps(__defProp({}, "__esModule", { value: true }), mod);
18
- var create_request_exports = {};
19
- __export(create_request_exports, {
18
+ var createRequest_exports = {};
19
+ __export(createRequest_exports, {
20
20
  createRequest: () => createRequest,
21
21
  getRequestUrl: () => getRequestUrl
22
22
  });
23
- module.exports = __toCommonJS(create_request_exports);
23
+ module.exports = __toCommonJS(createRequest_exports);
24
24
  var import_path_to_regexp = require("path-to-regexp");
25
25
  var import_react_router_dom = require("react-router-dom");
26
26
  var import_constants = require("../common/constants");
27
+ var import_data = require("./data");
27
28
  const getRequestUrl = ({
28
29
  params,
29
30
  request,
@@ -43,7 +44,14 @@ const handleRedirectResponse = (res) => {
43
44
  const { headers } = res;
44
45
  const location = headers.get("X-Modernjs-Redirect");
45
46
  if (location) {
46
- return (0, import_react_router_dom.redirect)(location);
47
+ throw (0, import_react_router_dom.redirect)(location);
48
+ }
49
+ return res;
50
+ };
51
+ const handleDeferredResponse = async (res) => {
52
+ var _a;
53
+ if (((_a = res.headers.get("Content-Type")) == null ? void 0 : _a.match(import_constants.CONTENT_TYPE_DEFERRED)) && res.body) {
54
+ return await (0, import_data.parseDeferredReadableStream)(res.body);
47
55
  }
48
56
  return res;
49
57
  };
@@ -53,14 +61,17 @@ const createRequest = (routeId, method = "get") => {
53
61
  request
54
62
  }) => {
55
63
  const url = getRequestUrl({ params, request, routeId });
56
- const res = await fetch(url, {
64
+ let res;
65
+ res = await fetch(url, {
57
66
  method,
58
67
  signal: request.signal
59
68
  });
60
69
  if (!res.ok) {
61
70
  throw res;
62
71
  }
63
- return handleRedirectResponse(res);
72
+ res = handleRedirectResponse(res);
73
+ res = await handleDeferredResponse(res);
74
+ return res;
64
75
  };
65
76
  };
66
77
  // Annotate the CommonJS export names for ESM import in node:
@@ -0,0 +1,162 @@
1
+ var __defProp = Object.defineProperty;
2
+ var __getOwnPropDesc = Object.getOwnPropertyDescriptor;
3
+ var __getOwnPropNames = Object.getOwnPropertyNames;
4
+ var __hasOwnProp = Object.prototype.hasOwnProperty;
5
+ var __export = (target, all) => {
6
+ for (var name in all)
7
+ __defProp(target, name, { get: all[name], enumerable: true });
8
+ };
9
+ var __copyProps = (to, from, except, desc) => {
10
+ if (from && typeof from === "object" || typeof from === "function") {
11
+ for (let key of __getOwnPropNames(from))
12
+ if (!__hasOwnProp.call(to, key) && key !== except)
13
+ __defProp(to, key, { get: () => from[key], enumerable: !(desc = __getOwnPropDesc(from, key)) || desc.enumerable });
14
+ }
15
+ return to;
16
+ };
17
+ var __toCommonJS = (mod) => __copyProps(__defProp({}, "__esModule", { value: true }), mod);
18
+ var data_exports = {};
19
+ __export(data_exports, {
20
+ parseDeferredReadableStream: () => parseDeferredReadableStream
21
+ });
22
+ module.exports = __toCommonJS(data_exports);
23
+ var import_remix_router = require("@modern-js/utils/remix-router");
24
+ const DEFERRED_VALUE_PLACEHOLDER_PREFIX = "__deferred_promise:";
25
+ async function parseDeferredReadableStream(stream) {
26
+ if (!stream) {
27
+ throw new Error("parseDeferredReadableStream requires stream argument");
28
+ }
29
+ let deferredData;
30
+ const deferredResolvers = {};
31
+ try {
32
+ const sectionReader = readStreamSections(stream);
33
+ const initialSectionResult = await sectionReader.next();
34
+ const initialSection = initialSectionResult.value;
35
+ if (!initialSection) {
36
+ throw new Error("no critical data");
37
+ }
38
+ const criticalData = JSON.parse(initialSection);
39
+ if (typeof criticalData === "object" && criticalData !== null) {
40
+ for (const [eventKey, value] of Object.entries(criticalData)) {
41
+ if (typeof value !== "string" || !value.startsWith(DEFERRED_VALUE_PLACEHOLDER_PREFIX)) {
42
+ continue;
43
+ }
44
+ deferredData = deferredData || {};
45
+ deferredData[eventKey] = new Promise((resolve, reject) => {
46
+ deferredResolvers[eventKey] = {
47
+ resolve: (value2) => {
48
+ resolve(value2);
49
+ delete deferredResolvers[eventKey];
50
+ },
51
+ reject: (error) => {
52
+ reject(error);
53
+ delete deferredResolvers[eventKey];
54
+ }
55
+ };
56
+ });
57
+ }
58
+ }
59
+ (async () => {
60
+ try {
61
+ for await (const section of sectionReader) {
62
+ const [event, ...sectionDataStrings] = section.split(":");
63
+ const sectionDataString = sectionDataStrings.join(":");
64
+ const data = JSON.parse(sectionDataString);
65
+ if (event === "data") {
66
+ for (const [key, value] of Object.entries(data)) {
67
+ if (deferredResolvers[key]) {
68
+ deferredResolvers[key].resolve(value);
69
+ }
70
+ }
71
+ } else if (event === "error") {
72
+ for (const [key, value] of Object.entries(data)) {
73
+ const err = new Error(value.message);
74
+ err.stack = value.stack;
75
+ if (deferredResolvers[key]) {
76
+ deferredResolvers[key].reject(err);
77
+ }
78
+ }
79
+ }
80
+ }
81
+ for (const [key, resolver] of Object.entries(deferredResolvers)) {
82
+ resolver.reject(
83
+ new import_remix_router.AbortedDeferredError(`Deferred ${key} will never resolved`)
84
+ );
85
+ }
86
+ } catch (error) {
87
+ for (const resolver of Object.values(deferredResolvers)) {
88
+ resolver.reject(error);
89
+ }
90
+ }
91
+ })();
92
+ return new import_remix_router.UNSAFE_DeferredData({ ...criticalData, ...deferredData });
93
+ } catch (error) {
94
+ for (const resolver of Object.values(deferredResolvers)) {
95
+ resolver.reject(error);
96
+ }
97
+ throw error;
98
+ }
99
+ }
100
+ async function* readStreamSections(stream) {
101
+ const reader = stream.getReader();
102
+ let buffer = [];
103
+ let sections = [];
104
+ let closed = false;
105
+ const encoder = new TextEncoder();
106
+ const decoder = new TextDecoder();
107
+ const readStreamSection = async () => {
108
+ if (sections.length > 0) {
109
+ return sections.shift();
110
+ }
111
+ while (!closed && sections.length === 0) {
112
+ const chunk = await reader.read();
113
+ if (chunk.done) {
114
+ closed = true;
115
+ break;
116
+ }
117
+ buffer.push(chunk.value);
118
+ try {
119
+ const bufferedString = decoder.decode(mergeArrays(...buffer));
120
+ const splitSections = bufferedString.split("\n\n");
121
+ if (splitSections.length >= 2) {
122
+ sections.push(...splitSections.slice(0, -1));
123
+ buffer = [encoder.encode(splitSections.slice(-1).join("\n\n"))];
124
+ }
125
+ if (sections.length > 0) {
126
+ break;
127
+ }
128
+ } catch {
129
+ continue;
130
+ }
131
+ }
132
+ if (sections.length > 0) {
133
+ return sections.shift();
134
+ }
135
+ if (buffer.length > 0) {
136
+ const bufferedString = decoder.decode(mergeArrays(...buffer));
137
+ sections = bufferedString.split("\n\n").filter((s) => s);
138
+ buffer = [];
139
+ }
140
+ return sections.shift();
141
+ };
142
+ let section = await readStreamSection();
143
+ while (section) {
144
+ yield section;
145
+ section = await readStreamSection();
146
+ }
147
+ }
148
+ function mergeArrays(...arrays) {
149
+ const out = new Uint8Array(
150
+ arrays.reduce((total, arr) => total + arr.length, 0)
151
+ );
152
+ let offset = 0;
153
+ for (const arr of arrays) {
154
+ out.set(arr, offset);
155
+ offset += arr.length;
156
+ }
157
+ return out;
158
+ }
159
+ // Annotate the CommonJS export names for ESM import in node:
160
+ 0 && (module.exports = {
161
+ parseDeferredReadableStream
162
+ });
@@ -25,11 +25,11 @@ var __toESM = (mod, isNodeMode, target) => (target = mod != null ? __create(__ge
25
25
  mod
26
26
  ));
27
27
  var __toCommonJS = (mod) => __copyProps(__defProp({}, "__esModule", { value: true }), mod);
28
- var generate_client_exports = {};
29
- __export(generate_client_exports, {
28
+ var generateClient_exports = {};
29
+ __export(generateClient_exports, {
30
30
  generateClient: () => generateClient
31
31
  });
32
- module.exports = __toCommonJS(generate_client_exports);
32
+ module.exports = __toCommonJS(generateClient_exports);
33
33
  var import_path = __toESM(require("path"));
34
34
  const generateClient = ({
35
35
  mapFile,
@@ -39,7 +39,7 @@ const generateClient = ({
39
39
  const loadersMap = require(mapFile);
40
40
  let requestCode = ``;
41
41
  let exportsCode = ``;
42
- const requestCreatorPath = import_path.default.join(__dirname, "./create-request").replace("/cjs/cli/", "/esm/cli/").replace(/\\/g, "/");
42
+ const requestCreatorPath = import_path.default.join(__dirname, "./createRequest").replace("/cjs/cli/", "/esm/cli/").replace(/\\/g, "/");
43
43
  const importCode = `
44
44
  import { createRequest } from '${requestCreatorPath}';
45
45
  `;
@@ -20,7 +20,7 @@ __export(loader_exports, {
20
20
  default: () => loader
21
21
  });
22
22
  module.exports = __toCommonJS(loader_exports);
23
- var import_generate_client = require("./generate-client");
23
+ var import_generateClient = require("./generateClient");
24
24
  async function loader(source) {
25
25
  var _a;
26
26
  this.cacheable();
@@ -32,7 +32,7 @@ async function loader(source) {
32
32
  return source;
33
33
  }
34
34
  const options = this.getOptions();
35
- const code = (0, import_generate_client.generateClient)({
35
+ const code = (0, import_generateClient.generateClient)({
36
36
  mapFile: options.mapFile,
37
37
  loaderId: options.loaderId
38
38
  });
@@ -17,14 +17,17 @@ var __copyProps = (to, from, except, desc) => {
17
17
  var __toCommonJS = (mod) => __copyProps(__defProp({}, "__esModule", { value: true }), mod);
18
18
  var constants_exports = {};
19
19
  __export(constants_exports, {
20
+ CONTENT_TYPE_DEFERRED: () => CONTENT_TYPE_DEFERRED,
20
21
  DIRECT_PARAM: () => DIRECT_PARAM,
21
22
  LOADER_ID_PARAM: () => LOADER_ID_PARAM
22
23
  });
23
24
  module.exports = __toCommonJS(constants_exports);
24
25
  const LOADER_ID_PARAM = "__loader";
25
26
  const DIRECT_PARAM = "__ssrDirect";
27
+ const CONTENT_TYPE_DEFERRED = "text/modernjs-deferred";
26
28
  // Annotate the CommonJS export names for ESM import in node:
27
29
  0 && (module.exports = {
30
+ CONTENT_TYPE_DEFERRED,
28
31
  DIRECT_PARAM,
29
32
  LOADER_ID_PARAM
30
33
  });
@@ -0,0 +1,180 @@
1
+ var __defProp = Object.defineProperty;
2
+ var __getOwnPropDesc = Object.getOwnPropertyDescriptor;
3
+ var __getOwnPropNames = Object.getOwnPropertyNames;
4
+ var __hasOwnProp = Object.prototype.hasOwnProperty;
5
+ var __export = (target, all) => {
6
+ for (var name in all)
7
+ __defProp(target, name, { get: all[name], enumerable: true });
8
+ };
9
+ var __copyProps = (to, from, except, desc) => {
10
+ if (from && typeof from === "object" || typeof from === "function") {
11
+ for (let key of __getOwnPropNames(from))
12
+ if (!__hasOwnProp.call(to, key) && key !== except)
13
+ __defProp(to, key, { get: () => from[key], enumerable: !(desc = __getOwnPropDesc(from, key)) || desc.enumerable });
14
+ }
15
+ return to;
16
+ };
17
+ var __toCommonJS = (mod) => __copyProps(__defProp({}, "__esModule", { value: true }), mod);
18
+ var runtime_exports = {};
19
+ __export(runtime_exports, {
20
+ handleRequest: () => handleRequest,
21
+ isRedirectResponse: () => isRedirectResponse,
22
+ isResponse: () => isResponse,
23
+ matchEntry: () => matchEntry
24
+ });
25
+ module.exports = __toCommonJS(runtime_exports);
26
+ var import_node = require("@remix-run/node");
27
+ var import_nestedRoutes = require("@modern-js/utils/nestedRoutes");
28
+ var import_remix_router = require("@modern-js/utils/remix-router");
29
+ var import_lodash = require("@modern-js/utils/lodash");
30
+ var import_constants = require("../common/constants");
31
+ var import_response = require("./response");
32
+ (0, import_node.installGlobals)();
33
+ const redirectStatusCodes = /* @__PURE__ */ new Set([301, 302, 303, 307, 308]);
34
+ function isRedirectResponse(status) {
35
+ return redirectStatusCodes.has(status);
36
+ }
37
+ function isResponse(value) {
38
+ return value != null && typeof value.status === "number" && typeof value.statusText === "string" && typeof value.headers === "object" && typeof value.body !== "undefined";
39
+ }
40
+ function sortByUrlPath(entries) {
41
+ entries.sort(function(a, b) {
42
+ const length1 = a.urlPath.length;
43
+ const length2 = b.urlPath.length;
44
+ if (length1 < length2) {
45
+ return 1;
46
+ }
47
+ if (length1 > length2) {
48
+ return -1;
49
+ }
50
+ return 0;
51
+ });
52
+ return entries;
53
+ }
54
+ function convertModernRedirectResponse(headers, basename) {
55
+ const newHeaders = new Headers(headers);
56
+ let redirectUrl = headers.get("Location");
57
+ if (basename !== "/") {
58
+ redirectUrl = redirectUrl.replace(basename, "");
59
+ }
60
+ newHeaders.set("X-Modernjs-Redirect", redirectUrl);
61
+ newHeaders.delete("Location");
62
+ return new import_node.Response(null, {
63
+ status: 204,
64
+ headers: newHeaders
65
+ });
66
+ }
67
+ const createLoaderHeaders = (requestHeaders) => {
68
+ const headers = new Headers();
69
+ for (const [key, values] of Object.entries(requestHeaders)) {
70
+ if (values) {
71
+ if (Array.isArray(values)) {
72
+ for (const value of values) {
73
+ headers.append(key, value);
74
+ }
75
+ } else {
76
+ headers.set(key, values);
77
+ }
78
+ }
79
+ }
80
+ return headers;
81
+ };
82
+ const createLoaderRequest = (context) => {
83
+ const origin = `${context.protocol}://${context.host}`;
84
+ const url = new URL(context.url, origin);
85
+ const controller = new AbortController();
86
+ const init = {
87
+ method: context.method,
88
+ headers: createLoaderHeaders(context.headers),
89
+ signal: controller.signal
90
+ };
91
+ return new Request(url.href, init);
92
+ };
93
+ const sendLoaderResponse = async (res, nodeResponse) => {
94
+ res.statusMessage = nodeResponse.statusText;
95
+ res.statusCode = nodeResponse.status;
96
+ for (const [key, value] of nodeResponse.headers.entries()) {
97
+ res.setHeader(key, value);
98
+ }
99
+ if (nodeResponse.body) {
100
+ await (0, import_node.writeReadableStreamToWritable)(nodeResponse.body, res);
101
+ } else {
102
+ res.end();
103
+ }
104
+ };
105
+ const matchEntry = (pathname, entries) => {
106
+ sortByUrlPath(entries);
107
+ return entries.find((entry) => pathname.startsWith(entry.urlPath));
108
+ };
109
+ const handleRequest = async ({
110
+ context,
111
+ serverRoutes,
112
+ routes
113
+ }) => {
114
+ const { method, query } = context;
115
+ const routeId = query[import_constants.LOADER_ID_PARAM];
116
+ const entry = matchEntry(context.path, serverRoutes);
117
+ if (!routeId || !entry) {
118
+ return;
119
+ }
120
+ if (method.toLowerCase() !== "get") {
121
+ throw new Error("CSR data loader request only support http GET method");
122
+ }
123
+ const basename = entry.urlPath;
124
+ const dataRoutes = (0, import_nestedRoutes.transformNestedRoutes)(routes);
125
+ const staticHandler = (0, import_remix_router.createStaticHandler)(dataRoutes, {
126
+ basename
127
+ });
128
+ const { res } = context;
129
+ const request = createLoaderRequest(context);
130
+ let response;
131
+ try {
132
+ response = await staticHandler.queryRoute(request, {
133
+ routeId,
134
+ requestContext: context
135
+ });
136
+ if (isResponse(response) && isRedirectResponse(response.status)) {
137
+ response = convertModernRedirectResponse(response.headers, basename);
138
+ } else if ((0, import_lodash.isPlainObject)(response) && import_remix_router.UNSAFE_DEFERRED_SYMBOL in response) {
139
+ const deferredData = response[import_remix_router.UNSAFE_DEFERRED_SYMBOL];
140
+ const body = (0, import_response.createDeferredReadableStream)(deferredData, request.signal);
141
+ const init = deferredData.init || {};
142
+ if (init.status && isRedirectResponse(init.status)) {
143
+ if (!init.headers) {
144
+ throw new Error("redirect response includes no headers");
145
+ }
146
+ response = convertModernRedirectResponse(
147
+ new Headers(init.headers),
148
+ basename
149
+ );
150
+ } else {
151
+ const headers = new Headers(init.headers);
152
+ headers.set("Content-Type", import_constants.CONTENT_TYPE_DEFERRED);
153
+ init.headers = headers;
154
+ response = new import_node.Response(body, init);
155
+ }
156
+ } else {
157
+ response = isResponse(response) ? response : new import_node.Response(JSON.stringify(response), {
158
+ headers: {
159
+ "Content-Type": "application/json; charset=utf-8"
160
+ }
161
+ });
162
+ }
163
+ } catch (error) {
164
+ const message = error instanceof import_remix_router.ErrorResponse ? error.data : String(error);
165
+ response = new import_node.Response(message, {
166
+ status: 500,
167
+ headers: {
168
+ "Content-Type": "text/plain"
169
+ }
170
+ });
171
+ }
172
+ sendLoaderResponse(res, response);
173
+ };
174
+ // Annotate the CommonJS export names for ESM import in node:
175
+ 0 && (module.exports = {
176
+ handleRequest,
177
+ isRedirectResponse,
178
+ isResponse,
179
+ matchEntry
180
+ });
@@ -0,0 +1,102 @@
1
+ var __defProp = Object.defineProperty;
2
+ var __getOwnPropDesc = Object.getOwnPropertyDescriptor;
3
+ var __getOwnPropNames = Object.getOwnPropertyNames;
4
+ var __hasOwnProp = Object.prototype.hasOwnProperty;
5
+ var __export = (target, all) => {
6
+ for (var name in all)
7
+ __defProp(target, name, { get: all[name], enumerable: true });
8
+ };
9
+ var __copyProps = (to, from, except, desc) => {
10
+ if (from && typeof from === "object" || typeof from === "function") {
11
+ for (let key of __getOwnPropNames(from))
12
+ if (!__hasOwnProp.call(to, key) && key !== except)
13
+ __defProp(to, key, { get: () => from[key], enumerable: !(desc = __getOwnPropDesc(from, key)) || desc.enumerable });
14
+ }
15
+ return to;
16
+ };
17
+ var __toCommonJS = (mod) => __copyProps(__defProp({}, "__esModule", { value: true }), mod);
18
+ var response_exports = {};
19
+ __export(response_exports, {
20
+ createDeferredReadableStream: () => createDeferredReadableStream
21
+ });
22
+ module.exports = __toCommonJS(response_exports);
23
+ var import_util = require("util");
24
+ var import_serialize = require("@modern-js/utils/serialize");
25
+ function isTrackedPromise(value) {
26
+ return value != null && typeof value.then === "function" && value._tracked === true;
27
+ }
28
+ const DEFERRED_VALUE_PLACEHOLDER_PREFIX = "__deferred_promise:";
29
+ function createDeferredReadableStream(deferredData, signal) {
30
+ const encoder = new import_util.TextEncoder();
31
+ const stream = new ReadableStream({
32
+ async start(controller) {
33
+ const criticalData = {};
34
+ const preresolvedKeys = [];
35
+ for (const [key, value] of Object.entries(deferredData.data)) {
36
+ if (isTrackedPromise(value)) {
37
+ criticalData[key] = `${DEFERRED_VALUE_PLACEHOLDER_PREFIX}${key}`;
38
+ if (typeof value._data !== "undefined" || typeof value._error !== "undefined") {
39
+ preresolvedKeys.push(key);
40
+ }
41
+ } else {
42
+ criticalData[key] = value;
43
+ }
44
+ }
45
+ controller.enqueue(encoder.encode(`${JSON.stringify(criticalData)}
46
+
47
+ `));
48
+ for (const preresolvedKey of preresolvedKeys) {
49
+ enqueueTrackedPromise(
50
+ controller,
51
+ encoder,
52
+ preresolvedKey,
53
+ deferredData.data[preresolvedKey]
54
+ );
55
+ }
56
+ const unsubscribe = deferredData.subscribe((aborted, settledKey) => {
57
+ if (settledKey) {
58
+ enqueueTrackedPromise(
59
+ controller,
60
+ encoder,
61
+ settledKey,
62
+ deferredData.data[settledKey]
63
+ );
64
+ }
65
+ });
66
+ await deferredData.resolveData(signal);
67
+ unsubscribe();
68
+ controller.close();
69
+ }
70
+ });
71
+ return stream;
72
+ }
73
+ function enqueueTrackedPromise(controller, encoder, settledKey, promise) {
74
+ var _a;
75
+ if ("_error" in promise) {
76
+ const { _error } = promise;
77
+ controller.enqueue(
78
+ encoder.encode(
79
+ `error:${(0, import_serialize.serializeJson)({
80
+ [settledKey]: {
81
+ message: _error.message,
82
+ stack: _error.stack
83
+ }
84
+ })}
85
+
86
+ `
87
+ )
88
+ );
89
+ } else {
90
+ controller.enqueue(
91
+ encoder.encode(
92
+ `data:${JSON.stringify({ [settledKey]: (_a = promise._data) != null ? _a : null })}
93
+
94
+ `
95
+ )
96
+ );
97
+ }
98
+ }
99
+ // Annotate the CommonJS export names for ESM import in node:
100
+ 0 && (module.exports = {
101
+ createDeferredReadableStream
102
+ });