2020-09-29 23:14:51 +03:00
|
|
|
/*
|
|
|
|
* Copyright 2020 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.
|
|
|
|
*/
|
|
|
|
|
2020-11-18 06:38:59 +03:00
|
|
|
import {
|
|
|
|
Capability,
|
2020-11-26 00:40:01 +03:00
|
|
|
EventDirection,
|
2020-11-24 00:10:14 +03:00
|
|
|
IOpenIDCredentials,
|
|
|
|
IOpenIDUpdate,
|
2020-11-18 06:38:59 +03:00
|
|
|
ISendEventDetails,
|
2020-11-19 21:24:17 +03:00
|
|
|
MatrixCapabilities,
|
2020-11-24 00:10:14 +03:00
|
|
|
OpenIDRequestState,
|
|
|
|
SimpleObservable,
|
2020-11-19 21:24:17 +03:00
|
|
|
Widget,
|
2020-11-18 06:38:59 +03:00
|
|
|
WidgetDriver,
|
2020-11-26 00:40:01 +03:00
|
|
|
WidgetEventCapability,
|
2020-11-19 21:24:17 +03:00
|
|
|
WidgetKind,
|
2020-11-18 06:38:59 +03:00
|
|
|
} from "matrix-widget-api";
|
|
|
|
import { iterableDiff, iterableUnion } from "../../utils/iterables";
|
2020-11-03 01:17:05 +03:00
|
|
|
import { MatrixClientPeg } from "../../MatrixClientPeg";
|
2020-11-03 07:32:49 +03:00
|
|
|
import ActiveRoomObserver from "../../ActiveRoomObserver";
|
2020-11-18 06:38:59 +03:00
|
|
|
import Modal from "../../Modal";
|
2020-11-24 00:10:14 +03:00
|
|
|
import WidgetOpenIDPermissionsDialog from "../../components/views/dialogs/WidgetOpenIDPermissionsDialog";
|
2020-11-19 22:12:00 +03:00
|
|
|
import WidgetCapabilitiesPromptDialog, {
|
|
|
|
getRememberedCapabilitiesForWidget,
|
|
|
|
} from "../../components/views/dialogs/WidgetCapabilitiesPromptDialog";
|
2020-11-23 21:59:38 +03:00
|
|
|
import { WidgetPermissionCustomisations } from "../../customisations/WidgetPermissions";
|
2020-11-26 04:39:11 +03:00
|
|
|
import { OIDCState, WidgetPermissionStore } from "./WidgetPermissionStore";
|
2020-11-26 00:40:01 +03:00
|
|
|
import { WidgetType } from "../../widgets/WidgetType";
|
|
|
|
import { EventType } from "matrix-js-sdk/src/@types/event";
|
2020-09-29 23:14:51 +03:00
|
|
|
|
|
|
|
// TODO: Purge this from the universe
|
|
|
|
|
|
|
|
export class StopGapWidgetDriver extends WidgetDriver {
|
2020-11-18 06:38:59 +03:00
|
|
|
private allowedCapabilities: Set<Capability>;
|
|
|
|
|
2020-11-19 21:24:17 +03:00
|
|
|
// TODO: Refactor widgetKind into the Widget class
|
2020-11-26 04:39:11 +03:00
|
|
|
constructor(
|
|
|
|
allowedCapabilities: Capability[],
|
|
|
|
private forWidget: Widget,
|
|
|
|
private forWidgetKind: WidgetKind,
|
|
|
|
private inRoomId?: string,
|
|
|
|
) {
|
2020-09-29 23:14:51 +03:00
|
|
|
super();
|
2020-11-18 06:38:59 +03:00
|
|
|
|
|
|
|
// Always allow screenshots to be taken because it's a client-induced flow. The widget can't
|
|
|
|
// spew screenshots at us and can't request screenshots of us, so it's up to us to provide the
|
|
|
|
// button if the widget says it supports screenshots.
|
|
|
|
this.allowedCapabilities = new Set([...allowedCapabilities, MatrixCapabilities.Screenshots]);
|
2020-11-26 00:40:01 +03:00
|
|
|
|
|
|
|
// Grant the permissions that are specific to given widget types
|
|
|
|
if (WidgetType.JITSI.matches(this.forWidget.type) && forWidgetKind === WidgetKind.Room) {
|
|
|
|
this.allowedCapabilities.add(MatrixCapabilities.AlwaysOnScreen);
|
|
|
|
} else if (WidgetType.STICKERPICKER.matches(this.forWidget.type) && forWidgetKind === WidgetKind.Account) {
|
|
|
|
const stickerSendingCap = WidgetEventCapability.forRoomEvent(EventDirection.Send, EventType.Sticker).raw;
|
|
|
|
this.allowedCapabilities.add(MatrixCapabilities.StickerSending); // legacy as far as MSC2762 is concerned
|
|
|
|
this.allowedCapabilities.add(stickerSendingCap);
|
|
|
|
}
|
2020-09-29 23:14:51 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
public async validateCapabilities(requested: Set<Capability>): Promise<Set<Capability>> {
|
2020-11-18 06:38:59 +03:00
|
|
|
// Check to see if any capabilities aren't automatically accepted (such as sticker pickers
|
|
|
|
// allowing stickers to be sent). If there are excess capabilities to be approved, the user
|
|
|
|
// will be prompted to accept them.
|
|
|
|
const diff = iterableDiff(requested, this.allowedCapabilities);
|
|
|
|
const missing = new Set(diff.removed); // "removed" is "in A (requested) but not in B (allowed)"
|
|
|
|
const allowedSoFar = new Set(this.allowedCapabilities);
|
2020-11-23 21:59:38 +03:00
|
|
|
getRememberedCapabilitiesForWidget(this.forWidget).forEach(cap => {
|
|
|
|
allowedSoFar.add(cap);
|
|
|
|
missing.delete(cap);
|
|
|
|
});
|
|
|
|
if (WidgetPermissionCustomisations.preapproveCapabilities) {
|
|
|
|
const approved = await WidgetPermissionCustomisations.preapproveCapabilities(this.forWidget, requested);
|
|
|
|
if (approved) {
|
|
|
|
approved.forEach(cap => {
|
|
|
|
allowedSoFar.add(cap);
|
|
|
|
missing.delete(cap);
|
|
|
|
});
|
|
|
|
}
|
|
|
|
}
|
2020-11-18 06:38:59 +03:00
|
|
|
// TODO: Do something when the widget requests new capabilities not yet asked for
|
|
|
|
if (missing.size > 0) {
|
|
|
|
try {
|
|
|
|
const [result] = await Modal.createTrackedDialog(
|
|
|
|
'Approve Widget Caps', '',
|
|
|
|
WidgetCapabilitiesPromptDialog,
|
|
|
|
{
|
|
|
|
requestedCapabilities: missing,
|
|
|
|
widget: this.forWidget,
|
2020-11-19 21:24:17 +03:00
|
|
|
widgetKind: this.forWidgetKind,
|
2020-11-18 06:38:59 +03:00
|
|
|
}).finished;
|
|
|
|
(result.approved || []).forEach(cap => allowedSoFar.add(cap));
|
|
|
|
} catch (e) {
|
|
|
|
console.error("Non-fatal error getting capabilities: ", e);
|
2020-11-03 07:32:49 +03:00
|
|
|
}
|
2020-11-03 01:17:05 +03:00
|
|
|
}
|
2020-11-18 06:38:59 +03:00
|
|
|
|
|
|
|
return new Set(iterableUnion(allowedSoFar, requested));
|
2020-09-29 23:14:51 +03:00
|
|
|
}
|
2020-11-03 07:32:49 +03:00
|
|
|
|
|
|
|
public async sendEvent(eventType: string, content: any, stateKey: string = null): Promise<ISendEventDetails> {
|
|
|
|
const client = MatrixClientPeg.get();
|
|
|
|
const roomId = ActiveRoomObserver.activeRoomId;
|
|
|
|
|
|
|
|
if (!client || !roomId) throw new Error("Not in a room or not attached to a client");
|
|
|
|
|
2020-11-24 00:10:14 +03:00
|
|
|
let r: { event_id: string } = null; // eslint-disable-line camelcase
|
2020-11-03 07:32:49 +03:00
|
|
|
if (stateKey !== null) {
|
|
|
|
// state event
|
|
|
|
r = await client.sendStateEvent(roomId, eventType, content, stateKey);
|
|
|
|
} else {
|
|
|
|
// message event
|
|
|
|
r = await client.sendEvent(roomId, eventType, content);
|
|
|
|
}
|
|
|
|
|
|
|
|
return {roomId, eventId: r.event_id};
|
|
|
|
}
|
2020-11-24 00:10:14 +03:00
|
|
|
|
|
|
|
public async askOpenID(observer: SimpleObservable<IOpenIDUpdate>) {
|
2020-11-26 04:39:11 +03:00
|
|
|
const oidcState = WidgetPermissionStore.instance.getOIDCState(
|
|
|
|
this.forWidget, this.forWidgetKind, this.inRoomId,
|
|
|
|
);
|
2020-11-24 00:10:14 +03:00
|
|
|
|
|
|
|
const getToken = (): Promise<IOpenIDCredentials> => {
|
|
|
|
return MatrixClientPeg.get().getOpenIdToken();
|
|
|
|
};
|
|
|
|
|
2020-11-26 04:39:11 +03:00
|
|
|
if (oidcState === OIDCState.Denied) {
|
2020-11-24 00:10:14 +03:00
|
|
|
return observer.update({state: OpenIDRequestState.Blocked});
|
|
|
|
}
|
2020-11-26 04:39:11 +03:00
|
|
|
if (oidcState === OIDCState.Allowed) {
|
2020-11-24 00:10:14 +03:00
|
|
|
return observer.update({state: OpenIDRequestState.Allowed, token: await getToken()});
|
|
|
|
}
|
|
|
|
|
|
|
|
observer.update({state: OpenIDRequestState.PendingUserConfirmation});
|
|
|
|
|
|
|
|
Modal.createTrackedDialog("OpenID widget permissions", '', WidgetOpenIDPermissionsDialog, {
|
2020-11-26 04:39:11 +03:00
|
|
|
widget: this.forWidget,
|
|
|
|
widgetKind: this.forWidgetKind,
|
|
|
|
inRoomId: this.inRoomId,
|
2020-11-24 00:10:14 +03:00
|
|
|
|
|
|
|
onFinished: async (confirm) => {
|
|
|
|
if (!confirm) {
|
|
|
|
return observer.update({state: OpenIDRequestState.Blocked});
|
|
|
|
}
|
|
|
|
|
|
|
|
return observer.update({state: OpenIDRequestState.Allowed, token: await getToken()});
|
|
|
|
},
|
|
|
|
});
|
|
|
|
}
|
2020-09-29 23:14:51 +03:00
|
|
|
}
|