/* 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. */ import React, {createRef} from "react"; import {AutoDiscovery} from "matrix-js-sdk/src/autodiscovery"; import AutoDiscoveryUtils, {ValidatedServerConfig} from "../../../utils/AutoDiscoveryUtils"; import BaseDialog from './BaseDialog'; import { _t } from '../../../languageHandler'; import AccessibleButton from "../elements/AccessibleButton"; import SdkConfig from "../../../SdkConfig"; import Field from "../elements/Field"; import StyledRadioButton from "../elements/StyledRadioButton"; import TextWithTooltip from "../elements/TextWithTooltip"; import withValidation, {IFieldState} from "../elements/Validation"; import {replaceableComponent} from "../../../utils/replaceableComponent"; interface IProps { title?: string; serverConfig: ValidatedServerConfig; onFinished(config?: ValidatedServerConfig): void; } interface IState { defaultChosen: boolean; otherHomeserver: string; } @replaceableComponent("views.dialogs.ServerPickerDialog") export default class ServerPickerDialog extends React.PureComponent { private readonly defaultServer: ValidatedServerConfig; private readonly fieldRef = createRef(); private validatedConf: ValidatedServerConfig; constructor(props) { super(props); const config = SdkConfig.get(); this.defaultServer = config["validated_server_config"] as ValidatedServerConfig; const { serverConfig } = this.props; let otherHomeserver = ""; if (!serverConfig.isDefault) { if (serverConfig.isNameResolvable && serverConfig.hsName) { otherHomeserver = serverConfig.hsName; } else { otherHomeserver = serverConfig.hsUrl; } } this.state = { defaultChosen: serverConfig.isDefault, otherHomeserver, }; } private onDefaultChosen = () => { this.setState({ defaultChosen: true }); }; private onOtherChosen = () => { this.setState({ defaultChosen: false }); }; private onHomeserverChange = (ev) => { this.setState({ otherHomeserver: ev.target.value }); }; // TODO: Do we want to support .well-known lookups here? // If for some reason someone enters "matrix.org" for a URL, we could do a lookup to // find their homeserver without demanding they use "https://matrix.org" private validate = withValidation({ deriveData: async ({ value }) => { let hsUrl = value.trim(); // trim to account for random whitespace // if the URL has no protocol, try validate it as a serverName via well-known if (!hsUrl.includes("://")) { try { const discoveryResult = await AutoDiscovery.findClientConfig(hsUrl); this.validatedConf = AutoDiscoveryUtils.buildValidatedConfigFromDiscovery(hsUrl, discoveryResult); return {}; // we have a validated config, we don't need to try the other paths } catch (e) { console.error(`Attempted ${hsUrl} as a server_name but it failed`, e); } } // if we got to this stage then either the well-known failed or the URL had a protocol specified, // so validate statically only. If the URL has no protocol, default to https. if (!hsUrl.includes("://")) { hsUrl = "https://" + hsUrl; } try { this.validatedConf = await AutoDiscoveryUtils.validateServerConfigWithStaticUrls(hsUrl); return {}; } catch (e) { console.error(e); const stateForError = AutoDiscoveryUtils.authComponentStateForError(e); if (stateForError.isFatalError) { let error = _t("Unable to validate homeserver"); if (e.translatedMessage) { error = e.translatedMessage; } return { error }; } // try to carry on anyway try { this.validatedConf = await AutoDiscoveryUtils.validateServerConfigWithStaticUrls(hsUrl, null, true); return {}; } catch (e) { console.error(e); return { error: _t("Invalid URL") }; } } }, rules: [ { key: "required", test: ({ value, allowEmpty }) => allowEmpty || !!value, invalid: () => _t("Specify a homeserver"), }, { key: "valid", test: async function({ value }, { error }) { if (!value) return true; return !error; }, invalid: function({ error }) { return error; }, }, ], }); private onHomeserverValidate = (fieldState: IFieldState) => this.validate(fieldState); private onSubmit = async (ev) => { ev.preventDefault(); const valid = await this.fieldRef.current.validate({ allowEmpty: false }); if (!valid && !this.state.defaultChosen) { this.fieldRef.current.focus(); this.fieldRef.current.validate({ allowEmpty: false, focused: true }); return; } this.props.onFinished(this.state.defaultChosen ? this.defaultServer : this.validatedConf); }; public render() { let text; if (this.defaultServer.hsName === "matrix.org") { text = _t("Matrix.org is the biggest public homeserver in the world, so it’s a good place for many."); } let defaultServerName = this.defaultServer.hsName; if (this.defaultServer.hsNameIsDifferent) { defaultServerName = ( ); } return

{_t("We call the places where you can host your account ‘homeservers’.")} {text}

{defaultServerName}

{_t("Use your preferred Matrix homeserver if you have one, or host your own.")}

{_t("Continue")}

{_t("Learn more")}

{_t("About homeservers")}
; } }