@ecopages/core 0.2.0-alpha.1
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/CHANGELOG.md +89 -0
- package/LICENSE +21 -0
- package/README.md +32 -0
- package/package.json +279 -0
- package/src/adapters/abstract/application-adapter.d.ts +168 -0
- package/src/adapters/abstract/application-adapter.js +109 -0
- package/src/adapters/abstract/application-adapter.ts +337 -0
- package/src/adapters/abstract/router-adapter.d.ts +26 -0
- package/src/adapters/abstract/router-adapter.js +5 -0
- package/src/adapters/abstract/router-adapter.ts +30 -0
- package/src/adapters/abstract/server-adapter.d.ts +69 -0
- package/src/adapters/abstract/server-adapter.js +15 -0
- package/src/adapters/abstract/server-adapter.ts +79 -0
- package/src/adapters/bun/client-bridge.d.ts +34 -0
- package/src/adapters/bun/client-bridge.js +48 -0
- package/src/adapters/bun/client-bridge.ts +62 -0
- package/src/adapters/bun/create-app.d.ts +60 -0
- package/src/adapters/bun/create-app.js +117 -0
- package/src/adapters/bun/create-app.ts +189 -0
- package/src/adapters/bun/define-api-handler.d.ts +61 -0
- package/src/adapters/bun/define-api-handler.js +15 -0
- package/src/adapters/bun/define-api-handler.ts +114 -0
- package/src/adapters/bun/hmr-manager.d.ts +84 -0
- package/src/adapters/bun/hmr-manager.js +227 -0
- package/src/adapters/bun/hmr-manager.ts +281 -0
- package/src/adapters/bun/index.d.ts +3 -0
- package/src/adapters/bun/index.js +8 -0
- package/src/adapters/bun/index.ts +3 -0
- package/src/adapters/bun/server-adapter.d.ts +155 -0
- package/src/adapters/bun/server-adapter.js +368 -0
- package/src/adapters/bun/server-adapter.ts +492 -0
- package/src/adapters/bun/server-lifecycle.d.ts +52 -0
- package/src/adapters/bun/server-lifecycle.js +120 -0
- package/src/adapters/bun/server-lifecycle.ts +154 -0
- package/src/adapters/index.d.ts +6 -0
- package/src/adapters/index.js +14 -0
- package/src/adapters/index.ts +6 -0
- package/src/adapters/node/create-app.d.ts +21 -0
- package/src/adapters/node/create-app.js +143 -0
- package/src/adapters/node/create-app.ts +179 -0
- package/src/adapters/node/index.d.ts +4 -0
- package/src/adapters/node/index.js +8 -0
- package/src/adapters/node/index.ts +9 -0
- package/src/adapters/node/node-client-bridge.d.ts +26 -0
- package/src/adapters/node/node-client-bridge.js +66 -0
- package/src/adapters/node/node-client-bridge.ts +79 -0
- package/src/adapters/node/node-hmr-manager.d.ts +62 -0
- package/src/adapters/node/node-hmr-manager.js +221 -0
- package/src/adapters/node/node-hmr-manager.ts +271 -0
- package/src/adapters/node/server-adapter.d.ts +190 -0
- package/src/adapters/node/server-adapter.js +420 -0
- package/src/adapters/node/server-adapter.ts +561 -0
- package/src/adapters/node/static-content-server.d.ts +24 -0
- package/src/adapters/node/static-content-server.js +166 -0
- package/src/adapters/node/static-content-server.ts +203 -0
- package/src/adapters/shared/api-response.d.ts +52 -0
- package/src/adapters/shared/api-response.js +96 -0
- package/src/adapters/shared/api-response.ts +104 -0
- package/src/adapters/shared/application-adapter.d.ts +18 -0
- package/src/adapters/shared/application-adapter.js +90 -0
- package/src/adapters/shared/application-adapter.ts +199 -0
- package/src/adapters/shared/explicit-static-route-matcher.d.ts +38 -0
- package/src/adapters/shared/explicit-static-route-matcher.js +100 -0
- package/src/adapters/shared/explicit-static-route-matcher.ts +134 -0
- package/src/adapters/shared/file-route-middleware-pipeline.d.ts +65 -0
- package/src/adapters/shared/file-route-middleware-pipeline.js +98 -0
- package/src/adapters/shared/file-route-middleware-pipeline.ts +123 -0
- package/src/adapters/shared/fs-server-response-factory.d.ts +19 -0
- package/src/adapters/shared/fs-server-response-factory.js +97 -0
- package/src/adapters/shared/fs-server-response-factory.ts +118 -0
- package/src/adapters/shared/fs-server-response-matcher.d.ts +71 -0
- package/src/adapters/shared/fs-server-response-matcher.js +155 -0
- package/src/adapters/shared/fs-server-response-matcher.ts +198 -0
- package/src/adapters/shared/render-context.d.ts +14 -0
- package/src/adapters/shared/render-context.js +69 -0
- package/src/adapters/shared/render-context.ts +105 -0
- package/src/adapters/shared/server-adapter.d.ts +87 -0
- package/src/adapters/shared/server-adapter.js +353 -0
- package/src/adapters/shared/server-adapter.ts +442 -0
- package/src/adapters/shared/server-route-handler.d.ts +89 -0
- package/src/adapters/shared/server-route-handler.js +120 -0
- package/src/adapters/shared/server-route-handler.ts +166 -0
- package/src/adapters/shared/server-static-builder.d.ts +38 -0
- package/src/adapters/shared/server-static-builder.js +46 -0
- package/src/adapters/shared/server-static-builder.ts +82 -0
- package/src/build/build-adapter.d.ts +74 -0
- package/src/build/build-adapter.js +54 -0
- package/src/build/build-adapter.ts +132 -0
- package/src/build/build-types.d.ts +57 -0
- package/src/build/build-types.js +0 -0
- package/src/build/build-types.ts +83 -0
- package/src/build/esbuild-build-adapter.d.ts +69 -0
- package/src/build/esbuild-build-adapter.js +390 -0
- package/src/build/esbuild-build-adapter.ts +510 -0
- package/src/config/config-builder.d.ts +227 -0
- package/src/config/config-builder.js +392 -0
- package/src/config/config-builder.ts +474 -0
- package/src/constants.d.ts +32 -0
- package/src/constants.js +21 -0
- package/src/constants.ts +39 -0
- package/src/create-app.d.ts +17 -0
- package/src/create-app.js +66 -0
- package/src/create-app.ts +87 -0
- package/src/declarations.d.ts +26 -0
- package/src/define-api-handler.d.ts +25 -0
- package/src/define-api-handler.js +15 -0
- package/src/define-api-handler.ts +66 -0
- package/src/dev/sc-server.d.ts +30 -0
- package/src/dev/sc-server.js +111 -0
- package/src/dev/sc-server.ts +143 -0
- package/src/eco/README.md +636 -0
- package/src/eco/component-render-context.d.ts +105 -0
- package/src/eco/component-render-context.js +77 -0
- package/src/eco/component-render-context.ts +202 -0
- package/src/eco/eco.d.ts +9 -0
- package/src/eco/eco.js +110 -0
- package/src/eco/eco.ts +221 -0
- package/src/eco/eco.types.d.ts +170 -0
- package/src/eco/eco.types.js +0 -0
- package/src/eco/eco.types.ts +202 -0
- package/src/eco/eco.utils.d.ts +40 -0
- package/src/eco/eco.utils.js +40 -0
- package/src/eco/eco.utils.ts +89 -0
- package/src/eco/global-injector-map.d.ts +16 -0
- package/src/eco/global-injector-map.js +80 -0
- package/src/eco/global-injector-map.ts +112 -0
- package/src/eco/lazy-injector-map.d.ts +8 -0
- package/src/eco/lazy-injector-map.js +70 -0
- package/src/eco/lazy-injector-map.ts +120 -0
- package/src/eco/module-dependencies.d.ts +18 -0
- package/src/eco/module-dependencies.js +49 -0
- package/src/eco/module-dependencies.ts +75 -0
- package/src/env.d.ts +20 -0
- package/src/errors/http-error.d.ts +31 -0
- package/src/errors/http-error.js +50 -0
- package/src/errors/http-error.ts +72 -0
- package/src/errors/index.d.ts +2 -0
- package/src/errors/index.js +4 -0
- package/src/errors/index.ts +2 -0
- package/src/errors/locals-access-error.d.ts +4 -0
- package/src/errors/locals-access-error.js +9 -0
- package/src/errors/locals-access-error.ts +7 -0
- package/src/global/app-logger.d.ts +2 -0
- package/src/global/app-logger.js +6 -0
- package/src/global/app-logger.ts +4 -0
- package/src/hmr/client/__screenshots__/hmr-runtime.test.browser.ts/HMR-Runtime-HMR-Server-Integration-should-have-HMR-script-injected-in-page-1.png +0 -0
- package/src/hmr/client/__screenshots__/hmr-runtime.test.browser.ts/HMR-Runtime-HMR-Server-Integration-should-load-fixture-app-page-1.png +0 -0
- package/src/hmr/client/__screenshots__/hmr-runtime.test.browser.ts/HMR-Runtime-WebSocket-Connection-should-connect-to-correct-HMR-endpoint-1.png +0 -0
- package/src/hmr/client/hmr-runtime.d.ts +10 -0
- package/src/hmr/client/hmr-runtime.js +86 -0
- package/src/hmr/client/hmr-runtime.ts +121 -0
- package/src/hmr/hmr-strategy.d.ts +159 -0
- package/src/hmr/hmr-strategy.js +29 -0
- package/src/hmr/hmr-strategy.ts +172 -0
- package/src/hmr/hmr.test.e2e.d.ts +1 -0
- package/src/hmr/hmr.test.e2e.js +50 -0
- package/src/hmr/hmr.test.e2e.ts +75 -0
- package/src/hmr/strategies/default-hmr-strategy.d.ts +43 -0
- package/src/hmr/strategies/default-hmr-strategy.js +34 -0
- package/src/hmr/strategies/default-hmr-strategy.ts +60 -0
- package/src/hmr/strategies/js-hmr-strategy.d.ts +136 -0
- package/src/hmr/strategies/js-hmr-strategy.js +179 -0
- package/src/hmr/strategies/js-hmr-strategy.ts +308 -0
- package/src/index.browser.d.ts +3 -0
- package/src/index.browser.js +4 -0
- package/src/index.browser.ts +3 -0
- package/src/index.d.ts +5 -0
- package/src/index.js +10 -0
- package/src/index.ts +5 -0
- package/src/integrations/ghtml/ghtml-renderer.d.ts +15 -0
- package/src/integrations/ghtml/ghtml-renderer.js +60 -0
- package/src/integrations/ghtml/ghtml-renderer.ts +93 -0
- package/src/integrations/ghtml/ghtml.plugin.d.ts +20 -0
- package/src/integrations/ghtml/ghtml.plugin.js +21 -0
- package/src/integrations/ghtml/ghtml.plugin.ts +32 -0
- package/src/internal-types.d.ts +200 -0
- package/src/internal-types.js +0 -0
- package/src/internal-types.ts +212 -0
- package/src/plugins/alias-resolver-plugin.d.ts +2 -0
- package/src/plugins/alias-resolver-plugin.js +39 -0
- package/src/plugins/alias-resolver-plugin.ts +45 -0
- package/src/plugins/eco-component-meta-plugin.d.ts +95 -0
- package/src/plugins/eco-component-meta-plugin.js +157 -0
- package/src/plugins/eco-component-meta-plugin.ts +474 -0
- package/src/plugins/integration-plugin.d.ts +102 -0
- package/src/plugins/integration-plugin.js +100 -0
- package/src/plugins/integration-plugin.ts +184 -0
- package/src/plugins/processor.d.ts +82 -0
- package/src/plugins/processor.js +122 -0
- package/src/plugins/processor.ts +220 -0
- package/src/public-types.d.ts +1094 -0
- package/src/public-types.js +0 -0
- package/src/public-types.ts +1255 -0
- package/src/route-renderer/GRAPH.md +387 -0
- package/src/route-renderer/README.md +135 -0
- package/src/route-renderer/component-graph-executor.d.ts +32 -0
- package/src/route-renderer/component-graph-executor.js +31 -0
- package/src/route-renderer/component-graph-executor.ts +84 -0
- package/src/route-renderer/component-graph.d.ts +42 -0
- package/src/route-renderer/component-graph.js +72 -0
- package/src/route-renderer/component-graph.ts +159 -0
- package/src/route-renderer/component-marker.d.ts +52 -0
- package/src/route-renderer/component-marker.js +46 -0
- package/src/route-renderer/component-marker.ts +117 -0
- package/src/route-renderer/dependency-resolver.d.ts +24 -0
- package/src/route-renderer/dependency-resolver.js +428 -0
- package/src/route-renderer/dependency-resolver.ts +596 -0
- package/src/route-renderer/html-post-processing.service.d.ts +40 -0
- package/src/route-renderer/html-post-processing.service.js +86 -0
- package/src/route-renderer/html-post-processing.service.ts +103 -0
- package/src/route-renderer/integration-renderer.d.ts +339 -0
- package/src/route-renderer/integration-renderer.js +526 -0
- package/src/route-renderer/integration-renderer.ts +696 -0
- package/src/route-renderer/marker-graph-resolver.d.ts +76 -0
- package/src/route-renderer/marker-graph-resolver.js +93 -0
- package/src/route-renderer/marker-graph-resolver.ts +153 -0
- package/src/route-renderer/page-module-loader.d.ts +61 -0
- package/src/route-renderer/page-module-loader.js +102 -0
- package/src/route-renderer/page-module-loader.ts +153 -0
- package/src/route-renderer/render-execution.service.d.ts +69 -0
- package/src/route-renderer/render-execution.service.js +91 -0
- package/src/route-renderer/render-execution.service.ts +158 -0
- package/src/route-renderer/render-preparation.service.d.ts +112 -0
- package/src/route-renderer/render-preparation.service.js +243 -0
- package/src/route-renderer/render-preparation.service.ts +358 -0
- package/src/route-renderer/route-renderer.d.ts +26 -0
- package/src/route-renderer/route-renderer.js +68 -0
- package/src/route-renderer/route-renderer.ts +80 -0
- package/src/router/fs-router-scanner.d.ts +41 -0
- package/src/router/fs-router-scanner.js +155 -0
- package/src/router/fs-router-scanner.ts +217 -0
- package/src/router/fs-router.d.ts +26 -0
- package/src/router/fs-router.js +100 -0
- package/src/router/fs-router.ts +122 -0
- package/src/services/asset-processing-service/asset-processing.service.d.ts +41 -0
- package/src/services/asset-processing-service/asset-processing.service.js +250 -0
- package/src/services/asset-processing-service/asset-processing.service.ts +306 -0
- package/src/services/asset-processing-service/asset.factory.d.ts +17 -0
- package/src/services/asset-processing-service/asset.factory.js +82 -0
- package/src/services/asset-processing-service/asset.factory.ts +105 -0
- package/src/services/asset-processing-service/assets.types.d.ts +88 -0
- package/src/services/asset-processing-service/assets.types.js +0 -0
- package/src/services/asset-processing-service/assets.types.ts +112 -0
- package/src/services/asset-processing-service/index.d.ts +3 -0
- package/src/services/asset-processing-service/index.js +3 -0
- package/src/services/asset-processing-service/index.ts +3 -0
- package/src/services/asset-processing-service/processor.interface.d.ts +22 -0
- package/src/services/asset-processing-service/processor.interface.js +6 -0
- package/src/services/asset-processing-service/processor.interface.ts +27 -0
- package/src/services/asset-processing-service/processor.registry.d.ts +8 -0
- package/src/services/asset-processing-service/processor.registry.js +15 -0
- package/src/services/asset-processing-service/processor.registry.ts +18 -0
- package/src/services/asset-processing-service/processors/base/base-processor.d.ts +24 -0
- package/src/services/asset-processing-service/processors/base/base-processor.js +59 -0
- package/src/services/asset-processing-service/processors/base/base-processor.ts +76 -0
- package/src/services/asset-processing-service/processors/base/base-script-processor.d.ts +16 -0
- package/src/services/asset-processing-service/processors/base/base-script-processor.js +80 -0
- package/src/services/asset-processing-service/processors/base/base-script-processor.ts +105 -0
- package/src/services/asset-processing-service/processors/index.d.ts +5 -0
- package/src/services/asset-processing-service/processors/index.js +5 -0
- package/src/services/asset-processing-service/processors/index.ts +5 -0
- package/src/services/asset-processing-service/processors/script/content-script.processor.d.ts +5 -0
- package/src/services/asset-processing-service/processors/script/content-script.processor.js +57 -0
- package/src/services/asset-processing-service/processors/script/content-script.processor.ts +66 -0
- package/src/services/asset-processing-service/processors/script/file-script.processor.d.ts +8 -0
- package/src/services/asset-processing-service/processors/script/file-script.processor.js +76 -0
- package/src/services/asset-processing-service/processors/script/file-script.processor.ts +88 -0
- package/src/services/asset-processing-service/processors/script/node-module-script.processor.d.ts +7 -0
- package/src/services/asset-processing-service/processors/script/node-module-script.processor.js +74 -0
- package/src/services/asset-processing-service/processors/script/node-module-script.processor.ts +84 -0
- package/src/services/asset-processing-service/processors/stylesheet/content-stylesheet.processor.d.ts +5 -0
- package/src/services/asset-processing-service/processors/stylesheet/content-stylesheet.processor.js +25 -0
- package/src/services/asset-processing-service/processors/stylesheet/content-stylesheet.processor.ts +27 -0
- package/src/services/asset-processing-service/processors/stylesheet/file-stylesheet.processor.d.ts +9 -0
- package/src/services/asset-processing-service/processors/stylesheet/file-stylesheet.processor.js +63 -0
- package/src/services/asset-processing-service/processors/stylesheet/file-stylesheet.processor.ts +77 -0
- package/src/services/cache/cache.types.d.ts +107 -0
- package/src/services/cache/cache.types.js +0 -0
- package/src/services/cache/cache.types.ts +126 -0
- package/src/services/cache/index.d.ts +7 -0
- package/src/services/cache/index.js +7 -0
- package/src/services/cache/index.ts +18 -0
- package/src/services/cache/memory-cache-store.d.ts +42 -0
- package/src/services/cache/memory-cache-store.js +98 -0
- package/src/services/cache/memory-cache-store.ts +130 -0
- package/src/services/cache/page-cache-service.d.ts +70 -0
- package/src/services/cache/page-cache-service.js +152 -0
- package/src/services/cache/page-cache-service.ts +202 -0
- package/src/services/html-transformer.service.d.ts +50 -0
- package/src/services/html-transformer.service.js +163 -0
- package/src/services/html-transformer.service.ts +217 -0
- package/src/services/page-module-import.service.d.ts +37 -0
- package/src/services/page-module-import.service.js +88 -0
- package/src/services/page-module-import.service.ts +129 -0
- package/src/services/page-request-cache-coordinator.service.d.ts +75 -0
- package/src/services/page-request-cache-coordinator.service.js +107 -0
- package/src/services/page-request-cache-coordinator.service.ts +128 -0
- package/src/services/schema-validation-service.d.ts +122 -0
- package/src/services/schema-validation-service.js +101 -0
- package/src/services/schema-validation-service.ts +204 -0
- package/src/services/validation/standard-schema.types.d.ts +65 -0
- package/src/services/validation/standard-schema.types.js +0 -0
- package/src/services/validation/standard-schema.types.ts +68 -0
- package/src/static-site-generator/static-site-generator.d.ts +57 -0
- package/src/static-site-generator/static-site-generator.js +272 -0
- package/src/static-site-generator/static-site-generator.ts +359 -0
- package/src/utils/css.d.ts +1 -0
- package/src/utils/css.js +7 -0
- package/src/utils/css.ts +5 -0
- package/src/utils/deep-merge.d.ts +14 -0
- package/src/utils/deep-merge.js +32 -0
- package/src/utils/deep-merge.ts +47 -0
- package/src/utils/hash.d.ts +1 -0
- package/src/utils/hash.js +7 -0
- package/src/utils/hash.ts +5 -0
- package/src/utils/html.d.ts +1 -0
- package/src/utils/html.js +4 -0
- package/src/utils/html.ts +1 -0
- package/src/utils/invariant.d.ts +5 -0
- package/src/utils/invariant.js +11 -0
- package/src/utils/invariant.ts +15 -0
- package/src/utils/locals-utils.d.ts +15 -0
- package/src/utils/locals-utils.js +24 -0
- package/src/utils/locals-utils.ts +37 -0
- package/src/utils/parse-cli-args.d.ts +24 -0
- package/src/utils/parse-cli-args.js +47 -0
- package/src/utils/parse-cli-args.ts +83 -0
- package/src/utils/path-utils.module.d.ts +5 -0
- package/src/utils/path-utils.module.js +14 -0
- package/src/utils/path-utils.module.ts +14 -0
- package/src/utils/runtime.d.ts +11 -0
- package/src/utils/runtime.js +40 -0
- package/src/utils/runtime.ts +44 -0
- package/src/utils/server-utils.module.d.ts +19 -0
- package/src/utils/server-utils.module.js +56 -0
- package/src/utils/server-utils.module.ts +67 -0
- package/src/watchers/project-watcher.d.ts +120 -0
- package/src/watchers/project-watcher.js +238 -0
- package/src/watchers/project-watcher.test-helpers.d.ts +4 -0
- package/src/watchers/project-watcher.test-helpers.js +51 -0
- package/src/watchers/project-watcher.test-helpers.ts +40 -0
- package/src/watchers/project-watcher.ts +306 -0
|
@@ -0,0 +1,89 @@
|
|
|
1
|
+
import type { EcoComponent } from '../public-types.ts';
|
|
2
|
+
import { buildInjectorMapScript } from './lazy-injector-map.ts';
|
|
3
|
+
|
|
4
|
+
/**
|
|
5
|
+
* Returns `true` when `value` is a thenable (Promise-like) object.
|
|
6
|
+
*
|
|
7
|
+
* Used to transparently handle both synchronous and asynchronous component
|
|
8
|
+
* render results without requiring every caller to branch on `instanceof Promise`.
|
|
9
|
+
*
|
|
10
|
+
* @typeParam T Expected resolved type of the thenable.
|
|
11
|
+
*/
|
|
12
|
+
export function isThenable<T>(value: unknown): value is PromiseLike<T> {
|
|
13
|
+
return (
|
|
14
|
+
typeof value === 'object' &&
|
|
15
|
+
value !== null &&
|
|
16
|
+
'then' in value &&
|
|
17
|
+
typeof (value as { then?: unknown }).then === 'function'
|
|
18
|
+
);
|
|
19
|
+
}
|
|
20
|
+
|
|
21
|
+
/**
|
|
22
|
+
* Injects `data-eco-trigger` into the first real HTML element opening tag of
|
|
23
|
+
* a component's rendered output string.
|
|
24
|
+
*
|
|
25
|
+
* The scan skips over leading whitespace, HTML comments (`<!-- -->`), CDATA
|
|
26
|
+
* sections, and doctype declarations so that the attribute is always placed on
|
|
27
|
+
* the first actual element — not spurious markup that can precede it.
|
|
28
|
+
*
|
|
29
|
+
* The insertion point is the end of the element's tag name, before any existing
|
|
30
|
+
* attributes or the closing `>`, which produces output like:
|
|
31
|
+
*
|
|
32
|
+
* ```html
|
|
33
|
+
* <my-element data-eco-trigger="eco-trigger-abc123" class="foo">…</my-element>
|
|
34
|
+
* ```
|
|
35
|
+
*
|
|
36
|
+
* When no eligible opening tag is found the original string is returned
|
|
37
|
+
* unchanged so callers never receive a broken fragment.
|
|
38
|
+
*
|
|
39
|
+
* @param content Rendered HTML string (or any value coercible to string).
|
|
40
|
+
* @param triggerId Stable trigger identifier produced by `buildResolvedLazyTriggers`.
|
|
41
|
+
*/
|
|
42
|
+
export function addTriggerAttribute(content: unknown, triggerId: string): string {
|
|
43
|
+
const str = String(content);
|
|
44
|
+
let i = 0;
|
|
45
|
+
|
|
46
|
+
while (i < str.length) {
|
|
47
|
+
if (str[i] !== '<') {
|
|
48
|
+
i++;
|
|
49
|
+
continue;
|
|
50
|
+
}
|
|
51
|
+
|
|
52
|
+
const next = str[i + 1];
|
|
53
|
+
|
|
54
|
+
if (next === '!' || next === '?') {
|
|
55
|
+
const end = str.indexOf('>', i);
|
|
56
|
+
if (end === -1) break;
|
|
57
|
+
i = end + 1;
|
|
58
|
+
continue;
|
|
59
|
+
}
|
|
60
|
+
|
|
61
|
+
if (next && /[a-zA-Z]/.test(next)) {
|
|
62
|
+
const tagSlice = str.slice(i + 1);
|
|
63
|
+
const nameEnd = tagSlice.search(/[\s/>]/);
|
|
64
|
+
if (nameEnd === -1) break;
|
|
65
|
+
const insertAt = i + 1 + nameEnd;
|
|
66
|
+
return `${str.slice(0, insertAt)} data-eco-trigger="${triggerId}"${str.slice(insertAt)}`;
|
|
67
|
+
}
|
|
68
|
+
|
|
69
|
+
break;
|
|
70
|
+
}
|
|
71
|
+
|
|
72
|
+
return str;
|
|
73
|
+
}
|
|
74
|
+
|
|
75
|
+
/**
|
|
76
|
+
* Wraps rendered component output in a `<scripts-injector>` element that
|
|
77
|
+
* carries an inline injector map for the legacy (non-global-injector) path.
|
|
78
|
+
*
|
|
79
|
+
* @param content Rendered component HTML.
|
|
80
|
+
* @param lazyGroups Resolved lazy script groups attached to the component config.
|
|
81
|
+
*/
|
|
82
|
+
export function wrapWithScriptsInjector(
|
|
83
|
+
content: unknown,
|
|
84
|
+
lazyGroups: NonNullable<EcoComponent['config']>['_resolvedLazyScripts'],
|
|
85
|
+
): string {
|
|
86
|
+
const wrappedContent = String(content);
|
|
87
|
+
const injectorMapScript = buildInjectorMapScript(lazyGroups ?? []);
|
|
88
|
+
return `<scripts-injector><script type="ecopages/injector-map">${injectorMapScript}</script>${wrappedContent}</scripts-injector>`;
|
|
89
|
+
}
|
|
@@ -0,0 +1,16 @@
|
|
|
1
|
+
import type { ResolvedLazyTrigger } from '../public-types.js';
|
|
2
|
+
/**
|
|
3
|
+
* Serializes resolved lazy triggers into a JSON string safe for embedding
|
|
4
|
+
* inside an inline `<script type="ecopages/global-injector-map">` tag.
|
|
5
|
+
*
|
|
6
|
+
* The `</script` sequence is escaped because an unescaped occurrence in inline
|
|
7
|
+
* script content causes the HTML parser to close the script tag prematurely,
|
|
8
|
+
* breaking page rendering. `JSON.stringify` does not perform this escape by
|
|
9
|
+
* default, so a targeted replacement is applied after serialization.
|
|
10
|
+
*/
|
|
11
|
+
export declare function buildGlobalInjectorMapScript(triggers: ResolvedLazyTrigger[]): string;
|
|
12
|
+
/**
|
|
13
|
+
* Builds the inline module script that boots the global lazy injector on the client.
|
|
14
|
+
* Emitted once per page alongside the `ecopages/global-injector-map` script block.
|
|
15
|
+
*/
|
|
16
|
+
export declare function buildGlobalInjectorBootstrapContent(globalInjectorModuleUrl: string): string;
|
|
@@ -0,0 +1,80 @@
|
|
|
1
|
+
function buildGlobalInjectorMap(triggers) {
|
|
2
|
+
const map = {};
|
|
3
|
+
for (const { triggerId, rules } of triggers) {
|
|
4
|
+
const entry = map[triggerId] ?? {};
|
|
5
|
+
for (const rule of rules) {
|
|
6
|
+
if ("on:idle" in rule) {
|
|
7
|
+
const { scripts } = rule["on:idle"];
|
|
8
|
+
entry["on:idle"] = {
|
|
9
|
+
scripts: Array.from(/* @__PURE__ */ new Set([...entry["on:idle"]?.scripts ?? [], ...scripts]))
|
|
10
|
+
};
|
|
11
|
+
continue;
|
|
12
|
+
}
|
|
13
|
+
if ("on:interaction" in rule) {
|
|
14
|
+
const { value, scripts } = rule["on:interaction"];
|
|
15
|
+
entry["on:interaction"] = {
|
|
16
|
+
value,
|
|
17
|
+
scripts: Array.from(/* @__PURE__ */ new Set([...entry["on:interaction"]?.scripts ?? [], ...scripts]))
|
|
18
|
+
};
|
|
19
|
+
continue;
|
|
20
|
+
}
|
|
21
|
+
if ("on:visible" in rule) {
|
|
22
|
+
const { value, scripts } = rule["on:visible"];
|
|
23
|
+
entry["on:visible"] = {
|
|
24
|
+
...value ? { value } : {},
|
|
25
|
+
scripts: Array.from(/* @__PURE__ */ new Set([...entry["on:visible"]?.scripts ?? [], ...scripts]))
|
|
26
|
+
};
|
|
27
|
+
continue;
|
|
28
|
+
}
|
|
29
|
+
}
|
|
30
|
+
if (Object.keys(entry).length > 0) {
|
|
31
|
+
map[triggerId] = entry;
|
|
32
|
+
}
|
|
33
|
+
}
|
|
34
|
+
return map;
|
|
35
|
+
}
|
|
36
|
+
function buildGlobalInjectorMapScript(triggers) {
|
|
37
|
+
const map = buildGlobalInjectorMap(triggers);
|
|
38
|
+
return JSON.stringify(map).replace(/<\/script/gi, "<\\/script");
|
|
39
|
+
}
|
|
40
|
+
function buildGlobalInjectorBootstrapContent(globalInjectorModuleUrl) {
|
|
41
|
+
return `import { initGlobalInjector } from ${JSON.stringify(globalInjectorModuleUrl)};
|
|
42
|
+
|
|
43
|
+
function pruneStaleTriggerMaps() {
|
|
44
|
+
const mapScripts = Array.from(document.querySelectorAll('script[type="ecopages/global-injector-map"]'));
|
|
45
|
+
if (mapScripts.length <= 1) {
|
|
46
|
+
return;
|
|
47
|
+
}
|
|
48
|
+
|
|
49
|
+
const latestMapScript = mapScripts[mapScripts.length - 1];
|
|
50
|
+
for (const mapScript of mapScripts) {
|
|
51
|
+
if (mapScript !== latestMapScript) {
|
|
52
|
+
mapScript.remove();
|
|
53
|
+
}
|
|
54
|
+
}
|
|
55
|
+
}
|
|
56
|
+
|
|
57
|
+
const globalScope = window;
|
|
58
|
+
if (typeof globalScope.__ecoGlobalInjectorCleanup === 'function') {
|
|
59
|
+
globalScope.__ecoGlobalInjectorCleanup();
|
|
60
|
+
}
|
|
61
|
+
|
|
62
|
+
const injector = initGlobalInjector();
|
|
63
|
+
|
|
64
|
+
const handleAfterSwap = () => {
|
|
65
|
+
pruneStaleTriggerMaps();
|
|
66
|
+
injector.refresh();
|
|
67
|
+
};
|
|
68
|
+
|
|
69
|
+
document.addEventListener('eco:after-swap', handleAfterSwap);
|
|
70
|
+
pruneStaleTriggerMaps();
|
|
71
|
+
|
|
72
|
+
globalScope.__ecoGlobalInjectorCleanup = () => {
|
|
73
|
+
document.removeEventListener('eco:after-swap', handleAfterSwap);
|
|
74
|
+
injector.cleanup();
|
|
75
|
+
};`;
|
|
76
|
+
}
|
|
77
|
+
export {
|
|
78
|
+
buildGlobalInjectorBootstrapContent,
|
|
79
|
+
buildGlobalInjectorMapScript
|
|
80
|
+
};
|
|
@@ -0,0 +1,112 @@
|
|
|
1
|
+
import type { ResolvedLazyTrigger } from '../public-types.ts';
|
|
2
|
+
|
|
3
|
+
type GlobalInjectorEntry = {
|
|
4
|
+
'on:idle'?: { scripts: string[] };
|
|
5
|
+
'on:interaction'?: { value: string; scripts: string[] };
|
|
6
|
+
'on:visible'?: { value?: string; scripts: string[] };
|
|
7
|
+
};
|
|
8
|
+
|
|
9
|
+
type GlobalInjectorMap = Record<string, GlobalInjectorEntry>;
|
|
10
|
+
|
|
11
|
+
/**
|
|
12
|
+
* Converts a flat list of resolved triggers into the map structure consumed
|
|
13
|
+
* by the `initGlobalInjector` bootstrap. Multiple triggers sharing the same
|
|
14
|
+
* `triggerId` are merged so their script lists are deduplicated.
|
|
15
|
+
*/
|
|
16
|
+
function buildGlobalInjectorMap(triggers: ResolvedLazyTrigger[]): GlobalInjectorMap {
|
|
17
|
+
const map: GlobalInjectorMap = {};
|
|
18
|
+
|
|
19
|
+
for (const { triggerId, rules } of triggers) {
|
|
20
|
+
const entry: GlobalInjectorEntry = map[triggerId] ?? {};
|
|
21
|
+
|
|
22
|
+
for (const rule of rules) {
|
|
23
|
+
if ('on:idle' in rule) {
|
|
24
|
+
const { scripts } = rule['on:idle'];
|
|
25
|
+
entry['on:idle'] = {
|
|
26
|
+
scripts: Array.from(new Set([...(entry['on:idle']?.scripts ?? []), ...scripts])),
|
|
27
|
+
};
|
|
28
|
+
continue;
|
|
29
|
+
}
|
|
30
|
+
|
|
31
|
+
if ('on:interaction' in rule) {
|
|
32
|
+
const { value, scripts } = rule['on:interaction'];
|
|
33
|
+
entry['on:interaction'] = {
|
|
34
|
+
value,
|
|
35
|
+
scripts: Array.from(new Set([...(entry['on:interaction']?.scripts ?? []), ...scripts])),
|
|
36
|
+
};
|
|
37
|
+
continue;
|
|
38
|
+
}
|
|
39
|
+
|
|
40
|
+
if ('on:visible' in rule) {
|
|
41
|
+
const { value, scripts } = rule['on:visible'];
|
|
42
|
+
entry['on:visible'] = {
|
|
43
|
+
...(value ? { value } : {}),
|
|
44
|
+
scripts: Array.from(new Set([...(entry['on:visible']?.scripts ?? []), ...scripts])),
|
|
45
|
+
};
|
|
46
|
+
continue;
|
|
47
|
+
}
|
|
48
|
+
}
|
|
49
|
+
|
|
50
|
+
if (Object.keys(entry).length > 0) {
|
|
51
|
+
map[triggerId] = entry;
|
|
52
|
+
}
|
|
53
|
+
}
|
|
54
|
+
|
|
55
|
+
return map;
|
|
56
|
+
}
|
|
57
|
+
|
|
58
|
+
/**
|
|
59
|
+
* Serializes resolved lazy triggers into a JSON string safe for embedding
|
|
60
|
+
* inside an inline `<script type="ecopages/global-injector-map">` tag.
|
|
61
|
+
*
|
|
62
|
+
* The `</script` sequence is escaped because an unescaped occurrence in inline
|
|
63
|
+
* script content causes the HTML parser to close the script tag prematurely,
|
|
64
|
+
* breaking page rendering. `JSON.stringify` does not perform this escape by
|
|
65
|
+
* default, so a targeted replacement is applied after serialization.
|
|
66
|
+
*/
|
|
67
|
+
export function buildGlobalInjectorMapScript(triggers: ResolvedLazyTrigger[]): string {
|
|
68
|
+
const map = buildGlobalInjectorMap(triggers);
|
|
69
|
+
return JSON.stringify(map).replace(/<\/script/gi, '<\\/script');
|
|
70
|
+
}
|
|
71
|
+
|
|
72
|
+
/**
|
|
73
|
+
* Builds the inline module script that boots the global lazy injector on the client.
|
|
74
|
+
* Emitted once per page alongside the `ecopages/global-injector-map` script block.
|
|
75
|
+
*/
|
|
76
|
+
export function buildGlobalInjectorBootstrapContent(globalInjectorModuleUrl: string): string {
|
|
77
|
+
return `import { initGlobalInjector } from ${JSON.stringify(globalInjectorModuleUrl)};
|
|
78
|
+
|
|
79
|
+
function pruneStaleTriggerMaps() {
|
|
80
|
+
const mapScripts = Array.from(document.querySelectorAll('script[type="ecopages/global-injector-map"]'));
|
|
81
|
+
if (mapScripts.length <= 1) {
|
|
82
|
+
return;
|
|
83
|
+
}
|
|
84
|
+
|
|
85
|
+
const latestMapScript = mapScripts[mapScripts.length - 1];
|
|
86
|
+
for (const mapScript of mapScripts) {
|
|
87
|
+
if (mapScript !== latestMapScript) {
|
|
88
|
+
mapScript.remove();
|
|
89
|
+
}
|
|
90
|
+
}
|
|
91
|
+
}
|
|
92
|
+
|
|
93
|
+
const globalScope = window;
|
|
94
|
+
if (typeof globalScope.__ecoGlobalInjectorCleanup === 'function') {
|
|
95
|
+
globalScope.__ecoGlobalInjectorCleanup();
|
|
96
|
+
}
|
|
97
|
+
|
|
98
|
+
const injector = initGlobalInjector();
|
|
99
|
+
|
|
100
|
+
const handleAfterSwap = () => {
|
|
101
|
+
pruneStaleTriggerMaps();
|
|
102
|
+
injector.refresh();
|
|
103
|
+
};
|
|
104
|
+
|
|
105
|
+
document.addEventListener('eco:after-swap', handleAfterSwap);
|
|
106
|
+
pruneStaleTriggerMaps();
|
|
107
|
+
|
|
108
|
+
globalScope.__ecoGlobalInjectorCleanup = () => {
|
|
109
|
+
document.removeEventListener('eco:after-swap', handleAfterSwap);
|
|
110
|
+
injector.cleanup();
|
|
111
|
+
};`;
|
|
112
|
+
}
|
|
@@ -0,0 +1,8 @@
|
|
|
1
|
+
import type { ResolvedLazyScriptGroup } from '../public-types.js';
|
|
2
|
+
/**
|
|
3
|
+
* Creates a safe JSON payload string for `<script type="ecopages/injector-map">`.
|
|
4
|
+
*
|
|
5
|
+
* @param lazyGroups Lazy script groups resolved during dependency processing.
|
|
6
|
+
* @returns Escaped JSON string safe for inline script embedding.
|
|
7
|
+
*/
|
|
8
|
+
export declare function buildInjectorMapScript(lazyGroups: ResolvedLazyScriptGroup[]): string;
|
|
@@ -0,0 +1,70 @@
|
|
|
1
|
+
function normalizeScripts(scripts) {
|
|
2
|
+
return scripts.split(",").map((scriptPath) => scriptPath.trim()).filter(Boolean).map((scriptPath) => {
|
|
3
|
+
if (scriptPath.startsWith("/")) {
|
|
4
|
+
return scriptPath;
|
|
5
|
+
}
|
|
6
|
+
const normalizedRelativePath = scriptPath.replace(/^\/+/, "").replace(/^(\.\/)+/, "");
|
|
7
|
+
return `/${normalizedRelativePath}`;
|
|
8
|
+
});
|
|
9
|
+
}
|
|
10
|
+
function dedupeCsvValues(values) {
|
|
11
|
+
const seen = /* @__PURE__ */ new Set();
|
|
12
|
+
for (const value of values) {
|
|
13
|
+
for (const token of value.split(",").map((part) => part.trim()).filter(Boolean)) {
|
|
14
|
+
seen.add(token);
|
|
15
|
+
}
|
|
16
|
+
}
|
|
17
|
+
return Array.from(seen).join(",");
|
|
18
|
+
}
|
|
19
|
+
function buildInjectorMap(lazyGroups) {
|
|
20
|
+
const map = {};
|
|
21
|
+
for (const group of lazyGroups) {
|
|
22
|
+
const scripts = normalizeScripts(group.scripts);
|
|
23
|
+
const lazy = group.lazy;
|
|
24
|
+
if ("on:idle" in lazy) {
|
|
25
|
+
map["on:idle"] = {
|
|
26
|
+
scripts: [...map["on:idle"]?.scripts ?? [], ...scripts]
|
|
27
|
+
};
|
|
28
|
+
continue;
|
|
29
|
+
}
|
|
30
|
+
if ("on:interaction" in lazy) {
|
|
31
|
+
const currentValue = map["on:interaction"]?.value;
|
|
32
|
+
const nextValue = lazy["on:interaction"];
|
|
33
|
+
map["on:interaction"] = {
|
|
34
|
+
value: typeof currentValue === "string" ? dedupeCsvValues([currentValue, nextValue]) : nextValue,
|
|
35
|
+
scripts: [...map["on:interaction"]?.scripts ?? [], ...scripts]
|
|
36
|
+
};
|
|
37
|
+
continue;
|
|
38
|
+
}
|
|
39
|
+
if ("on:visible" in lazy) {
|
|
40
|
+
const currentValue = map["on:visible"]?.value;
|
|
41
|
+
const currentThreshold = typeof currentValue === "string" ? currentValue : void 0;
|
|
42
|
+
const nextValue = lazy["on:visible"];
|
|
43
|
+
map["on:visible"] = {
|
|
44
|
+
...nextValue === true ? currentThreshold ? { value: currentThreshold } : {} : { value: String(nextValue) },
|
|
45
|
+
scripts: [...map["on:visible"]?.scripts ?? [], ...scripts]
|
|
46
|
+
};
|
|
47
|
+
continue;
|
|
48
|
+
}
|
|
49
|
+
throw new Error(
|
|
50
|
+
`Invalid lazy options: must specify on:idle, on:interaction, or on:visible. Received: ${JSON.stringify(lazy)}`
|
|
51
|
+
);
|
|
52
|
+
}
|
|
53
|
+
for (const key of Object.keys(map)) {
|
|
54
|
+
const scripts = map[key]?.scripts;
|
|
55
|
+
if (scripts) {
|
|
56
|
+
map[key] = {
|
|
57
|
+
...map[key],
|
|
58
|
+
scripts: Array.from(new Set(scripts))
|
|
59
|
+
};
|
|
60
|
+
}
|
|
61
|
+
}
|
|
62
|
+
return map;
|
|
63
|
+
}
|
|
64
|
+
function buildInjectorMapScript(lazyGroups) {
|
|
65
|
+
const map = buildInjectorMap(lazyGroups);
|
|
66
|
+
return JSON.stringify(map).replace(/<\/script/gi, "<\\/script");
|
|
67
|
+
}
|
|
68
|
+
export {
|
|
69
|
+
buildInjectorMapScript
|
|
70
|
+
};
|
|
@@ -0,0 +1,120 @@
|
|
|
1
|
+
import type { InjectorMapConfig } from '@ecopages/scripts-injector/types';
|
|
2
|
+
import type { ResolvedLazyScriptGroup } from '../public-types.ts';
|
|
3
|
+
|
|
4
|
+
/**
|
|
5
|
+
* Normalizes a comma-separated scripts string into absolute script URLs.
|
|
6
|
+
*
|
|
7
|
+
* @param scripts Comma-separated script paths.
|
|
8
|
+
* @returns Normalized script path list.
|
|
9
|
+
*/
|
|
10
|
+
function normalizeScripts(scripts: string): string[] {
|
|
11
|
+
return scripts
|
|
12
|
+
.split(',')
|
|
13
|
+
.map((scriptPath) => scriptPath.trim())
|
|
14
|
+
.filter(Boolean)
|
|
15
|
+
.map((scriptPath) => {
|
|
16
|
+
if (scriptPath.startsWith('/')) {
|
|
17
|
+
return scriptPath;
|
|
18
|
+
}
|
|
19
|
+
|
|
20
|
+
const normalizedRelativePath = scriptPath.replace(/^\/+/, '').replace(/^(\.\/)+/, '');
|
|
21
|
+
return `/${normalizedRelativePath}`;
|
|
22
|
+
});
|
|
23
|
+
}
|
|
24
|
+
|
|
25
|
+
/**
|
|
26
|
+
* Deduplicates comma-separated values while preserving first-seen order.
|
|
27
|
+
*
|
|
28
|
+
* @param values CSV value list.
|
|
29
|
+
* @returns A single normalized CSV string.
|
|
30
|
+
*/
|
|
31
|
+
function dedupeCsvValues(values: string[]): string {
|
|
32
|
+
const seen = new Set<string>();
|
|
33
|
+
|
|
34
|
+
for (const value of values) {
|
|
35
|
+
for (const token of value
|
|
36
|
+
.split(',')
|
|
37
|
+
.map((part) => part.trim())
|
|
38
|
+
.filter(Boolean)) {
|
|
39
|
+
seen.add(token);
|
|
40
|
+
}
|
|
41
|
+
}
|
|
42
|
+
|
|
43
|
+
return Array.from(seen).join(',');
|
|
44
|
+
}
|
|
45
|
+
|
|
46
|
+
/**
|
|
47
|
+
* Builds a scripts-injector map config from resolved lazy script groups.
|
|
48
|
+
*
|
|
49
|
+
* @param lazyGroups Lazy script groups resolved during dependency processing.
|
|
50
|
+
* @returns Injector map config keyed by lazy trigger.
|
|
51
|
+
*/
|
|
52
|
+
function buildInjectorMap(lazyGroups: ResolvedLazyScriptGroup[]): InjectorMapConfig {
|
|
53
|
+
const map: InjectorMapConfig = {};
|
|
54
|
+
|
|
55
|
+
for (const group of lazyGroups) {
|
|
56
|
+
const scripts = normalizeScripts(group.scripts);
|
|
57
|
+
const lazy = group.lazy;
|
|
58
|
+
|
|
59
|
+
if ('on:idle' in lazy) {
|
|
60
|
+
map['on:idle'] = {
|
|
61
|
+
scripts: [...(map['on:idle']?.scripts ?? []), ...scripts],
|
|
62
|
+
};
|
|
63
|
+
continue;
|
|
64
|
+
}
|
|
65
|
+
|
|
66
|
+
if ('on:interaction' in lazy) {
|
|
67
|
+
const currentValue = map['on:interaction']?.value;
|
|
68
|
+
const nextValue = lazy['on:interaction'];
|
|
69
|
+
|
|
70
|
+
map['on:interaction'] = {
|
|
71
|
+
value: typeof currentValue === 'string' ? dedupeCsvValues([currentValue, nextValue]) : nextValue,
|
|
72
|
+
scripts: [...(map['on:interaction']?.scripts ?? []), ...scripts],
|
|
73
|
+
};
|
|
74
|
+
continue;
|
|
75
|
+
}
|
|
76
|
+
|
|
77
|
+
if ('on:visible' in lazy) {
|
|
78
|
+
const currentValue = map['on:visible']?.value;
|
|
79
|
+
const currentThreshold = typeof currentValue === 'string' ? currentValue : undefined;
|
|
80
|
+
const nextValue = lazy['on:visible'];
|
|
81
|
+
|
|
82
|
+
map['on:visible'] = {
|
|
83
|
+
...(nextValue === true
|
|
84
|
+
? currentThreshold
|
|
85
|
+
? { value: currentThreshold }
|
|
86
|
+
: {}
|
|
87
|
+
: { value: String(nextValue) }),
|
|
88
|
+
scripts: [...(map['on:visible']?.scripts ?? []), ...scripts],
|
|
89
|
+
};
|
|
90
|
+
continue;
|
|
91
|
+
}
|
|
92
|
+
|
|
93
|
+
throw new Error(
|
|
94
|
+
`Invalid lazy options: must specify on:idle, on:interaction, or on:visible. Received: ${JSON.stringify(lazy)}`,
|
|
95
|
+
);
|
|
96
|
+
}
|
|
97
|
+
|
|
98
|
+
for (const key of Object.keys(map)) {
|
|
99
|
+
const scripts = map[key]?.scripts;
|
|
100
|
+
if (scripts) {
|
|
101
|
+
map[key] = {
|
|
102
|
+
...map[key],
|
|
103
|
+
scripts: Array.from(new Set(scripts)),
|
|
104
|
+
};
|
|
105
|
+
}
|
|
106
|
+
}
|
|
107
|
+
|
|
108
|
+
return map;
|
|
109
|
+
}
|
|
110
|
+
|
|
111
|
+
/**
|
|
112
|
+
* Creates a safe JSON payload string for `<script type="ecopages/injector-map">`.
|
|
113
|
+
*
|
|
114
|
+
* @param lazyGroups Lazy script groups resolved during dependency processing.
|
|
115
|
+
* @returns Escaped JSON string safe for inline script embedding.
|
|
116
|
+
*/
|
|
117
|
+
export function buildInjectorMapScript(lazyGroups: ResolvedLazyScriptGroup[]): string {
|
|
118
|
+
const map = buildInjectorMap(lazyGroups);
|
|
119
|
+
return JSON.stringify(map).replace(/<\/script/gi, '<\\/script');
|
|
120
|
+
}
|
|
@@ -0,0 +1,18 @@
|
|
|
1
|
+
export type ModuleDependencyDeclaration = {
|
|
2
|
+
from: string;
|
|
3
|
+
imports?: string[];
|
|
4
|
+
};
|
|
5
|
+
/**
|
|
6
|
+
* Parses a module grammar declaration into normalized object form.
|
|
7
|
+
*
|
|
8
|
+
* Supported input examples:
|
|
9
|
+
* - `react-aria-components`
|
|
10
|
+
* - `react-aria-components{Table,Select}`
|
|
11
|
+
*/
|
|
12
|
+
export declare function parseModuleDeclaration(value: string): ModuleDependencyDeclaration;
|
|
13
|
+
/**
|
|
14
|
+
* Normalizes and deduplicates module declarations from `modules`.
|
|
15
|
+
*
|
|
16
|
+
* Returns canonical object declarations suitable for downstream bundling logic.
|
|
17
|
+
*/
|
|
18
|
+
export declare function normalizeModuleDeclarations(modules?: string[]): ModuleDependencyDeclaration[];
|
|
@@ -0,0 +1,49 @@
|
|
|
1
|
+
function normalizeImports(imports) {
|
|
2
|
+
if (!imports || imports.length === 0) {
|
|
3
|
+
return void 0;
|
|
4
|
+
}
|
|
5
|
+
const normalized = Array.from(new Set(imports.map((entry) => entry.trim()).filter((entry) => entry.length > 0)));
|
|
6
|
+
return normalized.length > 0 ? normalized : void 0;
|
|
7
|
+
}
|
|
8
|
+
function parseModuleDeclaration(value) {
|
|
9
|
+
const source = value.trim();
|
|
10
|
+
if (source.length === 0) {
|
|
11
|
+
throw new Error("Module declaration cannot be empty");
|
|
12
|
+
}
|
|
13
|
+
const openBraceIndex = source.indexOf("{");
|
|
14
|
+
if (openBraceIndex < 0) {
|
|
15
|
+
return { from: source };
|
|
16
|
+
}
|
|
17
|
+
const closeBraceIndex = source.lastIndexOf("}");
|
|
18
|
+
if (closeBraceIndex !== source.length - 1 || closeBraceIndex <= openBraceIndex) {
|
|
19
|
+
throw new Error(`Invalid module declaration '${value}'`);
|
|
20
|
+
}
|
|
21
|
+
const from = source.slice(0, openBraceIndex).trim();
|
|
22
|
+
if (from.length === 0) {
|
|
23
|
+
throw new Error(`Invalid module declaration '${value}'`);
|
|
24
|
+
}
|
|
25
|
+
const importsSlice = source.slice(openBraceIndex + 1, closeBraceIndex);
|
|
26
|
+
const imports = normalizeImports(importsSlice.split(","));
|
|
27
|
+
return {
|
|
28
|
+
from,
|
|
29
|
+
imports
|
|
30
|
+
};
|
|
31
|
+
}
|
|
32
|
+
function normalizeModuleDeclarations(modules) {
|
|
33
|
+
const declarations = modules ?? [];
|
|
34
|
+
const output = [];
|
|
35
|
+
const seen = /* @__PURE__ */ new Set();
|
|
36
|
+
for (const declaration of declarations) {
|
|
37
|
+
const normalized = parseModuleDeclaration(declaration);
|
|
38
|
+
const key = `${normalized.from}::${(normalized.imports ?? []).join(",")}`;
|
|
39
|
+
if (!seen.has(key)) {
|
|
40
|
+
seen.add(key);
|
|
41
|
+
output.push(normalized);
|
|
42
|
+
}
|
|
43
|
+
}
|
|
44
|
+
return output;
|
|
45
|
+
}
|
|
46
|
+
export {
|
|
47
|
+
normalizeModuleDeclarations,
|
|
48
|
+
parseModuleDeclaration
|
|
49
|
+
};
|
|
@@ -0,0 +1,75 @@
|
|
|
1
|
+
export type ModuleDependencyDeclaration = {
|
|
2
|
+
from: string;
|
|
3
|
+
imports?: string[];
|
|
4
|
+
};
|
|
5
|
+
|
|
6
|
+
function normalizeImports(imports: string[] | undefined): string[] | undefined {
|
|
7
|
+
if (!imports || imports.length === 0) {
|
|
8
|
+
return undefined;
|
|
9
|
+
}
|
|
10
|
+
|
|
11
|
+
const normalized = Array.from(new Set(imports.map((entry) => entry.trim()).filter((entry) => entry.length > 0)));
|
|
12
|
+
|
|
13
|
+
return normalized.length > 0 ? normalized : undefined;
|
|
14
|
+
}
|
|
15
|
+
|
|
16
|
+
/**
|
|
17
|
+
* Parses a module grammar declaration into normalized object form.
|
|
18
|
+
*
|
|
19
|
+
* Supported input examples:
|
|
20
|
+
* - `react-aria-components`
|
|
21
|
+
* - `react-aria-components{Table,Select}`
|
|
22
|
+
*/
|
|
23
|
+
export function parseModuleDeclaration(value: string): ModuleDependencyDeclaration {
|
|
24
|
+
const source = value.trim();
|
|
25
|
+
if (source.length === 0) {
|
|
26
|
+
throw new Error('Module declaration cannot be empty');
|
|
27
|
+
}
|
|
28
|
+
|
|
29
|
+
const openBraceIndex = source.indexOf('{');
|
|
30
|
+
if (openBraceIndex < 0) {
|
|
31
|
+
return { from: source };
|
|
32
|
+
}
|
|
33
|
+
|
|
34
|
+
const closeBraceIndex = source.lastIndexOf('}');
|
|
35
|
+
if (closeBraceIndex !== source.length - 1 || closeBraceIndex <= openBraceIndex) {
|
|
36
|
+
throw new Error(`Invalid module declaration '${value}'`);
|
|
37
|
+
}
|
|
38
|
+
|
|
39
|
+
const from = source.slice(0, openBraceIndex).trim();
|
|
40
|
+
if (from.length === 0) {
|
|
41
|
+
throw new Error(`Invalid module declaration '${value}'`);
|
|
42
|
+
}
|
|
43
|
+
|
|
44
|
+
const importsSlice = source.slice(openBraceIndex + 1, closeBraceIndex);
|
|
45
|
+
const imports = normalizeImports(importsSlice.split(','));
|
|
46
|
+
|
|
47
|
+
return {
|
|
48
|
+
from,
|
|
49
|
+
imports,
|
|
50
|
+
};
|
|
51
|
+
}
|
|
52
|
+
|
|
53
|
+
/**
|
|
54
|
+
* Normalizes and deduplicates module declarations from `modules`.
|
|
55
|
+
*
|
|
56
|
+
* Returns canonical object declarations suitable for downstream bundling logic.
|
|
57
|
+
*/
|
|
58
|
+
export function normalizeModuleDeclarations(modules?: string[]): ModuleDependencyDeclaration[] {
|
|
59
|
+
const declarations = modules ?? [];
|
|
60
|
+
|
|
61
|
+
const output: ModuleDependencyDeclaration[] = [];
|
|
62
|
+
const seen = new Set<string>();
|
|
63
|
+
|
|
64
|
+
for (const declaration of declarations) {
|
|
65
|
+
const normalized = parseModuleDeclaration(declaration);
|
|
66
|
+
const key = `${normalized.from}::${(normalized.imports ?? []).join(',')}`;
|
|
67
|
+
|
|
68
|
+
if (!seen.has(key)) {
|
|
69
|
+
seen.add(key);
|
|
70
|
+
output.push(normalized);
|
|
71
|
+
}
|
|
72
|
+
}
|
|
73
|
+
|
|
74
|
+
return output;
|
|
75
|
+
}
|
package/src/env.d.ts
ADDED
|
@@ -0,0 +1,20 @@
|
|
|
1
|
+
interface EcopagesEnv {
|
|
2
|
+
ECOPAGES_BASE_URL: string;
|
|
3
|
+
ECOPAGES_HOSTNAME: string;
|
|
4
|
+
ECOPAGES_PORT: string;
|
|
5
|
+
ECOPAGES_LOGGER_DEBUG: 'true' | 'false';
|
|
6
|
+
}
|
|
7
|
+
|
|
8
|
+
declare global {
|
|
9
|
+
namespace NodeJS {
|
|
10
|
+
interface ProcessEnv extends EcopagesEnv {}
|
|
11
|
+
}
|
|
12
|
+
|
|
13
|
+
namespace Bun {
|
|
14
|
+
interface Env extends EcopagesEnv {}
|
|
15
|
+
}
|
|
16
|
+
|
|
17
|
+
interface ImportMetaEnv extends EcopagesEnv {}
|
|
18
|
+
}
|
|
19
|
+
|
|
20
|
+
export {};
|
|
@@ -0,0 +1,31 @@
|
|
|
1
|
+
export interface HttpErrorDetails {
|
|
2
|
+
[key: string]: unknown;
|
|
3
|
+
}
|
|
4
|
+
export interface HttpErrorJson {
|
|
5
|
+
error: string;
|
|
6
|
+
status: number;
|
|
7
|
+
details?: HttpErrorDetails;
|
|
8
|
+
}
|
|
9
|
+
/**
|
|
10
|
+
* HTTP error class for structured error handling in route handlers.
|
|
11
|
+
* Provides static factory methods for common HTTP error responses.
|
|
12
|
+
*/
|
|
13
|
+
export declare class HttpError extends Error {
|
|
14
|
+
readonly status: number;
|
|
15
|
+
readonly details?: HttpErrorDetails;
|
|
16
|
+
constructor(status: number, message: string, details?: HttpErrorDetails);
|
|
17
|
+
/**
|
|
18
|
+
* Serialize error to JSON for API responses.
|
|
19
|
+
*/
|
|
20
|
+
toJSON(): HttpErrorJson;
|
|
21
|
+
/**
|
|
22
|
+
* Create a Response object from this error.
|
|
23
|
+
*/
|
|
24
|
+
toResponse(): Response;
|
|
25
|
+
static BadRequest(message?: string, details?: HttpErrorDetails): HttpError;
|
|
26
|
+
static Unauthorized(message?: string): HttpError;
|
|
27
|
+
static Forbidden(message?: string): HttpError;
|
|
28
|
+
static NotFound(message?: string): HttpError;
|
|
29
|
+
static Conflict(message?: string, details?: HttpErrorDetails): HttpError;
|
|
30
|
+
static InternalServerError(message?: string): HttpError;
|
|
31
|
+
}
|