2020-01-21 03:52:11 +01:00
|
|
|
/*
|
|
|
|
Copyright 2015, 2016 OpenMarket Ltd
|
|
|
|
Copyright 2017 Vector Creations Ltd
|
|
|
|
Copyright 2018, 2019 New Vector Ltd
|
|
|
|
Copyright 2019 Michael Telatynski <7t3chguy@gmail.com>
|
|
|
|
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 from 'react';
|
|
|
|
import * as sdk from 'matrix-react-sdk';
|
|
|
|
import PlatformPeg from 'matrix-react-sdk/src/PlatformPeg';
|
2021-06-30 14:28:31 +02:00
|
|
|
import { _td, newTranslatableError } from 'matrix-react-sdk/src/languageHandler';
|
2020-01-21 03:52:11 +01:00
|
|
|
import AutoDiscoveryUtils from 'matrix-react-sdk/src/utils/AutoDiscoveryUtils';
|
2021-06-30 14:28:31 +02:00
|
|
|
import { AutoDiscovery } from "matrix-js-sdk/src/autodiscovery";
|
2020-01-21 03:52:11 +01:00
|
|
|
import * as Lifecycle from "matrix-react-sdk/src/Lifecycle";
|
2021-10-21 16:35:05 +02:00
|
|
|
import SdkConfig, { parseSsoRedirectOptions } from "matrix-react-sdk/src/SdkConfig";
|
2021-10-15 17:04:34 +02:00
|
|
|
import { logger } from "matrix-js-sdk/src/logger";
|
2021-12-09 23:57:46 +01:00
|
|
|
import { createClient } from "matrix-js-sdk/src/matrix";
|
2020-01-21 03:52:11 +01:00
|
|
|
|
2021-12-09 23:57:46 +01:00
|
|
|
import type MatrixChatType from "matrix-react-sdk/src/components/structures/MatrixChat";
|
2021-06-30 14:28:31 +02:00
|
|
|
import { parseQs, parseQsFromFragment } from './url_utils';
|
2020-07-03 00:16:03 +02:00
|
|
|
import VectorBasePlatform from "./platform/VectorBasePlatform";
|
2021-12-09 23:57:46 +01:00
|
|
|
|
|
|
|
// add React and ReactPerf to the global namespace, to make them easier to access via the console
|
|
|
|
// this incidentally means we can forget our React imports in JSX files without penalty.
|
|
|
|
window.React = React;
|
2020-07-03 00:16:03 +02:00
|
|
|
|
2020-04-22 14:29:12 +02:00
|
|
|
let lastLocationHashSet: string = null;
|
2020-01-21 03:52:11 +01:00
|
|
|
|
2021-10-15 16:59:13 +02:00
|
|
|
logger.log(`Application is running in ${process.env.NODE_ENV} mode`);
|
2021-08-23 12:35:43 +02:00
|
|
|
|
2021-10-15 17:04:34 +02:00
|
|
|
window.matrixLogger = logger;
|
|
|
|
|
2020-01-21 03:52:11 +01:00
|
|
|
// Parse the given window.location and return parameters that can be used when calling
|
|
|
|
// MatrixChat.showScreen(screen, params)
|
2020-04-22 14:29:12 +02:00
|
|
|
function getScreenFromLocation(location: Location) {
|
2020-01-21 03:52:11 +01:00
|
|
|
const fragparts = parseQsFromFragment(location);
|
|
|
|
return {
|
|
|
|
screen: fragparts.location.substring(1),
|
|
|
|
params: fragparts.params,
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
|
|
|
// Here, we do some crude URL analysis to allow
|
|
|
|
// deep-linking.
|
2020-04-22 14:29:12 +02:00
|
|
|
function routeUrl(location: Location) {
|
2020-01-21 03:52:11 +01:00
|
|
|
if (!window.matrixChat) return;
|
|
|
|
|
2021-10-15 16:59:13 +02:00
|
|
|
logger.log("Routing URL ", location.href);
|
2020-01-21 03:52:11 +01:00
|
|
|
const s = getScreenFromLocation(location);
|
2020-04-22 14:41:29 +02:00
|
|
|
(window.matrixChat as MatrixChatType).showScreen(s.screen, s.params);
|
2020-01-21 03:52:11 +01:00
|
|
|
}
|
|
|
|
|
2020-04-22 14:29:12 +02:00
|
|
|
function onHashChange(ev: HashChangeEvent) {
|
2020-01-21 03:52:11 +01:00
|
|
|
if (decodeURIComponent(window.location.hash) === lastLocationHashSet) {
|
|
|
|
// we just set this: no need to route it!
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
routeUrl(window.location);
|
|
|
|
}
|
|
|
|
|
|
|
|
// This will be called whenever the SDK changes screens,
|
|
|
|
// so a web page can update the URL bar appropriately.
|
2020-09-09 17:52:48 +02:00
|
|
|
function onNewScreen(screen: string, replaceLast = false) {
|
2021-10-15 16:59:13 +02:00
|
|
|
logger.log("newscreen " + screen);
|
2020-01-21 03:52:11 +01:00
|
|
|
const hash = '#/' + screen;
|
|
|
|
lastLocationHashSet = hash;
|
2020-09-09 17:52:48 +02:00
|
|
|
|
2021-07-19 10:40:08 +02:00
|
|
|
// if the new hash is a substring of the old one then we are stripping fields e.g `via` so replace history
|
2021-07-19 10:48:50 +02:00
|
|
|
if (screen.startsWith("room/") &&
|
|
|
|
window.location.hash.includes("/$") === hash.includes("/$") && // only if both did or didn't contain event link
|
|
|
|
window.location.hash.startsWith(hash)
|
|
|
|
) {
|
2021-07-19 10:40:08 +02:00
|
|
|
replaceLast = true;
|
|
|
|
}
|
|
|
|
|
2020-09-09 17:52:48 +02:00
|
|
|
if (replaceLast) {
|
|
|
|
window.location.replace(hash);
|
|
|
|
} else {
|
|
|
|
window.location.assign(hash);
|
|
|
|
}
|
2020-01-21 03:52:11 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
// We use this to work out what URL the SDK should
|
|
|
|
// pass through when registering to allow the user to
|
|
|
|
// click back to the client having registered.
|
|
|
|
// It's up to us to recognise if we're loaded with
|
|
|
|
// this URL and tell MatrixClient to resume registration.
|
|
|
|
//
|
|
|
|
// If we're in electron, we should never pass through a file:// URL otherwise
|
|
|
|
// the identity server will try to 302 the browser to it, which breaks horribly.
|
2020-07-13 18:02:20 +02:00
|
|
|
// so in that instance, hardcode to use app.element.io for now instead.
|
2020-04-22 14:29:12 +02:00
|
|
|
function makeRegistrationUrl(params: object) {
|
2020-01-21 03:52:11 +01:00
|
|
|
let url;
|
|
|
|
if (window.location.protocol === "vector:") {
|
2020-07-13 18:02:20 +02:00
|
|
|
url = 'https://app.element.io/#/register';
|
2020-01-21 03:52:11 +01:00
|
|
|
} else {
|
|
|
|
url = (
|
|
|
|
window.location.protocol + '//' +
|
|
|
|
window.location.host +
|
|
|
|
window.location.pathname +
|
|
|
|
'#/register'
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
const keys = Object.keys(params);
|
|
|
|
for (let i = 0; i < keys.length; ++i) {
|
|
|
|
if (i === 0) {
|
|
|
|
url += '?';
|
|
|
|
} else {
|
|
|
|
url += '&';
|
|
|
|
}
|
|
|
|
const k = keys[i];
|
|
|
|
url += k + '=' + encodeURIComponent(params[k]);
|
|
|
|
}
|
|
|
|
return url;
|
|
|
|
}
|
|
|
|
|
|
|
|
function onTokenLoginCompleted() {
|
|
|
|
// if we did a token login, we're now left with the token, hs and is
|
|
|
|
// url as query params in the url; a little nasty but let's redirect to
|
|
|
|
// clear them.
|
2021-02-11 15:47:50 +01:00
|
|
|
const url = new URL(window.location.href);
|
2021-02-09 10:08:40 +01:00
|
|
|
|
2021-02-11 17:27:55 +01:00
|
|
|
url.searchParams.delete("loginToken");
|
2021-02-09 10:08:40 +01:00
|
|
|
|
2021-10-15 16:59:13 +02:00
|
|
|
logger.log(`Redirecting to ${url.href} to drop loginToken from queryparams`);
|
2021-02-11 15:47:50 +01:00
|
|
|
window.history.replaceState(null, "", url.href);
|
2020-01-21 03:52:11 +01:00
|
|
|
}
|
|
|
|
|
2020-04-08 22:41:22 +02:00
|
|
|
export async function loadApp(fragParams: {}) {
|
2020-01-21 03:52:11 +01:00
|
|
|
window.addEventListener('hashchange', onHashChange);
|
|
|
|
|
|
|
|
const platform = PlatformPeg.get();
|
|
|
|
|
|
|
|
const params = parseQs(window.location);
|
|
|
|
|
|
|
|
const urlWithoutQuery = window.location.protocol + '//' + window.location.host + window.location.pathname;
|
2021-10-15 16:59:13 +02:00
|
|
|
logger.log("Vector starting at " + urlWithoutQuery);
|
2020-04-08 22:41:22 +02:00
|
|
|
|
2020-07-03 00:16:03 +02:00
|
|
|
(platform as VectorBasePlatform).startUpdater();
|
2020-04-08 22:41:22 +02:00
|
|
|
|
2020-04-09 11:32:04 +02:00
|
|
|
// Don't bother loading the app until the config is verified
|
|
|
|
const config = await verifyServerConfig();
|
2021-04-07 03:39:26 +02:00
|
|
|
|
|
|
|
// Before we continue, let's see if we're supposed to do an SSO redirect
|
|
|
|
const [userId] = await Lifecycle.getStoredSessionOwner();
|
|
|
|
const hasPossibleToken = !!userId;
|
|
|
|
const isReturningFromSso = !!params.loginToken;
|
2021-10-21 16:35:05 +02:00
|
|
|
const ssoRedirects = parseSsoRedirectOptions(config);
|
|
|
|
let autoRedirect = ssoRedirects.immediate === true;
|
|
|
|
// XXX: This path matching is a bit brittle, but better to do it early instead of in the app code.
|
|
|
|
const isWelcomeOrLanding = window.location.hash === '#/welcome' || window.location.hash === '#';
|
|
|
|
if (!autoRedirect && ssoRedirects.on_welcome_page && isWelcomeOrLanding) {
|
|
|
|
autoRedirect = true;
|
|
|
|
}
|
2021-04-07 03:39:26 +02:00
|
|
|
if (!hasPossibleToken && !isReturningFromSso && autoRedirect) {
|
2021-10-15 16:59:13 +02:00
|
|
|
logger.log("Bypassing app load to redirect to SSO");
|
2021-04-07 03:39:26 +02:00
|
|
|
const tempCli = createClient({
|
|
|
|
baseUrl: config['validated_server_config'].hsUrl,
|
|
|
|
idBaseUrl: config['validated_server_config'].isUrl,
|
|
|
|
});
|
|
|
|
PlatformPeg.get().startSingleSignOn(tempCli, "sso", `/${getScreenFromLocation(window.location).screen}`);
|
2021-04-07 03:43:24 +02:00
|
|
|
|
|
|
|
// We return here because startSingleSignOn() will asynchronously redirect us. We don't
|
|
|
|
// care to wait for it, and don't want to show any UI while we wait (not even half a welcome
|
|
|
|
// page). As such, just don't even bother loading the MatrixChat component.
|
2021-04-07 03:39:26 +02:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2022-01-28 16:52:43 +01:00
|
|
|
const defaultDeviceName = config['defaultDeviceDisplayName'] ?? platform.getDefaultDeviceDisplayName();
|
|
|
|
|
2020-04-09 11:32:04 +02:00
|
|
|
const MatrixChat = sdk.getComponent('structures.MatrixChat');
|
|
|
|
return <MatrixChat
|
|
|
|
onNewScreen={onNewScreen}
|
|
|
|
makeRegistrationUrl={makeRegistrationUrl}
|
|
|
|
config={config}
|
|
|
|
realQueryParams={params}
|
|
|
|
startingFragmentQueryParams={fragParams}
|
|
|
|
enableGuest={!config.disable_guests}
|
|
|
|
onTokenLoginCompleted={onTokenLoginCompleted}
|
|
|
|
initialScreenAfterLogin={getScreenFromLocation(window.location)}
|
2022-01-28 16:52:43 +01:00
|
|
|
defaultDeviceDisplayName={defaultDeviceName}
|
2020-04-09 11:32:04 +02:00
|
|
|
/>;
|
2020-01-21 03:52:11 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
async function verifyServerConfig() {
|
|
|
|
let validatedConfig;
|
|
|
|
try {
|
2021-10-15 16:59:13 +02:00
|
|
|
logger.log("Verifying homeserver configuration");
|
2020-01-21 03:52:11 +01:00
|
|
|
|
|
|
|
// Note: the query string may include is_url and hs_url - we only respect these in the
|
|
|
|
// context of email validation. Because we don't respect them otherwise, we do not need
|
|
|
|
// to parse or consider them here.
|
|
|
|
|
|
|
|
// Note: Although we throw all 3 possible configuration options through a .well-known-style
|
|
|
|
// verification, we do not care if the servers are online at this point. We do moderately
|
|
|
|
// care if they are syntactically correct though, so we shove them through the .well-known
|
|
|
|
// validators for that purpose.
|
|
|
|
|
|
|
|
const config = SdkConfig.get();
|
|
|
|
let wkConfig = config['default_server_config']; // overwritten later under some conditions
|
|
|
|
const serverName = config['default_server_name'];
|
|
|
|
const hsUrl = config['default_hs_url'];
|
|
|
|
const isUrl = config['default_is_url'];
|
|
|
|
|
|
|
|
const incompatibleOptions = [wkConfig, serverName, hsUrl].filter(i => !!i);
|
|
|
|
if (incompatibleOptions.length > 1) {
|
|
|
|
// noinspection ExceptionCaughtLocallyJS
|
|
|
|
throw newTranslatableError(_td(
|
|
|
|
"Invalid configuration: can only specify one of default_server_config, default_server_name, " +
|
|
|
|
"or default_hs_url.",
|
|
|
|
));
|
|
|
|
}
|
|
|
|
if (incompatibleOptions.length < 1) {
|
|
|
|
// noinspection ExceptionCaughtLocallyJS
|
|
|
|
throw newTranslatableError(_td("Invalid configuration: no default server specified."));
|
|
|
|
}
|
|
|
|
|
|
|
|
if (hsUrl) {
|
2021-10-15 16:59:13 +02:00
|
|
|
logger.log("Config uses a default_hs_url - constructing a default_server_config using this information");
|
2021-10-15 17:00:43 +02:00
|
|
|
logger.warn(
|
2020-01-21 03:52:11 +01:00
|
|
|
"DEPRECATED CONFIG OPTION: In the future, default_hs_url will not be accepted. Please use " +
|
|
|
|
"default_server_config instead.",
|
|
|
|
);
|
|
|
|
|
|
|
|
wkConfig = {
|
|
|
|
"m.homeserver": {
|
|
|
|
"base_url": hsUrl,
|
|
|
|
},
|
|
|
|
};
|
|
|
|
if (isUrl) {
|
|
|
|
wkConfig["m.identity_server"] = {
|
|
|
|
"base_url": isUrl,
|
|
|
|
};
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
let discoveryResult = null;
|
|
|
|
if (wkConfig) {
|
2021-10-15 16:59:13 +02:00
|
|
|
logger.log("Config uses a default_server_config - validating object");
|
2020-01-21 03:52:11 +01:00
|
|
|
discoveryResult = await AutoDiscovery.fromDiscoveryConfig(wkConfig);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (serverName) {
|
2021-10-15 16:59:13 +02:00
|
|
|
logger.log("Config uses a default_server_name - doing .well-known lookup");
|
2021-10-15 17:00:43 +02:00
|
|
|
logger.warn(
|
2020-01-21 03:52:11 +01:00
|
|
|
"DEPRECATED CONFIG OPTION: In the future, default_server_name will not be accepted. Please " +
|
|
|
|
"use default_server_config instead.",
|
|
|
|
);
|
|
|
|
discoveryResult = await AutoDiscovery.findClientConfig(serverName);
|
|
|
|
}
|
|
|
|
|
|
|
|
validatedConfig = AutoDiscoveryUtils.buildValidatedConfigFromDiscovery(serverName, discoveryResult, true);
|
|
|
|
} catch (e) {
|
2021-06-30 14:28:31 +02:00
|
|
|
const { hsUrl, isUrl, userId } = await Lifecycle.getStoredSessionVars();
|
2020-01-21 03:52:11 +01:00
|
|
|
if (hsUrl && userId) {
|
2021-10-15 16:56:22 +02:00
|
|
|
logger.error(e);
|
2021-10-15 17:00:43 +02:00
|
|
|
logger.warn("A session was found - suppressing config error and using the session's homeserver");
|
2020-01-21 03:52:11 +01:00
|
|
|
|
2021-10-15 16:59:13 +02:00
|
|
|
logger.log("Using pre-existing hsUrl and isUrl: ", { hsUrl, isUrl });
|
2020-01-21 03:52:11 +01:00
|
|
|
validatedConfig = await AutoDiscoveryUtils.validateServerConfigWithStaticUrls(hsUrl, isUrl, true);
|
|
|
|
} else {
|
|
|
|
// the user is not logged in, so scream
|
|
|
|
throw e;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
validatedConfig.isDefault = true;
|
|
|
|
|
|
|
|
// Just in case we ever have to debug this
|
2021-10-15 16:59:13 +02:00
|
|
|
logger.log("Using homeserver config:", validatedConfig);
|
2020-01-21 03:52:11 +01:00
|
|
|
|
|
|
|
// Add the newly built config to the actual config for use by the app
|
2021-10-15 16:59:13 +02:00
|
|
|
logger.log("Updating SdkConfig with validated discovery information");
|
2021-06-30 14:28:31 +02:00
|
|
|
SdkConfig.add({ "validated_server_config": validatedConfig });
|
2020-01-21 03:52:11 +01:00
|
|
|
|
|
|
|
return SdkConfig.get();
|
|
|
|
}
|