mirror of
https://github.com/vector-im/element-web.git
synced 2024-11-30 06:21:09 +08:00
Merge pull request #22 from matrix-org/bwindels/lltests3
Test LL with gappy syncs
This commit is contained in:
commit
f26c50cdaa
@ -27,10 +27,10 @@ module.exports = class RestSessionCreator {
|
|||||||
this.cwd = cwd;
|
this.cwd = cwd;
|
||||||
}
|
}
|
||||||
|
|
||||||
async createSessionRange(usernames, password) {
|
async createSessionRange(usernames, password, groupName) {
|
||||||
const sessionPromises = usernames.map((username) => this.createSession(username, password));
|
const sessionPromises = usernames.map((username) => this.createSession(username, password));
|
||||||
const sessions = await Promise.all(sessionPromises);
|
const sessions = await Promise.all(sessionPromises);
|
||||||
return new RestMultiSession(sessions);
|
return new RestMultiSession(sessions, groupName);
|
||||||
}
|
}
|
||||||
|
|
||||||
async createSession(username, password) {
|
async createSession(username, password) {
|
||||||
|
@ -17,14 +17,16 @@ limitations under the License.
|
|||||||
const request = require('request-promise-native');
|
const request = require('request-promise-native');
|
||||||
const RestRoom = require('./room');
|
const RestRoom = require('./room');
|
||||||
const {approveConsent} = require('./consent');
|
const {approveConsent} = require('./consent');
|
||||||
|
const Logger = require('../logger');
|
||||||
|
|
||||||
module.exports = class RestMultiSession {
|
module.exports = class RestMultiSession {
|
||||||
constructor(sessions) {
|
constructor(sessions, groupName) {
|
||||||
|
this.log = new Logger(groupName);
|
||||||
this.sessions = sessions;
|
this.sessions = sessions;
|
||||||
}
|
}
|
||||||
|
|
||||||
slice(start, end) {
|
slice(groupName, start, end) {
|
||||||
return new RestMultiSession(this.sessions.slice(start, end));
|
return new RestMultiSession(this.sessions.slice(start, end), groupName);
|
||||||
}
|
}
|
||||||
|
|
||||||
pop(userName) {
|
pop(userName) {
|
||||||
@ -37,25 +39,52 @@ module.exports = class RestMultiSession {
|
|||||||
}
|
}
|
||||||
|
|
||||||
async setDisplayName(fn) {
|
async setDisplayName(fn) {
|
||||||
await Promise.all(this.sessions.map((s) => s.setDisplayName(fn(s))));
|
this.log.step("set their display name")
|
||||||
|
await Promise.all(this.sessions.map(async (s) => {
|
||||||
|
s.log.mute();
|
||||||
|
await s.setDisplayName(fn(s));
|
||||||
|
s.log.unmute();
|
||||||
|
}));
|
||||||
|
this.log.done();
|
||||||
}
|
}
|
||||||
|
|
||||||
async join(roomId) {
|
async join(roomIdOrAlias) {
|
||||||
const rooms = await Promise.all(this.sessions.map((s) => s.join(roomId)));
|
this.log.step(`join ${roomIdOrAlias}`)
|
||||||
return new RestMultiRoom(rooms);
|
const rooms = await Promise.all(this.sessions.map(async (s) => {
|
||||||
|
s.log.mute();
|
||||||
|
const room = await s.join(roomIdOrAlias);
|
||||||
|
s.log.unmute();
|
||||||
|
return room;
|
||||||
|
}));
|
||||||
|
this.log.done();
|
||||||
|
return new RestMultiRoom(rooms, roomIdOrAlias, this.log);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
class RestMultiRoom {
|
class RestMultiRoom {
|
||||||
constructor(rooms) {
|
constructor(rooms, roomIdOrAlias, log) {
|
||||||
this.rooms = rooms;
|
this.rooms = rooms;
|
||||||
|
this.roomIdOrAlias = roomIdOrAlias;
|
||||||
|
this.log = log;
|
||||||
}
|
}
|
||||||
|
|
||||||
async talk(message) {
|
async talk(message) {
|
||||||
await Promise.all(this.rooms.map((r) => r.talk(message)));
|
this.log.step(`say "${message}" in ${this.roomIdOrAlias}`)
|
||||||
|
await Promise.all(this.rooms.map(async (r) => {
|
||||||
|
r.log.mute();
|
||||||
|
await r.talk(message);
|
||||||
|
r.log.unmute();
|
||||||
|
}));
|
||||||
|
this.log.done();
|
||||||
}
|
}
|
||||||
|
|
||||||
async leave() {
|
async leave() {
|
||||||
await Promise.all(this.rooms.map((r) => r.leave()));
|
this.log.step(`leave ${this.roomIdOrAlias}`)
|
||||||
|
await Promise.all(this.rooms.map(async (r) => {
|
||||||
|
r.log.mute();
|
||||||
|
await r.leave(message);
|
||||||
|
r.log.unmute();
|
||||||
|
}));
|
||||||
|
this.log.done();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -18,22 +18,27 @@ const uuidv4 = require('uuid/v4');
|
|||||||
|
|
||||||
/* no pun intented */
|
/* no pun intented */
|
||||||
module.exports = class RestRoom {
|
module.exports = class RestRoom {
|
||||||
constructor(session, roomId) {
|
constructor(session, roomId, log) {
|
||||||
this.session = session;
|
this.session = session;
|
||||||
this._roomId = roomId;
|
this._roomId = roomId;
|
||||||
|
this.log = log;
|
||||||
}
|
}
|
||||||
|
|
||||||
async talk(message) {
|
async talk(message) {
|
||||||
|
this.log.step(`says "${message}" in ${this._roomId}`)
|
||||||
const txId = uuidv4();
|
const txId = uuidv4();
|
||||||
await this.session._put(`/rooms/${this._roomId}/send/m.room.message/${txId}`, {
|
await this.session._put(`/rooms/${this._roomId}/send/m.room.message/${txId}`, {
|
||||||
"msgtype": "m.text",
|
"msgtype": "m.text",
|
||||||
"body": message
|
"body": message
|
||||||
});
|
});
|
||||||
|
this.log.done();
|
||||||
return txId;
|
return txId;
|
||||||
}
|
}
|
||||||
|
|
||||||
async leave() {
|
async leave() {
|
||||||
|
this.log.step(`leaves ${this._roomId}`)
|
||||||
await this.session._post(`/rooms/${this._roomId}/leave`);
|
await this.session._post(`/rooms/${this._roomId}/leave`);
|
||||||
|
this.log.done();
|
||||||
}
|
}
|
||||||
|
|
||||||
roomId() {
|
roomId() {
|
||||||
|
@ -15,11 +15,13 @@ limitations under the License.
|
|||||||
*/
|
*/
|
||||||
|
|
||||||
const request = require('request-promise-native');
|
const request = require('request-promise-native');
|
||||||
|
const Logger = require('../logger');
|
||||||
const RestRoom = require('./room');
|
const RestRoom = require('./room');
|
||||||
const {approveConsent} = require('./consent');
|
const {approveConsent} = require('./consent');
|
||||||
|
|
||||||
module.exports = class RestSession {
|
module.exports = class RestSession {
|
||||||
constructor(credentials) {
|
constructor(credentials) {
|
||||||
|
this.log = new Logger(credentials.userId);
|
||||||
this._credentials = credentials;
|
this._credentials = credentials;
|
||||||
this._displayName = null;
|
this._displayName = null;
|
||||||
}
|
}
|
||||||
@ -37,18 +39,23 @@ module.exports = class RestSession {
|
|||||||
}
|
}
|
||||||
|
|
||||||
async setDisplayName(displayName) {
|
async setDisplayName(displayName) {
|
||||||
|
this.log.step(`sets their display name to ${displayName}`);
|
||||||
this._displayName = displayName;
|
this._displayName = displayName;
|
||||||
await this._put(`/profile/${this._credentials.userId}/displayname`, {
|
await this._put(`/profile/${this._credentials.userId}/displayname`, {
|
||||||
displayname: displayName
|
displayname: displayName
|
||||||
});
|
});
|
||||||
|
this.log.done();
|
||||||
}
|
}
|
||||||
|
|
||||||
async join(roomIdOrAlias) {
|
async join(roomIdOrAlias) {
|
||||||
|
this.log.step(`joins ${roomIdOrAlias}`);
|
||||||
const {room_id} = await this._post(`/join/${encodeURIComponent(roomIdOrAlias)}`);
|
const {room_id} = await this._post(`/join/${encodeURIComponent(roomIdOrAlias)}`);
|
||||||
return new RestRoom(this, room_id);
|
this.log.done();
|
||||||
|
return new RestRoom(this, room_id, this.log);
|
||||||
}
|
}
|
||||||
|
|
||||||
async createRoom(name, options) {
|
async createRoom(name, options) {
|
||||||
|
this.log.step(`creates room ${name}`);
|
||||||
const body = {
|
const body = {
|
||||||
name,
|
name,
|
||||||
};
|
};
|
||||||
@ -68,7 +75,8 @@ module.exports = class RestSession {
|
|||||||
}
|
}
|
||||||
|
|
||||||
const {room_id} = await this._post(`/createRoom`, body);
|
const {room_id} = await this._post(`/createRoom`, body);
|
||||||
return new RestRoom(this, room_id);
|
this.log.done();
|
||||||
|
return new RestRoom(this, room_id, this.log);
|
||||||
}
|
}
|
||||||
|
|
||||||
_post(csApiPath, body) {
|
_post(csApiPath, body) {
|
||||||
|
@ -41,7 +41,7 @@ module.exports = async function scenario(createSession, restCreator) {
|
|||||||
|
|
||||||
async function createRestUsers(restCreator) {
|
async function createRestUsers(restCreator) {
|
||||||
const usernames = range(1, 10).map((i) => `charly-${i}`);
|
const usernames = range(1, 10).map((i) => `charly-${i}`);
|
||||||
const charlies = await restCreator.createSessionRange(usernames, 'testtest');
|
const charlies = await restCreator.createSessionRange(usernames, "testtest", "charly-1..10");
|
||||||
await charlies.setDisplayName((s) => `Charly #${s.userName().split('-')[1]}`);
|
await charlies.setDisplayName((s) => `Charly #${s.userName().split('-')[1]}`);
|
||||||
return charlies;
|
return charlies;
|
||||||
}
|
}
|
||||||
|
@ -31,9 +31,15 @@ const assert = require('assert');
|
|||||||
module.exports = async function lazyLoadingScenarios(alice, bob, charlies) {
|
module.exports = async function lazyLoadingScenarios(alice, bob, charlies) {
|
||||||
console.log(" creating a room for lazy loading member scenarios:");
|
console.log(" creating a room for lazy loading member scenarios:");
|
||||||
await enableLazyLoading(alice);
|
await enableLazyLoading(alice);
|
||||||
await setupRoomWithBobAliceAndCharlies(alice, bob, charlies);
|
const charly1to5 = charlies.slice("charly-1..5", 0, 5);
|
||||||
await checkPaginatedDisplayNames(alice, charlies);
|
const charly6to10 = charlies.slice("charly-6..10", 5);
|
||||||
await checkMemberList(alice, charlies);
|
assert(charly1to5.sessions.length, 5);
|
||||||
|
assert(charly6to10.sessions.length, 5);
|
||||||
|
await setupRoomWithBobAliceAndCharlies(alice, bob, charly1to5);
|
||||||
|
await checkPaginatedDisplayNames(alice, charly1to5);
|
||||||
|
await checkMemberList(alice, charly1to5);
|
||||||
|
await joinCharliesWhileAliceIsOffline(alice, charly6to10);
|
||||||
|
await checkMemberList(alice, charly6to10);
|
||||||
}
|
}
|
||||||
|
|
||||||
const room = "Lazy Loading Test";
|
const room = "Lazy Loading Test";
|
||||||
@ -70,11 +76,11 @@ async function checkPaginatedDisplayNames(alice, charlies) {
|
|||||||
});
|
});
|
||||||
}, messages);
|
}, messages);
|
||||||
}, []);
|
}, []);
|
||||||
await checkTimelineContains(alice, expectedMessages, "Charly #1-10");
|
await checkTimelineContains(alice, expectedMessages, charlies.log.username);
|
||||||
}
|
}
|
||||||
|
|
||||||
async function checkMemberList(alice, charlies) {
|
async function checkMemberList(alice, charlies) {
|
||||||
alice.log.step("checks the memberlist contains herself, bob and all charlies");
|
alice.log.step(`checks the memberlist contains herself, bob and ${charlies.log.username}`);
|
||||||
const displayNames = (await getMembersInMemberlist(alice)).map((m) => m.displayName);
|
const displayNames = (await getMembersInMemberlist(alice)).map((m) => m.displayName);
|
||||||
assert(displayNames.includes("alice"));
|
assert(displayNames.includes("alice"));
|
||||||
assert(displayNames.includes("bob"));
|
assert(displayNames.includes("bob"));
|
||||||
@ -85,3 +91,19 @@ async function checkMemberList(alice, charlies) {
|
|||||||
});
|
});
|
||||||
alice.log.done();
|
alice.log.done();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
async function joinCharliesWhileAliceIsOffline(alice, charly6to10) {
|
||||||
|
await alice.setOffline(true);
|
||||||
|
await delay(1000);
|
||||||
|
const members6to10 = await charly6to10.join(alias);
|
||||||
|
const member6 = members6to10.rooms[0];
|
||||||
|
member6.log.step("sends 20 messages").mute();
|
||||||
|
for(let i = 20; i >= 1; --i) {
|
||||||
|
await member6.talk("where is charly?");
|
||||||
|
}
|
||||||
|
member6.log.unmute().done();
|
||||||
|
const catchupPromise = alice.waitForNextSuccessfulSync();
|
||||||
|
await alice.setOffline(false);
|
||||||
|
await catchupPromise;
|
||||||
|
await delay(2000);
|
||||||
|
}
|
||||||
|
@ -161,6 +161,33 @@ module.exports = class RiotSession {
|
|||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
|
waitForSyncResponseWith(predicate) {
|
||||||
|
return this.page.waitForResponse(async (response) => {
|
||||||
|
if (response.request().url().indexOf("/sync") === -1) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
return predicate(response);
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
/** wait for a /sync request started after this call that gets a 200 response */
|
||||||
|
async waitForNextSuccessfulSync() {
|
||||||
|
const syncUrls = [];
|
||||||
|
function onRequest(request) {
|
||||||
|
if (request.url().indexOf("/sync") !== -1) {
|
||||||
|
syncUrls.push(request.url());
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
this.page.on('request', onRequest);
|
||||||
|
|
||||||
|
await this.page.waitForResponse((response) => {
|
||||||
|
return syncUrls.includes(response.request().url()) && response.status() === 200;
|
||||||
|
});
|
||||||
|
|
||||||
|
this.page.removeListener('request', onRequest);
|
||||||
|
}
|
||||||
|
|
||||||
goto(url) {
|
goto(url) {
|
||||||
return this.page.goto(url);
|
return this.page.goto(url);
|
||||||
}
|
}
|
||||||
@ -173,6 +200,13 @@ module.exports = class RiotSession {
|
|||||||
return delay(ms);
|
return delay(ms);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
async setOffline(enabled) {
|
||||||
|
const description = enabled ? "offline" : "back online";
|
||||||
|
this.log.step(`goes ${description}`);
|
||||||
|
await this.page.setOfflineMode(enabled);
|
||||||
|
this.log.done();
|
||||||
|
}
|
||||||
|
|
||||||
close() {
|
close() {
|
||||||
return this.browser.close();
|
return this.browser.close();
|
||||||
}
|
}
|
||||||
|
@ -64,10 +64,7 @@ module.exports.receiveMessage = async function(session, expectedMessage) {
|
|||||||
if (isExpectedMessage) {
|
if (isExpectedMessage) {
|
||||||
assertMessage(lastMessage, expectedMessage);
|
assertMessage(lastMessage, expectedMessage);
|
||||||
} else {
|
} else {
|
||||||
await session.page.waitForResponse(async (response) => {
|
await session.waitForSyncResponseWith(async (response) => {
|
||||||
if (response.request().url().indexOf("/sync") === -1) {
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
const body = await response.text();
|
const body = await response.text();
|
||||||
if (expectedMessage.encrypted) {
|
if (expectedMessage.encrypted) {
|
||||||
return body.indexOf(expectedMessage.sender) !== -1 &&
|
return body.indexOf(expectedMessage.sender) !== -1 &&
|
||||||
|
@ -1,6 +1,6 @@
|
|||||||
#!/bin/bash
|
#!/bin/bash
|
||||||
# config
|
# config
|
||||||
SYNAPSE_BRANCH=master
|
SYNAPSE_BRANCH=develop
|
||||||
INSTALLATION_NAME=consent
|
INSTALLATION_NAME=consent
|
||||||
SERVER_DIR=installations/$INSTALLATION_NAME
|
SERVER_DIR=installations/$INSTALLATION_NAME
|
||||||
CONFIG_TEMPLATE=consent
|
CONFIG_TEMPLATE=consent
|
||||||
|
Loading…
Reference in New Issue
Block a user