使用jQuery和jQuery UI运行任何Jest测试的问题

时间:2019-03-12 18:16:54

标签: jquery angular jestjs slickgrid angular-test

因此,我有一个名为Angular-Slickgrid的开放源代码库,该库尚未进行测试,并且我正尝试将其与Jest一起使用,但是使用它很难。该库是旧jQuery数据网格库(SlickGrid)的包装,该库也使用jQuery UI。我认为我部分解决了jQuery问题(甚至不确定),但是jQuery UI仍然抱怨。还请注意,我是Angular的Jest和Unit Testing的新手,但我真的希望它能正常工作并使我的lib安全。

您可以在GitHub上看到我尝试使用我的开放源代码库实现Jest所做的所有代码更改的提交。提交为here。如果更容易,请随意创建PR。我使用的是Jest(23.6.0)的先前版本,而不是最新版本,因为我对最新版本还有其他疑问。

这是我目前遇到的错误

FAIL  src/app/modules/angular-slickgrid/components/angular-slickgrid.component.spec.ts
Test suite failed to run 
TypeError: Cannot read property 'ui' of undefined
  at node_modules/jquery-ui-dist/jquery-ui.js:18:10
  at Object.<anonymous>.$.ui (node_modules/jquery-ui-dist/jquery-ui.js:14:3)
  at Object.<anonymous> (node_modules/jquery-ui-dist/jquery-ui.js:16:2)
  at Object.<anonymous> (src/app/modules/angular-slickgrid/components/angular-slickgrid.component.ts:11193:1)
  at Object.<anonymous> (src/app/modules/angular-slickgrid/components/angular-slickgrid.component.spec.ts:7:37)

我尝试使用unmock('jquery')unmock('jquery-ui'),但这似乎无济于事。这是测试失败的

jest.unmock('jquery');
jest.unmock('jquery-ui');
import { TestBed, async } from '@angular/core/testing';
import { RouterTestingModule } from '@angular/router/testing';

import { AngularSlickgridComponent } from './angular-slickgrid.component';

describe('AppComponent', () => {
  beforeEach(async(() => {
    TestBed.configureTestingModule({
      declarations: [
        AngularSlickgridComponent
      ],
      providers: [],
      imports: [RouterTestingModule]
    }).compileComponents();
  }));

  it('should create the app', async(() => {
    const fixture = TestBed.createComponent(AngularSlickgridComponent);
    const app = fixture.debugElement.componentInstance;
    expect(app).toBeTruthy();
  }));
});

还有我的jest.config.js

module.exports = {
  globals: {
    'ts-jest': {
      tsConfigFile: './src/tsconfig.spec.json',
    },
    __TRANSFORM_HTML__: true,
  },
  testMatch: ['**/__tests__/**/*.+(ts|js)', '**/+(*.)+(spec|test).+(ts|js)'],
  setupFiles: ['<rootDir>/test-env.ts'],
  setupTestFrameworkScriptFile: '<rootDir>/node_modules/@angular-builders/jest/src/jest-config/setup.js',
  transform: {
    '^.+\\.(ts|html)$': '<rootDir>/node_modules/jest-preset-angular/preprocessor.js',
  },
  transformIgnorePatterns: ['node_modules/(?!@ngrx)'],
  moduleDirectories: [
    "node_modules",
    "src/app",
  ],
  collectCoverage: true,
  moduleFileExtensions: [
    'ts',
    'json',
    'js'
  ],
  testResultsProcessor: 'jest-sonar-reporter',
  moduleNameMapper: {
    "app/(.*)": "<rootDir>/src/app/$1",
    "@common/(.*)": "<rootDir>/src/app/common/$1",
  }
};

最后是一个测试设置,用于为Jest全局导入jQuery

import jQuery from 'jquery';
declare var window: any;
declare var global: any;
window.$ = window.jQuery = jQuery;
global.$ = global.jQuery = jQuery;

我希望完成的工作是至少测试我的Angular Services和Component的创建,这将是一个很好的开始,但是即使我不想测试任何jQuery和jQuery UI问题,我也无法通过的核心库(SlickGrid),既不是jQuery,也不是jQuery UI。

编辑

感谢@ brian-lives-outdoors在jQueryjQuery-UI上的回答,我走得更远。现在,我有一个小问题,直接在@Inject()中使用了Constructor(即将配置传递到我的组件库),如果有人知道如何解决,我不确定该如何解决请帮忙。

constructor(
  private elm: ElementRef,
  // ... more Services import
  //
  @Inject('config') private forRootConfig: GridOption
) {}

,错误是

StaticInjectorError(DynamicTestModule)[config]:
  StaticInjectorError(Platform: core)[config]:
    NullInjectorError: No provider for config!

at NullInjector.get (../packages/core/src/di/injector.ts:43:13)
at resolveToken (../packages/core/src/di/injector.ts:346:20)
...

上一个编辑问题的答案

我找到了如何修复Constructor@Inject()的方法,如下所示,我可以在overrideComponent()内用beforeEach进行修复

beforeEach(async(() => {
  TestBed.configureTestingModule({
    declarations: [
      AngularSlickgridComponent,
      SlickPaginationComponent
    ],
    providers: [
      // ... all Services
    ],
    imports: [
      RouterTestingModule,
      TranslateModule.forRoot()
    ]
  })
  // THIS LINE is for the @Inject('config')
  .overrideComponent(AngularSlickgridComponent, {
    set: { providers: [{ provide: 'config', useValue: {} }] },
  })
  .compileComponents();
}));

最后我现在可以说我在开玩笑!!!

2 个答案:

答案 0 :(得分:3)

问题在于jQuery的导入方式如下:

import jQuery from 'jquery';

...无法正常工作,导致jQueryundefined

由于jQuery被导入为undefined,因此全局$设置为undefined,并且当jQuery UI尝试加载时会引发错误。


这个问题很奇怪,因为即使在官方TypeScript文档中以jQuery语法为例,也显示了导入import的语法。


无论如何,您可以通过使用以下语法导入jQuery来解决此问题:

import * as jQuery from 'jquery';


将您的test-env.ts更改为此:

import * as jQuery from 'jquery';
declare var window: any;
declare var global: any;
window.$ = window.jQuery = jQuery;
global.$ = global.jQuery = jQuery;

...将angular-slickgrid.components.ts的顶部更改为此:

// import 3rd party vendor libs
// only import the necessary core lib, each will be imported on demand when enabled (via require)
import 'jquery-ui-dist/jquery-ui';
import 'slickgrid/lib/jquery.event.drag-2.3.0';
import 'slickgrid/slick.core';
import 'slickgrid/slick.grid';
import 'slickgrid/slick.dataview';

// ...then everything else...
import { AfterViewInit, Component, ElementRef, EventEmitter, Inject, Injectable, Input, Output, OnDestroy, OnInit } from '@angular/core';
import { TranslateService } from '@ngx-translate/core';
import { GlobalGridOptions } from './../global-grid-options';
// ...

...并将您的angular-slickgrid.component.spec.ts更改为此:

import { TestBed, async } from '@angular/core/testing';
import { RouterTestingModule } from '@angular/router/testing';

import { AngularSlickgridComponent } from './angular-slickgrid.component';

describe('AppComponent', () => {
  beforeEach(async(() => {
    TestBed.configureTestingModule({
      declarations: [
        AngularSlickgridComponent
      ],
      providers: [],
      imports: [RouterTestingModule]
    }).compileComponents();
  }));

  it('should create the app', async(() => {
    const fixture = TestBed.createComponent(AngularSlickgridComponent);
    const app = fixture.debugElement.componentInstance;
    expect(app).toBeTruthy();
  }));

  it(`should have as title 'Angular SlickGrid Demo'`, async(() => {
    const fixture = TestBed.createComponent(AngularSlickgridComponent);
    const app = fixture.debugElement.componentInstance;
    expect(app.title).toEqual('Angular SlickGrid Demo');
  }));
});

...这将使您克服最初的jQuery错误。

答案 1 :(得分:-1)

在您的 jest.config.js 中添加这一行:

setupFiles: ['<rootDir>/src/test-env.js']

将您的 test-env.ts 更改为:

import * as jQuery from 'jquery';
window.$ = window.jQuery = jQuery;
global.$ = global.jQuery = jQuery;

import 'jquery-ui/ui/version';
import 'jquery-ui/ui/plugin';
import 'jquery-ui/ui/widget';
import 'jquery-ui/ui/widgets/mouse';