We already intercepted most of the cases where no integration manager was present, though there was a bug in many components where openAll() would be called regardless of an integration manager being available. The integration manager being disabled by the user is handled in the IntegrationManager classes rather than on click because we have quite a few calls to these functions. The StickerPicker is an exception because it does slightly different behaviour. This also removes the old "no integration manager configured" state from the IntegrationManager component as it is now replaced by a dialog.
261 lines
9.7 KiB
JavaScript
261 lines
9.7 KiB
JavaScript
/*
|
|
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.
|
|
*/
|
|
|
|
import SdkConfig from '../SdkConfig';
|
|
import sdk from "../index";
|
|
import Modal from '../Modal';
|
|
import {IntegrationManagerInstance, KIND_ACCOUNT, KIND_CONFIG, KIND_HOMESERVER} from "./IntegrationManagerInstance";
|
|
import type {MatrixClient, MatrixEvent, Room} from "matrix-js-sdk";
|
|
import WidgetUtils from "../utils/WidgetUtils";
|
|
import MatrixClientPeg from "../MatrixClientPeg";
|
|
import {AutoDiscovery} from "matrix-js-sdk";
|
|
import {_t} from "../languageHandler";
|
|
import dis from "../dispatcher";
|
|
import React from 'react';
|
|
import SettingsStore from "../settings/SettingsStore";
|
|
|
|
const HS_MANAGERS_REFRESH_INTERVAL = 8 * 60 * 60 * 1000; // 8 hours
|
|
const KIND_PREFERENCE = [
|
|
// Ordered: first is most preferred, last is least preferred.
|
|
KIND_ACCOUNT,
|
|
KIND_HOMESERVER,
|
|
KIND_CONFIG,
|
|
];
|
|
|
|
export class IntegrationManagers {
|
|
static _instance;
|
|
|
|
static sharedInstance(): IntegrationManagers {
|
|
if (!IntegrationManagers._instance) {
|
|
IntegrationManagers._instance = new IntegrationManagers();
|
|
}
|
|
return IntegrationManagers._instance;
|
|
}
|
|
|
|
_managers: IntegrationManagerInstance[] = [];
|
|
_client: MatrixClient;
|
|
_wellknownRefreshTimerId: number = null;
|
|
_primaryManager: IntegrationManagerInstance;
|
|
|
|
constructor() {
|
|
this._compileManagers();
|
|
}
|
|
|
|
startWatching(): void {
|
|
this.stopWatching();
|
|
this._client = MatrixClientPeg.get();
|
|
this._client.on("accountData", this._onAccountData.bind(this));
|
|
this._compileManagers();
|
|
setInterval(() => this._setupHomeserverManagers(), HS_MANAGERS_REFRESH_INTERVAL);
|
|
}
|
|
|
|
stopWatching(): void {
|
|
if (!this._client) return;
|
|
this._client.removeListener("accountData", this._onAccountData.bind(this));
|
|
if (this._wellknownRefreshTimerId !== null) clearInterval(this._wellknownRefreshTimerId);
|
|
}
|
|
|
|
_compileManagers() {
|
|
this._managers = [];
|
|
this._setupConfiguredManager();
|
|
this._setupHomeserverManagers();
|
|
this._setupAccountManagers();
|
|
}
|
|
|
|
_setupConfiguredManager() {
|
|
const apiUrl = SdkConfig.get()['integrations_rest_url'];
|
|
const uiUrl = SdkConfig.get()['integrations_ui_url'];
|
|
|
|
if (apiUrl && uiUrl) {
|
|
this._managers.push(new IntegrationManagerInstance(KIND_CONFIG, apiUrl, uiUrl));
|
|
this._primaryManager = null; // reset primary
|
|
}
|
|
}
|
|
|
|
async _setupHomeserverManagers() {
|
|
try {
|
|
console.log("Updating homeserver-configured integration managers...");
|
|
const homeserverDomain = MatrixClientPeg.getHomeserverName();
|
|
const discoveryResponse = await AutoDiscovery.getRawClientConfig(homeserverDomain);
|
|
if (discoveryResponse && discoveryResponse['m.integrations']) {
|
|
let managers = discoveryResponse['m.integrations']['managers'];
|
|
if (!Array.isArray(managers)) managers = []; // make it an array so we can wipe the HS managers
|
|
|
|
console.log(`Homeserver has ${managers.length} integration managers`);
|
|
|
|
// Clear out any known managers for the homeserver
|
|
// TODO: Log out of the scalar clients
|
|
this._managers = this._managers.filter(m => m.kind !== KIND_HOMESERVER);
|
|
|
|
// Now add all the managers the homeserver wants us to have
|
|
for (const hsManager of managers) {
|
|
if (!hsManager["api_url"]) continue;
|
|
this._managers.push(new IntegrationManagerInstance(
|
|
KIND_HOMESERVER,
|
|
hsManager["api_url"],
|
|
hsManager["ui_url"], // optional
|
|
));
|
|
}
|
|
|
|
this._primaryManager = null; // reset primary
|
|
} else {
|
|
console.log("Homeserver has no integration managers");
|
|
}
|
|
} catch (e) {
|
|
console.error(e);
|
|
// Errors during discovery are non-fatal
|
|
}
|
|
}
|
|
|
|
_setupAccountManagers() {
|
|
if (!this._client || !this._client.getUserId()) return; // not logged in
|
|
const widgets = WidgetUtils.getIntegrationManagerWidgets();
|
|
widgets.forEach(w => {
|
|
const data = w.content['data'];
|
|
if (!data) return;
|
|
|
|
const uiUrl = w.content['url'];
|
|
const apiUrl = data['api_url'];
|
|
if (!apiUrl || !uiUrl) return;
|
|
|
|
const manager = new IntegrationManagerInstance(KIND_ACCOUNT, apiUrl, uiUrl);
|
|
manager.id = w['id'] || w['state_key'] || '';
|
|
this._managers.push(manager);
|
|
});
|
|
this._primaryManager = null; // reset primary
|
|
}
|
|
|
|
_onAccountData(ev: MatrixEvent): void {
|
|
if (ev.getType() === 'm.widgets') {
|
|
this._compileManagers();
|
|
}
|
|
}
|
|
|
|
hasManager(): boolean {
|
|
return this._managers.length > 0;
|
|
}
|
|
|
|
getOrderedManagers(): IntegrationManagerInstance[] {
|
|
const ordered = [];
|
|
for (const kind of KIND_PREFERENCE) {
|
|
const managers = this._managers.filter(m => m.kind === kind);
|
|
if (!managers || !managers.length) continue;
|
|
|
|
if (kind === KIND_ACCOUNT) {
|
|
// Order by state_keys (IDs)
|
|
managers.sort((a, b) => a.id.localeCompare(b.id));
|
|
}
|
|
|
|
ordered.push(...managers);
|
|
}
|
|
return ordered;
|
|
}
|
|
|
|
getPrimaryManager(): IntegrationManagerInstance {
|
|
if (this.hasManager()) {
|
|
if (this._primaryManager) return this._primaryManager;
|
|
|
|
this._primaryManager = this.getOrderedManagers()[0];
|
|
return this._primaryManager;
|
|
} else {
|
|
return null;
|
|
}
|
|
}
|
|
|
|
openNoManagerDialog(): void {
|
|
const IntegrationsImpossibleDialog = sdk.getComponent("dialogs.IntegrationsImpossibleDialog");
|
|
Modal.createTrackedDialog('Integrations impossible', '', IntegrationsImpossibleDialog);
|
|
}
|
|
|
|
openAll(room: Room = null, screen: string = null, integrationId: string = null): void {
|
|
if (!SettingsStore.getValue("integrationProvisioning")) {
|
|
return this.showDisabledDialog();
|
|
}
|
|
|
|
if (this._managers.length === 0) {
|
|
return this.openNoManagerDialog();
|
|
}
|
|
|
|
const TabbedIntegrationManagerDialog = sdk.getComponent("views.dialogs.TabbedIntegrationManagerDialog");
|
|
Modal.createTrackedDialog(
|
|
'Tabbed Integration Manager', '', TabbedIntegrationManagerDialog,
|
|
{room, screen, integrationId}, 'mx_TabbedIntegrationManagerDialog',
|
|
);
|
|
}
|
|
|
|
showDisabledDialog(): void {
|
|
const IntegrationsDisabledDialog = sdk.getComponent("dialogs.IntegrationsDisabledDialog");
|
|
Modal.createTrackedDialog('Integrations disabled', '', IntegrationsDisabledDialog);
|
|
}
|
|
|
|
async overwriteManagerOnAccount(manager: IntegrationManagerInstance) {
|
|
// TODO: TravisR - We should be logging out of scalar clients.
|
|
await WidgetUtils.removeIntegrationManagerWidgets();
|
|
|
|
// TODO: TravisR - We should actually be carrying over the discovery response verbatim.
|
|
await WidgetUtils.addIntegrationManagerWidget(manager.name, manager.uiUrl, manager.apiUrl);
|
|
}
|
|
|
|
/**
|
|
* Attempts to discover an integration manager using only its name. This will not validate that
|
|
* the integration manager is functional - that is the caller's responsibility.
|
|
* @param {string} domainName The domain name to look up.
|
|
* @returns {Promise<IntegrationManagerInstance>} Resolves to an integration manager instance,
|
|
* or null if none was found.
|
|
*/
|
|
async tryDiscoverManager(domainName: string): IntegrationManagerInstance {
|
|
console.log("Looking up integration manager via .well-known");
|
|
if (domainName.startsWith("http:") || domainName.startsWith("https:")) {
|
|
// trim off the scheme and just use the domain
|
|
const url = url.parse(domainName);
|
|
domainName = url.host;
|
|
}
|
|
|
|
let wkConfig;
|
|
try {
|
|
const result = await fetch(`https://${domainName}/.well-known/matrix/integrations`);
|
|
wkConfig = await result.json();
|
|
} catch (e) {
|
|
console.error(e);
|
|
console.warn("Failed to locate integration manager");
|
|
return null;
|
|
}
|
|
|
|
if (!wkConfig || !wkConfig["m.integrations_widget"]) {
|
|
console.warn("Missing integrations widget on .well-known response");
|
|
return null;
|
|
}
|
|
|
|
const widget = wkConfig["m.integrations_widget"];
|
|
if (!widget["url"] || !widget["data"] || !widget["data"]["api_url"]) {
|
|
console.warn("Malformed .well-known response for integrations widget");
|
|
return null;
|
|
}
|
|
|
|
// All discovered managers are per-user managers
|
|
const manager = new IntegrationManagerInstance(KIND_ACCOUNT, widget["data"]["api_url"], widget["url"]);
|
|
console.log("Got an integration manager (untested)");
|
|
|
|
// We don't test the manager because the caller may need to do extra
|
|
// checks or similar with it. For instance, they may need to deal with
|
|
// terms of service or want to call something particular.
|
|
|
|
return manager;
|
|
}
|
|
}
|
|
|
|
// For debugging
|
|
global.mxIntegrationManagers = IntegrationManagers;
|