145 lines
5.2 KiB
JavaScript
145 lines
5.2 KiB
JavaScript
/*
|
|
Copyright 2018 New Vector Ltd
|
|
Copyright 2019 The Matrix.org Foundation C.I.C.
|
|
|
|
Licensed under the Apache License, Version 2.0 (the "License");
|
|
you may not use this file except in compliance with the License.
|
|
You may obtain a copy of the License at
|
|
|
|
http://www.apache.org/licenses/LICENSE-2.0
|
|
|
|
Unless required by applicable law or agreed to in writing, software
|
|
distributed under the License is distributed on an "AS IS" BASIS,
|
|
WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
See the License for the specific language governing permissions and
|
|
limitations under the License.
|
|
*/
|
|
|
|
const assert = require('assert');
|
|
|
|
module.exports.scrollToTimelineTop = async function(session) {
|
|
session.log.step(`scrolls to the top of the timeline`);
|
|
await session.page.evaluate(() => {
|
|
return Promise.resolve().then(async () => {
|
|
let timedOut = false;
|
|
let timeoutHandle = null;
|
|
// set scrollTop to 0 in a loop and check every 50ms
|
|
// if content became available (scrollTop not being 0 anymore),
|
|
// assume everything is loaded after 3s
|
|
do {
|
|
const timelineScrollView = document.querySelector(".mx_RoomView_timeline .mx_ScrollPanel");
|
|
if (timelineScrollView && timelineScrollView.scrollTop !== 0) {
|
|
if (timeoutHandle) {
|
|
clearTimeout(timeoutHandle);
|
|
}
|
|
timeoutHandle = setTimeout(() => timedOut = true, 3000);
|
|
timelineScrollView.scrollTop = 0;
|
|
} else {
|
|
await new Promise((resolve) => setTimeout(resolve, 50));
|
|
}
|
|
} while (!timedOut);
|
|
});
|
|
});
|
|
session.log.done();
|
|
};
|
|
|
|
module.exports.receiveMessage = async function(session, expectedMessage) {
|
|
session.log.step(`receives message "${expectedMessage.body}" from ${expectedMessage.sender}`);
|
|
// wait for a response to come in that contains the message
|
|
// crude, but effective
|
|
|
|
async function getLastMessage() {
|
|
const lastTile = await getLastEventTile(session);
|
|
return getMessageFromEventTile(lastTile);
|
|
}
|
|
|
|
let lastMessage;
|
|
await session.poll(async () => {
|
|
try {
|
|
lastMessage = await getLastMessage();
|
|
} catch (err) {
|
|
return false;
|
|
}
|
|
// stop polling when found the expected message
|
|
return lastMessage &&
|
|
lastMessage.body === expectedMessage.body &&
|
|
lastMessage.sender === expectedMessage.sender;
|
|
});
|
|
assertMessage(lastMessage, expectedMessage);
|
|
session.log.done();
|
|
};
|
|
|
|
|
|
module.exports.checkTimelineContains = async function(session, expectedMessages, sendersDescription) {
|
|
session.log.step(`checks timeline contains ${expectedMessages.length} ` +
|
|
`given messages${sendersDescription ? ` from ${sendersDescription}`:""}`);
|
|
const eventTiles = await getAllEventTiles(session);
|
|
let timelineMessages = await Promise.all(eventTiles.map((eventTile) => {
|
|
return getMessageFromEventTile(eventTile);
|
|
}));
|
|
//filter out tiles that were not messages
|
|
timelineMessages = timelineMessages.filter((m) => !!m);
|
|
timelineMessages.reduce((prevSender, m) => {
|
|
if (m.continuation) {
|
|
m.sender = prevSender;
|
|
return prevSender;
|
|
} else {
|
|
return m.sender;
|
|
}
|
|
});
|
|
|
|
expectedMessages.forEach((expectedMessage) => {
|
|
const foundMessage = timelineMessages.find((message) => {
|
|
return message.sender === expectedMessage.sender &&
|
|
message.body === expectedMessage.body;
|
|
});
|
|
try {
|
|
assertMessage(foundMessage, expectedMessage);
|
|
} catch (err) {
|
|
console.log("timelineMessages", timelineMessages);
|
|
throw err;
|
|
}
|
|
});
|
|
|
|
session.log.done();
|
|
};
|
|
|
|
function assertMessage(foundMessage, expectedMessage) {
|
|
assert(foundMessage, `message ${JSON.stringify(expectedMessage)} not found in timeline`);
|
|
assert.equal(foundMessage.body, expectedMessage.body);
|
|
assert.equal(foundMessage.sender, expectedMessage.sender);
|
|
if (expectedMessage.hasOwnProperty("encrypted")) {
|
|
assert.equal(foundMessage.encrypted, expectedMessage.encrypted);
|
|
}
|
|
}
|
|
|
|
function getLastEventTile(session) {
|
|
return session.query(".mx_EventTile_last");
|
|
}
|
|
|
|
function getAllEventTiles(session) {
|
|
return session.queryAll(".mx_RoomView_MessageList .mx_EventTile");
|
|
}
|
|
|
|
async function getMessageFromEventTile(eventTile) {
|
|
const senderElement = await eventTile.$(".mx_SenderProfile_name");
|
|
const className = await (await eventTile.getProperty("className")).jsonValue();
|
|
const classNames = className.split(" ");
|
|
const bodyElement = await eventTile.$(".mx_EventTile_body");
|
|
let sender = null;
|
|
if (senderElement) {
|
|
sender = await(await senderElement.getProperty("innerText")).jsonValue();
|
|
}
|
|
if (!bodyElement) {
|
|
return null;
|
|
}
|
|
const body = await(await bodyElement.getProperty("innerText")).jsonValue();
|
|
|
|
return {
|
|
sender,
|
|
body,
|
|
encrypted: classNames.includes("mx_EventTile_verified"),
|
|
continuation: classNames.includes("mx_EventTile_continuation"),
|
|
};
|
|
}
|