loopback-datasource-juggler/test/async-observer.test.js

336 lines
9.7 KiB
JavaScript

// Copyright IBM Corp. 2015,2018. All Rights Reserved.
// Node module: loopback-datasource-juggler
// This file is licensed under the MIT License.
// License text available at https://opensource.org/licenses/MIT
'use strict';
const ModelBuilder = require('../').ModelBuilder;
const should = require('./init');
describe('async observer', function() {
let TestModel;
beforeEach(function defineTestModel() {
const modelBuilder = new ModelBuilder();
TestModel = modelBuilder.define('TestModel', {name: String});
});
it('calls registered async observers', function(done) {
const notifications = [];
TestModel.observe('before', pushAndNext(notifications, 'before'));
TestModel.observe('after', pushAndNext(notifications, 'after'));
TestModel.notifyObserversOf('before', {}, function(err) {
if (err) return done(err);
notifications.push('call');
TestModel.notifyObserversOf('after', {}, function(err) {
if (err) return done(err);
notifications.should.eql(['before', 'call', 'after']);
done();
});
});
});
it('allows multiple observers for the same operation', function(done) {
const notifications = [];
TestModel.observe('event', pushAndNext(notifications, 'one'));
TestModel.observe('event', pushAndNext(notifications, 'two'));
TestModel.notifyObserversOf('event', {}, function(err) {
if (err) return done(err);
notifications.should.eql(['one', 'two']);
done();
});
});
it('allows multiple operations to be notified in one call', function(done) {
const notifications = [];
TestModel.observe('event1', pushAndNext(notifications, 'one'));
TestModel.observe('event2', pushAndNext(notifications, 'two'));
TestModel.notifyObserversOf(['event1', 'event2'], {}, function(err) {
if (err) return done(err);
notifications.should.eql(['one', 'two']);
done();
});
});
it('inherits observers from base model', function(done) {
const notifications = [];
TestModel.observe('event', pushAndNext(notifications, 'base'));
const Child = TestModel.extend('Child');
Child.observe('event', pushAndNext(notifications, 'child'));
Child.notifyObserversOf('event', {}, function(err) {
if (err) return done(err);
notifications.should.eql(['base', 'child']);
done();
});
});
it('allow multiple operations to be notified with base models', function(done) {
const notifications = [];
TestModel.observe('event1', pushAndNext(notifications, 'base1'));
TestModel.observe('event2', pushAndNext(notifications, 'base2'));
const Child = TestModel.extend('Child');
Child.observe('event1', pushAndNext(notifications, 'child1'));
Child.observe('event2', pushAndNext(notifications, 'child2'));
Child.notifyObserversOf(['event1', 'event2'], {}, function(err) {
if (err) return done(err);
notifications.should.eql(['base1', 'child1', 'base2', 'child2']);
done();
});
});
it('does not modify observers in the base model', function(done) {
const notifications = [];
TestModel.observe('event', pushAndNext(notifications, 'base'));
const Child = TestModel.extend('Child');
Child.observe('event', pushAndNext(notifications, 'child'));
TestModel.notifyObserversOf('event', {}, function(err) {
if (err) return done(err);
notifications.should.eql(['base']);
done();
});
});
it('always calls inherited observers', function(done) {
const notifications = [];
TestModel.observe('event', pushAndNext(notifications, 'base'));
const Child = TestModel.extend('Child');
// Important: there are no observers on the Child model
Child.notifyObserversOf('event', {}, function(err) {
if (err) return done(err);
notifications.should.eql(['base']);
done();
});
});
it('can remove observers', function(done) {
const notifications = [];
function call(ctx, next) {
notifications.push('call');
process.nextTick(next);
}
TestModel.observe('event', call);
TestModel.removeObserver('event', call);
TestModel.notifyObserversOf('event', {}, function(err) {
if (err) return done(err);
notifications.should.eql([]);
done();
});
});
it('can clear all observers', function(done) {
const notifications = [];
function call(ctx, next) {
notifications.push('call');
process.nextTick(next);
}
TestModel.observe('event', call);
TestModel.observe('event', call);
TestModel.observe('event', call);
TestModel.clearObservers('event');
TestModel.notifyObserversOf('event', {}, function(err) {
if (err) return done(err);
notifications.should.eql([]);
done();
});
});
it('handles no observers', function(done) {
TestModel.notifyObserversOf('no-observers', {}, function(err) {
// the test passes when no error was raised
done(err);
});
});
it('passes context to final callback', function(done) {
const context = {};
TestModel.notifyObserversOf('event', context, function(err, ctx) {
(ctx || 'null').should.equal(context);
done();
});
});
describe('notifyObserversAround', function() {
let notifications;
beforeEach(function() {
notifications = [];
TestModel.observe('before execute',
pushAndNext(notifications, 'before execute'));
TestModel.observe('after execute',
pushAndNext(notifications, 'after execute'));
});
it('should notify before/after observers', function(done) {
const context = {};
function work(done) {
process.nextTick(function() {
done(null, 1);
});
}
TestModel.notifyObserversAround('execute', context, work,
function(err, result) {
notifications.should.eql(['before execute', 'after execute']);
result.should.eql(1);
done();
});
});
it('should allow work with context', function(done) {
const context = {};
function work(context, done) {
process.nextTick(function() {
done(null, 1);
});
}
TestModel.notifyObserversAround('execute', context, work,
function(err, result) {
notifications.should.eql(['before execute', 'after execute']);
result.should.eql(1);
done();
});
});
it('should notify before/after observers with multiple results',
function(done) {
const context = {};
function work(done) {
process.nextTick(function() {
done(null, 1, 2);
});
}
TestModel.notifyObserversAround('execute', context, work,
function(err, r1, r2) {
r1.should.eql(1);
r2.should.eql(2);
notifications.should.eql(['before execute', 'after execute']);
done();
});
});
it('should allow observers to skip other ones',
function(done) {
TestModel.observe('before invoke',
function(context, next) {
notifications.push('before invoke');
context.end(null, 0);
});
TestModel.observe('after invoke',
pushAndNext(notifications, 'after invoke'));
const context = {};
function work(done) {
process.nextTick(function() {
done(null, 1, 2);
});
}
TestModel.notifyObserversAround('invoke', context, work,
function(err, r1) {
r1.should.eql(0);
notifications.should.eql(['before invoke']);
done();
});
});
it('should allow observers to tweak results',
function(done) {
TestModel.observe('after invoke',
function(context, next) {
notifications.push('after invoke');
context.results = [3];
next();
});
const context = {};
function work(done) {
process.nextTick(function() {
done(null, 1, 2);
});
}
TestModel.notifyObserversAround('invoke', context, work,
function(err, r1) {
r1.should.eql(3);
notifications.should.eql(['after invoke']);
done();
});
});
});
it('resolves promises returned by observers', function(done) {
TestModel.observe('event', function(ctx) {
return Promise.resolve('value-to-ignore');
});
TestModel.notifyObserversOf('event', {}, function(err, ctx) {
// the test times out when the promises are not supported
done();
});
});
it('handles rejected promise returned by an observer', function(done) {
const testError = new Error('expected test error');
TestModel.observe('event', function(ctx) {
return Promise.reject(testError);
});
TestModel.notifyObserversOf('event', {}, function(err, ctx) {
err.should.eql(testError);
done();
});
});
it('returns a promise when no callback is provided', function() {
const context = {value: 'a-test-context'};
const p = TestModel.notifyObserversOf('event', context);
(p !== undefined).should.be.true;
return p.then(function(result) {
result.should.eql(context);
});
});
it('returns a rejected promise when no callback is provided', function() {
const testError = new Error('expected test error');
TestModel.observe('event', function(ctx, next) { next(testError); });
const p = TestModel.notifyObserversOf('event', context);
return p.then(
function(result) {
throw new Error('The promise should have been rejected.');
},
function(err) {
err.should.eql(testError);
}
);
});
});
function pushAndNext(array, value) {
return function(ctx, next) {
array.push(value);
process.nextTick(next);
};
}