Как провести модульное тестирование реактивного компонента, который использует наследование, а не композицию, используя шут и фермент? - PullRequest
0 голосов
/ 08 мая 2018

У меня есть компонент, который расширяет общий компонент (компонент пользовательской реакции). Как мне выполнить юнит-тест с использованием ферментного крепления? Вот код:

Container.js

import React from 'react';
import request from 'API/request';

export default class Container extends React.Component {
  componentDidMount() {
    this.populateData();
  }

  componentDidUpdate() {
    if (this.isDataStale()) {
      return this.populateData();
    }
  }

  populateData() {
    const url = this.getUrl();
    const params = this.getParams();

return request(this.props.dispatch, {
  url,
  params,
  method: 'GET'
})
  .then(response => {
    this.dispatchData(response.data);
  })
  .catch(error => {
    this.dispatchError(error);
  });


}
render() {
    if (this.getData() && !this.isDataStale()) {
      return this.renderChildren();
    } else {
      return null;
    }
  }

  getError() {}
  getParams() {}
  isDataStale() {
    return false;
  }
}

Вот функциональный компонент:

import React from 'react';
import { connect } from 'react-redux';
import { Route, withRouter } from 'react-router-dom';
import CustomPage from 'Client/customPage';
import Container from 'Client/container';
import TestPanel from 'Client/testPanel/TestPanel';
import Greeting from 'Client/greeting';
import Button from '../Button';

export class Test extends Container {
  constructor({ match }) {
    super();
    this.state = {
      match: match,
      id: match.params.id,
      location: ''
    };
  }

  isDataStale() {
    return (
      this.props.data.id !== this.props.match.params.id
    );
  }

  getData() {
    return this.props.data.values;
  }

  dispatchData(data) {
    this.props.dispatch({
      type: 'DATA_FOUND',
      data: data,
      id: this.state.id
    });
  }

  dispatchError(error) {
    this.props.dispatch({ type: 'CUSTOM_ERROR', data: error });
  }

  showDetails(url) {
    this.props.history.push(url);
  }

  renderChildren() {
    const match = this.state.match;
    const testUrl = `/test/values/${this.state.id}`;
    const verifyUrl = `${testUrl}/verify`;

    return (
      <div className="test-plan">
        <Route path={verifyUrl} render={() => (
          <Greeting show={true} />
        )} />
        <TestPanel
          data={this.props.data}
          id={this.props.match.params.id}
        />
        <Route exact path={testUrl} render={() => (
          <CustomPage id={this.state.id} />
        )} />
        <Route path={verifyUrl} render={() => (
          <div className="next-button" ><Button label = {' NEXT '} onButtonClick = { this.showDetails.bind(this, loanUrl) } /> </div>
        )} />
      </div>
    );
  }
}

const mapStateToProps = state => {
  return {
    data: state.data
  };
};

export default withRouter(connect(mapStateToProps)(Test));

Как мне протестировать этот тестовый (функциональный) компонент с помощью энзима? Я не уверен, как этот макет работает, поскольку метод render находится внутри контейнера, который, в свою очередь, вызывает метод renderChildren в нашем функциональном компоненте. Многие функциональные компоненты расширяют этот контейнер, поэтому мы не можем изменить его на шаблон композиции на данный момент.

Вот контрольный пример (не работает)

import jsdomWindow from '../../test.setup';
import React from 'react';
import {BrowserRouter} from 'react-router-dom';
import { Test } from 'Client/containers/test';
import { createStore } from 'redux';
import { Provider } from 'react-redux';
import Enzyme, {mount} from 'enzyme';
import Adapter from 'enzyme-adapter-react-16';
import sinon from 'sinon';
import * as api from 'API/request';
import { populateData, isDataStale } from '../../../../src/client/app/container';

Enzyme.configure({ adapter: new Adapter() });
const sandbox = sinon.sandbox.create();

describe('<Test />',() => {
  beforeEach(function() {
    sandbox.stub(api, 'default').returns(Promise.resolve({data: { saySomething: 'Yay Tests!'}}));
  });

  afterEach(function() {
    sandbox.restore();
  });
  const match = {
    params: {
      id: 100
    }
  };
  const testState = {
    data : {
      test: {
        id:'100',
        email:'someone@something.com',
        first_name:'John',
        home_phone:'325-555-6564',
        last_name:'Doe'
      }
    }
  };
  const wrapper = mount(
    <BrowserRouter>
      <Provider store={createStore(state => state, { data: testState.data })}>
        <Test data={testState.data} dispatch={ sinon.spy() }
          match={match} />
      </Provider>
    </BrowserRouter>
  );
  jest.mock('../../../../src/client/app/container', () => ({
    populateData: jest.fn(),
    isDataStale: jest.fn(),
    getError: jest.fn(),
    getParams: jest.fn(),

  }));

  it('should render', () => {
    console.log(wrapper.debug());
    populateData.mockReturnValueOnce({});
    isDataStale.mockReturnValueOnce(false);
    const greetingContainer = wrapper.find('.greeting-container').find('.quick-info').find('.quick-info-heading');
    expect(greetingContainer.text()).toContain('PROPER RESPONSE');
  });

  // it('should render greeting component by default', () => {
  //   expect(wrapper.find('div.greeting-container')).toBeDefined();
  //   expect(wrapper.find('div.info-container')).toBeDefined();
  //   expect(wrapper.find('div.next-button')).toBeDefined();
  // });

  // it('should not render greeting and next-button components on clicking next', () => {
  //   console.log(`Test**** ${JSON.stringify(wrapper.find('div.plan'))}`);
  // });
});

Может ли кто-нибудь помочь мне провести модульное тестирование с использованием метода монтирования энзима. Я не хочу использовать мелководье здесь, так как я должен проверить свойства вложенных компонентов.

Спасибо.

Добро пожаловать на сайт PullRequest, где вы можете задавать вопросы и получать ответы от других членов сообщества.
...