如何使用 TypeScript 在 Jest 的 globalSetup 函数中使用模块导入 (@)?

hyp*_*ipe 6 typescript jestjs ts-jest

使用 JS 挂钩调用 TS setupGlobal 函数(例如在模块导入中使用@app/blablabla.ts),我收到错误Cannot find module '@config/config'。我已经描述过moduleNameMapper并且它在测试中工作,但似乎不起作用setupGlobal。我该如何解决这个问题?

正如我所写,moduleNameMapper已描述并且 Jest 可以理解测试中的那些(在模块中)导入。

笑话配置.js

module.exports = {
  roots: ['<rootDir>/src'],
  transform: {
    '^.+\\.tsx?$': 'ts-jest',
  },
  testRegex: '(/__tests__/.*|(\\.|/)(test|spec))\\.tsx?$',
  moduleFileExtensions: ['ts', 'tsx', 'js', 'jsx', 'json', 'node'],
  moduleDirectories: ['node_modules', 'src'],
  moduleNameMapper: {
    '@tests-suite/(.*)': '<rootDir>/src/tests/tests-suite/$1',
    '@config/(.*)': '<rootDir>/config/$1',
    '@tests/(.*)': '<rootDir>/src/tests/$1',
    '@src/(.*)': '<rootDir>/src/$1',
  },
  globalSetup: "<rootDir>/src/tests/unit/jestGlobalSetup.js"
}

Run Code Online (Sandbox Code Playgroud)

jestGlobalSetup.js

require("ts-node/register");
module.exports = require('./setupTestEnvironment').default;
Run Code Online (Sandbox Code Playgroud)

setupTestEnvironment.ts

module.exports = {
  roots: ['<rootDir>/src'],
  transform: {
    '^.+\\.tsx?$': 'ts-jest',
  },
  testRegex: '(/__tests__/.*|(\\.|/)(test|spec))\\.tsx?$',
  moduleFileExtensions: ['ts', 'tsx', 'js', 'jsx', 'json', 'node'],
  moduleDirectories: ['node_modules', 'src'],
  moduleNameMapper: {
    '@tests-suite/(.*)': '<rootDir>/src/tests/tests-suite/$1',
    '@config/(.*)': '<rootDir>/config/$1',
    '@tests/(.*)': '<rootDir>/src/tests/$1',
    '@src/(.*)': '<rootDir>/src/$1',
  },
  globalSetup: "<rootDir>/src/tests/unit/jestGlobalSetup.js"
}

Run Code Online (Sandbox Code Playgroud)

我期望“at module”导入将在setupGlobal函数中执行,但它会引发错误。

hyp*_*ipe 4

好的。问题是我们使用的是绝对路径,通过安装ts-jest和添加下一个解决了jest.config.js

    module.exports = {
  roots: [
    '<rootDir>/src',
  ],
  testRegex: '(/__tests__/.*|(\\.|/)(test|spec))\\.tsx?$',
  moduleFileExtensions: [
    'js',
    'json',
    'jsx',
    'node',
    'ts',
    'tsx',
  ],
  moduleDirectories: [
    'node_modules',
    'src',
  ],
  moduleNameMapper: {
    '^@tests\\-suite/(.*)$': '<rootDir>/src/tests/tests-suite/$1',
    '^@admin/(.*)$': '<rootDir>/src/admin/$1',
    '^@common/(.*)$': '<rootDir>/src/common/$1',
    '^@app/(.*)$': '<rootDir>/src/app/$1',
    '^@partners/(.*)$': '<rootDir>/src/partners/$1',
    '^@config/(.*)$': '<rootDir>/config/$1',
    '^@tests/(.*)$': '<rootDir>/src/tests/$1',
    '^@src/(.*)$': '<rootDir>/src/$1',
  },
  setupFilesAfterEnv: [`<rootDir>/src/tests/setup/GlobalSetup.ts`],
  preset: 'ts-jest',
  testMatch: null,
};
Run Code Online (Sandbox Code Playgroud)

我们的运行脚本如下所示"test:jest": "NODE_ENV=test yarn ts-node -r tsconfig-paths/register ./src/tests/setup/testInit.ts"

testInit 看起来像这样;

import { runCLI } from 'jest'

// globalSetup
async function init() {
  console.log('Initialization')

  await runCLI({ config: './jest.config.js', watch: true } as any, [__dirname])

  // Do all your initialization stuff
  // I use a setTimeout to simulate true async
  return new Promise<void>((resolve, _reject) => {
    setTimeout(() => {
      console.log('Init finished')
      resolve()
    }, 1000)
  })
}

// globalTeardown
async function afterTests(): Promise<void> {
  console.log('End of tests - Execute something')
}

init()
  .then(() => {
    afterTests()
  })
  // tslint:disable-next-line:no-console
  .catch(e => console.error(e))
Run Code Online (Sandbox Code Playgroud)

GlobalSetup.ts 看起来像这样:

import { DBConnectionManager } from '@src/DBConnectionManager'

beforeAll(async () => {
  await DBConnectionManager
  .awaitConnection()
})
Run Code Online (Sandbox Code Playgroud)

所以它DBConnectionManager只是 TypeORM 的包装,它为我们提供了一个连接。在每次测试之前连接到真正的测试数据库。目前有效。

  • 我想说这不是正确的解决方案。我遇到了同样的情况,当我使用 globalSetup 时找不到别名,但它与“setupFilesAfterEnv”一起工作正常,这与“globalSetup”完全不同。setupFilesAfterEnv 在每个测试之前运行,而 globalSetup 在所有测试之前仅运行一次。你成功解决这个问题了吗? (3认同)
  • 或多或少......解决方案是不幸的是,根据对此问题的评论,这是不可能的:https://github.com/facebook/jest/issues/6048 (2认同)