1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
438
439
440
441
442
443
444
445
446
447
448
449
450
451
452
453
454
455
456
457
458
459
460
461
462
463
464
465
466
467
468
469
470
471
472
473
474
475
476
477
478
479
480
481
482
483
484
485
486
487
488
489
490
491
492
493
494
495
496
497
498
499
500
501
502
503
504
505
506
507
508
509
510
511
512
513
514
|
/* This Source Code Form is subject to the terms of the Mozilla Public
* License, v. 2.0. If a copy of the MPL was not distributed with this
* file, You can obtain one at <http://mozilla.org/MPL/2.0/>. */
/**
* Helper methods for finding messages in the virtualized output of the
* webconsole. This file can be safely required from other panel test
* files.
*/
"use strict";
/* eslint-disable no-unused-vars */
// Assume that shared-head is always imported before this file
/* import-globals-from ../../../shared/test/shared-head.js */
/**
* Find a message with given messageId in the output, scrolling through the
* output from top to bottom in order to make sure the messages are actually
* rendered.
*
* @param object hud
* The web console.
* @param messageId
* A message ID to look for. This could be baked into the selector, but
* is provided as a convenience.
* @return {Node} the node corresponding the found message
*/
async function findMessageVirtualizedById({ hud, messageId }) {
if (!messageId) {
throw new Error("messageId parameter is required");
}
const elements = await findMessagesVirtualized({
hud,
expectedCount: 1,
messageId,
});
return elements.at(-1);
}
/**
* Find the last message with given message type in the output, scrolling
* through the output from top to bottom in order to make sure the messages are
* actually rendered.
*
* @param object hud
* The web console.
* @param string text
* A substring that can be found in the message.
* @param string typeSelector
* A part of selector for the message, to specify the message type.
* @return {Node} the node corresponding the found message
*/
async function findMessageVirtualizedByType({ hud, text, typeSelector }) {
const elements = await findMessagesVirtualizedByType({
hud,
text,
typeSelector,
expectedCount: 1,
});
return elements.at(-1);
}
/**
* Find all messages in the output, scrolling through the output from top
* to bottom in order to make sure the messages are actually rendered.
*
* @param object hud
* The web console.
* @return {Array} all of the message nodes in the console output. Some of
* these may be stale from having been scrolled out of view.
*/
async function findAllMessagesVirtualized(hud) {
return findMessagesVirtualized({ hud });
}
// This is just a reentrancy guard. Because findMessagesVirtualized mucks
// around with the scroll position, if we do something like
// let promise1 = findMessagesVirtualized(...);
// let promise2 = findMessagesVirtualized(...);
// await promise1;
// await promise2;
// then the two calls will end up messing up each other's expected scroll
// position, at which point they could get stuck. This lets us throw an
// error when that happens.
let gInFindMessagesVirtualized = false;
// And this lets us get a little more information in the error - it just holds
// the stack of the prior call.
let gFindMessagesVirtualizedStack = null;
/**
* Find multiple messages in the output, scrolling through the output from top
* to bottom in order to make sure the messages are actually rendered.
*
* @param object options
* @param object options.hud
* The web console.
* @param options.text [optional]
* A substring that can be found in the message.
* @param options.typeSelector
* A part of selector for the message, to specify the message type.
* @param options.expectedCount [optional]
* The number of messages to get. This lets us stop scrolling early if
* we find that number of messages.
* @return {Array} all of the message nodes in the console output matching the
* provided filters. If expectedCount is greater than 1, or equal to -1,
* some of these may be stale from having been scrolled out of view.
*/
async function findMessagesVirtualizedByType({
hud,
text,
typeSelector,
expectedCount,
}) {
if (!typeSelector) {
throw new Error("typeSelector parameter is required");
}
if (!typeSelector.startsWith(".")) {
throw new Error("typeSelector should start with a dot e.g. `.result`");
}
return findMessagesVirtualized({
hud,
text,
selector: ".message" + typeSelector,
expectedCount,
});
}
/**
* Find multiple messages in the output, scrolling through the output from top
* to bottom in order to make sure the messages are actually rendered.
*
* @param object options
* @param object options.hud
* The web console.
* @param options.text [optional]
* A substring that can be found in the message.
* @param options.selector [optional]
* The selector to use in finding the message.
* @param options.expectedCount [optional]
* The number of messages to get. This lets us stop scrolling early if
* we find that number of messages.
* @param options.messageId [optional]
* A message ID to look for. This could be baked into the selector, but
* is provided as a convenience.
* @return {Array} all of the message nodes in the console output matching the
* provided filters. If expectedCount is greater than 1, or equal to -1,
* some of these may be stale from having been scrolled out of view.
*/
async function findMessagesVirtualized({
hud,
text,
selector,
expectedCount,
messageId,
}) {
if (text === undefined) {
text = "";
}
if (selector === undefined) {
selector = ".message";
}
if (expectedCount === undefined) {
expectedCount = -1;
}
const outputNode = hud.ui.outputNode;
const scrollport = outputNode.querySelector(".webconsole-output");
function getVisibleMessageIds() {
return JSON.parse(scrollport.getAttribute("data-visible-messages"));
}
function getVisibleMessageMap() {
return new Map(
JSON.parse(scrollport.getAttribute("data-visible-messages")).map(
(id, i) => [id, i]
)
);
}
function getMessageIndex(message) {
return getVisibleMessageIds().indexOf(
message.getAttribute("data-message-id")
);
}
function getNextMessageId(prevMessage) {
const visible = getVisibleMessageIds();
let index = 0;
if (prevMessage) {
const lastId = prevMessage.getAttribute("data-message-id");
index = visible.indexOf(lastId);
if (index === -1) {
throw new Error(
`Tried to get next message ID for message that doesn't exist. Last seen ID: ${lastId}, all visible ids: [${visible.join(
", "
)}]`
);
}
}
if (index + 1 >= visible.length) {
return null;
}
return visible[index + 1];
}
if (gInFindMessagesVirtualized) {
throw new Error(
`findMessagesVirtualized was re-entered somehow. This is not allowed. Other stack: [${gFindMessagesVirtualizedStack}]`
);
}
try {
gInFindMessagesVirtualized = true;
gFindMessagesVirtualizedStack = new Error().stack;
// The console output will automatically scroll to the bottom of the
// scrollport in certain circumstances. Because we need to scroll the
// output to find all messages, we need to disable this. This attribute
// controls that.
scrollport.setAttribute("disable-autoscroll", "");
// This array is here purely for debugging purposes. We collect the indices
// of every element we see in order to validate that we don't have any gaps
// in the list.
const allIndices = [];
const allElements = [];
const seenIds = new Set();
let lastItem = null;
while (true) {
if (scrollport.scrollHeight > scrollport.clientHeight) {
if (!lastItem && scrollport.scrollTop != 0) {
// For simplicity's sake, we always start from the top of the output.
scrollport.scrollTop = 0;
} else if (!lastItem && scrollport.scrollTop == 0) {
// We want to make sure that we actually change the scroll position
// here, because we're going to wait for an update below regardless,
// just to flush out any changes that may have just happened. If we
// don't do this, and there were no changes before this function was
// called, then we'll just hang on the promise below.
scrollport.scrollTop = 1;
} else {
// This is the core of the loop. Scroll down to the bottom of the
// current scrollport, wait until we see the element after the last
// one we've seen, and then harvest the messages that are displayed.
scrollport.scrollTop = scrollport.scrollTop + scrollport.clientHeight;
}
// Wait for something to happen in the output before checking for our
// expected next message.
await new Promise(resolve =>
hud.ui.once("lazy-message-list-updated-or-noop", resolve)
);
try {
await waitFor(async () => {
const nextMessageId = getNextMessageId(lastItem);
if (
nextMessageId === undefined ||
scrollport.querySelector(`[data-message-id="${nextMessageId}"]`)
) {
return true;
}
// After a scroll, we typically expect to get an updated list of
// elements. However, we have some slack at the top of the list,
// because we draw elements above and below the actual scrollport to
// avoid white flashes when async scrolling.
const scrollTarget = scrollport.scrollTop + scrollport.clientHeight;
scrollport.scrollTop = scrollTarget;
await new Promise(resolve =>
hud.ui.once("lazy-message-list-updated-or-noop", resolve)
);
return false;
});
} catch (e) {
throw new Error(
`Failed waiting for next message ID (${getNextMessageId(
lastItem
)}) Visible messages: [${[
...scrollport.querySelectorAll(".message"),
].map(el => el.getAttribute("data-message-id"))}]`
);
}
}
const bottomPlaceholder = scrollport.querySelector(
".lazy-message-list-bottom"
);
if (!bottomPlaceholder) {
// When there are no messages in the output, there is also no
// top/bottom placeholder. There's nothing more to do at this point,
// so break and return.
break;
}
lastItem = bottomPlaceholder.previousSibling;
// This chunk is just validating that we have no gaps in our output so
// far.
const indices = [...scrollport.querySelectorAll("[data-message-id]")]
.filter(
el => el !== scrollport.firstChild && el !== scrollport.lastChild
)
.map(el => getMessageIndex(el));
allIndices.push(...indices);
allIndices.sort((lhs, rhs) => lhs - rhs);
for (let i = 1; i < allIndices.length; i++) {
if (
allIndices[i] != allIndices[i - 1] &&
allIndices[i] != allIndices[i - 1] + 1
) {
throw new Error(
`Gap detected in virtualized webconsole output between ${
allIndices[i - 1]
} and ${allIndices[i]}. Indices: ${allIndices.join(",")}`
);
}
}
const messages = scrollport.querySelectorAll(selector);
const filtered = [...messages].filter(
el =>
// Core user filters:
el.textContent.includes(text) &&
(!messageId || el.getAttribute("data-message-id") === messageId) &&
// Make sure we don't collect duplicate messages:
!seenIds.has(el.getAttribute("data-message-id"))
);
allElements.push(...filtered);
for (const message of filtered) {
seenIds.add(message.getAttribute("data-message-id"));
}
if (expectedCount >= 0 && allElements.length >= expectedCount) {
break;
}
// If the bottom placeholder has 0 height, it means we've scrolled to the
// bottom and output all the items.
if (bottomPlaceholder.getBoundingClientRect().height == 0) {
break;
}
await waitForTime(0);
}
// Finally, we get the map of message IDs to indices within the output, and
// sort the message nodes according to that index. They can come in out of
// order for a number of reasons (we continue rendering any messages that
// have been expanded, and we always render the topmost and bottommost
// messages for a11y reasons.)
const idsToIndices = getVisibleMessageMap();
allElements.sort(
(lhs, rhs) =>
idsToIndices.get(lhs.getAttribute("data-message-id")) -
idsToIndices.get(rhs.getAttribute("data-message-id"))
);
return allElements;
} finally {
scrollport.removeAttribute("disable-autoscroll");
gInFindMessagesVirtualized = false;
gFindMessagesVirtualizedStack = null;
}
}
/**
* Find the last message with given message type in the output.
*
* @param object hud
* The web console.
* @param string text
* A substring that can be found in the message.
* @param string typeSelector
* A part of selector for the message, to specify the message type.
* @return {Node} the node corresponding the found message, otherwise undefined
*/
function findMessageByType(hud, text, typeSelector) {
const elements = findMessagesByType(hud, text, typeSelector);
return elements.at(-1);
}
/**
* Find multiple messages with given message type in the output.
*
* @param object hud
* The web console.
* @param string text
* A substring that can be found in the message.
* @param string typeSelector
* A part of selector for the message, to specify the message type.
* @return {Array} The nodes corresponding the found messages
*/
function findMessagesByType(hud, text, typeSelector) {
if (!typeSelector) {
throw new Error("typeSelector parameter is required");
}
if (!typeSelector.startsWith(".")) {
throw new Error("typeSelector should start with a dot e.g. `.result`");
}
const selector = ".message" + typeSelector;
const messages = hud.ui.outputNode.querySelectorAll(selector);
const elements = Array.from(messages).filter(el =>
el.textContent.includes(text)
);
return elements;
}
/**
* Find all messages in the output.
*
* @param object hud
* The web console.
* @return {Array} The nodes corresponding the found messages
*/
function findAllMessages(hud) {
const messages = hud.ui.outputNode.querySelectorAll(".message");
return Array.from(messages);
}
/**
* Find a part of the last message with given message type in the output.
*
* @param object hud
* The web console.
* @param object options
* - text : {String} A substring that can be found in the message.
* - typeSelector: {String} A part of selector for the message,
* to specify the message type.
* - partSelector: {String} A selector for the part of the message.
* @return {Node} the node corresponding the found part, otherwise undefined
*/
function findMessagePartByType(hud, options) {
const elements = findMessagePartsByType(hud, options);
return elements.at(-1);
}
/**
* Find parts of multiple messages with given message type in the output.
*
* @param object hud
* The web console.
* @param object options
* - text : {String} A substring that can be found in the message.
* - typeSelector: {String} A part of selector for the message,
* to specify the message type.
* - partSelector: {String} A selector for the part of the message.
* @return {Array} The nodes corresponding the found parts
*/
function findMessagePartsByType(hud, { text, typeSelector, partSelector }) {
if (!typeSelector) {
throw new Error("typeSelector parameter is required");
}
if (!typeSelector.startsWith(".")) {
throw new Error("typeSelector should start with a dot e.g. `.result`");
}
if (!partSelector) {
throw new Error("partSelector parameter is required");
}
const selector = ".message" + typeSelector + " " + partSelector;
const parts = hud.ui.outputNode.querySelectorAll(selector);
const elements = Array.from(parts).filter(el =>
el.textContent.includes(text)
);
return elements;
}
/**
* Type-specific wrappers for findMessageByType and findMessagesByType.
*
* @param object hud
* The web console.
* @param string text
* A substring that can be found in the message.
* @param string extraSelector [optional]
* An extra part of selector for the message, in addition to
* type-specific selector.
* @return {Node|Array} See findMessageByType or findMessagesByType.
*/
function findEvaluationResultMessage(hud, text, extraSelector = "") {
return findMessageByType(hud, text, ".result" + extraSelector);
}
function findEvaluationResultMessages(hud, text, extraSelector = "") {
return findMessagesByType(hud, text, ".result" + extraSelector);
}
function findErrorMessage(hud, text, extraSelector = "") {
return findMessageByType(hud, text, ".error" + extraSelector);
}
function findErrorMessages(hud, text, extraSelector = "") {
return findMessagesByType(hud, text, ".error" + extraSelector);
}
function findWarningMessage(hud, text, extraSelector = "") {
return findMessageByType(hud, text, ".warn" + extraSelector);
}
function findWarningMessages(hud, text, extraSelector = "") {
return findMessagesByType(hud, text, ".warn" + extraSelector);
}
function findConsoleAPIMessage(hud, text, extraSelector = "") {
return findMessageByType(hud, text, ".console-api" + extraSelector);
}
function findConsoleAPIMessages(hud, text, extraSelector = "") {
return findMessagesByType(hud, text, ".console-api" + extraSelector);
}
function findNetworkMessage(hud, text, extraSelector = "") {
return findMessageByType(hud, text, ".network" + extraSelector);
}
function findNetworkMessages(hud, text, extraSelector = "") {
return findMessagesByType(hud, text, ".network" + extraSelector);
}
|