@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,306 @@
|
|
|
1
|
+
import path from 'node:path';
|
|
2
|
+
import chokidar, { type FSWatcher } from 'chokidar';
|
|
3
|
+
import { fileSystem } from '@ecopages/file-system';
|
|
4
|
+
import { appLogger } from '../global/app-logger.ts';
|
|
5
|
+
import type { EcoPagesAppConfig, IHmrManager, IClientBridge } from '../internal-types.ts';
|
|
6
|
+
import type { ProcessorWatchContext } from '../plugins/processor.ts';
|
|
7
|
+
|
|
8
|
+
/**
|
|
9
|
+
* Configuration options for the ProjectWatcher
|
|
10
|
+
* @interface ProjectWatcherConfig
|
|
11
|
+
* @property {EcoPagesAppConfig} config - The application configuration
|
|
12
|
+
* @property {() => void} refreshRouterRoutesCallback - Callback to refresh router routes
|
|
13
|
+
* @property {IHmrManager} hmrManager - The HMR manager instance
|
|
14
|
+
* @property {ClientBridge} bridge - The client bridge instance
|
|
15
|
+
*/
|
|
16
|
+
export interface ProjectWatcherConfig {
|
|
17
|
+
config: EcoPagesAppConfig;
|
|
18
|
+
refreshRouterRoutesCallback: () => void;
|
|
19
|
+
hmrManager: IHmrManager;
|
|
20
|
+
bridge: IClientBridge;
|
|
21
|
+
}
|
|
22
|
+
|
|
23
|
+
/**
|
|
24
|
+
* ProjectWatcher handles file system changes for hot module replacement (HMR).
|
|
25
|
+
* It uses chokidar to watch for file changes and triggers appropriate actions:
|
|
26
|
+
* - Uncaches modules when files change
|
|
27
|
+
* - Refreshes router routes for page files
|
|
28
|
+
* - Triggers HMR server reload
|
|
29
|
+
* - Handles processor-specific file changes
|
|
30
|
+
*
|
|
31
|
+
* The watcher uses chokidar's built-in debouncing through `awaitWriteFinish`
|
|
32
|
+
* to handle rapid file changes efficiently:
|
|
33
|
+
* - stabilityThreshold: 50ms - Time to wait for writes to stabilize
|
|
34
|
+
* - pollInterval: 50ms - Interval to poll for file changes
|
|
35
|
+
*
|
|
36
|
+
* @class ProjectWatcher
|
|
37
|
+
*/
|
|
38
|
+
export class ProjectWatcher {
|
|
39
|
+
/**
|
|
40
|
+
* Duplicate identical watcher events within this window are ignored.
|
|
41
|
+
*
|
|
42
|
+
* Some editors or save pipelines emit two near-identical filesystem change
|
|
43
|
+
* notifications for the same file. Ecopages should treat those as one logical
|
|
44
|
+
* update so HMR and route refresh work are not repeated unnecessarily.
|
|
45
|
+
*/
|
|
46
|
+
private static readonly duplicateChangeWindowMs = 150;
|
|
47
|
+
private appConfig: EcoPagesAppConfig;
|
|
48
|
+
private refreshRouterRoutesCallback: () => void;
|
|
49
|
+
private hmrManager: IHmrManager;
|
|
50
|
+
private bridge: IClientBridge;
|
|
51
|
+
private watcher: FSWatcher | null = null;
|
|
52
|
+
private lastHandledChange = new Map<string, number>();
|
|
53
|
+
|
|
54
|
+
constructor({ config, refreshRouterRoutesCallback, hmrManager, bridge }: ProjectWatcherConfig) {
|
|
55
|
+
this.appConfig = config;
|
|
56
|
+
this.refreshRouterRoutesCallback = refreshRouterRoutesCallback;
|
|
57
|
+
this.hmrManager = hmrManager;
|
|
58
|
+
this.bridge = bridge;
|
|
59
|
+
this.triggerRouterRefresh = this.triggerRouterRefresh.bind(this);
|
|
60
|
+
this.handleError = this.handleError.bind(this);
|
|
61
|
+
this.handleFileChange = this.handleFileChange.bind(this);
|
|
62
|
+
}
|
|
63
|
+
|
|
64
|
+
/**
|
|
65
|
+
* Uncaches modules in the source directory to ensure fresh imports.
|
|
66
|
+
* This is necessary for hot module replacement to work correctly.
|
|
67
|
+
* @private
|
|
68
|
+
*/
|
|
69
|
+
private uncacheModules(): void {
|
|
70
|
+
if (typeof require === 'undefined') return;
|
|
71
|
+
|
|
72
|
+
const { srcDir, rootDir } = this.appConfig;
|
|
73
|
+
const regex = new RegExp(`${rootDir}/${srcDir}/.*`);
|
|
74
|
+
|
|
75
|
+
for (const key in require.cache) {
|
|
76
|
+
if (regex.test(key)) {
|
|
77
|
+
delete require.cache[key];
|
|
78
|
+
}
|
|
79
|
+
}
|
|
80
|
+
}
|
|
81
|
+
|
|
82
|
+
/**
|
|
83
|
+
* Handles public directory file changes by copying only the changed file.
|
|
84
|
+
* @param filePath - Absolute path of the changed file
|
|
85
|
+
*/
|
|
86
|
+
private async handlePublicDirFileChange(filePath: string): Promise<void> {
|
|
87
|
+
try {
|
|
88
|
+
const relativePath = path.relative(this.appConfig.absolutePaths.publicDir, filePath);
|
|
89
|
+
const destPath = path.join(this.appConfig.absolutePaths.distDir, relativePath);
|
|
90
|
+
|
|
91
|
+
if (fileSystem.exists(filePath)) {
|
|
92
|
+
const destDir = path.dirname(destPath);
|
|
93
|
+
fileSystem.ensureDir(destDir);
|
|
94
|
+
await fileSystem.copyFileAsync(filePath, destPath);
|
|
95
|
+
}
|
|
96
|
+
|
|
97
|
+
this.bridge.reload();
|
|
98
|
+
} catch (error) {
|
|
99
|
+
appLogger.error(`Failed to copy public file: ${error instanceof Error ? error.message : String(error)}`);
|
|
100
|
+
this.bridge.reload();
|
|
101
|
+
}
|
|
102
|
+
}
|
|
103
|
+
|
|
104
|
+
/**
|
|
105
|
+
* Handles file changes by uncaching modules, refreshing routes, and delegating appropriately.
|
|
106
|
+
* Follows 4-rule priority:
|
|
107
|
+
* 0. Public directory match? → copy file and reload
|
|
108
|
+
* 1. additionalWatchPaths match? → reload
|
|
109
|
+
* 2. Processor extension match? → processor handles (skip HMR)
|
|
110
|
+
* 3. Otherwise → HMR strategies
|
|
111
|
+
*
|
|
112
|
+
* Duplicate identical watcher events for the same file are coalesced within a
|
|
113
|
+
* short window before any of the priority rules run.
|
|
114
|
+
* @param rawPath - Path of the changed file
|
|
115
|
+
*/
|
|
116
|
+
private async handleFileChange(rawPath: string): Promise<void> {
|
|
117
|
+
const filePath = path.resolve(rawPath);
|
|
118
|
+
const now = Date.now();
|
|
119
|
+
const lastHandledAt = this.lastHandledChange.get(filePath);
|
|
120
|
+
if (lastHandledAt !== undefined && now - lastHandledAt < ProjectWatcher.duplicateChangeWindowMs) {
|
|
121
|
+
return;
|
|
122
|
+
}
|
|
123
|
+
this.lastHandledChange.set(filePath, now);
|
|
124
|
+
|
|
125
|
+
try {
|
|
126
|
+
if (this.isPublicDirFile(filePath)) {
|
|
127
|
+
await this.handlePublicDirFileChange(filePath);
|
|
128
|
+
return;
|
|
129
|
+
}
|
|
130
|
+
|
|
131
|
+
this.uncacheModules();
|
|
132
|
+
const isPageFile = filePath.startsWith(this.appConfig.absolutePaths.pagesDir);
|
|
133
|
+
|
|
134
|
+
if (isPageFile) {
|
|
135
|
+
this.refreshRouterRoutesCallback();
|
|
136
|
+
}
|
|
137
|
+
|
|
138
|
+
if (this.matchesAdditionalWatchPaths(filePath)) {
|
|
139
|
+
this.bridge.reload();
|
|
140
|
+
return;
|
|
141
|
+
}
|
|
142
|
+
|
|
143
|
+
if (this.isHandledByProcessor(filePath)) {
|
|
144
|
+
return;
|
|
145
|
+
}
|
|
146
|
+
|
|
147
|
+
await this.hmrManager.handleFileChange(filePath);
|
|
148
|
+
} catch (error) {
|
|
149
|
+
if (error instanceof Error) {
|
|
150
|
+
this.bridge.error(error.message);
|
|
151
|
+
this.handleError(error);
|
|
152
|
+
}
|
|
153
|
+
}
|
|
154
|
+
}
|
|
155
|
+
|
|
156
|
+
/**
|
|
157
|
+
* Checks if a file is in the public directory.
|
|
158
|
+
*/
|
|
159
|
+
private isPublicDirFile(filePath: string): boolean {
|
|
160
|
+
return filePath.startsWith(this.appConfig.absolutePaths.publicDir);
|
|
161
|
+
}
|
|
162
|
+
|
|
163
|
+
/**
|
|
164
|
+
* Checks if file path matches any additionalWatchPaths patterns.
|
|
165
|
+
*/
|
|
166
|
+
private matchesAdditionalWatchPaths(filePath: string): boolean {
|
|
167
|
+
const patterns = this.appConfig.additionalWatchPaths;
|
|
168
|
+
if (!patterns.length) return false;
|
|
169
|
+
|
|
170
|
+
for (const pattern of patterns) {
|
|
171
|
+
if (pattern.includes('*')) {
|
|
172
|
+
const ext = pattern.replace(/\*\*?\/\*/, '');
|
|
173
|
+
if (filePath.endsWith(ext)) return true;
|
|
174
|
+
} else {
|
|
175
|
+
if (filePath.endsWith(pattern) || filePath === path.resolve(pattern)) return true;
|
|
176
|
+
}
|
|
177
|
+
}
|
|
178
|
+
return false;
|
|
179
|
+
}
|
|
180
|
+
|
|
181
|
+
/**
|
|
182
|
+
* Checks if a file is handled by a processor.
|
|
183
|
+
* Processors that declare extensions own those file types.
|
|
184
|
+
*/
|
|
185
|
+
private isHandledByProcessor(filePath: string): boolean {
|
|
186
|
+
for (const processor of this.appConfig.processors.values()) {
|
|
187
|
+
const watchConfig = processor.getWatchConfig();
|
|
188
|
+
if (!watchConfig) continue;
|
|
189
|
+
|
|
190
|
+
const { extensions = [] } = watchConfig;
|
|
191
|
+
if (extensions.length && extensions.some((ext) => filePath.endsWith(ext))) {
|
|
192
|
+
return true;
|
|
193
|
+
}
|
|
194
|
+
}
|
|
195
|
+
return false;
|
|
196
|
+
}
|
|
197
|
+
|
|
198
|
+
/**
|
|
199
|
+
* Triggers router refresh for page directory changes.
|
|
200
|
+
* This ensures the router is updated when pages are added or removed.
|
|
201
|
+
*
|
|
202
|
+
* @param {string} path - Path of the changed directory
|
|
203
|
+
*/
|
|
204
|
+
triggerRouterRefresh(path: string) {
|
|
205
|
+
const isPageDir = path.startsWith(this.appConfig.absolutePaths.pagesDir);
|
|
206
|
+
if (isPageDir) {
|
|
207
|
+
this.refreshRouterRoutesCallback();
|
|
208
|
+
}
|
|
209
|
+
}
|
|
210
|
+
|
|
211
|
+
/**
|
|
212
|
+
* Handles and logs errors that occur during file watching.
|
|
213
|
+
*
|
|
214
|
+
* @param {unknown} error - The error to handle
|
|
215
|
+
*/
|
|
216
|
+
handleError(error: unknown) {
|
|
217
|
+
if (error instanceof Error) {
|
|
218
|
+
this.hmrManager.broadcast({ type: 'error', message: error.message });
|
|
219
|
+
}
|
|
220
|
+
appLogger.error(`Watcher error: ${error}`);
|
|
221
|
+
}
|
|
222
|
+
|
|
223
|
+
/**
|
|
224
|
+
* Processes file changes for specific file extensions.
|
|
225
|
+
* Used by processors to handle their specific file types.
|
|
226
|
+
*
|
|
227
|
+
* @private
|
|
228
|
+
* @param {string} path - Path of the changed file
|
|
229
|
+
* @param {string[]} extensions - File extensions to process
|
|
230
|
+
* @param {(ctx: ProcessorWatchContext) => void} handler - Handler function for the file change
|
|
231
|
+
*/
|
|
232
|
+
private shouldProcess(path: string, extensions: string[], handler: (ctx: ProcessorWatchContext) => void) {
|
|
233
|
+
if (!extensions.length || extensions.some((ext) => path.endsWith(ext))) {
|
|
234
|
+
handler({ path, bridge: this.bridge });
|
|
235
|
+
}
|
|
236
|
+
}
|
|
237
|
+
|
|
238
|
+
/**
|
|
239
|
+
* Creates and configures the file system watcher.
|
|
240
|
+
* This sets up:
|
|
241
|
+
* 1. Processor-specific file watching
|
|
242
|
+
* 2. Page file watching
|
|
243
|
+
* 3. Directory watching
|
|
244
|
+
* 4. Error handling
|
|
245
|
+
*
|
|
246
|
+
* Uses chokidar's built-in debouncing through `awaitWriteFinish` to handle
|
|
247
|
+
* rapid file changes efficiently.
|
|
248
|
+
*/
|
|
249
|
+
public async createWatcherSubscription() {
|
|
250
|
+
if (!this.watcher) {
|
|
251
|
+
const processorPaths: string[] = [];
|
|
252
|
+
for (const processor of this.appConfig.processors.values()) {
|
|
253
|
+
const watchConfig = processor.getWatchConfig();
|
|
254
|
+
if (!watchConfig) continue;
|
|
255
|
+
processorPaths.push(...watchConfig.paths);
|
|
256
|
+
}
|
|
257
|
+
|
|
258
|
+
if (fileSystem.exists(this.appConfig.absolutePaths.pagesDir)) {
|
|
259
|
+
processorPaths.push(this.appConfig.absolutePaths.pagesDir);
|
|
260
|
+
}
|
|
261
|
+
|
|
262
|
+
if (fileSystem.exists(this.appConfig.absolutePaths.publicDir)) {
|
|
263
|
+
processorPaths.push(this.appConfig.absolutePaths.publicDir);
|
|
264
|
+
}
|
|
265
|
+
|
|
266
|
+
if (this.appConfig.additionalWatchPaths.length) {
|
|
267
|
+
processorPaths.push(...this.appConfig.additionalWatchPaths);
|
|
268
|
+
}
|
|
269
|
+
|
|
270
|
+
this.watcher = chokidar.watch(processorPaths, {
|
|
271
|
+
ignoreInitial: true,
|
|
272
|
+
ignorePermissionErrors: true,
|
|
273
|
+
awaitWriteFinish: {
|
|
274
|
+
stabilityThreshold: 50,
|
|
275
|
+
pollInterval: 50,
|
|
276
|
+
},
|
|
277
|
+
});
|
|
278
|
+
}
|
|
279
|
+
|
|
280
|
+
for (const processor of this.appConfig.processors.values()) {
|
|
281
|
+
const watchConfig = processor.getWatchConfig();
|
|
282
|
+
if (!watchConfig) continue;
|
|
283
|
+
const { extensions = [], onCreate, onChange, onDelete, onError } = watchConfig;
|
|
284
|
+
|
|
285
|
+
if (onCreate) this.watcher.on('add', (path) => this.shouldProcess(path, extensions, onCreate));
|
|
286
|
+
if (onChange) this.watcher.on('change', (path) => this.shouldProcess(path, extensions, onChange));
|
|
287
|
+
if (onDelete) this.watcher.on('unlink', (path) => this.shouldProcess(path, extensions, onDelete));
|
|
288
|
+
if (onError) this.watcher.on('error', onError as (error: unknown) => void);
|
|
289
|
+
}
|
|
290
|
+
|
|
291
|
+
this.watcher.add(this.appConfig.absolutePaths.srcDir);
|
|
292
|
+
|
|
293
|
+
this.watcher
|
|
294
|
+
.on('change', (path) => this.handleFileChange(path))
|
|
295
|
+
.on('add', (path) => {
|
|
296
|
+
this.handleFileChange(path);
|
|
297
|
+
this.triggerRouterRefresh(path);
|
|
298
|
+
})
|
|
299
|
+
.on('addDir', (path) => this.triggerRouterRefresh(path))
|
|
300
|
+
.on('unlink', (path) => this.triggerRouterRefresh(path))
|
|
301
|
+
.on('unlinkDir', (path) => this.triggerRouterRefresh(path))
|
|
302
|
+
.on('error', (error) => this.handleError(error));
|
|
303
|
+
|
|
304
|
+
return this.watcher;
|
|
305
|
+
}
|
|
306
|
+
}
|