2019-08-12 13:51:44 +03:00
|
|
|
|
/*
|
|
|
|
|
Copyright 2019 New Vector Ltd
|
|
|
|
|
|
|
|
|
|
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 request from 'browser-request';
|
|
|
|
|
import url from 'url';
|
|
|
|
|
import React from 'react';
|
|
|
|
|
import {_t} from "../../../languageHandler";
|
|
|
|
|
import sdk from '../../../index';
|
|
|
|
|
import MatrixClientPeg from "../../../MatrixClientPeg";
|
|
|
|
|
import SdkConfig from "../../../SdkConfig";
|
|
|
|
|
import Field from "../elements/Field";
|
2019-08-12 16:36:48 +03:00
|
|
|
|
import Modal from '../../../Modal';
|
2019-08-12 13:51:44 +03:00
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* If a url has no path component, etc. abbreviate it to just the hostname
|
|
|
|
|
*
|
|
|
|
|
* @param {string} u The url to be abbreviated
|
|
|
|
|
* @returns {string} The abbreviated url
|
|
|
|
|
*/
|
|
|
|
|
function abbreviateUrl(u) {
|
|
|
|
|
if (!u) return '';
|
|
|
|
|
|
|
|
|
|
const parsedUrl = url.parse(u);
|
|
|
|
|
// if it's something we can't parse as a url then just return it
|
|
|
|
|
if (!parsedUrl) return u;
|
|
|
|
|
|
|
|
|
|
if (parsedUrl.path == '/') {
|
|
|
|
|
// we ignore query / hash parts: these aren't relevant for IS server URLs
|
|
|
|
|
return parsedUrl.host;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return u;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
function unabbreviateUrl(u) {
|
|
|
|
|
if (!u) return '';
|
|
|
|
|
|
|
|
|
|
let longUrl = u;
|
|
|
|
|
if (!u.startsWith('https://')) longUrl = 'https://' + u;
|
|
|
|
|
const parsed = url.parse(longUrl);
|
|
|
|
|
if (parsed.hostname === null) return u;
|
|
|
|
|
|
|
|
|
|
return longUrl;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* Check an IS URL is valid, including liveness check
|
|
|
|
|
*
|
|
|
|
|
* @param {string} isUrl The url to check
|
|
|
|
|
* @returns {string} null if url passes all checks, otherwise i18ned error string
|
|
|
|
|
*/
|
|
|
|
|
async function checkIsUrl(isUrl) {
|
|
|
|
|
const parsedUrl = url.parse(isUrl);
|
|
|
|
|
|
|
|
|
|
if (parsedUrl.protocol !== 'https:') return _t("Identity Server URL must be HTTPS");
|
|
|
|
|
|
|
|
|
|
// XXX: duplicated logic from js-sdk but it's quite tied up in the validation logic in the
|
|
|
|
|
// js-sdk so probably as easy to duplicate it than to separate it out so we can reuse it
|
|
|
|
|
return new Promise((resolve) => {
|
|
|
|
|
request(
|
2019-08-12 15:31:31 +03:00
|
|
|
|
// also XXX: we don't really know whether to hit /v1 or /v2 for this: we
|
|
|
|
|
// probably want a /versions endpoint like the C/S API.
|
2019-08-12 13:51:44 +03:00
|
|
|
|
{ method: "GET", url: isUrl + '/_matrix/identity/api/v1' },
|
|
|
|
|
(err, response, body) => {
|
|
|
|
|
if (err) {
|
|
|
|
|
resolve(_t("Could not connect to ID Server"));
|
|
|
|
|
} else if (response.status < 200 || response.status >= 300) {
|
|
|
|
|
resolve(_t("Not a valid ID Server (status code %(code)s)", {code: response.status}));
|
|
|
|
|
} else {
|
|
|
|
|
resolve(null);
|
|
|
|
|
}
|
|
|
|
|
},
|
|
|
|
|
);
|
|
|
|
|
});
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
export default class SetIdServer extends React.Component {
|
|
|
|
|
constructor() {
|
|
|
|
|
super();
|
|
|
|
|
|
2019-08-12 15:11:05 +03:00
|
|
|
|
let defaultIdServer = abbreviateUrl(MatrixClientPeg.get().getIdentityServerUrl());
|
2019-08-12 13:51:44 +03:00
|
|
|
|
if (!defaultIdServer) {
|
2019-08-12 15:11:05 +03:00
|
|
|
|
defaultIdServer = abbreviateUrl(SdkConfig.get()['validated_server_config']['idServer']) || '';
|
2019-08-12 13:51:44 +03:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
this.state = {
|
|
|
|
|
currentClientIdServer: MatrixClientPeg.get().getIdentityServerUrl(),
|
|
|
|
|
idServer: defaultIdServer,
|
|
|
|
|
error: null,
|
|
|
|
|
busy: false,
|
|
|
|
|
};
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
_onIdentityServerChanged = (ev) => {
|
|
|
|
|
const u = ev.target.value;
|
|
|
|
|
|
|
|
|
|
this.setState({idServer: u});
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
_getTooltip = () => {
|
|
|
|
|
if (this.state.busy) {
|
|
|
|
|
const InlineSpinner = sdk.getComponent('views.elements.InlineSpinner');
|
|
|
|
|
return <div>
|
|
|
|
|
<InlineSpinner />
|
|
|
|
|
{ _t("Checking Server") }
|
|
|
|
|
</div>;
|
|
|
|
|
} else if (this.state.error) {
|
|
|
|
|
return this.state.error;
|
|
|
|
|
} else {
|
|
|
|
|
return null;
|
|
|
|
|
}
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
_idServerChangeEnabled = () => {
|
|
|
|
|
return !!this.state.idServer && !this.state.busy;
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
_saveIdServer = async () => {
|
|
|
|
|
this.setState({busy: true});
|
|
|
|
|
|
|
|
|
|
const fullUrl = unabbreviateUrl(this.state.idServer);
|
|
|
|
|
|
|
|
|
|
const errStr = await checkIsUrl(fullUrl);
|
2019-08-12 15:29:52 +03:00
|
|
|
|
|
|
|
|
|
let newFormValue = this.state.idServer;
|
2019-08-12 13:51:44 +03:00
|
|
|
|
if (!errStr) {
|
|
|
|
|
MatrixClientPeg.get().setIdentityServerUrl(fullUrl);
|
2019-08-12 16:38:49 +03:00
|
|
|
|
localStorage.removeItem("mx_is_access_token", fullUrl);
|
2019-08-12 13:51:44 +03:00
|
|
|
|
localStorage.setItem("mx_is_url", fullUrl);
|
2019-08-12 15:29:52 +03:00
|
|
|
|
newFormValue = '';
|
2019-08-12 13:51:44 +03:00
|
|
|
|
}
|
|
|
|
|
this.setState({
|
|
|
|
|
busy: false,
|
|
|
|
|
error: errStr,
|
|
|
|
|
currentClientIdServer: MatrixClientPeg.get().getIdentityServerUrl(),
|
2019-08-12 15:29:52 +03:00
|
|
|
|
idServer: newFormValue,
|
2019-08-12 13:51:44 +03:00
|
|
|
|
});
|
|
|
|
|
};
|
|
|
|
|
|
2019-08-12 16:36:48 +03:00
|
|
|
|
_onDisconnectClicked = () => {
|
|
|
|
|
const QuestionDialog = sdk.getComponent("dialogs.QuestionDialog");
|
|
|
|
|
Modal.createTrackedDialog('ID Server Disconnect Warning', '', QuestionDialog, {
|
|
|
|
|
title: _t("Disconnect ID Server"),
|
|
|
|
|
description:
|
|
|
|
|
<div>
|
|
|
|
|
{_t(
|
|
|
|
|
"Disconnect from the ID Server <idserver></idserver>?", {},
|
|
|
|
|
{idserver: sub => <b>{abbreviateUrl(this.state.currentClientIdServer)}</b>},
|
|
|
|
|
)},
|
|
|
|
|
</div>,
|
|
|
|
|
button: _t("Disconnect"),
|
|
|
|
|
onFinished: (confirmed) => {
|
|
|
|
|
if (confirmed) {
|
|
|
|
|
this._disconnectIdServer();
|
|
|
|
|
}
|
|
|
|
|
},
|
|
|
|
|
});
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
_disconnectIdServer = () => {
|
|
|
|
|
MatrixClientPeg.get().setIdentityServerUrl(null);
|
2019-08-12 16:45:15 +03:00
|
|
|
|
localStorage.removeItem("mx_is_access_token");
|
2019-08-12 16:36:48 +03:00
|
|
|
|
localStorage.removeItem("mx_is_url");
|
|
|
|
|
this.setState({
|
|
|
|
|
busy: false,
|
|
|
|
|
error: null,
|
|
|
|
|
currentClientIdServer: MatrixClientPeg.get().getIdentityServerUrl(),
|
|
|
|
|
idServer: '',
|
|
|
|
|
});
|
|
|
|
|
};
|
|
|
|
|
|
2019-08-12 13:51:44 +03:00
|
|
|
|
render() {
|
2019-08-12 16:36:48 +03:00
|
|
|
|
const AccessibleButton = sdk.getComponent('views.elements.AccessibleButton');
|
2019-08-12 13:51:44 +03:00
|
|
|
|
const idServerUrl = this.state.currentClientIdServer;
|
|
|
|
|
let sectionTitle;
|
|
|
|
|
let bodyText;
|
|
|
|
|
if (idServerUrl) {
|
|
|
|
|
sectionTitle = _t("Identity Server (%(server)s)", { server: abbreviateUrl(idServerUrl) });
|
|
|
|
|
bodyText = _t(
|
|
|
|
|
"You are currently using <server></server> to discover and be discoverable by " +
|
|
|
|
|
"existing contacts you know. You can change your identity server below.",
|
|
|
|
|
{},
|
|
|
|
|
{ server: sub => <b>{abbreviateUrl(idServerUrl)}</b> },
|
|
|
|
|
);
|
|
|
|
|
} else {
|
|
|
|
|
sectionTitle = _t("Identity Server");
|
|
|
|
|
bodyText = _t(
|
|
|
|
|
"You are not currently using an Identity Server. " +
|
|
|
|
|
"To discover and be discoverable by existing contacts you know, " +
|
|
|
|
|
"add one below",
|
|
|
|
|
);
|
|
|
|
|
}
|
|
|
|
|
|
2019-08-12 16:36:48 +03:00
|
|
|
|
let discoSection;
|
|
|
|
|
if (idServerUrl) {
|
|
|
|
|
discoSection = <div>
|
|
|
|
|
<span className="mx_SettingsTab_subsectionText">{_t(
|
|
|
|
|
"Disconnecting from your identity server will mean you " +
|
|
|
|
|
"won’t be discoverable by other users and you won’t be " +
|
|
|
|
|
"able to invite others by email or phone.",
|
|
|
|
|
)}</span>
|
|
|
|
|
<AccessibleButton onClick={this._onDisconnectClicked} kind="danger">
|
|
|
|
|
{_t("Disconnect")}
|
|
|
|
|
</AccessibleButton>
|
|
|
|
|
</div>;
|
|
|
|
|
}
|
|
|
|
|
|
2019-08-12 13:51:44 +03:00
|
|
|
|
return (
|
|
|
|
|
<form className="mx_SettingsTab_section mx_SetIdServer" onSubmit={this._saveIdServer}>
|
|
|
|
|
<span className="mx_SettingsTab_subheading">
|
|
|
|
|
{sectionTitle}
|
|
|
|
|
</span>
|
|
|
|
|
<span className="mx_SettingsTab_subsectionText">
|
|
|
|
|
{bodyText}
|
|
|
|
|
</span>
|
|
|
|
|
<Field label={_t("Identity Server")}
|
|
|
|
|
id="mx_SetIdServer_idServer"
|
|
|
|
|
type="text" value={this.state.idServer} autoComplete="off"
|
|
|
|
|
onChange={this._onIdentityServerChanged}
|
|
|
|
|
tooltip={this._getTooltip()}
|
|
|
|
|
/>
|
|
|
|
|
<input className="mx_Dialog_primary"
|
|
|
|
|
type="submit" value={_t("Change")}
|
|
|
|
|
disabled={!this._idServerChangeEnabled()}
|
|
|
|
|
/>
|
2019-08-12 16:36:48 +03:00
|
|
|
|
{discoSection}
|
2019-08-12 13:51:44 +03:00
|
|
|
|
</form>
|
|
|
|
|
);
|
|
|
|
|
}
|
|
|
|
|
}
|