Revision 6d430e9d2dbc770f66c40e944928f5ec4f6cbccb authored by Henrik Skupin on 15 August 2018, 17:44:30 UTC, committed by jgraham on 17 August 2018, 09:47:16 UTC
Those tests make sure that an "invalid argument" error is raised
if the body doesn't contain any data.

bugzilla-url: https://bugzilla.mozilla.org/show_bug.cgi?id=1469601
gecko-commit: 7915b80eca091d53838f20594e84ed543b1752f3
gecko-integration-branch: mozilla-inbound
gecko-reviewers: ato
1 parent e27c34b
Raw File
query.tentative.https.any.js
// META: title=Web Locks API: navigator.locks.query method
// META: script=resources/helpers.js

'use strict';

// Returns an array of the modes for the locks with matching name.
function modes(list, name) {
  return list.filter(item => item.name === name).map(item => item.mode);
}
// Returns an array of the clientIds for the locks with matching name.
function clients(list, name) {
  return list.filter(item => item.name === name).map(item => item.clientId);
}

promise_test(async t => {
  const res = uniqueName(t);

  await navigator.locks.request(res, async lock1 => {
    // Attempt to request this again - should be blocked.
    let lock2_acquired = false;
    navigator.locks.request(res, lock2 => { lock2_acquired = true; });

    // Verify that it was blocked.
    await navigator.locks.request(res, {ifAvailable: true}, async lock3 => {
      assert_false(lock2_acquired, 'second request should be blocked');
      assert_equals(lock3, null, 'third request should have failed');

      const state = await navigator.locks.query();

      assert_own_property(state, 'pending', 'State has `pending` property');
      assert_true(Array.isArray(state.pending),
                  'State `pending` property is an array');
      const pending_info = state.pending[0];
      assert_own_property(pending_info, 'name',
                          'Pending info dictionary has `name` property');
      assert_own_property(pending_info, 'mode',
                          'Pending info dictionary has `mode` property');
      assert_own_property(pending_info, 'clientId',
                          'Pending info dictionary has `clientId` property');

      assert_own_property(state, 'held', 'State has `held` property');
      assert_true(Array.isArray(state.held),
                  'State `held` property is an array');
      const held_info = state.held[0];
      assert_own_property(held_info, 'name',
                          'Held info dictionary has `name` property');
      assert_own_property(held_info, 'mode',
                          'Held info dictionary has `mode` property');
      assert_own_property(held_info, 'clientId',
                          'Held info dictionary has `clientId` property');
    });
  });
}, 'query() returns dictionaries with expected properties');



promise_test(async t => {
  const res = uniqueName(t);

  await navigator.locks.request(res, async lock1 => {
    const state = await navigator.locks.query();
    assert_array_equals(modes(state.held, res), ['exclusive'],
                        'Held lock should appear once');
  });

  await navigator.locks.request(res, {mode: 'shared'}, async lock1 => {
    const state = await navigator.locks.query();
    assert_array_equals(modes(state.held, res), ['shared'],
                        'Held lock should appear once');
  });
}, 'query() reports individual held locks');

promise_test(async t => {
  const res1 = uniqueName(t);
  const res2 = uniqueName(t);

  await navigator.locks.request(res1, async lock1 => {
    await navigator.locks.request(res2, {mode: 'shared'}, async lock2 => {
      const state = await navigator.locks.query();
      assert_array_equals(modes(state.held, res1), ['exclusive'],
                          'Held lock should appear once');
      assert_array_equals(modes(state.held, res2), ['shared'],
                          'Held lock should appear once');
    });
  });
}, 'query() reports multiple held locks');

promise_test(async t => {
  const res = uniqueName(t);

  await navigator.locks.request(res, async lock1 => {
    // Attempt to request this again - should be blocked.
    let lock2_acquired = false;
    navigator.locks.request(res, lock2 => { lock2_acquired = true; });

    // Verify that it was blocked.
    await navigator.locks.request(res, {ifAvailable: true}, async lock3 => {
      assert_false(lock2_acquired, 'second request should be blocked');
      assert_equals(lock3, null, 'third request should have failed');

      const state = await navigator.locks.query();
      assert_array_equals(modes(state.pending, res), ['exclusive'],
                          'Pending lock should appear once');
      assert_array_equals(modes(state.held, res), ['exclusive'],
                          'Held lock should appear once');
    });
  });
}, 'query() reports pending and held locks');

promise_test(async t => {
  const res = uniqueName(t);

  await navigator.locks.request(res, {mode: 'shared'}, async lock1 => {
    await navigator.locks.request(res, {mode: 'shared'}, async lock2 => {
      const state = await navigator.locks.query();
      assert_array_equals(modes(state.held, res), ['shared', 'shared'],
                          'Held lock should appear twice');
    });
  });
}, 'query() reports held shared locks with appropriate count');

promise_test(async t => {
  const res = uniqueName(t);

  await navigator.locks.request(res, async lock1 => {
    let lock2_acquired = false, lock3_acquired = false;
    navigator.locks.request(res, {mode: 'shared'},
                            lock2 => { lock2_acquired = true; });
    navigator.locks.request(res, {mode: 'shared'},
                            lock3 => { lock3_acquired = true; });

    await navigator.locks.request(res, {ifAvailable: true}, async lock4 => {
      assert_equals(lock4, null, 'lock should not be available');
      assert_false(lock2_acquired, 'second attempt should be blocked');
      assert_false(lock3_acquired, 'third attempt should be blocked');

      const state = await navigator.locks.query();
      assert_array_equals(modes(state.held, res), ['exclusive'],
                          'Held lock should appear once');

      assert_array_equals(modes(state.pending, res), ['shared', 'shared'],
                          'Pending lock should appear twice');
    });
  });
}, 'query() reports pending shared locks with appropriate count');

promise_test(async t => {
  const res1 = uniqueName(t);
  const res2 = uniqueName(t);

  await navigator.locks.request(res1, async lock1 => {
    await navigator.locks.request(res2, async lock2 => {
      const state = await navigator.locks.query();

      const res1_clients = clients(state.held, res1);
      const res2_clients = clients(state.held, res2);

      assert_equals(res1_clients.length, 1, 'Each lock should have one holder');
      assert_equals(res2_clients.length, 1, 'Each lock should have one holder');

      assert_array_equals(res1_clients, res2_clients,
                          'Both locks should have same clientId');
    });
  });
}, 'query() reports the same clientId for held locks from the same context');

promise_test(async t => {
  const res = uniqueName(t);

  const worker = new Worker('resources/worker.js');
  t.add_cleanup(() => { worker.terminate(); });

  await postToWorkerAndWait(
    worker, {op: 'request', name: res, mode: 'shared'});

  await navigator.locks.request(res, {mode: 'shared'}, async lock => {
    const state = await navigator.locks.query();
    const res_clients = clients(state.held, res);
    assert_equals(res_clients.length, 2, 'Clients should have same resource');
    assert_not_equals(res_clients[0], res_clients[1],
                      'Clients should have different ids');
  });
}, 'query() reports different ids for held locks from different contexts');

promise_test(async t => {
  const res1 = uniqueName(t);
  const res2 = uniqueName(t);

  const worker = new Worker('resources/worker.js');
  t.add_cleanup(() => { worker.terminate(); });

  // Acquire 1 in the worker.
  await postToWorkerAndWait(worker, {op: 'request', name: res1})

  // Acquire 2 here.
  await new Promise(resolve => {
    navigator.locks.request(res2, lock => {
      resolve();
      return new Promise(() => {}); // Never released.
    });
  });

  // Request 2 in the worker.
  postToWorkerAndWait(worker, {op: 'request', name: res2});
  assert_true((await postToWorkerAndWait(worker, {
    op: 'request', name: res2, ifAvailable: true
  })).failed, 'Lock request should have failed');

  // Request 1 here.
  navigator.locks.request(
    res1, t.unreached_func('Lock should not be acquired'));

  // Verify that we're seeing a deadlock.
  const state = await navigator.locks.query();
  const res1_held_clients = clients(state.held, res1);
  const res2_held_clients = clients(state.held, res2);
  const res1_pending_clients = clients(state.pending, res1);
  const res2_pending_clients = clients(state.pending, res2);

  assert_equals(res1_held_clients.length, 1);
  assert_equals(res2_held_clients.length, 1);
  assert_equals(res1_pending_clients.length, 1);
  assert_equals(res2_pending_clients.length, 1);

  assert_equals(res1_held_clients[0], res2_pending_clients[0]);
  assert_equals(res2_held_clients[0], res1_pending_clients[0]);
}, 'query() can observe a deadlock');
back to top