不要怂,就是干,撸起袖子干!

instance-validator.test.js 6.05 KB
'use strict';

const chai = require('chai');
const expect = chai.expect;
const Support = require(__dirname + '/support');
const InstanceValidator = require('../../lib/instance-validator');
const sinon = require('sinon');
const Promise = Support.sequelize.Promise;
const SequelizeValidationError = require('../../lib/errors').ValidationError;

describe(Support.getTestDialectTeaser('InstanceValidator'), () => {
  beforeEach(() => {
    this.User = Support.sequelize.define('user', {
      fails: {
        type: Support.Sequelize.BOOLEAN,
        validate: {
          isNotTrue(value) {
            if (value) {
              throw Error('Manual model validation failure');
            }
          }
        }
      }
    });
  });

  it('configures itself to run hooks by default', () => {
    const instanceValidator = new InstanceValidator();
    expect(instanceValidator.options.hooks).to.equal(true);
  });

  describe('validate', () => {
    it('runs the validation sequence and hooks when the hooks option is true', () => {
      const instanceValidator = new InstanceValidator(this.User.build(), {hooks: true});
      const _validate = sinon.spy(instanceValidator, '_validate');
      const _validateAndRunHooks = sinon.spy(instanceValidator, '_validateAndRunHooks');

      instanceValidator.validate();

      expect(_validateAndRunHooks).to.have.been.calledOnce;
      expect(_validate).to.not.have.been.called;
    });

    it('runs the validation sequence but skips hooks if the hooks option is false', () => {
      const instanceValidator = new InstanceValidator(this.User.build(), {hooks: false});
      const _validate = sinon.spy(instanceValidator, '_validate');
      const _validateAndRunHooks = sinon.spy(instanceValidator, '_validateAndRunHooks');

      instanceValidator.validate();

      expect(_validate).to.have.been.calledOnce;
      expect(_validateAndRunHooks).to.not.have.been.called;
    });

    it('fulfills when validation is successful', () => {
      const instanceValidator = new InstanceValidator(this.User.build());
      const result = instanceValidator.validate();

      return expect(result).to.be.fulfilled;
    });

    it('rejects with a validation error when validation fails', () => {
      const instanceValidator = new InstanceValidator(this.User.build({ fails: true }));
      const result = instanceValidator.validate();

      return expect(result).to.be.rejectedWith(SequelizeValidationError);
    });
  });

  describe('_validateAndRunHooks', () => {
    beforeEach(() => {
      this.successfulInstanceValidator = new InstanceValidator(this.User.build());
      sinon.stub(this.successfulInstanceValidator, '_validate').returns(Promise.resolve());
    });

    it('should run beforeValidate and afterValidate hooks when _validate is successful', () => {
      const beforeValidate = sinon.spy();
      const afterValidate = sinon.spy();
      this.User.beforeValidate(beforeValidate);
      this.User.afterValidate(afterValidate);

      return expect(this.successfulInstanceValidator._validateAndRunHooks()).to.be.fulfilled.then(() => {
        expect(beforeValidate).to.have.been.calledOnce;
        expect(afterValidate).to.have.been.calledOnce;
      });
    });

    it('should run beforeValidate hook but not afterValidate hook when _validate is unsuccessful', () => {
      const failingInstanceValidator = new InstanceValidator(this.User.build());
      sinon.stub(failingInstanceValidator, '_validate', () => {
        return Promise.reject();
      });
      const beforeValidate = sinon.spy();
      const afterValidate = sinon.spy();
      this.User.beforeValidate(beforeValidate);
      this.User.afterValidate(afterValidate);

      return expect(failingInstanceValidator._validateAndRunHooks()).to.be.rejected.then(() => {
        expect(beforeValidate).to.have.been.calledOnce;
        expect(afterValidate).to.not.have.been.called;
      });
    });

    it('should emit an error from after hook when afterValidate fails', () => {
      this.User.afterValidate(() => {
        throw new Error('after validation error');
      });

      return expect(this.successfulInstanceValidator._validateAndRunHooks()).to.be.rejectedWith('after validation error');
    });

    describe('validatedFailed hook', () => {
      it('should call validationFailed hook when validation fails', () => {
        const failingInstanceValidator = new InstanceValidator(this.User.build());
        sinon.stub(failingInstanceValidator, '_validate', () => {
          return Promise.reject();
        });
        const validationFailedHook = sinon.spy();
        this.User.validationFailed(validationFailedHook);

        return expect(failingInstanceValidator._validateAndRunHooks()).to.be.rejected.then(() => {
          expect(validationFailedHook).to.have.been.calledOnce;
        });
      });

      it('should not replace the validation error in validationFailed hook by default', () => {
        const failingInstanceValidator = new InstanceValidator(this.User.build());
        sinon.stub(failingInstanceValidator, '_validate', () => {
          return Promise.reject(new SequelizeValidationError());
        });
        const validationFailedHook = sinon.stub().returns(Promise.resolve());
        this.User.validationFailed(validationFailedHook);

        return expect(failingInstanceValidator._validateAndRunHooks()).to.be.rejected.then(err => {
          expect(err.name).to.equal('SequelizeValidationError');
        });
      });

      it('should replace the validation error if validationFailed hook creates a new error', () => {
        const failingInstanceValidator = new InstanceValidator(this.User.build());
        sinon.stub(failingInstanceValidator, '_validate', () => {
          return Promise.reject(new SequelizeValidationError());
        });
        const validationFailedHook = sinon.stub().throws(new Error('validation failed hook error'));
        this.User.validationFailed(validationFailedHook);

        return expect(failingInstanceValidator._validateAndRunHooks()).to.be.rejected.then(err => {
          expect(err.message).to.equal('validation failed hook error');
        });
      });
    });
  });
});