https://github.com/angular/angular
Raw File
Tip revision: 818f4a751ecbb9b637389909f01373208dd16f0d authored by Matias Niemelä on 15 September 2017, 22:35:25 UTC
release: cut the 4.4.1 release
Tip revision: 818f4a7
watch.spec.js
/**
 * @license
 * Copyright Google Inc. All Rights Reserved.
 *
 * Use of this source code is governed by an MIT-style license that can be
 * found in the LICENSE file at https://angular.io/license
 */

var rewire = require('rewire');

describe('watch()', function() {
  var timeout;
  var watch;
  var watcher;
  var unmock;

  beforeEach(function() {
    timeout = mockTimeout();
    var watchModule = rewire('./watch');
    unmock = watchModule.__set__(timeout.mocks);
    watch = watchModule.__get__('watch');
  });

  afterEach(function() {
    // Shouldn't really be necessary, but...
    if (watcher) {
      watcher.close();
      watcher = null;
    }
    unmock();
    expect(timeout.pending).toBe(0);
    timeout = null;
  });


  it('should fire callback once for events which occur within `delay` window', function() {
    var cb = jasmine.createSpy('callback');
    watcher = watch('./$$fake_path/**/*', {delay: 10, log: false}, cb);

    watcher._emit('add', './$$fake_path/test.txt');
    timeout.flush(9);
    expect(cb).not.toHaveBeenCalled();

    watcher._emit('change', './$$fake_path/test.txt');
    watcher._emit('add', './$$fake_path/test2.txt');
    watcher._emit('change', './$$fake_path/test2.txt');
    watcher._emit('add', './$$fake_path/test3.txt');
    watcher._emit('change', './$$fake_path/test3.txt');
    expect(cb).not.toHaveBeenCalled();

    timeout.flush(1);
    expect(cb.calls.count()).toBe(1);
  });


  it('should trigger callback if events are collected during task running', function() {
    var calls = 0;
    function cb(done) {
      if (++calls !== 1) return done();

      watcher._emit('change', './$$fake_path/test1.txt');
      watcher._emit('change', './$$fake_path/test2.txt');

      // Before the done callback, there are no pending timer events
      expect(timeout.pending).toBe(0);
      done();

      // Afterwards, there is one
      expect(timeout.pending).toBe(1);
    }

    var watcher = watch('./$$fake_path/**/*', {delay: 10, log: false}, cb);

    watcher._emit('change', './$$fake_path/test1.txt');
    expect(timeout.pending).toBe(1);
    expect(calls).toBe(0);

    timeout.flush(10);
    expect(calls).toBe(2);
  });


  it('should continue to trigger callbacks if task throws', function() {
    var calls = 0;
    spyOn(console, 'log');
    function cb(done) {
      calls += 1;
      if (calls === 1) throw new Error('oops!');
      done();
    }

    var watcher = watch('./$$fake_path/**/*', {delay: 10, log: false}, cb);

    watcher._emit('change', './$$fake_path/test1.txt');
    timeout.flush();
    expect(calls).toBe(1);
    expect(console.log).toHaveBeenCalledWith('Watch task error:', 'Error: oops!');

    watcher._emit('change', './$$fake_path/test2.txt');
    timeout.flush();
    expect(calls).toBe(2);
  });


  it('should cancel pending callback if FSWatcher is closed', function() {
    var cb = jasmine.createSpy('callback');
    var watcher = watch('./$$fake_path/**/*', {delay: 10, log: false}, cb);

    watcher._emit('change', './$$fake_path/test1.txt');
    expect(timeout.pending).toBe(1);
    expect(cb).not.toHaveBeenCalled();

    watcher.close();
    expect(timeout.pending).toBe(0);
  });


  it('should cancel followup pending callback if FSWatcher is closed during task', function() {
    var calls = 0;
    function cb(done) {
      if (++calls !== 1) return done();

      watcher._emit('change', './$$fake_path/test2.txt');
      done();
      expect(timeout.pending).toBe(1);
      watcher.close();
      expect(timeout.pending).toBe(0);
    }

    var watcher = watch('./$$fake_path/**/*', {delay: 10, log: false}, cb);
    watcher._emit('change', './$$fake_path/test1.txt');

    timeout.flush(10);

    expect(calls).toBe(1);
  });
});


// setTimeout/clearTimeout mocking, mostly stolen from angular-mocks.js
function mockTimeout() {
  var events = [];
  var id = 0;
  var now = 0;

  return {
    mocks: {setTimeout: mockSetTimeout, clearTimeout: mockClearTimeout},
    flush: flush, get pending() { return events.length; }
  };

  function mockSetTimeout(fn, delay) {
    delay = delay || 0;
    events.push({time: now + delay, fn: fn, id: id});
    events.sort(function(a, b) { return a.time - b.time; });
    return id++;
  }

  function mockClearTimeout(id) {
    for (var i = 0; i < events.length; ++i) {
      if (events[i].id === id) {
        events.splice(i, 1);
        break;
      }
    }
  }

  function flush(delay) {
    if (delay !== undefined)
      now += delay;
    else if (events.length)
      now = events[events.length - 1].time;
    else
      throw new Error('No timer events registered');

    while (events.length && events[0].time <= now) {
      events.shift().fn();
    }
  }
}
back to top