-
Notifications
You must be signed in to change notification settings - Fork 460
/
ts-jest-transformer.ts
353 lines (316 loc) · 13.2 KB
/
ts-jest-transformer.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
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
import { existsSync, readFileSync, statSync, writeFileSync, mkdirSync } from 'fs'
import path from 'path'
import type { SyncTransformer, TransformedSource } from '@jest/transform'
import type { Logger } from 'bs-logger'
import { TsJestCompiler } from '../compiler'
import { ConfigSet } from '../config'
import { DECLARATION_TYPE_EXT, JS_JSX_REGEX, TS_TSX_REGEX } from '../constants'
import type { CompilerInstance, DepGraphInfo, ProjectConfigTsJest, TransformOptionsTsJest } from '../types'
import { parse, stringify, JsonableValue, rootLogger } from '../utils'
import { importer } from '../utils/importer'
import { Deprecations, Errors, interpolate } from '../utils/messages'
import { sha1 } from '../utils/sha1'
import { VersionCheckers } from '../utils/version-checkers'
interface CachedConfigSet {
configSet: ConfigSet
jestConfig: JsonableValue<ProjectConfigTsJest>
transformerCfgStr: string
compiler: CompilerInstance
depGraphs: Map<string, DepGraphInfo>
tsResolvedModulesCachePath: string | undefined
watchMode: boolean
}
interface TsJestHooksMap {
// eslint-disable-next-line @typescript-eslint/no-explicit-any
afterProcess?(args: any[], result: TransformedSource): TransformedSource | void
}
/**
* @internal
*/
export const CACHE_KEY_EL_SEPARATOR = '\x00'
export class TsJestTransformer implements SyncTransformer {
/**
* cache ConfigSet between test runs
*
* @internal
*/
private static readonly _cachedConfigSets: CachedConfigSet[] = []
private readonly _logger: Logger
protected _compiler!: CompilerInstance
private _tsResolvedModulesCachePath: string | undefined
private _transformCfgStr!: string
private _depGraphs: Map<string, DepGraphInfo> = new Map<string, DepGraphInfo>()
private _watchMode = false
constructor(isLegacy = false) {
this._logger = rootLogger.child({ namespace: 'ts-jest-transformer' })
VersionCheckers.jest.warn()
if (isLegacy) {
this._logger.warn(Deprecations.LegacyTransformerEntry)
}
/**
* For some unknown reasons, `this` is undefined in `getCacheKey` and `process`
* when running Jest in ESM mode
*/
this.getCacheKey = this.getCacheKey.bind(this)
this.getCacheKeyAsync = this.getCacheKeyAsync.bind(this)
this.process = this.process.bind(this)
this.processAsync = this.processAsync.bind(this)
this._logger.debug('created new transformer')
process.env.TS_JEST = '1'
}
private _configsFor(transformOptions: TransformOptionsTsJest): ConfigSet {
const { config, cacheFS } = transformOptions
const ccs: CachedConfigSet | undefined = TsJestTransformer._cachedConfigSets.find(
(cs) => cs.jestConfig.value === config,
)
let configSet: ConfigSet
if (ccs) {
this._transformCfgStr = ccs.transformerCfgStr
this._compiler = ccs.compiler
this._depGraphs = ccs.depGraphs
this._tsResolvedModulesCachePath = ccs.tsResolvedModulesCachePath
this._watchMode = ccs.watchMode
configSet = ccs.configSet
} else {
// try to look-it up by stringified version
const serializedJestCfg = stringify(config)
const serializedCcs = TsJestTransformer._cachedConfigSets.find(
(cs) => cs.jestConfig.serialized === serializedJestCfg,
)
if (serializedCcs) {
// update the object so that we can find it later
// this happens because jest first calls getCacheKey with stringified version of
// the config, and then it calls the transformer with the proper object
serializedCcs.jestConfig.value = config
this._transformCfgStr = serializedCcs.transformerCfgStr
this._compiler = serializedCcs.compiler
this._depGraphs = serializedCcs.depGraphs
this._tsResolvedModulesCachePath = serializedCcs.tsResolvedModulesCachePath
this._watchMode = serializedCcs.watchMode
configSet = serializedCcs.configSet
} else {
// create the new record in the index
this._logger.info('no matching config-set found, creating a new one')
configSet = this._createConfigSet(config)
const jest = { ...config }
// we need to remove some stuff from jest config
// this which does not depend on config
jest.cacheDirectory = undefined as any // eslint-disable-line @typescript-eslint/no-explicit-any
this._transformCfgStr = `${new JsonableValue(jest).serialized}${configSet.cacheSuffix}`
this._createCompiler(configSet, cacheFS)
this._getFsCachedResolvedModules(configSet)
this._watchMode = process.argv.includes('--watch')
TsJestTransformer._cachedConfigSets.push({
jestConfig: new JsonableValue(config),
configSet,
transformerCfgStr: this._transformCfgStr,
compiler: this._compiler,
depGraphs: this._depGraphs,
tsResolvedModulesCachePath: this._tsResolvedModulesCachePath,
watchMode: this._watchMode,
})
}
}
return configSet
}
// eslint-disable-next-line class-methods-use-this
protected _createConfigSet(config: ProjectConfigTsJest | undefined): ConfigSet {
return new ConfigSet(config)
}
protected _createCompiler(configSet: ConfigSet, cacheFS: Map<string, string>): void {
this._compiler = new TsJestCompiler(configSet, cacheFS)
}
/**
* @public
*/
process(sourceText: string, sourcePath: string, transformOptions: TransformOptionsTsJest): TransformedSource {
this._logger.debug({ fileName: sourcePath, transformOptions }, 'processing', sourcePath)
const configs = this._configsFor(transformOptions)
const shouldStringifyContent = configs.shouldStringifyContent(sourcePath)
const babelJest = shouldStringifyContent ? undefined : configs.babelJestTransformer
let result = this.processWithTs(sourceText, sourcePath, transformOptions)
if (babelJest) {
this._logger.debug({ fileName: sourcePath }, 'calling babel-jest processor')
// do not instrument here, jest will do it anyway afterwards
result = babelJest.process(result.code, sourcePath, {
...transformOptions,
instrument: false,
})
}
result = this.runTsJestHook(sourcePath, sourceText, transformOptions, result)
return result
}
async processAsync(
sourceText: string,
sourcePath: string,
transformOptions: TransformOptionsTsJest,
): Promise<TransformedSource> {
this._logger.debug({ fileName: sourcePath, transformOptions }, 'processing', sourcePath)
return new Promise(async (resolve) => {
const configs = this._configsFor(transformOptions)
const shouldStringifyContent = configs.shouldStringifyContent(sourcePath)
const babelJest = shouldStringifyContent ? undefined : configs.babelJestTransformer
let result = this.processWithTs(sourceText, sourcePath, transformOptions)
if (babelJest) {
this._logger.debug({ fileName: sourcePath }, 'calling babel-jest processor')
// do not instrument here, jest will do it anyway afterwards
result = await babelJest.processAsync(result.code, sourcePath, {
...transformOptions,
instrument: false,
})
}
result = this.runTsJestHook(sourcePath, sourceText, transformOptions, result)
resolve(result)
})
}
private processWithTs(sourceText: string, sourcePath: string, transformOptions: TransformOptionsTsJest) {
let result: TransformedSource
const configs = this._configsFor(transformOptions)
const shouldStringifyContent = configs.shouldStringifyContent(sourcePath)
const babelJest = shouldStringifyContent ? undefined : configs.babelJestTransformer
const isDefinitionFile = sourcePath.endsWith(DECLARATION_TYPE_EXT)
const isJsFile = JS_JSX_REGEX.test(sourcePath)
const isTsFile = !isDefinitionFile && TS_TSX_REGEX.test(sourcePath)
if (shouldStringifyContent) {
// handles here what we should simply stringify
result = {
code: `module.exports=${stringify(sourceText)}`,
}
} else if (isDefinitionFile) {
// do not try to compile declaration files
result = {
code: '',
}
} else if (!configs.parsedTsConfig.options.allowJs && isJsFile) {
// we've got a '.js' but the compiler option `allowJs` is not set or set to false
this._logger.warn({ fileName: sourcePath }, interpolate(Errors.GotJsFileButAllowJsFalse, { path: sourcePath }))
result = {
code: sourceText,
}
} else if (isJsFile || isTsFile) {
// transpile TS code (source maps are included)
result = this._compiler.getCompiledOutput(sourceText, sourcePath, {
depGraphs: this._depGraphs,
supportsStaticESM: transformOptions.supportsStaticESM,
watchMode: this._watchMode,
})
} else {
// we should not get called for files with other extension than js[x], ts[x] and d.ts,
// TypeScript will bail if we try to compile, and if it was to call babel, users can
// define the transform value with `babel-jest` for this extension instead
const message = babelJest ? Errors.GotUnknownFileTypeWithBabel : Errors.GotUnknownFileTypeWithoutBabel
this._logger.warn({ fileName: sourcePath }, interpolate(message, { path: sourcePath }))
result = {
code: sourceText,
}
}
return result
}
private runTsJestHook(
sourcePath: string,
sourceText: string,
transformOptions: TransformOptionsTsJest,
compiledOutput: TransformedSource,
) {
let hooksFile = process.env.TS_JEST_HOOKS
let hooks: TsJestHooksMap | undefined
/* istanbul ignore next (cover by e2e) */
if (hooksFile) {
hooksFile = path.resolve(this._configsFor(transformOptions).cwd, hooksFile)
hooks = importer.tryTheseOr(hooksFile, {})
}
// This is not supposed to be a public API but we keep it as some people use it
if (hooks?.afterProcess) {
this._logger.debug({ fileName: sourcePath, hookName: 'afterProcess' }, 'calling afterProcess hook')
const newResult = hooks.afterProcess(
[sourceText, sourcePath, transformOptions.config, transformOptions],
compiledOutput,
)
if (newResult) {
return newResult
}
}
return compiledOutput
}
/**
* Jest uses this to cache the compiled version of a file
*
* @see https://github.com/facebook/jest/blob/v23.5.0/packages/jest-runtime/src/script_transformer.js#L61-L90
*
* @public
*/
getCacheKey(fileContent: string, filePath: string, transformOptions: TransformOptionsTsJest): string {
const configs = this._configsFor(transformOptions)
this._logger.debug({ fileName: filePath, transformOptions }, 'computing cache key for', filePath)
// we do not instrument, ensure it is false all the time
const { instrument = false } = transformOptions
const constructingCacheKeyElements = [
this._transformCfgStr,
CACHE_KEY_EL_SEPARATOR,
configs.rootDir,
CACHE_KEY_EL_SEPARATOR,
`instrument:${instrument ? 'on' : 'off'}`,
CACHE_KEY_EL_SEPARATOR,
fileContent,
CACHE_KEY_EL_SEPARATOR,
filePath,
]
if (!configs.isolatedModules && this._tsResolvedModulesCachePath) {
let resolvedModuleNames: string[]
if (this._depGraphs.get(filePath)?.fileContent === fileContent) {
this._logger.debug(
{ fileName: filePath, transformOptions },
'getting resolved modules from disk caching or memory caching for',
filePath,
)
// eslint-disable-next-line @typescript-eslint/no-non-null-assertion
resolvedModuleNames = this._depGraphs
.get(filePath)!
.resolvedModuleNames.filter((moduleName) => existsSync(moduleName))
} else {
this._logger.debug(
{ fileName: filePath, transformOptions },
'getting resolved modules from TypeScript API for',
filePath,
)
resolvedModuleNames = this._compiler.getResolvedModules(fileContent, filePath, transformOptions.cacheFS)
this._depGraphs.set(filePath, {
fileContent,
resolvedModuleNames,
})
writeFileSync(this._tsResolvedModulesCachePath, stringify([...this._depGraphs]))
}
resolvedModuleNames.forEach((moduleName) => {
constructingCacheKeyElements.push(
CACHE_KEY_EL_SEPARATOR,
moduleName,
CACHE_KEY_EL_SEPARATOR,
statSync(moduleName).mtimeMs.toString(),
)
})
}
return sha1(...constructingCacheKeyElements)
}
async getCacheKeyAsync(
sourceText: string,
sourcePath: string,
transformOptions: TransformOptionsTsJest,
): Promise<string> {
return new Promise((resolve) => resolve(this.getCacheKey(sourceText, sourcePath, transformOptions)))
}
/**
* Subclasses extends `TsJestTransformer` can call this method to get resolved module disk cache
*/
private _getFsCachedResolvedModules(configSet: ConfigSet): void {
const cacheDir = configSet.tsCacheDir
if (!configSet.isolatedModules && cacheDir) {
// Make sure the cache directory exists before continuing.
mkdirSync(cacheDir, { recursive: true })
this._tsResolvedModulesCachePath = path.join(cacheDir, sha1('ts-jest-resolved-modules', CACHE_KEY_EL_SEPARATOR))
try {
const cachedTSResolvedModules = readFileSync(this._tsResolvedModulesCachePath, 'utf-8')
this._depGraphs = new Map(parse(cachedTSResolvedModules))
} catch (e) {}
}
}
}