repository-observer.spec.js 19 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455
  1. import {expect} from 'chai';
  2. import {format} from 'util';
  3. import {Schema} from '../schema.js';
  4. import {RepositoryEvent} from './repository-observer.js';
  5. import {RepositoryObserver} from './repository-observer.js';
  6. const MODEL_NAME = 'model';
  7. const EVENT_NAME = RepositoryEvent.AFTER_CREATE;
  8. const METHOD_NAME = 'methodName';
  9. const HANDLER_FN = () => undefined;
  10. describe('RepositoryObserver', function () {
  11. describe('observe', function () {
  12. describe('2 arguments', function () {
  13. it('adds an event handler of the root scope', function () {
  14. const schema = new Schema();
  15. schema.defineDatasource({name: 'datasource', adapter: 'memory'});
  16. schema.defineModel({name: MODEL_NAME, datasource: 'datasource'});
  17. const observer = schema.get(RepositoryObserver);
  18. observer.observe(EVENT_NAME, HANDLER_FN);
  19. const eventsMap = observer._handlersMap.get(null);
  20. const listeners = eventsMap.get(EVENT_NAME);
  21. expect(listeners).to.have.lengthOf(1);
  22. expect(listeners).to.include(HANDLER_FN);
  23. });
  24. it('requires the "eventName" argument to be RepositoryEvent', function () {
  25. const schema = new Schema();
  26. const observer = schema.get(RepositoryObserver);
  27. const throwable = v => () => observer.observe(v, HANDLER_FN);
  28. const error = v =>
  29. format(
  30. 'The parameter "eventName" of RepositoryObserver.observe ' +
  31. 'must be a non-empty String, but %s given.',
  32. v,
  33. );
  34. expect(throwable()).to.throw(error('undefined'));
  35. expect(throwable('')).to.throw(error('""'));
  36. expect(throwable(10)).to.throw(error('10'));
  37. expect(throwable(true)).to.throw(error('true'));
  38. expect(throwable(false)).to.throw(error('false'));
  39. expect(throwable([])).to.throw(error('Array'));
  40. expect(throwable({})).to.throw(error('Object'));
  41. expect(throwable(null)).to.throw(error('null'));
  42. Object.values(RepositoryEvent).forEach(e => throwable(e)());
  43. });
  44. it('requires the "handler" argument to be a function', function () {
  45. const schema = new Schema();
  46. const observer = schema.get(RepositoryObserver);
  47. const throwable = v => () => observer.observe(EVENT_NAME, v);
  48. const error = v =>
  49. format(
  50. 'The parameter "handler" of RepositoryObserver.observe ' +
  51. 'must be a Function, but %s given.',
  52. v,
  53. );
  54. expect(throwable()).to.throw(error('undefined'));
  55. expect(throwable('')).to.throw(error('""'));
  56. expect(throwable('str')).to.throw(error('"str"'));
  57. expect(throwable(10)).to.throw(error('10'));
  58. expect(throwable(true)).to.throw(error('true'));
  59. expect(throwable(false)).to.throw(error('false'));
  60. expect(throwable([])).to.throw(error('Array'));
  61. expect(throwable({})).to.throw(error('Object'));
  62. expect(throwable(null)).to.throw(error('null'));
  63. throwable(HANDLER_FN)();
  64. });
  65. });
  66. describe('3 arguments', function () {
  67. it('adds an event handler of a model scope', function () {
  68. const schema = new Schema();
  69. schema.defineDatasource({name: 'datasource', adapter: 'memory'});
  70. schema.defineModel({name: MODEL_NAME, datasource: 'datasource'});
  71. const observer = schema.get(RepositoryObserver);
  72. observer.observe(MODEL_NAME, EVENT_NAME, HANDLER_FN);
  73. const eventsMap = observer._handlersMap.get(MODEL_NAME);
  74. const listeners = eventsMap.get(EVENT_NAME);
  75. expect(listeners).to.have.lengthOf(1);
  76. expect(listeners).to.include(HANDLER_FN);
  77. });
  78. it('requires the "modelName" argument to be a non-empty string', function () {
  79. const schema = new Schema();
  80. schema.defineDatasource({name: 'datasource', adapter: 'memory'});
  81. schema.defineModel({name: MODEL_NAME, datasource: 'datasource'});
  82. const observer = schema.get(RepositoryObserver);
  83. const throwable = v => () =>
  84. observer.observe(v, EVENT_NAME, HANDLER_FN);
  85. const error = v =>
  86. format(
  87. 'The parameter "modelName" of RepositoryObserver.observe ' +
  88. 'must be a non-empty String, but %s given.',
  89. v,
  90. );
  91. expect(throwable()).to.throw(error('undefined'));
  92. expect(throwable('')).to.throw(error('""'));
  93. expect(throwable(10)).to.throw(error('10'));
  94. expect(throwable(true)).to.throw(error('true'));
  95. expect(throwable(false)).to.throw(error('false'));
  96. expect(throwable([])).to.throw(error('Array'));
  97. expect(throwable({})).to.throw(error('Object'));
  98. expect(throwable(null)).to.throw(error('null'));
  99. throwable(MODEL_NAME)();
  100. });
  101. it('requires the "modelName" argument to be a defined model', function () {
  102. const schema = new Schema();
  103. const observer = schema.get(RepositoryObserver);
  104. const throwable = () =>
  105. observer.observe('unknown', EVENT_NAME, HANDLER_FN);
  106. expect(throwable).to.throw(
  107. 'Cannot observe repository of a not defined model "unknown".',
  108. );
  109. schema.defineDatasource({name: 'datasource', adapter: 'memory'});
  110. schema.defineModel({name: MODEL_NAME, datasource: 'datasource'});
  111. observer.observe(MODEL_NAME, EVENT_NAME, HANDLER_FN);
  112. });
  113. it('requires the "eventName" argument to be RepositoryEvent', function () {
  114. const schema = new Schema();
  115. schema.defineDatasource({name: 'datasource', adapter: 'memory'});
  116. schema.defineModel({name: MODEL_NAME, datasource: 'datasource'});
  117. const observer = schema.get(RepositoryObserver);
  118. const throwable = v => () =>
  119. observer.observe(MODEL_NAME, v, HANDLER_FN);
  120. const error = v =>
  121. format(
  122. 'The parameter "eventName" of RepositoryObserver.observe ' +
  123. 'must be a non-empty String, but %s given.',
  124. v,
  125. );
  126. expect(throwable()).to.throw(error('undefined'));
  127. expect(throwable('')).to.throw(error('""'));
  128. expect(throwable(10)).to.throw(error('10'));
  129. expect(throwable(true)).to.throw(error('true'));
  130. expect(throwable(false)).to.throw(error('false'));
  131. expect(throwable([])).to.throw(error('Array'));
  132. expect(throwable({})).to.throw(error('Object'));
  133. expect(throwable(null)).to.throw(error('null'));
  134. Object.values(RepositoryEvent).forEach(e => throwable(e)());
  135. });
  136. it('requires the "handler" argument to be a function', function () {
  137. const schema = new Schema();
  138. schema.defineDatasource({name: 'datasource', adapter: 'memory'});
  139. schema.defineModel({name: MODEL_NAME, datasource: 'datasource'});
  140. const observer = schema.get(RepositoryObserver);
  141. const throwable = v => () =>
  142. observer.observe(MODEL_NAME, EVENT_NAME, v);
  143. const error = v =>
  144. format(
  145. 'The parameter "handler" of RepositoryObserver.observe ' +
  146. 'must be a Function, but %s given.',
  147. v,
  148. );
  149. expect(throwable()).to.throw(error('undefined'));
  150. expect(throwable('')).to.throw(error('""'));
  151. expect(throwable('str')).to.throw(error('"str"'));
  152. expect(throwable(10)).to.throw(error('10'));
  153. expect(throwable(true)).to.throw(error('true'));
  154. expect(throwable(false)).to.throw(error('false'));
  155. expect(throwable([])).to.throw(error('Array'));
  156. expect(throwable({})).to.throw(error('Object'));
  157. expect(throwable(null)).to.throw(error('null'));
  158. throwable(HANDLER_FN)();
  159. });
  160. });
  161. });
  162. describe('_getHandlers', function () {
  163. it('returns an empty array if no handlers', function () {
  164. const schema = new Schema();
  165. const observer = schema.get(RepositoryObserver);
  166. const result = observer._getHandlers(MODEL_NAME, EVENT_NAME);
  167. expect(result).to.have.lengthOf(0);
  168. });
  169. it('returns model handlers that includes root handlers', function () {
  170. const schema = new Schema();
  171. schema.defineDatasource({name: 'datasource', adapter: 'memory'});
  172. schema.defineModel({name: 'modelA', datasource: 'datasource'});
  173. schema.defineModel({name: 'modelB', datasource: 'datasource'});
  174. const observer = schema.get(RepositoryObserver);
  175. const handler1 = () => undefined;
  176. const handler2 = () => undefined;
  177. const handler3 = () => undefined;
  178. const handler4 = () => undefined;
  179. const handler5 = () => undefined;
  180. const handler6 = () => undefined;
  181. const handler7 = () => undefined;
  182. const event1 = 'event1';
  183. const event2 = 'event2';
  184. observer.observe(event1, handler1);
  185. observer.observe(event1, handler2);
  186. observer.observe(event2, handler3);
  187. observer.observe('modelA', event1, handler4);
  188. observer.observe('modelA', event1, handler5);
  189. observer.observe('modelA', event2, handler6);
  190. observer.observe('modelB', event1, handler7);
  191. const result = observer._getHandlers('modelA', event1);
  192. expect(result).to.be.eql([handler1, handler2, handler4, handler5]);
  193. });
  194. it('requires the "modelName" argument to be a non-empty string', function () {
  195. const schema = new Schema();
  196. schema.defineDatasource({name: 'datasource', adapter: 'memory'});
  197. schema.defineModel({name: MODEL_NAME, datasource: 'datasource'});
  198. const observer = schema.get(RepositoryObserver);
  199. const throwable = v => () => observer._getHandlers(v, EVENT_NAME);
  200. const error = v =>
  201. format(
  202. 'The parameter "modelName" of RepositoryObserver._getHandlers ' +
  203. 'must be a non-empty String, but %s given.',
  204. v,
  205. );
  206. expect(throwable()).to.throw(error('undefined'));
  207. expect(throwable('')).to.throw(error('""'));
  208. expect(throwable(10)).to.throw(error('10'));
  209. expect(throwable(true)).to.throw(error('true'));
  210. expect(throwable(false)).to.throw(error('false'));
  211. expect(throwable([])).to.throw(error('Array'));
  212. expect(throwable({})).to.throw(error('Object'));
  213. expect(throwable(null)).to.throw(error('null'));
  214. throwable(MODEL_NAME)();
  215. });
  216. it('requires the "eventName" argument to be a non-empty string', function () {
  217. const schema = new Schema();
  218. schema.defineDatasource({name: 'datasource', adapter: 'memory'});
  219. schema.defineModel({name: MODEL_NAME, datasource: 'datasource'});
  220. const observer = schema.get(RepositoryObserver);
  221. const throwable = v => () => observer._getHandlers(MODEL_NAME, v);
  222. const error = v =>
  223. format(
  224. 'The parameter "eventName" of RepositoryObserver._getHandlers ' +
  225. 'must be a non-empty String, but %s given.',
  226. v,
  227. );
  228. expect(throwable()).to.throw(error('undefined'));
  229. expect(throwable('')).to.throw(error('""'));
  230. expect(throwable(10)).to.throw(error('10'));
  231. expect(throwable(true)).to.throw(error('true'));
  232. expect(throwable(false)).to.throw(error('false'));
  233. expect(throwable([])).to.throw(error('Array'));
  234. expect(throwable({})).to.throw(error('Object'));
  235. expect(throwable(null)).to.throw(error('null'));
  236. throwable('eventName')();
  237. });
  238. });
  239. describe('emit', function () {
  240. it('requires the "modelName" argument to be a non-empty string', function () {
  241. const schema = new Schema();
  242. schema.defineDatasource({name: 'datasource', adapter: 'memory'});
  243. schema.defineModel({name: MODEL_NAME, datasource: 'datasource'});
  244. const observer = schema.get(RepositoryObserver);
  245. const throwable = v => () =>
  246. observer.emit(v, EVENT_NAME, METHOD_NAME, {});
  247. const error = v =>
  248. format(
  249. 'The parameter "modelName" of RepositoryObserver.emit ' +
  250. 'must be a non-empty String, but %s given.',
  251. v,
  252. );
  253. expect(throwable()).to.throw(error('undefined'));
  254. expect(throwable('')).to.throw(error('""'));
  255. expect(throwable(10)).to.throw(error('10'));
  256. expect(throwable(true)).to.throw(error('true'));
  257. expect(throwable(false)).to.throw(error('false'));
  258. expect(throwable([])).to.throw(error('Array'));
  259. expect(throwable({})).to.throw(error('Object'));
  260. expect(throwable(null)).to.throw(error('null'));
  261. throwable(MODEL_NAME)();
  262. });
  263. it('requires the "eventName" argument to be a non-empty string', function () {
  264. const schema = new Schema();
  265. schema.defineDatasource({name: 'datasource', adapter: 'memory'});
  266. schema.defineModel({name: MODEL_NAME, datasource: 'datasource'});
  267. const observer = schema.get(RepositoryObserver);
  268. const throwable = v => () =>
  269. observer.emit(MODEL_NAME, v, METHOD_NAME, {});
  270. const error = v =>
  271. format(
  272. 'The parameter "eventName" of RepositoryObserver.emit ' +
  273. 'must be a non-empty String, but %s given.',
  274. v,
  275. );
  276. expect(throwable()).to.throw(error('undefined'));
  277. expect(throwable('')).to.throw(error('""'));
  278. expect(throwable(10)).to.throw(error('10'));
  279. expect(throwable(true)).to.throw(error('true'));
  280. expect(throwable(false)).to.throw(error('false'));
  281. expect(throwable([])).to.throw(error('Array'));
  282. expect(throwable({})).to.throw(error('Object'));
  283. expect(throwable(null)).to.throw(error('null'));
  284. throwable('eventName')();
  285. });
  286. it('requires the "methodName" argument to be a non-empty string', function () {
  287. const schema = new Schema();
  288. schema.defineDatasource({name: 'datasource', adapter: 'memory'});
  289. schema.defineModel({name: MODEL_NAME, datasource: 'datasource'});
  290. const observer = schema.get(RepositoryObserver);
  291. const throwable = v => () => observer.emit(MODEL_NAME, EVENT_NAME, v, {});
  292. const error = v =>
  293. format(
  294. 'The parameter "methodName" of RepositoryObserver.emit ' +
  295. 'must be a non-empty String, but %s given.',
  296. v,
  297. );
  298. expect(throwable()).to.throw(error('undefined'));
  299. expect(throwable('')).to.throw(error('""'));
  300. expect(throwable(10)).to.throw(error('10'));
  301. expect(throwable(true)).to.throw(error('true'));
  302. expect(throwable(false)).to.throw(error('false'));
  303. expect(throwable([])).to.throw(error('Array'));
  304. expect(throwable({})).to.throw(error('Object'));
  305. expect(throwable(null)).to.throw(error('null'));
  306. throwable(METHOD_NAME)();
  307. });
  308. it('requires the "context" argument to be a non-empty string', function () {
  309. const schema = new Schema();
  310. schema.defineDatasource({name: 'datasource', adapter: 'memory'});
  311. schema.defineModel({name: MODEL_NAME, datasource: 'datasource'});
  312. const observer = schema.get(RepositoryObserver);
  313. const throwable = v => () =>
  314. observer.emit(MODEL_NAME, EVENT_NAME, METHOD_NAME, v);
  315. const error = v =>
  316. format(
  317. 'The parameter "context" of RepositoryObserver.emit ' +
  318. 'must be an Object, but %s given.',
  319. v,
  320. );
  321. expect(throwable()).to.throw(error('undefined'));
  322. expect(throwable('')).to.throw(error('""'));
  323. expect(throwable(10)).to.throw(error('10'));
  324. expect(throwable(true)).to.throw(error('true'));
  325. expect(throwable(false)).to.throw(error('false'));
  326. expect(throwable([])).to.throw(error('Array'));
  327. expect(throwable(null)).to.throw(error('null'));
  328. throwable({})();
  329. });
  330. it('returns a promise for synchronous handlers', async function () {
  331. const schema = new Schema();
  332. schema.defineDatasource({name: 'datasource', adapter: 'memory'});
  333. schema.defineModel({name: MODEL_NAME, datasource: 'datasource'});
  334. schema.defineModel({name: 'modelB', datasource: 'datasource'});
  335. const observer = schema.get(RepositoryObserver);
  336. const order = [];
  337. const handler1 = () => order.push(handler1);
  338. const handler2 = () => order.push(handler2);
  339. const handler3 = () => order.push(handler3);
  340. const handler4 = () => order.push(handler4);
  341. const throwable = () => {
  342. throw new Error();
  343. };
  344. observer.observe(EVENT_NAME, handler1);
  345. observer.observe(EVENT_NAME, handler2);
  346. observer.observe('event2', throwable);
  347. observer.observe(MODEL_NAME, EVENT_NAME, handler3);
  348. observer.observe(MODEL_NAME, EVENT_NAME, handler4);
  349. observer.observe(MODEL_NAME, 'event2', throwable);
  350. observer.observe('modelB', EVENT_NAME, throwable);
  351. const promise = observer.emit(MODEL_NAME, EVENT_NAME, METHOD_NAME, {});
  352. expect(promise).to.be.instanceof(Promise);
  353. await promise;
  354. expect(order).to.be.eql([handler1, handler2, handler3, handler4]);
  355. });
  356. it('returns a promise of asynchronous handlers', async function () {
  357. const schema = new Schema();
  358. schema.defineDatasource({name: 'datasource', adapter: 'memory'});
  359. schema.defineModel({name: MODEL_NAME, datasource: 'datasource'});
  360. schema.defineModel({name: 'modelB', datasource: 'datasource'});
  361. const observer = schema.get(RepositoryObserver);
  362. const order = [];
  363. const handler0ms = () => {
  364. order.push(handler0ms);
  365. return Promise.resolve();
  366. };
  367. const handler5ms = () => {
  368. return new Promise(res => {
  369. setTimeout(() => {
  370. order.push(handler5ms);
  371. res();
  372. }, 5);
  373. });
  374. };
  375. const syncHandler = () => order.push(syncHandler);
  376. const handler3ms = () => {
  377. return new Promise(res => {
  378. setTimeout(() => {
  379. order.push(handler3ms);
  380. res();
  381. }, 3);
  382. });
  383. };
  384. const throwable = () => {
  385. throw new Error();
  386. };
  387. observer.observe(EVENT_NAME, handler0ms);
  388. observer.observe(EVENT_NAME, handler5ms);
  389. observer.observe('event2', throwable);
  390. observer.observe(MODEL_NAME, EVENT_NAME, syncHandler);
  391. observer.observe(MODEL_NAME, EVENT_NAME, handler3ms);
  392. observer.observe(MODEL_NAME, 'event2', throwable);
  393. observer.observe('modelB', EVENT_NAME, throwable);
  394. const promise = observer.emit(MODEL_NAME, EVENT_NAME, METHOD_NAME, {});
  395. expect(promise).to.be.instanceof(Promise);
  396. await promise;
  397. expect(order).to.have.lengthOf(4);
  398. expect(order).to.include(handler0ms);
  399. expect(order).to.include(handler5ms);
  400. expect(order).to.include(syncHandler);
  401. expect(order).to.include(handler3ms);
  402. });
  403. it('executes handlers with the context object', async function () {
  404. const schema = new Schema();
  405. schema.defineDatasource({name: 'datasource', adapter: 'memory'});
  406. schema.defineModel({name: MODEL_NAME, datasource: 'datasource'});
  407. const observer = schema.get(RepositoryObserver);
  408. const inputContext = {customProp: 'customProp'};
  409. const context = {
  410. modelName: MODEL_NAME,
  411. eventName: EVENT_NAME,
  412. methodName: METHOD_NAME,
  413. ...inputContext,
  414. };
  415. let counter = 0;
  416. const handler1 = ctx => {
  417. expect(ctx).to.be.eql(context);
  418. counter++;
  419. };
  420. const handler2 = ctx => {
  421. return new Promise(res => {
  422. setTimeout(() => {
  423. expect(ctx).to.be.eql(context);
  424. counter++;
  425. res();
  426. }, 3);
  427. });
  428. };
  429. observer.observe(EVENT_NAME, handler1);
  430. observer.observe(MODEL_NAME, EVENT_NAME, handler2);
  431. await observer.emit(MODEL_NAME, EVENT_NAME, METHOD_NAME, inputContext);
  432. expect(counter).to.be.eq(2);
  433. });
  434. });
  435. });