-
-
Notifications
You must be signed in to change notification settings - Fork 6.6k
/
Copy pathserverModuleRunner.ts
138 lines (130 loc) · 3.51 KB
/
serverModuleRunner.ts
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
import { existsSync, readFileSync } from 'node:fs'
import { ESModulesEvaluator, ModuleRunner } from 'vite/module-runner'
import type {
ModuleEvaluator,
ModuleRunnerHmr,
ModuleRunnerOptions,
} from 'vite/module-runner'
import type { HotPayload } from 'types/hmrPayload'
import type { DevEnvironment } from '../../server/environment'
import type {
HotChannelClient,
NormalizedServerHotChannel,
} from '../../server/hmr'
import type { ModuleRunnerTransport } from '../../../shared/moduleRunnerTransport'
/**
* @experimental
*/
export interface ServerModuleRunnerOptions
extends Omit<
ModuleRunnerOptions,
'root' | 'fetchModule' | 'hmr' | 'transport'
> {
/**
* Disable HMR or configure HMR logger.
*/
hmr?:
| false
| {
logger?: ModuleRunnerHmr['logger']
}
/**
* Provide a custom module evaluator. This controls how the code is executed.
*/
evaluator?: ModuleEvaluator
}
function createHMROptions(
environment: DevEnvironment,
options: ServerModuleRunnerOptions,
) {
if (environment.config.server.hmr === false || options.hmr === false) {
return false
}
if (!('api' in environment.hot)) return false
return {
logger: options.hmr?.logger,
}
}
const prepareStackTrace = {
retrieveFile(id: string) {
if (existsSync(id)) {
return readFileSync(id, 'utf-8')
}
},
}
function resolveSourceMapOptions(options: ServerModuleRunnerOptions) {
if (options.sourcemapInterceptor != null) {
if (options.sourcemapInterceptor === 'prepareStackTrace') {
return prepareStackTrace
}
if (typeof options.sourcemapInterceptor === 'object') {
return { ...prepareStackTrace, ...options.sourcemapInterceptor }
}
return options.sourcemapInterceptor
}
if (typeof process !== 'undefined' && 'setSourceMapsEnabled' in process) {
return 'node'
}
return prepareStackTrace
}
export const createServerModuleRunnerTransport = (options: {
channel: NormalizedServerHotChannel
}): ModuleRunnerTransport => {
const hmrClient: HotChannelClient = {
send: (payload: HotPayload) => {
if (payload.type !== 'custom') {
throw new Error(
'Cannot send non-custom events from the client to the server.',
)
}
options.channel.send(payload)
},
}
let handler: ((data: HotPayload) => void) | undefined
return {
connect({ onMessage }) {
options.channel.api!.outsideEmitter.on('send', onMessage)
onMessage({ type: 'connected' })
handler = onMessage
},
disconnect() {
if (handler) {
options.channel.api!.outsideEmitter.off('send', handler)
}
},
send(payload) {
if (payload.type !== 'custom') {
throw new Error(
'Cannot send non-custom events from the server to the client.',
)
}
options.channel.api!.innerEmitter.emit(
payload.event,
payload.data,
hmrClient,
)
},
}
}
/**
* Create an instance of the Vite SSR runtime that support HMR.
* @experimental
*/
export function createServerModuleRunner(
environment: DevEnvironment,
options: ServerModuleRunnerOptions = {},
): ModuleRunner {
const hmr = createHMROptions(environment, options)
return new ModuleRunner(
{
...options,
root: environment.config.root,
transport: createServerModuleRunnerTransport({
channel: environment.hot as NormalizedServerHotChannel,
}),
hmr,
sourcemapInterceptor: resolveSourceMapOptions(options),
},
options.evaluator || new ESModulesEvaluator(),
)
}