blob: 14fdeca7a426c44ac9027a3a2494e235d0ac1844 [file] [log] [blame]
// 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');