2022-05-10 19:09:31 +02:00
|
|
|
/*
|
|
|
|
Copyright 2022 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.
|
|
|
|
*/
|
|
|
|
|
|
|
|
/// <reference types="cypress" />
|
|
|
|
|
2022-07-04 21:37:48 +02:00
|
|
|
import type { FileType, UploadContentResponseType } from "matrix-js-sdk/src/http-api";
|
|
|
|
import type { IAbortablePromise } from "matrix-js-sdk/src/@types/partials";
|
|
|
|
import type { ICreateRoomOpts, ISendEventResponse, IUploadOpts } from "matrix-js-sdk/src/@types/requests";
|
2022-05-10 19:09:31 +02:00
|
|
|
import type { MatrixClient } from "matrix-js-sdk/src/client";
|
|
|
|
import type { Room } from "matrix-js-sdk/src/models/room";
|
2022-07-04 21:37:48 +02:00
|
|
|
import type { IContent } from "matrix-js-sdk/src/models/event";
|
2022-05-10 19:09:31 +02:00
|
|
|
import Chainable = Cypress.Chainable;
|
|
|
|
|
|
|
|
declare global {
|
|
|
|
// eslint-disable-next-line @typescript-eslint/no-namespace
|
|
|
|
namespace Cypress {
|
|
|
|
interface Chainable {
|
|
|
|
/**
|
|
|
|
* Returns the MatrixClient from the MatrixClientPeg
|
|
|
|
*/
|
|
|
|
getClient(): Chainable<MatrixClient | undefined>;
|
|
|
|
/**
|
|
|
|
* Create a room with given options.
|
|
|
|
* @param options the options to apply when creating the room
|
|
|
|
* @return the ID of the newly created room
|
|
|
|
*/
|
|
|
|
createRoom(options: ICreateRoomOpts): Chainable<string>;
|
2022-05-26 11:19:00 +02:00
|
|
|
/**
|
|
|
|
* Create a space with given options.
|
|
|
|
* @param options the options to apply when creating the space
|
|
|
|
* @return the ID of the newly created space (room)
|
|
|
|
*/
|
|
|
|
createSpace(options: ICreateRoomOpts): Chainable<string>;
|
2022-05-10 19:09:31 +02:00
|
|
|
/**
|
|
|
|
* Invites the given user to the given room.
|
|
|
|
* @param roomId the id of the room to invite to
|
|
|
|
* @param userId the id of the user to invite
|
|
|
|
*/
|
|
|
|
inviteUser(roomId: string, userId: string): Chainable<{}>;
|
2022-06-10 17:24:15 +02:00
|
|
|
/**
|
|
|
|
* Sets account data for the user.
|
|
|
|
* @param type The type of account data.
|
|
|
|
* @param data The data to store.
|
|
|
|
*/
|
|
|
|
setAccountData(type: string, data: object): Chainable<{}>;
|
2022-07-04 21:37:48 +02:00
|
|
|
/**
|
|
|
|
* @param {string} roomId
|
|
|
|
* @param {string} threadId
|
|
|
|
* @param {string} eventType
|
|
|
|
* @param {Object} content
|
|
|
|
* @return {module:http-api.MatrixError} Rejects: with an error response.
|
|
|
|
*/
|
|
|
|
sendEvent(
|
|
|
|
roomId: string,
|
|
|
|
threadId: string | null,
|
|
|
|
eventType: string,
|
|
|
|
content: IContent
|
|
|
|
): Chainable<ISendEventResponse>;
|
|
|
|
/**
|
|
|
|
* @param {string} name
|
|
|
|
* @param {module:client.callback} callback Optional.
|
|
|
|
* @return {Promise} Resolves: {} an empty object.
|
|
|
|
* @return {module:http-api.MatrixError} Rejects: with an error response.
|
|
|
|
*/
|
|
|
|
setDisplayName(name: string): Chainable<{}>;
|
|
|
|
/**
|
|
|
|
* @param {string} url
|
|
|
|
* @param {module:client.callback} callback Optional.
|
|
|
|
* @return {Promise} Resolves: {} an empty object.
|
|
|
|
* @return {module:http-api.MatrixError} Rejects: with an error response.
|
|
|
|
*/
|
|
|
|
setAvatarUrl(url: string): Chainable<{}>;
|
|
|
|
/**
|
|
|
|
* Upload a file to the media repository on the homeserver.
|
|
|
|
*
|
|
|
|
* @param {object} file The object to upload. On a browser, something that
|
|
|
|
* can be sent to XMLHttpRequest.send (typically a File). Under node.js,
|
|
|
|
* a a Buffer, String or ReadStream.
|
|
|
|
*/
|
|
|
|
uploadContent<O extends IUploadOpts>(
|
|
|
|
file: FileType,
|
|
|
|
opts?: O,
|
|
|
|
): IAbortablePromise<UploadContentResponseType<O>>;
|
|
|
|
/**
|
|
|
|
* Turn an MXC URL into an HTTP one. <strong>This method is experimental and
|
|
|
|
* may change.</strong>
|
|
|
|
* @param {string} mxcUrl The MXC URL
|
|
|
|
* @param {Number} width The desired width of the thumbnail.
|
|
|
|
* @param {Number} height The desired height of the thumbnail.
|
|
|
|
* @param {string} resizeMethod The thumbnail resize method to use, either
|
|
|
|
* "crop" or "scale".
|
|
|
|
* @param {Boolean} allowDirectLinks If true, return any non-mxc URLs
|
|
|
|
* directly. Fetching such URLs will leak information about the user to
|
|
|
|
* anyone they share a room with. If false, will return null for such URLs.
|
|
|
|
* @return {?string} the avatar URL or null.
|
|
|
|
*/
|
|
|
|
mxcUrlToHttp(
|
|
|
|
mxcUrl: string,
|
|
|
|
width?: number,
|
|
|
|
height?: number,
|
|
|
|
resizeMethod?: string,
|
|
|
|
allowDirectLinks?: boolean,
|
|
|
|
): string | null;
|
2022-06-30 11:44:56 +02:00
|
|
|
/**
|
|
|
|
* Gets the list of DMs with a given user
|
|
|
|
* @param userId The ID of the user
|
|
|
|
* @return the list of DMs with that user
|
|
|
|
*/
|
|
|
|
getDmRooms(userId: string): Chainable<string[]>;
|
2022-06-30 10:59:25 +02:00
|
|
|
/**
|
|
|
|
* Boostraps cross-signing.
|
|
|
|
*/
|
|
|
|
bootstrapCrossSigning(): Chainable<void>;
|
2022-05-10 19:09:31 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
Cypress.Commands.add("getClient", (): Chainable<MatrixClient | undefined> => {
|
2022-05-17 16:38:45 +02:00
|
|
|
return cy.window({ log: false }).then(win => win.mxMatrixClientPeg.matrixClient);
|
2022-05-10 19:09:31 +02:00
|
|
|
});
|
|
|
|
|
2022-06-30 11:44:56 +02:00
|
|
|
Cypress.Commands.add("getDmRooms", (userId: string): Chainable<string[]> => {
|
|
|
|
return cy.getClient()
|
|
|
|
.then(cli => cli.getAccountData("m.direct")?.getContent<Record<string, string[]>>())
|
|
|
|
.then(dmRoomMap => dmRoomMap[userId] ?? []);
|
|
|
|
});
|
|
|
|
|
2022-05-10 19:09:31 +02:00
|
|
|
Cypress.Commands.add("createRoom", (options: ICreateRoomOpts): Chainable<string> => {
|
2022-05-17 16:38:45 +02:00
|
|
|
return cy.window({ log: false }).then(async win => {
|
2022-05-10 19:09:31 +02:00
|
|
|
const cli = win.mxMatrixClientPeg.matrixClient;
|
|
|
|
const resp = await cli.createRoom(options);
|
|
|
|
const roomId = resp.room_id;
|
|
|
|
|
|
|
|
if (!cli.getRoom(roomId)) {
|
|
|
|
await new Promise<void>(resolve => {
|
|
|
|
const onRoom = (room: Room) => {
|
|
|
|
if (room.roomId === roomId) {
|
|
|
|
cli.off(win.matrixcs.ClientEvent.Room, onRoom);
|
|
|
|
resolve();
|
|
|
|
}
|
|
|
|
};
|
|
|
|
cli.on(win.matrixcs.ClientEvent.Room, onRoom);
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
return roomId;
|
|
|
|
});
|
|
|
|
});
|
|
|
|
|
2022-05-26 11:19:00 +02:00
|
|
|
Cypress.Commands.add("createSpace", (options: ICreateRoomOpts): Chainable<string> => {
|
|
|
|
return cy.createRoom({
|
|
|
|
...options,
|
|
|
|
creation_content: {
|
|
|
|
"type": "m.space",
|
|
|
|
},
|
|
|
|
});
|
|
|
|
});
|
|
|
|
|
2022-05-10 19:09:31 +02:00
|
|
|
Cypress.Commands.add("inviteUser", (roomId: string, userId: string): Chainable<{}> => {
|
|
|
|
return cy.getClient().then(async (cli: MatrixClient) => {
|
|
|
|
return cli.invite(roomId, userId);
|
|
|
|
});
|
|
|
|
});
|
2022-06-10 17:24:15 +02:00
|
|
|
|
|
|
|
Cypress.Commands.add("setAccountData", (type: string, data: object): Chainable<{}> => {
|
|
|
|
return cy.getClient().then(async (cli: MatrixClient) => {
|
|
|
|
return cli.setAccountData(type, data);
|
|
|
|
});
|
|
|
|
});
|
2022-06-30 10:59:25 +02:00
|
|
|
|
2022-07-04 21:37:48 +02:00
|
|
|
Cypress.Commands.add("sendEvent", (
|
|
|
|
roomId: string,
|
|
|
|
threadId: string | null,
|
|
|
|
eventType: string,
|
|
|
|
content: IContent,
|
|
|
|
): Chainable<ISendEventResponse> => {
|
|
|
|
return cy.getClient().then(async (cli: MatrixClient) => {
|
|
|
|
return cli.sendEvent(roomId, threadId, eventType, content);
|
|
|
|
});
|
|
|
|
});
|
|
|
|
|
|
|
|
Cypress.Commands.add("setDisplayName", (name: string): Chainable<{}> => {
|
|
|
|
return cy.getClient().then(async (cli: MatrixClient) => {
|
|
|
|
return cli.setDisplayName(name);
|
|
|
|
});
|
|
|
|
});
|
|
|
|
|
|
|
|
Cypress.Commands.add("uploadContent", (file: FileType): Chainable<{}> => {
|
|
|
|
return cy.getClient().then(async (cli: MatrixClient) => {
|
|
|
|
return cli.uploadContent(file);
|
|
|
|
});
|
|
|
|
});
|
|
|
|
|
|
|
|
Cypress.Commands.add("setAvatarUrl", (url: string): Chainable<{}> => {
|
|
|
|
return cy.getClient().then(async (cli: MatrixClient) => {
|
|
|
|
return cli.setAvatarUrl(url);
|
|
|
|
});
|
|
|
|
});
|
|
|
|
|
2022-06-30 10:59:25 +02:00
|
|
|
Cypress.Commands.add("bootstrapCrossSigning", () => {
|
|
|
|
cy.window({ log: false }).then(win => {
|
|
|
|
win.mxMatrixClientPeg.matrixClient.bootstrapCrossSigning({
|
|
|
|
authUploadDeviceSigningKeys: async func => { await func({}); },
|
|
|
|
});
|
|
|
|
});
|
|
|
|
});
|