vafast 0.3.2 → 0.3.4
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.
- package/dist/auth/token.d.ts +13 -11
- package/dist/auth/token.js +118 -111
- package/dist/auth/token.js.map +1 -0
- package/dist/defineRoute.d.ts +5 -2
- package/dist/defineRoute.js +7 -2
- package/dist/defineRoute.js.map +1 -0
- package/dist/index.d.ts +30 -14
- package/dist/index.js +2247 -15
- package/dist/index.js.map +1 -0
- package/dist/middleware/auth.d.ts +8 -6
- package/dist/middleware/auth.js +198 -99
- package/dist/middleware/auth.js.map +1 -0
- package/dist/middleware/authMiddleware.d.ts +5 -2
- package/dist/middleware/authMiddleware.js +55 -11
- package/dist/middleware/authMiddleware.js.map +1 -0
- package/dist/middleware/component-renderer.d.ts +4 -2
- package/dist/middleware/component-renderer.js +87 -80
- package/dist/middleware/component-renderer.js.map +1 -0
- package/dist/middleware/component-router.d.ts +8 -3
- package/dist/middleware/component-router.js +33 -39
- package/dist/middleware/component-router.js.map +1 -0
- package/dist/middleware/cors.d.ts +6 -3
- package/dist/middleware/cors.js +42 -29
- package/dist/middleware/cors.js.map +1 -0
- package/dist/middleware/rateLimit.d.ts +5 -3
- package/dist/middleware/rateLimit.js +45 -29
- package/dist/middleware/rateLimit.js.map +1 -0
- package/dist/middleware.d.ts +6 -3
- package/dist/middleware.js +97 -51
- package/dist/middleware.js.map +1 -0
- package/dist/monitoring/index.d.ts +11 -4
- package/dist/monitoring/index.js +1299 -17
- package/dist/monitoring/index.js.map +1 -0
- package/dist/monitoring/native-monitor.d.ts +12 -6
- package/dist/monitoring/native-monitor.js +1258 -161
- package/dist/monitoring/native-monitor.js.map +1 -0
- package/dist/monitoring/types.d.ts +8 -6
- package/dist/monitoring/types.js +1 -1
- package/dist/monitoring/types.js.map +1 -0
- package/dist/node-server/index.d.ts +4 -22
- package/dist/node-server/index.js +254 -21
- package/dist/node-server/index.js.map +1 -0
- package/dist/node-server/request.d.ts +6 -2
- package/dist/node-server/request.js +102 -134
- package/dist/node-server/request.js.map +1 -0
- package/dist/node-server/response.d.ts +7 -3
- package/dist/node-server/response.js +67 -89
- package/dist/node-server/response.js.map +1 -0
- package/dist/node-server/serve.d.ts +11 -7
- package/dist/node-server/serve.js +231 -82
- package/dist/node-server/serve.js.map +1 -0
- package/dist/router/index.d.ts +3 -5
- package/dist/router/index.js +228 -7
- package/dist/router/index.js.map +1 -0
- package/dist/router/radix-tree.d.ts +7 -4
- package/dist/router/radix-tree.js +186 -218
- package/dist/router/radix-tree.js.map +1 -0
- package/dist/router.d.ts +7 -3
- package/dist/router.js +37 -83
- package/dist/router.js.map +1 -0
- package/dist/serve.d.ts +2 -12
- package/dist/serve.js +237 -11
- package/dist/serve.js.map +1 -0
- package/dist/server/base-server.d.ts +5 -2
- package/dist/server/base-server.js +124 -135
- package/dist/server/base-server.js.map +1 -0
- package/dist/server/component-server.d.ts +9 -4
- package/dist/server/component-server.js +481 -139
- package/dist/server/component-server.js.map +1 -0
- package/dist/server/index.d.ts +8 -7
- package/dist/server/index.js +985 -11
- package/dist/server/index.js.map +1 -0
- package/dist/server/server-factory.d.ts +11 -5
- package/dist/server/server-factory.js +979 -67
- package/dist/server/server-factory.js.map +1 -0
- package/dist/server/server.d.ts +7 -3
- package/dist/server/server.js +553 -112
- package/dist/server/server.js.map +1 -0
- package/dist/types/component-route.d.ts +8 -4
- package/dist/types/component-route.js +1 -1
- package/dist/types/component-route.js.map +1 -0
- package/dist/types/index.d.ts +5 -5
- package/dist/types/index.js +21 -4
- package/dist/types/index.js.map +1 -0
- package/dist/types/route.d.ts +13 -10
- package/dist/types/route.js +10 -9
- package/dist/types/route.js.map +1 -0
- package/dist/types/schema.d.ts +11 -7
- package/dist/types/schema.js +1 -1
- package/dist/types/schema.js.map +1 -0
- package/dist/types/types.d.ts +11 -9
- package/dist/types/types.js +1 -1
- package/dist/types/types.js.map +1 -0
- package/dist/utils/base64url.d.ts +4 -2
- package/dist/utils/base64url.js +12 -9
- package/dist/utils/base64url.js.map +1 -0
- package/dist/utils/create-handler.d.ts +11 -7
- package/dist/utils/create-handler.js +393 -217
- package/dist/utils/create-handler.js.map +1 -0
- package/dist/utils/dependency-manager.d.ts +3 -1
- package/dist/utils/dependency-manager.js +67 -69
- package/dist/utils/dependency-manager.js.map +1 -0
- package/dist/utils/go-await.d.ts +3 -1
- package/dist/utils/go-await.js +8 -22
- package/dist/utils/go-await.js.map +1 -0
- package/dist/utils/handle.d.ts +6 -4
- package/dist/utils/handle.js +44 -25
- package/dist/utils/handle.js.map +1 -0
- package/dist/utils/html-renderer.d.ts +3 -1
- package/dist/utils/html-renderer.js +25 -24
- package/dist/utils/html-renderer.js.map +1 -0
- package/dist/utils/index.d.ts +13 -13
- package/dist/utils/index.js +832 -21
- package/dist/utils/index.js.map +1 -0
- package/dist/utils/parsers.d.ts +15 -13
- package/dist/utils/parsers.js +138 -188
- package/dist/utils/parsers.js.map +1 -0
- package/dist/utils/path-matcher.d.ts +3 -1
- package/dist/utils/path-matcher.js +68 -78
- package/dist/utils/path-matcher.js.map +1 -0
- package/dist/utils/request-validator.d.ts +13 -10
- package/dist/utils/request-validator.js +234 -84
- package/dist/utils/request-validator.js.map +1 -0
- package/dist/utils/response.d.ts +9 -7
- package/dist/utils/response.js +93 -102
- package/dist/utils/response.js.map +1 -0
- package/dist/utils/validators/schema-validator.d.ts +13 -9
- package/dist/utils/validators/schema-validator.js +228 -209
- package/dist/utils/validators/schema-validator.js.map +1 -0
- package/dist/utils/validators/schema-validators-ultra.d.ts +15 -12
- package/dist/utils/validators/schema-validators-ultra.js +233 -256
- package/dist/utils/validators/schema-validators-ultra.js.map +1 -0
- package/dist/utils/validators/validators.d.ts +15 -12
- package/dist/utils/validators/validators.js +81 -122
- package/dist/utils/validators/validators.js.map +1 -0
- package/package.json +5 -4
|
@@ -1,146 +1,488 @@
|
|
|
1
|
-
|
|
2
|
-
|
|
3
|
-
|
|
4
|
-
|
|
5
|
-
|
|
6
|
-
|
|
7
|
-
|
|
8
|
-
|
|
9
|
-
|
|
10
|
-
|
|
11
|
-
|
|
12
|
-
|
|
13
|
-
|
|
14
|
-
|
|
15
|
-
|
|
16
|
-
|
|
17
|
-
|
|
18
|
-
|
|
19
|
-
|
|
20
|
-
|
|
21
|
-
|
|
22
|
-
|
|
23
|
-
|
|
24
|
-
|
|
25
|
-
|
|
26
|
-
|
|
27
|
-
|
|
28
|
-
|
|
29
|
-
|
|
30
|
-
|
|
31
|
-
|
|
32
|
-
|
|
33
|
-
|
|
34
|
-
|
|
35
|
-
|
|
36
|
-
|
|
37
|
-
|
|
38
|
-
|
|
39
|
-
|
|
1
|
+
// src/middleware/component-router.ts
|
|
2
|
+
function flattenComponentRoutes(routes) {
|
|
3
|
+
const flattened = [];
|
|
4
|
+
function processRoute(route, parentPath = "", parentMiddleware = []) {
|
|
5
|
+
const currentPath = parentPath + route.path;
|
|
6
|
+
const currentMiddleware = [
|
|
7
|
+
...parentMiddleware,
|
|
8
|
+
...route.middleware || []
|
|
9
|
+
];
|
|
10
|
+
if ("component" in route) {
|
|
11
|
+
flattened.push({
|
|
12
|
+
...route,
|
|
13
|
+
fullPath: currentPath,
|
|
14
|
+
middlewareChain: currentMiddleware
|
|
15
|
+
});
|
|
16
|
+
} else if ("children" in route && route.children) {
|
|
17
|
+
for (const child of route.children) {
|
|
18
|
+
processRoute(child, currentPath, currentMiddleware);
|
|
19
|
+
}
|
|
20
|
+
}
|
|
21
|
+
}
|
|
22
|
+
for (const route of routes) {
|
|
23
|
+
processRoute(route);
|
|
24
|
+
}
|
|
25
|
+
return flattened;
|
|
26
|
+
}
|
|
27
|
+
|
|
28
|
+
// src/server/base-server.ts
|
|
29
|
+
var BaseServer = class {
|
|
30
|
+
globalMiddleware = [];
|
|
31
|
+
use(mw) {
|
|
32
|
+
this.globalMiddleware.push(mw);
|
|
33
|
+
}
|
|
34
|
+
/**
|
|
35
|
+
* 打印扁平化后的路由信息,用于调试
|
|
36
|
+
*/
|
|
37
|
+
logFlattenedRoutes(routes, type = "\u8DEF\u7531") {
|
|
38
|
+
console.log(`\u{1F680} \u6241\u5E73\u5316\u540E\u7684${type}:`);
|
|
39
|
+
for (const route of routes) {
|
|
40
|
+
const method = route.method || "GET";
|
|
41
|
+
const path = route.fullPath || route.path;
|
|
42
|
+
console.log(` ${method} ${path}`);
|
|
43
|
+
if (route.middlewareChain && route.middlewareChain.length > 0) {
|
|
44
|
+
console.log(` \u4E2D\u95F4\u4EF6\u94FE: ${route.middlewareChain.length} \u4E2A`);
|
|
45
|
+
}
|
|
46
|
+
}
|
|
47
|
+
console.log("");
|
|
48
|
+
}
|
|
49
|
+
/**
|
|
50
|
+
* 检测路由冲突
|
|
51
|
+
* 检查是否有路径相同但方法不同的路由,以及潜在的路径冲突
|
|
52
|
+
*/
|
|
53
|
+
detectRouteConflicts(routes) {
|
|
54
|
+
const pathGroups = /* @__PURE__ */ new Map();
|
|
55
|
+
for (const route of routes) {
|
|
56
|
+
const path = route.fullPath || route.path;
|
|
57
|
+
const method = route.method || "GET";
|
|
58
|
+
if (!pathGroups.has(path)) {
|
|
59
|
+
pathGroups.set(path, []);
|
|
60
|
+
}
|
|
61
|
+
pathGroups.get(path).push({ ...route, method });
|
|
62
|
+
}
|
|
63
|
+
for (const [path, routeList] of pathGroups) {
|
|
64
|
+
if (routeList.length > 1) {
|
|
65
|
+
const methods = routeList.map((r) => r.method);
|
|
66
|
+
const uniqueMethods = [...new Set(methods)];
|
|
67
|
+
if (uniqueMethods.length === 1) {
|
|
68
|
+
console.warn(
|
|
69
|
+
`\u26A0\uFE0F \u8DEF\u7531\u51B2\u7A81: ${uniqueMethods[0]} ${path} \u5B9A\u4E49\u4E86 ${routeList.length} \u6B21`
|
|
70
|
+
);
|
|
71
|
+
routeList.forEach((route, index) => {
|
|
72
|
+
console.warn(` ${index + 1}. ${route.method} ${path}`);
|
|
73
|
+
});
|
|
74
|
+
} else {
|
|
75
|
+
console.log(`\u2139\uFE0F \u8DEF\u5F84 ${path} \u652F\u6301\u65B9\u6CD5: ${uniqueMethods.join(", ")}`);
|
|
40
76
|
}
|
|
41
|
-
|
|
42
|
-
|
|
77
|
+
}
|
|
78
|
+
}
|
|
79
|
+
this.detectDynamicRouteConflicts(routes);
|
|
80
|
+
}
|
|
81
|
+
/**
|
|
82
|
+
* 检测动态路由的潜在冲突
|
|
83
|
+
*/
|
|
84
|
+
detectDynamicRouteConflicts(routes) {
|
|
85
|
+
const dynamicRoutes = routes.filter((r) => {
|
|
86
|
+
const path = r.fullPath || r.path;
|
|
87
|
+
return path.includes(":") || path.includes("*");
|
|
88
|
+
});
|
|
89
|
+
for (let i = 0; i < dynamicRoutes.length; i++) {
|
|
90
|
+
for (let j = i + 1; j < dynamicRoutes.length; j++) {
|
|
91
|
+
const route1 = dynamicRoutes[i];
|
|
92
|
+
const route2 = dynamicRoutes[j];
|
|
93
|
+
const method1 = route1.method || "GET";
|
|
94
|
+
const method2 = route2.method || "GET";
|
|
95
|
+
if (method1 === method2) {
|
|
96
|
+
const path1 = route1.fullPath || route1.path;
|
|
97
|
+
const path2 = route2.fullPath || route2.path;
|
|
98
|
+
if (this.pathsMayConflict(path1, path2)) {
|
|
99
|
+
console.warn(
|
|
100
|
+
`\u26A0\uFE0F \u6F5C\u5728\u8DEF\u7531\u51B2\u7A81: ${method1} ${path1} \u53EF\u80FD\u4E0E ${path2} \u51B2\u7A81`
|
|
101
|
+
);
|
|
102
|
+
}
|
|
43
103
|
}
|
|
44
|
-
|
|
45
|
-
|
|
46
|
-
|
|
47
|
-
|
|
48
|
-
|
|
49
|
-
|
|
50
|
-
|
|
104
|
+
}
|
|
105
|
+
}
|
|
106
|
+
}
|
|
107
|
+
/**
|
|
108
|
+
* 判断两个路径是否可能冲突
|
|
109
|
+
*/
|
|
110
|
+
pathsMayConflict(path1, path2) {
|
|
111
|
+
const parts1 = path1.split("/").filter(Boolean);
|
|
112
|
+
const parts2 = path2.split("/").filter(Boolean);
|
|
113
|
+
if (parts1.length !== parts2.length) return false;
|
|
114
|
+
for (let i = 0; i < parts1.length; i++) {
|
|
115
|
+
const p1 = parts1[i];
|
|
116
|
+
const p2 = parts2[i];
|
|
117
|
+
if (!p1.startsWith(":") && !p1.startsWith("*") && !p2.startsWith(":") && !p2.startsWith("*") && p1 !== p2) {
|
|
118
|
+
return false;
|
|
119
|
+
}
|
|
120
|
+
if (p1 === "*" && p2.startsWith(":") || p2 === "*" && p1.startsWith(":")) {
|
|
121
|
+
return true;
|
|
122
|
+
}
|
|
123
|
+
}
|
|
124
|
+
return false;
|
|
125
|
+
}
|
|
126
|
+
/**
|
|
127
|
+
* 路径匹配
|
|
128
|
+
*/
|
|
129
|
+
matchPath(pattern, path) {
|
|
130
|
+
const patternParts = pattern.split("/").filter(Boolean);
|
|
131
|
+
const pathParts = path.split("/").filter(Boolean);
|
|
132
|
+
if (patternParts.length !== pathParts.length) {
|
|
133
|
+
return false;
|
|
134
|
+
}
|
|
135
|
+
for (let i = 0; i < patternParts.length; i++) {
|
|
136
|
+
if (patternParts[i] !== pathParts[i] && !patternParts[i].startsWith(":")) {
|
|
137
|
+
return false;
|
|
138
|
+
}
|
|
139
|
+
}
|
|
140
|
+
return true;
|
|
141
|
+
}
|
|
142
|
+
/**
|
|
143
|
+
* 提取路径参数
|
|
144
|
+
*/
|
|
145
|
+
extractParams(pattern, path) {
|
|
146
|
+
const params = {};
|
|
147
|
+
const patternParts = pattern.split("/").filter(Boolean);
|
|
148
|
+
const pathParts = path.split("/").filter(Boolean);
|
|
149
|
+
for (let i = 0; i < patternParts.length; i++) {
|
|
150
|
+
if (patternParts[i].startsWith(":")) {
|
|
151
|
+
const paramName = patternParts[i].slice(1);
|
|
152
|
+
params[paramName] = pathParts[i];
|
|
153
|
+
}
|
|
154
|
+
}
|
|
155
|
+
return params;
|
|
156
|
+
}
|
|
157
|
+
};
|
|
158
|
+
|
|
159
|
+
// src/utils/path-matcher.ts
|
|
160
|
+
var PathMatcher = class {
|
|
161
|
+
/**
|
|
162
|
+
* 路径匹配
|
|
163
|
+
*/
|
|
164
|
+
static matchPath(pattern, path) {
|
|
165
|
+
const patternParts = pattern.split("/").filter(Boolean);
|
|
166
|
+
const pathParts = path.split("/").filter(Boolean);
|
|
167
|
+
if (patternParts.length !== pathParts.length) {
|
|
168
|
+
return false;
|
|
169
|
+
}
|
|
170
|
+
for (let i = 0; i < patternParts.length; i++) {
|
|
171
|
+
if (patternParts[i] !== pathParts[i] && !patternParts[i].startsWith(":")) {
|
|
172
|
+
return false;
|
|
173
|
+
}
|
|
174
|
+
}
|
|
175
|
+
return true;
|
|
176
|
+
}
|
|
177
|
+
/**
|
|
178
|
+
* 提取路径参数
|
|
179
|
+
*/
|
|
180
|
+
static extractParams(pattern, path) {
|
|
181
|
+
const params = {};
|
|
182
|
+
const patternParts = pattern.split("/").filter(Boolean);
|
|
183
|
+
const pathParts = path.split("/").filter(Boolean);
|
|
184
|
+
for (let i = 0; i < patternParts.length; i++) {
|
|
185
|
+
if (patternParts[i].startsWith(":")) {
|
|
186
|
+
const paramName = patternParts[i].slice(1);
|
|
187
|
+
params[paramName] = pathParts[i];
|
|
188
|
+
}
|
|
189
|
+
}
|
|
190
|
+
return params;
|
|
191
|
+
}
|
|
192
|
+
/**
|
|
193
|
+
* 计算路径特异性分数
|
|
194
|
+
* 用于路由排序:静态 > 动态(:param) > 通配符(*)
|
|
195
|
+
*/
|
|
196
|
+
static calculatePathScore(path) {
|
|
197
|
+
const parts = path.split("/").filter(Boolean);
|
|
198
|
+
let score = 0;
|
|
199
|
+
for (const p of parts) {
|
|
200
|
+
if (p === "*")
|
|
201
|
+
score += 1;
|
|
202
|
+
else if (p.startsWith(":"))
|
|
203
|
+
score += 2;
|
|
204
|
+
else score += 3;
|
|
205
|
+
}
|
|
206
|
+
return score * 10 + parts.length;
|
|
207
|
+
}
|
|
208
|
+
/**
|
|
209
|
+
* 判断两个路径是否可能冲突
|
|
210
|
+
*/
|
|
211
|
+
static pathsMayConflict(path1, path2) {
|
|
212
|
+
const parts1 = path1.split("/").filter(Boolean);
|
|
213
|
+
const parts2 = path2.split("/").filter(Boolean);
|
|
214
|
+
if (parts1.length !== parts2.length) return false;
|
|
215
|
+
for (let i = 0; i < parts1.length; i++) {
|
|
216
|
+
const p1 = parts1[i];
|
|
217
|
+
const p2 = parts2[i];
|
|
218
|
+
if (!p1.startsWith(":") && !p1.startsWith("*") && !p2.startsWith(":") && !p2.startsWith("*") && p1 !== p2) {
|
|
219
|
+
return false;
|
|
220
|
+
}
|
|
221
|
+
if (p1 === "*" && p2.startsWith(":") || p2 === "*" && p1.startsWith(":")) {
|
|
222
|
+
return true;
|
|
223
|
+
}
|
|
224
|
+
}
|
|
225
|
+
return false;
|
|
226
|
+
}
|
|
227
|
+
};
|
|
228
|
+
|
|
229
|
+
// src/utils/html-renderer.ts
|
|
230
|
+
var HtmlRenderer = class {
|
|
231
|
+
/**
|
|
232
|
+
* 生成基础HTML模板
|
|
233
|
+
*/
|
|
234
|
+
static generateBaseHtml(content, context, clientScriptPath = "/client.js") {
|
|
235
|
+
return `
|
|
236
|
+
<!doctype html>
|
|
237
|
+
<html>
|
|
238
|
+
<head>
|
|
239
|
+
<meta charset="utf-8">
|
|
240
|
+
<title>Vafast SSR App</title>
|
|
241
|
+
<meta name="viewport" content="width=device-width, initial-scale=1">
|
|
242
|
+
</head>
|
|
243
|
+
<body>
|
|
244
|
+
<div id="app">${content}</div>
|
|
245
|
+
<script>
|
|
246
|
+
window.__ROUTE_INFO__ = {
|
|
247
|
+
params: ${JSON.stringify(context.params || {})},
|
|
248
|
+
query: ${JSON.stringify(context.query || {})},
|
|
249
|
+
pathname: '${context.pathname}'
|
|
51
250
|
};
|
|
52
|
-
|
|
53
|
-
|
|
54
|
-
|
|
55
|
-
|
|
56
|
-
|
|
57
|
-
|
|
58
|
-
|
|
251
|
+
</script>
|
|
252
|
+
<script type="module" src="${clientScriptPath}"></script>
|
|
253
|
+
</body>
|
|
254
|
+
</html>
|
|
255
|
+
`;
|
|
256
|
+
}
|
|
257
|
+
/**
|
|
258
|
+
* 生成Vue组件HTML
|
|
259
|
+
*/
|
|
260
|
+
static generateVueHtml(content, context, clientScriptPath = "/client.js") {
|
|
261
|
+
return this.generateBaseHtml(content, context, clientScriptPath);
|
|
262
|
+
}
|
|
263
|
+
/**
|
|
264
|
+
* 生成React组件HTML
|
|
265
|
+
*/
|
|
266
|
+
static generateReactHtml(content, context, clientScriptPath = "/client.js") {
|
|
267
|
+
return `
|
|
268
|
+
<!doctype html>
|
|
269
|
+
<html>
|
|
270
|
+
<head>
|
|
271
|
+
<meta charset="utf-8">
|
|
272
|
+
<title>Vafast SSR App</title>
|
|
273
|
+
<meta name="viewport" content="width=device-width, initial-scale=1">
|
|
274
|
+
</head>
|
|
275
|
+
<body>
|
|
276
|
+
<div id="root">${content}</div>
|
|
277
|
+
<script>
|
|
278
|
+
window.__ROUTE_INFO__ = {
|
|
279
|
+
params: ${JSON.stringify(context.params || {})},
|
|
280
|
+
query: ${JSON.stringify(context.query || {})},
|
|
281
|
+
pathname: '${context.pathname}'
|
|
282
|
+
};
|
|
283
|
+
</script>
|
|
284
|
+
<script type="module" src="${clientScriptPath}"></script>
|
|
285
|
+
</body>
|
|
286
|
+
</html>
|
|
287
|
+
`;
|
|
288
|
+
}
|
|
289
|
+
};
|
|
290
|
+
|
|
291
|
+
// src/utils/dependency-manager.ts
|
|
292
|
+
var DependencyManager = class {
|
|
293
|
+
dependencyCache = /* @__PURE__ */ new Map();
|
|
294
|
+
/**
|
|
295
|
+
* 按需获取框架依赖
|
|
296
|
+
*/
|
|
297
|
+
async getFrameworkDeps(framework) {
|
|
298
|
+
if (this.dependencyCache.has(framework)) {
|
|
299
|
+
return this.dependencyCache.get(framework);
|
|
59
300
|
}
|
|
60
|
-
|
|
61
|
-
|
|
62
|
-
|
|
63
|
-
|
|
64
|
-
|
|
65
|
-
|
|
66
|
-
|
|
67
|
-
|
|
68
|
-
|
|
69
|
-
|
|
70
|
-
|
|
71
|
-
|
|
72
|
-
|
|
73
|
-
|
|
74
|
-
|
|
75
|
-
|
|
76
|
-
|
|
77
|
-
|
|
78
|
-
|
|
79
|
-
|
|
80
|
-
|
|
81
|
-
|
|
82
|
-
|
|
83
|
-
|
|
84
|
-
|
|
85
|
-
const next = async () => {
|
|
86
|
-
if (index >= middlewareChain.length) {
|
|
87
|
-
return await renderComponent();
|
|
88
|
-
}
|
|
89
|
-
const middleware = middlewareChain[index++];
|
|
90
|
-
return await middleware(context.req, next);
|
|
91
|
-
};
|
|
92
|
-
return await next();
|
|
93
|
-
}
|
|
94
|
-
/**
|
|
95
|
-
* 渲染 Vue 组件
|
|
96
|
-
*/
|
|
97
|
-
async renderVueComponent(component, context, deps) {
|
|
98
|
-
try {
|
|
99
|
-
const [vue, renderer] = deps;
|
|
100
|
-
const app = vue.createSSRApp(component);
|
|
101
|
-
// 提供路由信息
|
|
102
|
-
app.provide("routeInfo", {
|
|
103
|
-
params: context.params || {},
|
|
104
|
-
query: context.query || {},
|
|
105
|
-
pathname: context.pathname,
|
|
106
|
-
});
|
|
107
|
-
const html = await renderer.renderToString(app);
|
|
108
|
-
const fullHtml = HtmlRenderer.generateVueHtml(html, context);
|
|
109
|
-
return new Response(fullHtml, {
|
|
110
|
-
headers: { "Content-Type": "text/html; charset=utf-8" },
|
|
111
|
-
});
|
|
112
|
-
}
|
|
113
|
-
catch (error) {
|
|
114
|
-
console.error("Vue 组件渲染失败:", error);
|
|
115
|
-
return new Response("Vue Component Render Error", { status: 500 });
|
|
116
|
-
}
|
|
301
|
+
console.log(`\u{1F4E6} \u6309\u9700\u52A0\u8F7D ${framework} \u4F9D\u8D56...`);
|
|
302
|
+
try {
|
|
303
|
+
let deps;
|
|
304
|
+
switch (framework) {
|
|
305
|
+
case "vue":
|
|
306
|
+
deps = await Promise.all([
|
|
307
|
+
import("vue"),
|
|
308
|
+
import("@vue/server-renderer")
|
|
309
|
+
]);
|
|
310
|
+
break;
|
|
311
|
+
case "react":
|
|
312
|
+
deps = await Promise.all([
|
|
313
|
+
import("react"),
|
|
314
|
+
import("react-dom/server")
|
|
315
|
+
]);
|
|
316
|
+
break;
|
|
317
|
+
default:
|
|
318
|
+
throw new Error(`\u4E0D\u652F\u6301\u7684\u6846\u67B6: ${framework}`);
|
|
319
|
+
}
|
|
320
|
+
this.dependencyCache.set(framework, deps);
|
|
321
|
+
console.log(`\u2705 ${framework} \u4F9D\u8D56\u52A0\u8F7D\u5B8C\u6210`);
|
|
322
|
+
return deps;
|
|
323
|
+
} catch (error) {
|
|
324
|
+
console.error(`\u274C ${framework} \u4F9D\u8D56\u52A0\u8F7D\u5931\u8D25:`, error);
|
|
325
|
+
throw error;
|
|
117
326
|
}
|
|
118
|
-
|
|
119
|
-
|
|
120
|
-
|
|
121
|
-
|
|
122
|
-
|
|
123
|
-
|
|
124
|
-
|
|
125
|
-
req: context.req,
|
|
126
|
-
params: context.params || {},
|
|
127
|
-
query: context.query || {},
|
|
128
|
-
});
|
|
129
|
-
const html = renderer.renderToString(content);
|
|
130
|
-
const fullHtml = HtmlRenderer.generateReactHtml(html, context);
|
|
131
|
-
return new Response(fullHtml, {
|
|
132
|
-
headers: { "Content-Type": "text/html; charset=utf-8" },
|
|
133
|
-
});
|
|
134
|
-
}
|
|
135
|
-
catch (error) {
|
|
136
|
-
console.error("React 组件渲染失败:", error);
|
|
137
|
-
return new Response("React Component Render Error", { status: 500 });
|
|
138
|
-
}
|
|
327
|
+
}
|
|
328
|
+
/**
|
|
329
|
+
* 检测组件类型
|
|
330
|
+
*/
|
|
331
|
+
detectComponentType(component) {
|
|
332
|
+
if (component.render && typeof component.render === "function") {
|
|
333
|
+
return "vue";
|
|
139
334
|
}
|
|
140
|
-
|
|
141
|
-
|
|
142
|
-
*/
|
|
143
|
-
getDependencyManager() {
|
|
144
|
-
return this.dependencyManager;
|
|
335
|
+
if (component.$$typeof) {
|
|
336
|
+
return "react";
|
|
145
337
|
}
|
|
146
|
-
|
|
338
|
+
return "vue";
|
|
339
|
+
}
|
|
340
|
+
/**
|
|
341
|
+
* 清除缓存
|
|
342
|
+
*/
|
|
343
|
+
clearCache() {
|
|
344
|
+
this.dependencyCache.clear();
|
|
345
|
+
console.log("\u{1F9F9} \u4F9D\u8D56\u7F13\u5B58\u5DF2\u6E05\u9664");
|
|
346
|
+
}
|
|
347
|
+
/**
|
|
348
|
+
* 获取缓存状态
|
|
349
|
+
*/
|
|
350
|
+
getCacheStatus() {
|
|
351
|
+
const status = {};
|
|
352
|
+
for (const [framework] of this.dependencyCache) {
|
|
353
|
+
status[framework] = true;
|
|
354
|
+
}
|
|
355
|
+
return status;
|
|
356
|
+
}
|
|
357
|
+
};
|
|
358
|
+
|
|
359
|
+
// src/server/component-server.ts
|
|
360
|
+
var ComponentServer = class extends BaseServer {
|
|
361
|
+
routes;
|
|
362
|
+
dependencyManager;
|
|
363
|
+
constructor(routes) {
|
|
364
|
+
super();
|
|
365
|
+
this.routes = flattenComponentRoutes(routes);
|
|
366
|
+
this.dependencyManager = new DependencyManager();
|
|
367
|
+
this.detectRouteConflicts(this.routes);
|
|
368
|
+
this.logFlattenedRoutes(this.routes, "\u7EC4\u4EF6\u8DEF\u7531");
|
|
369
|
+
console.log("\u{1F680} \u4F9D\u8D56\u6309\u9700\u52A0\u8F7D\uFF0C\u670D\u52A1\u5668\u542F\u52A8\u5B8C\u6210");
|
|
370
|
+
}
|
|
371
|
+
/**
|
|
372
|
+
* 处理请求
|
|
373
|
+
*/
|
|
374
|
+
async fetch(req) {
|
|
375
|
+
const url = new URL(req.url);
|
|
376
|
+
const pathname = url.pathname;
|
|
377
|
+
const method = req.method;
|
|
378
|
+
if (method !== "GET") {
|
|
379
|
+
return new Response("Method Not Allowed", { status: 405 });
|
|
380
|
+
}
|
|
381
|
+
let matchedRoute = null;
|
|
382
|
+
for (const route of this.routes) {
|
|
383
|
+
if (PathMatcher.matchPath(route.fullPath, pathname)) {
|
|
384
|
+
matchedRoute = route;
|
|
385
|
+
break;
|
|
386
|
+
}
|
|
387
|
+
}
|
|
388
|
+
if (!matchedRoute) {
|
|
389
|
+
return new Response("Not Found", { status: 404 });
|
|
390
|
+
}
|
|
391
|
+
try {
|
|
392
|
+
const context = {
|
|
393
|
+
req,
|
|
394
|
+
params: PathMatcher.extractParams(matchedRoute.fullPath, pathname),
|
|
395
|
+
query: Object.fromEntries(url.searchParams),
|
|
396
|
+
pathname
|
|
397
|
+
};
|
|
398
|
+
return await this.executeMiddlewareChain(
|
|
399
|
+
matchedRoute.middlewareChain,
|
|
400
|
+
context,
|
|
401
|
+
matchedRoute.component
|
|
402
|
+
);
|
|
403
|
+
} catch (error) {
|
|
404
|
+
console.error("\u7EC4\u4EF6\u6E32\u67D3\u5931\u8D25:", error);
|
|
405
|
+
return new Response("Internal Server Error", { status: 500 });
|
|
406
|
+
}
|
|
407
|
+
}
|
|
408
|
+
/**
|
|
409
|
+
* 执行中间件链
|
|
410
|
+
*/
|
|
411
|
+
async executeMiddlewareChain(middlewareChain, context, componentImport) {
|
|
412
|
+
const renderComponent = async () => {
|
|
413
|
+
const componentModule = await componentImport();
|
|
414
|
+
const component = componentModule.default || componentModule;
|
|
415
|
+
const componentType = this.dependencyManager.detectComponentType(component);
|
|
416
|
+
const deps = await this.dependencyManager.getFrameworkDeps(componentType);
|
|
417
|
+
if (componentType === "vue") {
|
|
418
|
+
return await this.renderVueComponent(component, context, deps);
|
|
419
|
+
} else if (componentType === "react") {
|
|
420
|
+
return await this.renderReactComponent(component, context, deps);
|
|
421
|
+
} else {
|
|
422
|
+
throw new Error(`\u4E0D\u652F\u6301\u7684\u7EC4\u4EF6\u7C7B\u578B: ${componentType}`);
|
|
423
|
+
}
|
|
424
|
+
};
|
|
425
|
+
let index = 0;
|
|
426
|
+
const next = async () => {
|
|
427
|
+
if (index >= middlewareChain.length) {
|
|
428
|
+
return await renderComponent();
|
|
429
|
+
}
|
|
430
|
+
const middleware = middlewareChain[index++];
|
|
431
|
+
return await middleware(context.req, next);
|
|
432
|
+
};
|
|
433
|
+
return await next();
|
|
434
|
+
}
|
|
435
|
+
/**
|
|
436
|
+
* 渲染 Vue 组件
|
|
437
|
+
*/
|
|
438
|
+
async renderVueComponent(component, context, deps) {
|
|
439
|
+
try {
|
|
440
|
+
const [vue, renderer] = deps;
|
|
441
|
+
const app = vue.createSSRApp(component);
|
|
442
|
+
app.provide("routeInfo", {
|
|
443
|
+
params: context.params || {},
|
|
444
|
+
query: context.query || {},
|
|
445
|
+
pathname: context.pathname
|
|
446
|
+
});
|
|
447
|
+
const html = await renderer.renderToString(app);
|
|
448
|
+
const fullHtml = HtmlRenderer.generateVueHtml(html, context);
|
|
449
|
+
return new Response(fullHtml, {
|
|
450
|
+
headers: { "Content-Type": "text/html; charset=utf-8" }
|
|
451
|
+
});
|
|
452
|
+
} catch (error) {
|
|
453
|
+
console.error("Vue \u7EC4\u4EF6\u6E32\u67D3\u5931\u8D25:", error);
|
|
454
|
+
return new Response("Vue Component Render Error", { status: 500 });
|
|
455
|
+
}
|
|
456
|
+
}
|
|
457
|
+
/**
|
|
458
|
+
* 渲染 React 组件
|
|
459
|
+
*/
|
|
460
|
+
async renderReactComponent(component, context, deps) {
|
|
461
|
+
try {
|
|
462
|
+
const [react, renderer] = deps;
|
|
463
|
+
const content = react.createElement(component, {
|
|
464
|
+
req: context.req,
|
|
465
|
+
params: context.params || {},
|
|
466
|
+
query: context.query || {}
|
|
467
|
+
});
|
|
468
|
+
const html = renderer.renderToString(content);
|
|
469
|
+
const fullHtml = HtmlRenderer.generateReactHtml(html, context);
|
|
470
|
+
return new Response(fullHtml, {
|
|
471
|
+
headers: { "Content-Type": "text/html; charset=utf-8" }
|
|
472
|
+
});
|
|
473
|
+
} catch (error) {
|
|
474
|
+
console.error("React \u7EC4\u4EF6\u6E32\u67D3\u5931\u8D25:", error);
|
|
475
|
+
return new Response("React Component Render Error", { status: 500 });
|
|
476
|
+
}
|
|
477
|
+
}
|
|
478
|
+
/**
|
|
479
|
+
* 获取依赖管理器(用于外部访问)
|
|
480
|
+
*/
|
|
481
|
+
getDependencyManager() {
|
|
482
|
+
return this.dependencyManager;
|
|
483
|
+
}
|
|
484
|
+
};
|
|
485
|
+
export {
|
|
486
|
+
ComponentServer
|
|
487
|
+
};
|
|
488
|
+
//# sourceMappingURL=component-server.js.map
|