round-core 0.0.7 → 0.0.8

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 (75) hide show
  1. package/README.md +21 -0
  2. package/dist/index.d.ts +341 -341
  3. package/dist/vite-plugin.js +52 -3
  4. package/package.json +7 -3
  5. package/.github/workflows/benchmarks.yml +0 -44
  6. package/Round.png +0 -0
  7. package/benchmarks/apps/react/index.html +0 -9
  8. package/benchmarks/apps/react/main.jsx +0 -25
  9. package/benchmarks/apps/react/vite.config.js +0 -12
  10. package/benchmarks/apps/round/index.html +0 -11
  11. package/benchmarks/apps/round/main.jsx +0 -22
  12. package/benchmarks/apps/round/vite.config.js +0 -15
  13. package/benchmarks/bun.lock +0 -497
  14. package/benchmarks/dist-bench/react/assets/index-9KGqIPOU.js +0 -8
  15. package/benchmarks/dist-bench/react/index.html +0 -10
  16. package/benchmarks/dist-bench/round/assets/index-CBBIRhox.js +0 -52
  17. package/benchmarks/dist-bench/round/index.html +0 -8
  18. package/benchmarks/package.json +0 -22
  19. package/benchmarks/scripts/measure-build.js +0 -64
  20. package/benchmarks/tests/runtime.bench.js +0 -51
  21. package/benchmarks/vitest.config.js +0 -8
  22. package/bun.lock +0 -425
  23. package/cli.js +0 -2
  24. package/extension/.vscodeignore +0 -5
  25. package/extension/LICENSE +0 -21
  26. package/extension/cgmanifest.json +0 -45
  27. package/extension/extension.js +0 -163
  28. package/extension/images/round-config-dark.svg +0 -10
  29. package/extension/images/round-config-light.svg +0 -10
  30. package/extension/images/round-dark.svg +0 -10
  31. package/extension/images/round-light.svg +0 -10
  32. package/extension/javascript-language-configuration.json +0 -241
  33. package/extension/package-lock.json +0 -97
  34. package/extension/package.json +0 -119
  35. package/extension/package.nls.json +0 -4
  36. package/extension/round-0.1.0.vsix +0 -0
  37. package/extension/round-lsp/package-lock.json +0 -185
  38. package/extension/round-lsp/package.json +0 -21
  39. package/extension/round-lsp/src/round-transformer-lsp.js +0 -248
  40. package/extension/round-lsp/src/server.js +0 -396
  41. package/extension/snippets/javascript.code-snippets +0 -266
  42. package/extension/snippets/round.code-snippets +0 -109
  43. package/extension/syntaxes/JavaScript.tmLanguage.json +0 -6001
  44. package/extension/syntaxes/JavaScriptReact.tmLanguage.json +0 -6066
  45. package/extension/syntaxes/Readme.md +0 -12
  46. package/extension/syntaxes/Regular Expressions (JavaScript).tmLanguage +0 -237
  47. package/extension/syntaxes/Round.tmLanguage.json +0 -290
  48. package/extension/syntaxes/RoundInject.tmLanguage.json +0 -20
  49. package/extension/tags-language-configuration.json +0 -152
  50. package/extension/temp_astro/package-lock.json +0 -912
  51. package/extension/temp_astro/package.json +0 -16
  52. package/extension/types/round-core.d.ts +0 -326
  53. package/index.js +0 -2
  54. package/logo.svg +0 -10
  55. package/src/cli.js +0 -608
  56. package/src/compiler/index.js +0 -2
  57. package/src/compiler/transformer.js +0 -443
  58. package/src/compiler/vite-plugin.js +0 -472
  59. package/src/index.d.ts +0 -341
  60. package/src/index.js +0 -45
  61. package/src/runtime/context.js +0 -101
  62. package/src/runtime/dom.js +0 -403
  63. package/src/runtime/error-boundary.js +0 -48
  64. package/src/runtime/error-reporter.js +0 -13
  65. package/src/runtime/error-store.js +0 -85
  66. package/src/runtime/errors.js +0 -152
  67. package/src/runtime/lifecycle.js +0 -142
  68. package/src/runtime/markdown.js +0 -72
  69. package/src/runtime/router.js +0 -468
  70. package/src/runtime/signals.js +0 -548
  71. package/src/runtime/store.js +0 -215
  72. package/src/runtime/suspense.js +0 -128
  73. package/vite.config.build.js +0 -48
  74. package/vite.config.js +0 -10
  75. package/vitest.config.js +0 -8
@@ -1,472 +0,0 @@
1
- import { transform } from './transformer.js';
2
- import fs from 'node:fs';
3
- import path from 'node:path';
4
-
5
- function normalizePath(p) {
6
- return p.replaceAll('\\', '/');
7
- }
8
-
9
- function isMdRawRequest(id) {
10
- return typeof id === 'string' && id.includes('.md') && id.includes('?raw');
11
- }
12
-
13
- function stripQuery(id) {
14
- return id.split('?')[0];
15
- }
16
-
17
- function escapeForJsString(s) {
18
- return String(s)
19
- .replaceAll('\\', '\\\\')
20
- .replaceAll('`', '\\`')
21
- .replaceAll('${', '\\${');
22
- }
23
-
24
- function resolveMaybeRelative(baseDir, p) {
25
- if (!p) return null;
26
- if (path.isAbsolute(p)) return p;
27
- return path.resolve(baseDir, p);
28
- }
29
-
30
- function inlineMarkdownInRound(code, fileAbs, addWatchFile) {
31
- if (typeof code !== 'string' || typeof fileAbs !== 'string') return code;
32
-
33
- // Only handle simple self-closing tags with literal src: <Markdown src="./x.md" ... />
34
- // This runs before the .round transformer, so it's safe string-level rewriting.
35
- const dir = path.dirname(fileAbs);
36
-
37
- // Match src="..." or src='...'
38
- const re = /<Markdown\b([^>]*?)\bsrc\s*=\s*("([^"]+)"|'([^']+)')([^>]*)\/>/g;
39
- return code.replace(re, (full, beforeAttrs, _q, dbl, sgl, afterAttrs) => {
40
- const src = dbl ?? sgl;
41
- if (!src || typeof src !== 'string') return full;
42
-
43
- // Only inline relative paths; absolute/public URLs should remain runtime-resolved.
44
- if (!src.startsWith('./') && !src.startsWith('../')) return full;
45
-
46
- const mdAbs = path.resolve(dir, src);
47
- try {
48
- const raw = fs.readFileSync(mdAbs, 'utf8');
49
- if (typeof addWatchFile === 'function') {
50
- try { addWatchFile(mdAbs); } catch { }
51
- }
52
-
53
- const content = escapeForJsString(raw);
54
-
55
- // Remove the src=... portion and inject content={...}
56
- const rebuilt = `<Markdown${beforeAttrs}content={\`${content}\`} ${afterAttrs} />`;
57
- return rebuilt.replace(/\s+\/>$/, ' />');
58
- } catch (e) {
59
- const msg = e instanceof Error ? e.message : String(e);
60
- throw new Error(`Markdown file not found: ${src} (resolved: ${mdAbs})\n${msg}`);
61
- }
62
- });
63
- }
64
-
65
- function isExcluded(fileAbsPath, excludeAbs) {
66
- const file = normalizePath(fileAbsPath);
67
- for (const pat of excludeAbs) {
68
- const patNorm = normalizePath(pat);
69
- const prefix = patNorm.endsWith('/**') ? patNorm.slice(0, -3) : patNorm;
70
- if (file.startsWith(prefix)) return true;
71
- }
72
- return false;
73
- }
74
-
75
- function isIncluded(fileAbsPath, includeAbs) {
76
- if (!includeAbs.length) return true;
77
- const file = normalizePath(fileAbsPath);
78
- for (const pat of includeAbs) {
79
- const patNorm = normalizePath(pat);
80
- const prefix = patNorm.endsWith('/**') ? patNorm.slice(0, -3) : patNorm;
81
- if (file.startsWith(prefix)) return true;
82
- }
83
- return false;
84
- }
85
-
86
- export default function RoundPlugin(pluginOptions = {}) {
87
- const state = {
88
- rootDir: process.cwd(),
89
- includeAbs: [],
90
- excludeAbs: [],
91
- configLoaded: false,
92
- routingTrailingSlash: true,
93
- configPathAbs: null,
94
- configDir: null,
95
- entryAbs: null,
96
- entryRel: null,
97
- name: 'Round',
98
- startHead: null,
99
- startHeadHtml: null
100
- };
101
-
102
- let lastRuntimeErrorKey = null;
103
- let lastRuntimeErrorAt = 0;
104
-
105
- const runtimeImport = pluginOptions.runtimeImport ?? 'round-core';
106
- const restartOnConfigChange = pluginOptions.restartOnConfigChange !== undefined
107
- ? Boolean(pluginOptions.restartOnConfigChange)
108
- : true;
109
-
110
- function loadConfigOnce(rootDir) {
111
- if (state.configLoaded) return;
112
- state.configLoaded = true;
113
-
114
- const configPath = pluginOptions.configPath
115
- ? resolveMaybeRelative(rootDir, pluginOptions.configPath)
116
- : resolveMaybeRelative(rootDir, './round.config.json');
117
-
118
- state.configPathAbs = configPath;
119
-
120
- const configDir = configPath ? path.dirname(configPath) : rootDir;
121
- state.configDir = configDir;
122
-
123
- let config = null;
124
- if (configPath && fs.existsSync(configPath)) {
125
- try {
126
- const raw = fs.readFileSync(configPath, 'utf8');
127
- config = JSON.parse(raw);
128
- } catch {
129
- config = null;
130
- }
131
- }
132
-
133
- const trailingSlash = config?.routing?.trailingSlash;
134
- state.routingTrailingSlash = trailingSlash !== undefined ? Boolean(trailingSlash) : true;
135
-
136
- const customTags = config?.htmlTags;
137
- state.customTags = Array.isArray(customTags) ? customTags : [];
138
-
139
- state.name = config?.name ?? 'Round';
140
-
141
- const entryRel = config?.entry;
142
- state.entryRel = entryRel;
143
- state.entryAbs = entryRel ? resolveMaybeRelative(configDir, entryRel) : null;
144
-
145
- const include = pluginOptions.include ?? config?.include ?? [];
146
- const exclude = pluginOptions.exclude ?? config?.exclude ?? ['./node_modules', './dist'];
147
-
148
- const includeBase = pluginOptions.include ? rootDir : configDir;
149
- const excludeBase = pluginOptions.exclude ? rootDir : configDir;
150
-
151
- state.includeAbs = Array.isArray(include) ? include.map(p => resolveMaybeRelative(includeBase, p)).filter(Boolean) : [];
152
- state.excludeAbs = Array.isArray(exclude) ? exclude.map(p => resolveMaybeRelative(excludeBase, p)).filter(Boolean) : [];
153
- }
154
-
155
- function findBlock(str, startIndex) {
156
- let open = 0;
157
- let inSingle = false;
158
- let inDouble = false;
159
- let inTemplate = false;
160
-
161
- let start = -1;
162
- for (let i = startIndex; i < str.length; i++) {
163
- const ch = str[i];
164
- const prev = i > 0 ? str[i - 1] : '';
165
-
166
- if (!inDouble && !inTemplate && ch === '\'' && prev !== '\\') inSingle = !inSingle;
167
- else if (!inSingle && !inTemplate && ch === '"' && prev !== '\\') inDouble = !inDouble;
168
- else if (!inSingle && !inDouble && ch === '`' && prev !== '\\') inTemplate = !inTemplate;
169
-
170
- if (inSingle || inDouble || inTemplate) continue;
171
-
172
- if (ch === '{') {
173
- if (open === 0) start = i;
174
- open++;
175
- } else if (ch === '}') {
176
- open--;
177
- if (open === 0 && start !== -1) {
178
- return { start, end: i };
179
- }
180
- }
181
- }
182
- return null;
183
- }
184
-
185
- function parseStartHeadCallArgument(str, fromIndex) {
186
- const idx = str.indexOf('startHead', fromIndex);
187
- if (idx === -1) return null;
188
-
189
- const callIdx = str.indexOf('(', idx);
190
- if (callIdx === -1) return null;
191
-
192
- let i = callIdx;
193
- let paren = 0;
194
- let inSingle = false;
195
- let inDouble = false;
196
- let inTemplate = false;
197
-
198
- for (; i < str.length; i++) {
199
- const ch = str[i];
200
- const prev = i > 0 ? str[i - 1] : '';
201
-
202
- if (!inDouble && !inTemplate && ch === '\'' && prev !== '\\') inSingle = !inSingle;
203
- else if (!inSingle && !inTemplate && ch === '"' && prev !== '\\') inDouble = !inDouble;
204
- else if (!inSingle && !inDouble && ch === '`' && prev !== '\\') inTemplate = !inTemplate;
205
-
206
- if (inSingle || inDouble || inTemplate) continue;
207
-
208
- if (ch === '(') paren++;
209
- else if (ch === ')') {
210
- paren--;
211
- if (paren === 0) {
212
- const inner = str.slice(callIdx + 1, i).trim();
213
- return { arg: inner, start: idx, end: i + 1 };
214
- }
215
- }
216
- }
217
-
218
- return null;
219
- }
220
-
221
- function parseStartHeadInDefaultExport(code) {
222
- // Find `export default function ... { ... }`
223
- const m = code.match(/export\s+default\s+function\b/);
224
- const hasAnyCall = /\bstartHead\s*\(/.test(code);
225
- if (!m || typeof m.index !== 'number') return { headExpr: null, hasAny: hasAnyCall };
226
-
227
- const fnStart = m.index;
228
- const braceIdx = code.indexOf('{', fnStart);
229
- if (braceIdx === -1) return { headExpr: null, hasAny: hasAnyCall };
230
-
231
- const block = findBlock(code, braceIdx);
232
- if (!block) return { headExpr: null, hasAny: hasAnyCall };
233
-
234
- const body = code.slice(block.start + 1, block.end);
235
- const call = parseStartHeadCallArgument(body, 0);
236
- return { headExpr: call ? call.arg : null, hasAny: hasAnyCall, hasOutside: hasAnyCall && !call };
237
- }
238
-
239
- function headToHtml(head) {
240
- if (!head || typeof head !== 'object') return '';
241
-
242
- let out = '';
243
- if (typeof head.title === 'string') {
244
- out += `\n <title>${head.title}</title>`;
245
- }
246
-
247
- const meta = head.meta;
248
- const links = head.links;
249
-
250
- const renderAttrs = (attrs) => {
251
- if (!attrs || typeof attrs !== 'object') return '';
252
- return Object.entries(attrs)
253
- .filter(([, v]) => v !== null && v !== undefined)
254
- .map(([k, v]) => ` ${k}="${String(v).replaceAll('"', '&quot;')}"`)
255
- .join('');
256
- };
257
-
258
- if (Array.isArray(meta)) {
259
- meta.forEach((m) => {
260
- if (!m) return;
261
- if (Array.isArray(m) && m.length >= 2) {
262
- out += `\n <meta name="${String(m[0]).replaceAll('"', '&quot;')}" content="${String(m[1] ?? '').replaceAll('"', '&quot;')}">`;
263
- return;
264
- }
265
- if (typeof m === 'object') {
266
- out += `\n <meta${renderAttrs(m)}>`;
267
- }
268
- });
269
- } else if (meta && typeof meta === 'object') {
270
- Object.entries(meta).forEach(([name, content]) => {
271
- out += `\n <meta name="${String(name).replaceAll('"', '&quot;')}" content="${String(content ?? '').replaceAll('"', '&quot;')}">`;
272
- });
273
- }
274
-
275
- if (Array.isArray(links)) {
276
- links.forEach((l) => {
277
- if (!l || typeof l !== 'object') return;
278
- out += `\n <link${renderAttrs(l)}>`;
279
- });
280
- }
281
-
282
- // allow raw html injection (advanced escape hatch)
283
- if (typeof head.raw === 'string' && head.raw.trim()) {
284
- out += `\n${head.raw}`;
285
- }
286
-
287
- return out;
288
- }
289
-
290
- return {
291
- name: 'vite-plugin-round',
292
- enforce: 'pre',
293
-
294
- transformIndexHtml(html) {
295
- if (!state.startHeadHtml) return html;
296
- if (!html.includes('</head>')) return html;
297
-
298
- // Remove existing <title> to avoid duplicates if we set it.
299
- let next = html;
300
- if (state.startHead && typeof state.startHead.title === 'string') {
301
- next = next.replace(/<title>[\s\S]*?<\/title>/i, '');
302
- }
303
-
304
- return next.replace('</head>', `${state.startHeadHtml}\n</head>`);
305
- },
306
-
307
- config(userConfig, env) {
308
- const rootDir = path.resolve(process.cwd(), userConfig.root ?? '.');
309
- state.rootDir = rootDir;
310
- loadConfigOnce(rootDir);
311
-
312
- return {
313
- define: {
314
- __ROUND_ROUTING_TRAILING_SLASH__: JSON.stringify(state.routingTrailingSlash),
315
- __ROUND_CUSTOM_TAGS__: JSON.stringify(state.customTags ?? [])
316
- },
317
- esbuild: {
318
- include: /\.(round|js|jsx|ts|tsx)$/,
319
- loader: 'jsx',
320
- jsxFactory: 'createElement',
321
- jsxFragment: 'Fragment'
322
- // NOTE: Inject the runtime import in transform() to avoid
323
- },
324
- // Ensure .round files are treated as JS/JSX
325
- resolve: {
326
- extensions: ['.mjs', '.js', '.ts', '.jsx', '.tsx', '.json', '.round']
327
- }
328
- };
329
- },
330
-
331
- resolveId(id) {
332
- return null;
333
- },
334
- load(id) {
335
- if (!isMdRawRequest(id)) return;
336
-
337
- const fileAbs = stripQuery(id);
338
- try {
339
- const raw = fs.readFileSync(fileAbs, 'utf8');
340
- this.addWatchFile(fileAbs);
341
- return `export default \`${escapeForJsString(raw)}\`;`;
342
- } catch {
343
- this.addWatchFile(fileAbs);
344
- return 'export default ``;';
345
- }
346
- },
347
-
348
- configureServer(server) {
349
- loadConfigOnce(server.config.root ?? process.cwd());
350
-
351
- if (state.configPathAbs) {
352
- server.watcher.add(state.configPathAbs);
353
- }
354
-
355
- server.middlewares.use((req, res, next) => {
356
- if (!req.url) return next();
357
- const [urlPath] = req.url.split('?');
358
- if (urlPath && urlPath.endsWith('.md')) {
359
- res.setHeader('Content-Type', 'text/plain; charset=utf-8');
360
- }
361
- next();
362
- });
363
-
364
- server.ws.on('round:runtime-error', (payload = {}) => {
365
- try {
366
- const message = typeof payload.message === 'string' ? payload.message : 'Runtime error';
367
- const phase = typeof payload.phase === 'string' && payload.phase ? ` (${payload.phase})` : '';
368
- const component = typeof payload.component === 'string' && payload.component ? ` in ${payload.component}` : '';
369
- const header = `[round] Runtime error${component}${phase}: ${message}`;
370
-
371
- const stack = payload.stack ? String(payload.stack) : '';
372
- const key = `${header}\n${stack}`;
373
- const now = Date.now();
374
- if (lastRuntimeErrorKey === key && (now - lastRuntimeErrorAt) < 2000) return;
375
- lastRuntimeErrorKey = key;
376
- lastRuntimeErrorAt = now;
377
-
378
- server.config.logger.error(header);
379
- if (stack) server.config.logger.error(stack);
380
- } catch {
381
- server.config.logger.error('[round] Runtime error');
382
- }
383
- });
384
- },
385
-
386
- handleHotUpdate(ctx) {
387
- if (state.configPathAbs && ctx.file === state.configPathAbs) {
388
- if (!restartOnConfigChange) return [];
389
- try {
390
- if (typeof ctx.server.restart === 'function') {
391
- ctx.server.restart();
392
- } else {
393
- ctx.server.ws.send({ type: 'full-reload' });
394
- }
395
- } catch {
396
- ctx.server.ws.send({ type: 'full-reload' });
397
- }
398
- return [];
399
- }
400
- },
401
-
402
- configurePreviewServer(server) {
403
- server.middlewares.use((req, res, next) => {
404
- if (!req.url) return next();
405
- const [urlPath] = req.url.split('?');
406
- if (urlPath && urlPath.endsWith('.md')) {
407
- res.setHeader('Content-Type', 'text/plain; charset=utf-8');
408
- }
409
- next();
410
- });
411
- },
412
-
413
- transform(code, id) {
414
- if (id.endsWith('.round')) {
415
- const fileAbs = path.isAbsolute(id) ? id : path.resolve(state.rootDir, id);
416
- if (!isIncluded(fileAbs, state.includeAbs)) return;
417
- if (isExcluded(fileAbs, state.excludeAbs)) return;
418
-
419
- const isEntry = state.entryAbs && normalizePath(fileAbs) === normalizePath(state.entryAbs);
420
- const parsedHead = parseStartHeadInDefaultExport(code);
421
-
422
- if (parsedHead.hasAny && !isEntry) {
423
- this.error(new Error(`startHead() can only be used in the entry module's export default function: ${state.entryAbs ?? '(unknown entry)'}\nFound in: ${fileAbs}`));
424
- }
425
-
426
- if (isEntry && parsedHead.hasOutside) {
427
- this.error(new Error(`startHead() must be called inside the entry module's export default function body (not at top-level).\nEntry: ${fileAbs}`));
428
- }
429
-
430
- if (isEntry && parsedHead.headExpr) {
431
- const trimmed = parsedHead.headExpr.trim();
432
- if (!trimmed.startsWith('{')) {
433
- this.error(new Error(`startHead(...) expects an object literal. Example: startHead({ title: 'Home' })\nFound: ${trimmed.slice(0, 60)}...`));
434
- }
435
-
436
- if (/\bfunction\b|=>|\bimport\b|\brequire\b|\bprocess\b|\bglobal\b/.test(trimmed)) {
437
- this.error(new Error('startHead object must be static data (no functions/imports).'));
438
- }
439
-
440
- let headObj = null;
441
- try {
442
- headObj = Function(`"use strict"; return (${trimmed});`)();
443
- } catch (e) {
444
- this.error(new Error(`Failed to parse startHead(...) object in ${fileAbs}: ${String(e?.message ?? e)}`));
445
- }
446
-
447
- state.startHead = headObj;
448
- state.startHeadHtml = headToHtml(headObj);
449
- }
450
-
451
- let nextCode = code;
452
- try {
453
- nextCode = inlineMarkdownInRound(nextCode, fileAbs, (p) => this.addWatchFile(p));
454
- } catch (e) {
455
- // Fail fast in build and show the file that triggered the problem.
456
- this.error(e);
457
- }
458
-
459
- let transformedCode = transform(nextCode);
460
-
461
- if (!/^\s*import\s+\{\s*createElement\s*,\s*Fragment\s*\}\s+from\s+['"][^'"]+['"];?/m.test(transformedCode)) {
462
- transformedCode = `import { createElement, Fragment } from '${runtimeImport}';\n` + transformedCode;
463
- }
464
-
465
- return {
466
- code: transformedCode,
467
- map: null
468
- };
469
- }
470
- }
471
- };
472
- }