Merge pull request #22 from matrix-org/bwindels/lltests3

Test LL with gappy syncs
This commit is contained in:
Bruno Windels 2018-09-17 12:14:13 +02:00 committed by GitHub
commit f26c50cdaa
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
9 changed files with 121 additions and 26 deletions

View File

@ -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) {

View File

@ -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();
} }
} }

View File

@ -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() {

View File

@ -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) {

View File

@ -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;
} }

View File

@ -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);
}

View File

@ -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();
} }

View File

@ -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 &&

View File

@ -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