如何配置Typescript,Karma,Webpack在浏览器中运行测试

时间:2019-03-17 16:34:55

标签: typescript webpack karma-runner

我正在使用Phaser3开发一款游戏,并决定要为游戏的某些组件添加一些测试。

我使用Typescript 3.3和webpack 4作为主要工具。我想在浏览器上运行测试,其中某些测试可能取决于Phaser游戏库。过去,我曾使用Karma + Jasmine测试Angular.js应用程序,所以我认为我会这样做。

完成一些设置后,我运行了测试,但是由于某种原因,测试文件无法导入其他模块,从而导致错误

Compiled with warnings.
ℹ 「wdm」: Compiling...
✖ 「wdm」: 
ERROR in ./src/test/core/EntityFactory.spec.ts
Module not found: Error: Can't resolve './foo' in '<path to the project>/src/test/core'
 @ ./src/test/core/EntityFactory.spec.ts 1:0-26
ℹ 「wdm」: Failed to compile.

Module not found: Error: Can't resolve './foo' in '/<path to the project>/src/test/core'

提到的/test/core目录包含文件EntityFactory.spec.tsfoo.ts。导入在套件中写为:

import thing from './foo';

以下是我的业力配置文件及其使用的webpack配置:

// Karma configuration
// Generated on Sun Mar 17 2019 17:03:14 GMT+0200 (EET)
const webpackConfig = require('./webpack.config').test;

module.exports = function(config) {
  config.set({

    // base path that will be used to resolve all patterns (eg. files, exclude)
    basePath: '',


    // frameworks to use
    // available frameworks: https://npmjs.org/browse/keyword/karma-adapter
    frameworks: ['jasmine'],


    // list of files / patterns to load in the browser
    files: [
      '**/*.spec.ts'
      // each file acts as entry point for the webpack configuration
    ],

    mime: {
      "text/x-typescript": ["ts", "tsx"],
    },


    // list of files / patterns to exclude
    exclude: [
      // '**/*.spec.ts'
    ],


    // preprocess matching files before serving them to the browser
    // available preprocessors: https://npmjs.org/browse/keyword/karma-preprocessor
    preprocessors: {
       // add webpack as preprocessor
      '**/*.spec.ts': [ 'webpack' ],

    },

    webpack: webpackConfig,

    webpackMiddleware: {
      // webpack-dev-middleware configuration
      // i. e.
      stats: 'errors-only'
    },


    // test results reporter to use
    // possible values: 'dots', 'progress'
    // available reporters: https://npmjs.org/browse/keyword/karma-reporter
    reporters: ['progress'],


    // web server port
    port: 9876,


    // enable / disable colors in the output (reporters and logs)
    colors: true,


    // level of logging
    // possible values: config.LOG_DISABLE || config.LOG_ERROR || config.LOG_WARN || config.LOG_INFO || config.LOG_DEBUG
    logLevel: config.LOG_INFO,


    // enable / disable watching file and executing tests whenever any file changes
    autoWatch: false,


    // start these browsers
    // available browser launchers: https://npmjs.org/browse/keyword/karma-launcher
    browsers: ['Chrome'],


    // Continuous Integration mode
    // if true, Karma captures browsers, runs the tests and exits
    singleRun: true,

    // Concurrency level
    // how many browser should be started simultaneous
    concurrency: Infinity
  })
}

和Webpack配置:

const baseConfig = {
  entry: './src/index.ts',
  mode: 'development',
  output: {
    path: path.resolve(__dirname, 'build'),
    publicPath: '/',
    chunkFilename: '[name].js',
    filename: '[name].js'
  },

  resolve: {
    extensions: ['.ts', '.tsx', '.js']
  },

  // Source maps support ('inline-source-map' also works)
  devtool: 'source-map',

  module: {
    rules: [
      {
        test: /\.tsx?$/,
        loader: 'ts-loader'
      },
      {
        test: /\.js$/,
        use: ['source-map-loader'],
        enforce: 'pre'
      }
    ]
  },

  optimization: {
    runtimeChunk: {
      name: 'manifest'
    },
    splitChunks: {
      cacheGroups: {
        commons: {
          test: /[\\/]node_modules[\\/]/,
          name: 'vendor',
          chunks: 'initial'
        }
      }
    }
  },

  plugins: [
    new CopyWebpackPlugin([
      {
        from: path.resolve(__dirname, 'assets'),
        to: path.resolve(__dirname, 'build', 'assets')
      }
    ]),
    new webpack.DefinePlugin({
      CANVAS_RENDERER: JSON.stringify(true),
      WEBGL_RENDERER: JSON.stringify(true)
    }),
    new MiniHtmlWebpackPlugin({
      context: {
        title: 'Phaser game'
      },
      // FIXME: CSS-loader and default minithmlwebpack template would probably handle this better
      template: ({js, title, publicPath } ) =>
        `
        <!DOCTYPE html>
          <html>
            <head>
              <meta charset="UTF-8">
              <title>${title}</title>
              <style type = "text/css">
                body {
                    padding: 0;
                    margin: 0;
                  }
                  canvas {
                    display:block;
                    margin: 0;
                    position: absolute;
                    top: 50%;
                    left: 50%;
                    transform: translate(-50%, -50%);
                  }
              </style>
            </head>
            <body>
             ${generateJSReferences(js, publicPath)}
            </body>
          </html>
        `
    })
  ]
};

function getTestConfig() {
  const config = _.cloneDeep(baseConfig);
  delete config.entry;
  return config;
}

getTestConfig()函数会稍微更改baseConfig并将结果传递给Karma配置。

最后,如果有某些东西可能会有所帮助,那么开发人员的依赖关系:

  "devDependencies": {
    "@types/jasmine": "^3.3.10",
    "browser-sync": "^2.26.3",
    "browser-sync-webpack-plugin": "^2.2.2",
    "copy-webpack-plugin": "^4.6.0",
    "jasmine": "^3.3.1",
    "karma": "^4.0.1",
    "karma-chrome-launcher": "^2.2.0",
    "karma-jasmine": "^2.0.1",
    "karma-webpack": "^3.0.5",
    "mini-html-webpack-plugin": "^0.2.3",
    "prettier": "^1.16.4",
    "source-map-loader": "^0.2.4",
    "ts-loader": "^5.3.3",
    "tslint": "^5.12.1",
    "tslint-config-prettier": "^1.18.0",
    "typescript": "^3.3",
    "webpack": "^4.29.3",
    "webpack-cli": "^3.2.3",
    "webpack-dev-server": "^3.1.14"
  },

所以我的问题是,为什么测试似乎无法导入我要测试的模块? Webpack,Karma或Typescript中存在问题吗?

1 个答案:

答案 0 :(得分:0)

这听起来像Typescript没有编译您要包括的模块(foo)。整理一个可以重现该错误的小型测试用例会有所帮助,但是您应该检查tsconfig.json以确保它包含您希望能够测试的所有代码。

我有一个单独的tsconfig.test.json,涵盖了应用程序的主要入口点(files: ["src/index.ts"])和所有测试(include: ["src/**/*.spec.ts"])。如果您采用这种方法,则可以使用通用选项制作一个tsconfig.base.json,然后将其包括在您的产品tsconfig.json和测试版中。然后,您需要将Webpack配置指向正确的文件,例如

{
  test: /\.tsx?$/,
  use: {
    loader: 'ts-loader',
    options: {
      configFile: 'tsconfig.test.json',
    }
  }
},

您可以将configFile默认设置为tsconfig.json,然后您的getTestConfig()函数可以对其进行更改,以供karma-webpack使用。