2017-10-03 19:24:00 +00:00
|
|
|
// @flow
|
|
|
|
|
2018-05-10 04:45:24 +00:00
|
|
|
import { getAppProp } from '../app';
|
2018-05-17 15:45:51 +00:00
|
|
|
import { isLocalParticipantModerator } from '../base/participants';
|
2018-03-23 16:37:04 +00:00
|
|
|
import { doGetJSON } from '../base/util';
|
|
|
|
|
2017-06-14 18:41:22 +00:00
|
|
|
declare var $: Function;
|
2017-10-03 16:30:42 +00:00
|
|
|
declare var interfaceConfig: Object;
|
2017-06-14 18:41:22 +00:00
|
|
|
|
2018-03-23 16:37:04 +00:00
|
|
|
const logger = require('jitsi-meet-logger').getLogger(__filename);
|
|
|
|
|
2018-05-01 04:43:47 +00:00
|
|
|
/**
|
|
|
|
* Sends an ajax request to check if the phone number can be called.
|
|
|
|
*
|
|
|
|
* @param {string} dialNumber - The dial number to check for validity.
|
|
|
|
* @param {string} dialOutAuthUrl - The endpoint to use for checking validity.
|
|
|
|
* @returns {Promise} - The promise created by the request.
|
|
|
|
*/
|
|
|
|
export function checkDialNumber(
|
|
|
|
dialNumber: string,
|
|
|
|
dialOutAuthUrl: string
|
|
|
|
): Promise<Object> {
|
|
|
|
|
|
|
|
if (!dialOutAuthUrl) {
|
|
|
|
// no auth url, let's say it is valid
|
|
|
|
const response = {
|
|
|
|
allow: true,
|
|
|
|
phone: `+${dialNumber}`
|
|
|
|
};
|
|
|
|
|
|
|
|
return Promise.resolve(response);
|
|
|
|
}
|
|
|
|
|
|
|
|
const fullUrl = `${dialOutAuthUrl}?phone=${dialNumber}`;
|
|
|
|
|
|
|
|
return new Promise((resolve, reject) => {
|
|
|
|
$.getJSON(fullUrl)
|
|
|
|
.then(resolve)
|
|
|
|
.catch(reject);
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
2018-03-23 16:37:04 +00:00
|
|
|
/**
|
|
|
|
* Sends a GET request to obtain the conference ID necessary for identifying
|
|
|
|
* which conference to join after diaing the dial-in service.
|
|
|
|
*
|
|
|
|
* @param {string} baseUrl - The url for obtaining the conference ID (pin) for
|
|
|
|
* dialing into a conference.
|
|
|
|
* @param {string} roomName - The conference name to find the associated
|
|
|
|
* conference ID.
|
|
|
|
* @param {string} mucURL - In which MUC the conference exists.
|
|
|
|
* @returns {Promise} - The promise created by the request.
|
|
|
|
*/
|
|
|
|
export function getDialInConferenceID(
|
|
|
|
baseUrl: string,
|
|
|
|
roomName: string,
|
2018-05-01 04:43:47 +00:00
|
|
|
mucURL: string
|
|
|
|
): Promise<Object> {
|
|
|
|
|
2018-03-23 16:37:04 +00:00
|
|
|
const conferenceIDURL = `${baseUrl}?conference=${roomName}@${mucURL}`;
|
|
|
|
|
|
|
|
return doGetJSON(conferenceIDURL);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Sends a GET request for phone numbers used to dial into a conference.
|
|
|
|
*
|
|
|
|
* @param {string} url - The service that returns confernce dial-in numbers.
|
|
|
|
* @returns {Promise} - The promise created by the request. The returned numbers
|
|
|
|
* may be an array of numbers or an object with countries as keys and arrays of
|
|
|
|
* phone number strings.
|
|
|
|
*/
|
|
|
|
export function getDialInNumbers(url: string): Promise<*> {
|
|
|
|
return doGetJSON(url);
|
|
|
|
}
|
|
|
|
|
2018-03-28 00:49:22 +00:00
|
|
|
/**
|
|
|
|
* Removes all non-numeric characters from a string.
|
|
|
|
*
|
2018-05-03 01:20:31 +00:00
|
|
|
* @param {string} text - The string from which to remove all characters except
|
|
|
|
* numbers.
|
2018-03-28 00:49:22 +00:00
|
|
|
* @returns {string} A string with only numbers.
|
|
|
|
*/
|
2018-04-25 11:06:46 +00:00
|
|
|
export function getDigitsOnly(text: string = ''): string {
|
2018-03-28 00:49:22 +00:00
|
|
|
return text.replace(/\D/g, '');
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Type of the options to use when sending a search query.
|
|
|
|
*/
|
|
|
|
export type GetInviteResultsOptions = {
|
|
|
|
|
|
|
|
/**
|
|
|
|
* The endpoint to use for checking phone number validity.
|
|
|
|
*/
|
|
|
|
dialOutAuthUrl: string,
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Whether or not to search for people.
|
|
|
|
*/
|
2018-05-01 04:43:47 +00:00
|
|
|
addPeopleEnabled: boolean,
|
2018-03-28 00:49:22 +00:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Whether or not to check phone numbers.
|
|
|
|
*/
|
2018-05-01 04:43:47 +00:00
|
|
|
dialOutEnabled: boolean,
|
2018-03-28 00:49:22 +00:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Array with the query types that will be executed -
|
|
|
|
* "conferenceRooms" | "user" | "room".
|
|
|
|
*/
|
|
|
|
peopleSearchQueryTypes: Array<string>,
|
|
|
|
|
|
|
|
/**
|
|
|
|
* The url to query for people.
|
|
|
|
*/
|
|
|
|
peopleSearchUrl: string,
|
|
|
|
|
|
|
|
/**
|
|
|
|
* The jwt token to pass to the search service.
|
|
|
|
*/
|
|
|
|
jwt: string
|
|
|
|
};
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Combines directory search with phone number validation to produce a single
|
|
|
|
* set of invite search results.
|
|
|
|
*
|
2018-05-03 01:20:31 +00:00
|
|
|
* @param {string} query - Text to search.
|
|
|
|
* @param {GetInviteResultsOptions} options - Options to use when searching.
|
2018-03-28 00:49:22 +00:00
|
|
|
* @returns {Promise<*>}
|
|
|
|
*/
|
|
|
|
export function getInviteResultsForQuery(
|
|
|
|
query: string,
|
2018-05-01 04:43:47 +00:00
|
|
|
options: GetInviteResultsOptions
|
|
|
|
): Promise<*> {
|
|
|
|
|
2018-03-28 00:49:22 +00:00
|
|
|
const text = query.trim();
|
|
|
|
|
|
|
|
const {
|
|
|
|
dialOutAuthUrl,
|
2018-05-01 04:43:47 +00:00
|
|
|
addPeopleEnabled,
|
|
|
|
dialOutEnabled,
|
2018-03-28 00:49:22 +00:00
|
|
|
peopleSearchQueryTypes,
|
|
|
|
peopleSearchUrl,
|
|
|
|
jwt
|
|
|
|
} = options;
|
|
|
|
|
|
|
|
let peopleSearchPromise;
|
|
|
|
|
2018-05-01 04:43:47 +00:00
|
|
|
if (addPeopleEnabled && text) {
|
2018-03-28 00:49:22 +00:00
|
|
|
peopleSearchPromise = searchDirectory(
|
|
|
|
peopleSearchUrl,
|
|
|
|
jwt,
|
|
|
|
text,
|
|
|
|
peopleSearchQueryTypes);
|
|
|
|
} else {
|
|
|
|
peopleSearchPromise = Promise.resolve([]);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
const hasCountryCode = text.startsWith('+');
|
|
|
|
let phoneNumberPromise;
|
|
|
|
|
2018-05-01 04:43:47 +00:00
|
|
|
if (dialOutEnabled && isMaybeAPhoneNumber(text)) {
|
2018-03-28 00:49:22 +00:00
|
|
|
let numberToVerify = text;
|
|
|
|
|
|
|
|
// When the number to verify does not start with a +, we assume no
|
2018-05-03 01:20:31 +00:00
|
|
|
// proper country code has been entered. In such a case, prepend 1 for
|
|
|
|
// the country code. The service currently takes care of prepending the
|
|
|
|
// +.
|
2018-03-28 00:49:22 +00:00
|
|
|
if (!hasCountryCode && !text.startsWith('1')) {
|
|
|
|
numberToVerify = `1${numberToVerify}`;
|
|
|
|
}
|
|
|
|
|
2018-05-03 01:20:31 +00:00
|
|
|
// The validation service works properly when the query is digits only
|
|
|
|
// so ensure only digits get sent.
|
2018-03-28 00:49:22 +00:00
|
|
|
numberToVerify = getDigitsOnly(numberToVerify);
|
|
|
|
|
2018-05-03 01:20:31 +00:00
|
|
|
phoneNumberPromise = checkDialNumber(numberToVerify, dialOutAuthUrl);
|
2018-03-28 00:49:22 +00:00
|
|
|
} else {
|
|
|
|
phoneNumberPromise = Promise.resolve({});
|
|
|
|
}
|
|
|
|
|
|
|
|
return Promise.all([ peopleSearchPromise, phoneNumberPromise ])
|
|
|
|
.then(([ peopleResults, phoneResults ]) => {
|
|
|
|
const results = [
|
|
|
|
...peopleResults
|
|
|
|
];
|
|
|
|
|
|
|
|
/**
|
2018-05-03 01:20:31 +00:00
|
|
|
* This check for phone results is for the day the call to searching
|
|
|
|
* people might return phone results as well. When that day comes
|
|
|
|
* this check will make it so the server checks are honored and the
|
|
|
|
* local appending of the number is not done. The local appending of
|
|
|
|
* the phone number can then be cleaned up when convenient.
|
2018-03-28 00:49:22 +00:00
|
|
|
*/
|
2018-05-03 01:20:31 +00:00
|
|
|
const hasPhoneResult
|
|
|
|
= peopleResults.find(result => result.type === 'phone');
|
2018-03-28 00:49:22 +00:00
|
|
|
|
2018-05-03 01:20:31 +00:00
|
|
|
if (!hasPhoneResult && typeof phoneResults.allow === 'boolean') {
|
2018-03-28 00:49:22 +00:00
|
|
|
results.push({
|
|
|
|
allowed: phoneResults.allow,
|
|
|
|
country: phoneResults.country,
|
|
|
|
type: 'phone',
|
|
|
|
number: phoneResults.phone,
|
|
|
|
originalEntry: text,
|
|
|
|
showCountryCodeReminder: !hasCountryCode
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
return results;
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
2018-05-03 01:20:31 +00:00
|
|
|
/**
|
|
|
|
* Helper for determining how many of each type of user is being invited. Used
|
|
|
|
* for logging and sending analytics related to invites.
|
|
|
|
*
|
|
|
|
* @param {Array} inviteItems - An array with the invite items, as created in
|
|
|
|
* {@link _parseQueryResults}.
|
|
|
|
* @returns {Object} An object with keys as user types and values as the number
|
|
|
|
* of invites for that type.
|
|
|
|
*/
|
|
|
|
export function getInviteTypeCounts(inviteItems: Array<Object> = []) {
|
|
|
|
const inviteTypeCounts = {};
|
|
|
|
|
|
|
|
inviteItems.forEach(({ type }) => {
|
|
|
|
if (!inviteTypeCounts[type]) {
|
|
|
|
inviteTypeCounts[type] = 0;
|
|
|
|
}
|
|
|
|
inviteTypeCounts[type]++;
|
|
|
|
});
|
|
|
|
|
|
|
|
return inviteTypeCounts;
|
|
|
|
}
|
|
|
|
|
2018-05-01 04:43:47 +00:00
|
|
|
/**
|
|
|
|
* Sends a post request to an invite service.
|
|
|
|
*
|
|
|
|
* @param {string} inviteServiceUrl - The invite service that generates the
|
|
|
|
* invitation.
|
|
|
|
* @param {string} inviteUrl - The url to the conference.
|
|
|
|
* @param {string} jwt - The jwt token to pass to the search service.
|
2018-05-03 01:20:31 +00:00
|
|
|
* @param {Immutable.List} inviteItems - The list of the "user" or "room" type
|
|
|
|
* items to invite.
|
2018-05-01 04:43:47 +00:00
|
|
|
* @returns {Promise} - The promise created by the request.
|
|
|
|
*/
|
2018-04-25 11:06:46 +00:00
|
|
|
export function invitePeopleAndChatRooms( // eslint-disable-line max-params
|
2018-05-01 04:43:47 +00:00
|
|
|
inviteServiceUrl: string,
|
|
|
|
inviteUrl: string,
|
|
|
|
jwt: string,
|
|
|
|
inviteItems: Array<Object>
|
|
|
|
): Promise<void> {
|
|
|
|
|
|
|
|
if (!inviteItems || inviteItems.length === 0) {
|
|
|
|
return Promise.resolve();
|
|
|
|
}
|
|
|
|
|
|
|
|
return new Promise((resolve, reject) => {
|
|
|
|
$.post(
|
|
|
|
`${inviteServiceUrl}?token=${jwt}`,
|
|
|
|
JSON.stringify({
|
|
|
|
'invited': inviteItems,
|
|
|
|
'url': inviteUrl
|
|
|
|
}),
|
|
|
|
resolve,
|
|
|
|
'json')
|
|
|
|
.fail((jqxhr, textStatus, error) => reject(error));
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Determines if adding people is currently enabled.
|
|
|
|
*
|
|
|
|
* @param {boolean} state - Current state.
|
|
|
|
* @returns {boolean} Indication of whether adding people is currently enabled.
|
|
|
|
*/
|
|
|
|
export function isAddPeopleEnabled(state: Object): boolean {
|
|
|
|
const { isGuest } = state['features/base/jwt'];
|
|
|
|
|
|
|
|
if (!isGuest) {
|
|
|
|
// XXX The mobile/react-native app is capable of disabling the
|
|
|
|
// adding/inviting of people in the current conference. Anyway, the
|
|
|
|
// Web/React app does not have that capability so default appropriately.
|
2018-05-10 04:45:24 +00:00
|
|
|
const addPeopleEnabled = getAppProp(state, 'addPeopleEnabled');
|
2018-05-01 04:43:47 +00:00
|
|
|
|
|
|
|
return (
|
|
|
|
(typeof addPeopleEnabled === 'undefined')
|
|
|
|
|| Boolean(addPeopleEnabled));
|
|
|
|
}
|
|
|
|
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Determines if dial out is currently enabled or not.
|
|
|
|
*
|
|
|
|
* @param {boolean} state - Current state.
|
|
|
|
* @returns {boolean} Indication of whether dial out is currently enabled.
|
|
|
|
*/
|
|
|
|
export function isDialOutEnabled(state: Object): boolean {
|
|
|
|
const { conference } = state['features/base/conference'];
|
2018-05-17 15:45:51 +00:00
|
|
|
let dialOutEnabled = isLocalParticipantModerator(state)
|
|
|
|
&& conference
|
|
|
|
&& conference.isSIPCallingSupported();
|
2018-05-03 01:20:31 +00:00
|
|
|
|
|
|
|
if (dialOutEnabled) {
|
|
|
|
// XXX The mobile/react-native app is capable of disabling of dial-out.
|
|
|
|
// Anyway, the Web/React app does not have that capability so default
|
|
|
|
// appropriately.
|
2018-05-10 04:45:24 +00:00
|
|
|
dialOutEnabled = getAppProp(state, 'dialOutEnabled');
|
2018-05-03 01:20:31 +00:00
|
|
|
|
|
|
|
return (
|
|
|
|
(typeof dialOutEnabled === 'undefined') || Boolean(dialOutEnabled));
|
|
|
|
}
|
2018-05-01 04:43:47 +00:00
|
|
|
|
2018-05-03 01:20:31 +00:00
|
|
|
return false;
|
2018-05-01 04:43:47 +00:00
|
|
|
}
|
|
|
|
|
2018-03-28 00:49:22 +00:00
|
|
|
/**
|
|
|
|
* Checks whether a string looks like it could be for a phone number.
|
|
|
|
*
|
2018-05-03 01:20:31 +00:00
|
|
|
* @param {string} text - The text to check whether or not it could be a phone
|
2018-03-28 00:49:22 +00:00
|
|
|
* number.
|
2018-05-03 01:20:31 +00:00
|
|
|
* @private
|
|
|
|
* @returns {boolean} True if the string looks like it could be a phone number.
|
2018-03-28 00:49:22 +00:00
|
|
|
*/
|
|
|
|
function isMaybeAPhoneNumber(text: string): boolean {
|
|
|
|
if (!isPhoneNumberRegex().test(text)) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
const digits = getDigitsOnly(text);
|
|
|
|
|
|
|
|
return Boolean(digits.length);
|
|
|
|
}
|
|
|
|
|
2018-05-01 04:43:47 +00:00
|
|
|
/**
|
|
|
|
* RegExp to use to determine if some text might be a phone number.
|
|
|
|
*
|
|
|
|
* @returns {RegExp}
|
|
|
|
*/
|
|
|
|
function isPhoneNumberRegex(): RegExp {
|
|
|
|
let regexString = '^[0-9+()-\\s]*$';
|
|
|
|
|
|
|
|
if (typeof interfaceConfig !== 'undefined') {
|
|
|
|
regexString = interfaceConfig.PHONE_NUMBER_REGEX || regexString;
|
|
|
|
}
|
|
|
|
|
|
|
|
return new RegExp(regexString);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Sends an ajax request to a directory service.
|
|
|
|
*
|
|
|
|
* @param {string} serviceUrl - The service to query.
|
|
|
|
* @param {string} jwt - The jwt token to pass to the search service.
|
|
|
|
* @param {string} text - Text to search.
|
|
|
|
* @param {Array<string>} queryTypes - Array with the query types that will be
|
|
|
|
* executed - "conferenceRooms" | "user" | "room".
|
|
|
|
* @returns {Promise} - The promise created by the request.
|
|
|
|
*/
|
|
|
|
export function searchDirectory( // eslint-disable-line max-params
|
|
|
|
serviceUrl: string,
|
|
|
|
jwt: string,
|
|
|
|
text: string,
|
|
|
|
queryTypes: Array<string> = [ 'conferenceRooms', 'user', 'room' ]
|
|
|
|
): Promise<Array<Object>> {
|
|
|
|
|
|
|
|
const query = encodeURIComponent(text);
|
|
|
|
const queryTypesString = encodeURIComponent(JSON.stringify(queryTypes));
|
|
|
|
|
|
|
|
return fetch(`${serviceUrl}?query=${query}&queryTypes=${
|
|
|
|
queryTypesString}&jwt=${jwt}`)
|
|
|
|
.then(response => {
|
|
|
|
const jsonify = response.json();
|
|
|
|
|
|
|
|
if (response.ok) {
|
|
|
|
return jsonify;
|
|
|
|
}
|
|
|
|
|
|
|
|
return jsonify
|
|
|
|
.then(result => Promise.reject(result));
|
|
|
|
})
|
|
|
|
.catch(error => {
|
|
|
|
logger.error(
|
|
|
|
'Error searching directory:', error);
|
|
|
|
|
|
|
|
return Promise.reject(error);
|
|
|
|
});
|
|
|
|
}
|