2016-10-05 14:36:59 +00:00
|
|
|
import { isRoomValid } from '../base/conference';
|
2017-01-29 01:56:35 +00:00
|
|
|
import { RouteRegistry } from '../base/react';
|
2016-10-05 14:36:59 +00:00
|
|
|
import { Conference } from '../conference';
|
|
|
|
import { WelcomePage } from '../welcome';
|
|
|
|
|
2017-02-01 04:25:09 +00:00
|
|
|
/**
|
|
|
|
* The RegExp pattern of the authority of a URI.
|
|
|
|
*
|
|
|
|
* @private
|
|
|
|
* @type {string}
|
|
|
|
*/
|
|
|
|
const _URI_AUTHORITY_PATTERN = '(//[^/?#]+)';
|
|
|
|
|
|
|
|
/**
|
|
|
|
* The RegExp pattern of the path of a URI.
|
|
|
|
*
|
|
|
|
* @private
|
|
|
|
* @type {string}
|
|
|
|
*/
|
|
|
|
const _URI_PATH_PATTERN = '([^?#]*)';
|
|
|
|
|
|
|
|
/**
|
|
|
|
* The RegExp patther of the protocol of a URI.
|
|
|
|
*
|
|
|
|
* @private
|
|
|
|
* @type {string}
|
|
|
|
*/
|
|
|
|
const _URI_PROTOCOL_PATTERN = '([a-z][a-z0-9\\.\\+-]*:)';
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Fixes the hier-part of a specific URI (string) so that the URI is well-known.
|
|
|
|
* For example, certain Jitsi Meet deployments are not conventional but it is
|
|
|
|
* possible to translate their URLs into conventional.
|
|
|
|
*
|
|
|
|
* @param {string} uri - The URI (string) to fix the hier-part of.
|
|
|
|
* @private
|
|
|
|
* @returns {string}
|
|
|
|
*/
|
|
|
|
function _fixURIStringHierPart(uri) {
|
|
|
|
// Rewrite the specified URL in order to handle special cases such as
|
|
|
|
// hipchat.com and enso.me which do not follow the common pattern of most
|
|
|
|
// Jitsi Meet deployments.
|
|
|
|
|
|
|
|
// hipchat.com
|
|
|
|
let regex
|
|
|
|
= new RegExp(
|
|
|
|
`^${_URI_PROTOCOL_PATTERN}//hipchat\\.com/video/call/`,
|
|
|
|
'gi');
|
|
|
|
let match = regex.exec(uri);
|
|
|
|
|
|
|
|
if (!match) {
|
|
|
|
// enso.me
|
|
|
|
regex
|
|
|
|
= new RegExp(
|
|
|
|
`^${_URI_PROTOCOL_PATTERN}//enso\\.me/(?:call|meeting)/`,
|
|
|
|
'gi');
|
|
|
|
match = regex.exec(uri);
|
|
|
|
}
|
|
|
|
if (match) {
|
|
|
|
/* eslint-disable no-param-reassign, prefer-template */
|
|
|
|
|
|
|
|
uri
|
|
|
|
= match[1] /* protocol */
|
|
|
|
+ '//enso.hipchat.me/'
|
|
|
|
+ uri.substring(regex.lastIndex); /* room (name) */
|
|
|
|
|
|
|
|
/* eslint-enable no-param-reassign, prefer-template */
|
|
|
|
}
|
|
|
|
|
|
|
|
return uri;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Fixes the scheme part of a specific URI (string) so that it contains a
|
|
|
|
* well-known scheme such as HTTP(S). For example, the mobile app implements an
|
|
|
|
* app-specific URI scheme in addition to Universal Links. The app-specific
|
|
|
|
* scheme may precede or replace the well-known scheme. In such a case, dealing
|
|
|
|
* with the app-specific scheme only complicates the logic and it is simpler to
|
|
|
|
* get rid of it (by translating the app-specific scheme into a well-known
|
|
|
|
* scheme).
|
|
|
|
*
|
|
|
|
* @param {string} uri - The URI (string) to fix the scheme of.
|
|
|
|
* @private
|
|
|
|
* @returns {string}
|
|
|
|
*/
|
|
|
|
function _fixURIStringScheme(uri) {
|
|
|
|
const regex = new RegExp(`^${_URI_PROTOCOL_PATTERN}+`, 'gi');
|
|
|
|
const match = regex.exec(uri);
|
|
|
|
|
|
|
|
if (match) {
|
|
|
|
// As an implementation convenience, pick up the last scheme and make
|
|
|
|
// sure that it is a well-known one.
|
|
|
|
let protocol = match[match.length - 1].toLowerCase();
|
|
|
|
|
|
|
|
if (protocol !== 'http:' && protocol !== 'https:') {
|
|
|
|
protocol = 'https:';
|
|
|
|
}
|
|
|
|
|
|
|
|
/* eslint-disable no-param-reassign */
|
|
|
|
|
|
|
|
uri = uri.substring(regex.lastIndex);
|
|
|
|
if (uri.startsWith('//')) {
|
|
|
|
// The specified URL was not a room name only, it contained an
|
|
|
|
// authority.
|
|
|
|
uri = protocol + uri;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* eslint-enable no-param-reassign */
|
|
|
|
}
|
|
|
|
|
|
|
|
return uri;
|
|
|
|
}
|
|
|
|
|
2016-10-05 14:36:59 +00:00
|
|
|
/**
|
2017-05-09 05:15:43 +00:00
|
|
|
* Gets the (Web application) context root defined by a specific location (URI).
|
2016-10-05 14:36:59 +00:00
|
|
|
*
|
2017-05-09 05:15:43 +00:00
|
|
|
* @param {Object} location - The location (URI) which defines the (Web
|
|
|
|
* application) context root.
|
|
|
|
* @returns {string} - The (Web application) context root defined by the
|
|
|
|
* specified {@code location} (URI).
|
2016-10-05 14:36:59 +00:00
|
|
|
*/
|
2017-05-09 05:15:43 +00:00
|
|
|
export function getLocationContextRoot(location: Object) {
|
|
|
|
const pathname = location.pathname;
|
|
|
|
const contextRootEndIndex = pathname.lastIndexOf('/');
|
|
|
|
|
|
|
|
return (
|
|
|
|
contextRootEndIndex === -1
|
|
|
|
? '/'
|
|
|
|
: pathname.substring(0, contextRootEndIndex + 1));
|
2016-10-05 14:36:59 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Determines which route is to be rendered in order to depict a specific Redux
|
|
|
|
* store.
|
|
|
|
*
|
|
|
|
* @param {(Object|Function)} stateOrGetState - Redux state or Regux getState()
|
|
|
|
* method.
|
|
|
|
* @returns {Route}
|
|
|
|
*/
|
|
|
|
export function _getRouteToRender(stateOrGetState) {
|
|
|
|
const state
|
|
|
|
= typeof stateOrGetState === 'function'
|
2017-01-13 22:37:53 +00:00
|
|
|
? stateOrGetState()
|
|
|
|
: stateOrGetState;
|
2017-05-09 05:15:43 +00:00
|
|
|
const { room } = state['features/base/conference'];
|
2017-06-09 19:09:23 +00:00
|
|
|
let component;
|
|
|
|
|
|
|
|
if (isRoomValid(room)) {
|
|
|
|
component = Conference;
|
|
|
|
} else {
|
|
|
|
// The value of the App prop welcomePageEnabled was stored in redux in
|
|
|
|
// saghul's PR. But I removed the redux state, action, action type, etc.
|
|
|
|
// because I didn't like the name. We are not using the prop is a
|
|
|
|
// React-ive way anyway so it's all the same difference.
|
|
|
|
const { app } = state['features/app'];
|
|
|
|
|
|
|
|
component = app && app.props.welcomePageEnabled ? WelcomePage : null;
|
2017-06-09 10:30:59 +00:00
|
|
|
}
|
|
|
|
|
2016-10-05 14:36:59 +00:00
|
|
|
return RouteRegistry.getRouteByComponent(component);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2017-05-09 05:15:43 +00:00
|
|
|
* Parses a specific URI string into an object with the well-known properties of
|
|
|
|
* the {@link Location} and/or {@link URL} interfaces implemented by Web
|
|
|
|
* browsers. The parsing attempts to be in accord with IETF's RFC 3986.
|
2016-10-05 14:36:59 +00:00
|
|
|
*
|
2017-05-09 05:15:43 +00:00
|
|
|
* @param {string} str - The URI string to parse.
|
2017-02-01 04:25:09 +00:00
|
|
|
* @returns {{
|
2017-05-09 05:15:43 +00:00
|
|
|
* hash: string,
|
|
|
|
* host: (string|undefined),
|
|
|
|
* hostname: (string|undefined),
|
|
|
|
* pathname: string,
|
|
|
|
* port: (string|undefined),
|
|
|
|
* protocol: (string|undefined),
|
|
|
|
* search: string
|
2017-02-01 04:25:09 +00:00
|
|
|
* }}
|
2016-10-05 14:36:59 +00:00
|
|
|
*/
|
2017-05-09 05:15:43 +00:00
|
|
|
function _parseStandardURIString(str: string) {
|
|
|
|
/* eslint-disable no-param-reassign */
|
2016-10-05 14:36:59 +00:00
|
|
|
|
2017-05-09 05:15:43 +00:00
|
|
|
const obj = {};
|
|
|
|
|
|
|
|
let regex;
|
|
|
|
let match;
|
|
|
|
|
|
|
|
// protocol
|
|
|
|
regex = new RegExp(`^${_URI_PROTOCOL_PATTERN}`, 'gi');
|
|
|
|
match = regex.exec(str);
|
|
|
|
if (match) {
|
|
|
|
obj.protocol = match[1].toLowerCase();
|
|
|
|
str = str.substring(regex.lastIndex);
|
|
|
|
}
|
2017-02-01 04:25:09 +00:00
|
|
|
|
2017-05-09 05:15:43 +00:00
|
|
|
// authority
|
|
|
|
regex = new RegExp(`^${_URI_AUTHORITY_PATTERN}`, 'gi');
|
|
|
|
match = regex.exec(str);
|
|
|
|
if (match) {
|
|
|
|
let authority = match[1].substring(/* // */ 2);
|
2017-02-01 04:25:09 +00:00
|
|
|
|
2017-05-09 05:15:43 +00:00
|
|
|
str = str.substring(regex.lastIndex);
|
2017-02-01 04:25:09 +00:00
|
|
|
|
2017-05-09 05:15:43 +00:00
|
|
|
// userinfo
|
|
|
|
const userinfoEndIndex = authority.indexOf('@');
|
2017-02-01 04:25:09 +00:00
|
|
|
|
2017-05-09 05:15:43 +00:00
|
|
|
if (userinfoEndIndex !== -1) {
|
|
|
|
authority = authority.substring(userinfoEndIndex + 1);
|
2017-02-01 04:25:09 +00:00
|
|
|
}
|
|
|
|
|
2017-05-09 05:15:43 +00:00
|
|
|
obj.host = authority;
|
2017-02-01 04:25:09 +00:00
|
|
|
|
2017-05-09 05:15:43 +00:00
|
|
|
// port
|
|
|
|
const portBeginIndex = authority.lastIndexOf(':');
|
2017-02-01 04:25:09 +00:00
|
|
|
|
2017-05-09 05:15:43 +00:00
|
|
|
if (portBeginIndex !== -1) {
|
|
|
|
obj.port = authority.substring(portBeginIndex + 1);
|
|
|
|
authority = authority.substring(0, portBeginIndex);
|
|
|
|
}
|
2017-02-01 04:25:09 +00:00
|
|
|
|
2017-05-09 05:15:43 +00:00
|
|
|
// hostname
|
|
|
|
obj.hostname = authority;
|
|
|
|
}
|
2017-02-01 04:25:09 +00:00
|
|
|
|
2017-05-09 05:15:43 +00:00
|
|
|
// pathname
|
|
|
|
regex = new RegExp(`^${_URI_PATH_PATTERN}`, 'gi');
|
|
|
|
match = regex.exec(str);
|
2017-02-01 04:25:09 +00:00
|
|
|
|
2017-05-09 05:15:43 +00:00
|
|
|
let pathname;
|
2017-02-01 04:25:09 +00:00
|
|
|
|
2017-05-09 05:15:43 +00:00
|
|
|
if (match) {
|
|
|
|
pathname = match[1];
|
|
|
|
str = str.substring(regex.lastIndex);
|
|
|
|
}
|
|
|
|
if (pathname) {
|
|
|
|
if (!pathname.startsWith('/')) {
|
|
|
|
pathname = `/${pathname}`;
|
2017-02-01 04:25:09 +00:00
|
|
|
}
|
2017-05-09 05:15:43 +00:00
|
|
|
} else {
|
|
|
|
pathname = '/';
|
|
|
|
}
|
|
|
|
obj.pathname = pathname;
|
|
|
|
|
|
|
|
// query
|
|
|
|
if (str.startsWith('?')) {
|
|
|
|
let hashBeginIndex = str.indexOf('#', 1);
|
2017-02-01 04:25:09 +00:00
|
|
|
|
2017-05-09 05:15:43 +00:00
|
|
|
if (hashBeginIndex === -1) {
|
|
|
|
hashBeginIndex = str.length;
|
2016-10-05 14:36:59 +00:00
|
|
|
}
|
2017-05-09 05:15:43 +00:00
|
|
|
obj.search = str.substring(0, hashBeginIndex);
|
|
|
|
str = str.substring(hashBeginIndex);
|
|
|
|
} else {
|
|
|
|
obj.search = ''; // Google Chrome
|
|
|
|
}
|
|
|
|
|
|
|
|
// fragment
|
|
|
|
obj.hash = str.startsWith('#') ? str : '';
|
|
|
|
|
|
|
|
/* eslint-enable no-param-reassign */
|
|
|
|
|
|
|
|
return obj;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Parses a specific URI which (supposedly) references a Jitsi Meet resource
|
|
|
|
* (location).
|
|
|
|
*
|
|
|
|
* @param {(string|undefined)} uri - The URI to parse which (supposedly)
|
|
|
|
* references a Jitsi Meet resource (location).
|
|
|
|
* @returns {{
|
|
|
|
* room: (string|undefined)
|
|
|
|
* }}
|
|
|
|
*/
|
|
|
|
export function _parseURIString(uri: ?string) {
|
|
|
|
if (typeof uri !== 'string') {
|
|
|
|
return undefined;
|
2016-10-05 14:36:59 +00:00
|
|
|
}
|
|
|
|
|
2017-05-09 05:15:43 +00:00
|
|
|
const obj
|
|
|
|
= _parseStandardURIString(
|
|
|
|
_fixURIStringHierPart(_fixURIStringScheme(uri)));
|
|
|
|
|
|
|
|
// Add the properties that are specific to a Jitsi Meet resource (location)
|
|
|
|
// such as contextRoot, room:
|
|
|
|
|
|
|
|
// contextRoot
|
|
|
|
obj.contextRoot = getLocationContextRoot(obj);
|
|
|
|
|
|
|
|
// The room (name) is the last component of pathname.
|
|
|
|
const { pathname } = obj;
|
|
|
|
|
|
|
|
obj.room = pathname.substring(pathname.lastIndexOf('/') + 1) || undefined;
|
|
|
|
|
|
|
|
return obj;
|
2016-10-05 14:36:59 +00:00
|
|
|
}
|