2016-05-06 13:19:56 +00:00
|
|
|
/*
|
|
|
|
Copyright 2016 OpenMarket Ltd
|
2019-07-09 17:51:56 +00:00
|
|
|
Copyright 2019 The Matrix.org Foundation C.I.C.
|
2016-05-06 13:19:56 +00:00
|
|
|
|
|
|
|
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.
|
|
|
|
*/
|
|
|
|
|
2019-07-22 17:54:04 +00:00
|
|
|
import url from 'url';
|
2017-07-12 12:58:14 +00:00
|
|
|
import Promise from 'bluebird';
|
2017-12-17 03:13:27 +00:00
|
|
|
import SettingsStore from "./settings/SettingsStore";
|
2019-07-23 14:11:38 +00:00
|
|
|
import { Service, startTermsFlow, TermsNotSignedError } from './Terms';
|
2017-10-11 16:56:17 +00:00
|
|
|
const request = require('browser-request');
|
2016-05-06 13:19:56 +00:00
|
|
|
|
2017-10-11 16:56:17 +00:00
|
|
|
const SdkConfig = require('./SdkConfig');
|
|
|
|
const MatrixClientPeg = require('./MatrixClientPeg');
|
2016-05-06 13:19:56 +00:00
|
|
|
|
2019-07-09 17:51:56 +00:00
|
|
|
import * as Matrix from 'matrix-js-sdk';
|
|
|
|
|
2019-03-13 10:16:44 +00:00
|
|
|
// The version of the integration manager API we're intending to work with
|
|
|
|
const imApiVersion = "1.1";
|
|
|
|
|
2016-05-06 13:19:56 +00:00
|
|
|
class ScalarAuthClient {
|
2016-09-02 15:03:24 +00:00
|
|
|
constructor() {
|
|
|
|
this.scalarToken = null;
|
2019-07-23 14:11:38 +00:00
|
|
|
// `undefined` to allow `startTermsFlow` to fallback to a default
|
|
|
|
// callback if this is unset.
|
|
|
|
this.termsInteractionCallback = undefined;
|
2016-09-02 15:03:24 +00:00
|
|
|
}
|
|
|
|
|
2019-06-17 21:29:19 +00:00
|
|
|
/**
|
|
|
|
* Determines if setting up a ScalarAuthClient is even possible
|
|
|
|
* @returns {boolean} true if possible, false otherwise.
|
|
|
|
*/
|
|
|
|
static isPossible() {
|
|
|
|
return SdkConfig.get()['integrations_rest_url'] && SdkConfig.get()['integrations_ui_url'];
|
|
|
|
}
|
|
|
|
|
2019-07-23 14:11:38 +00:00
|
|
|
setTermsInteractionCallback(callback) {
|
|
|
|
this.termsInteractionCallback = callback;
|
|
|
|
}
|
|
|
|
|
2016-09-02 15:03:24 +00:00
|
|
|
connect() {
|
|
|
|
return this.getScalarToken().then((tok) => {
|
|
|
|
this.scalarToken = tok;
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
hasCredentials() {
|
|
|
|
return this.scalarToken != null; // undef or null
|
|
|
|
}
|
|
|
|
|
2019-07-11 15:00:24 +00:00
|
|
|
// Returns a promise that resolves to a scalar_token string
|
2016-09-02 15:03:24 +00:00
|
|
|
getScalarToken() {
|
2019-06-17 21:29:19 +00:00
|
|
|
let token = this.scalarToken;
|
|
|
|
if (!token) token = window.localStorage.getItem("mx_scalar_token");
|
2016-09-02 15:03:24 +00:00
|
|
|
|
2017-12-17 03:04:32 +00:00
|
|
|
if (!token) {
|
|
|
|
return this.registerForToken();
|
|
|
|
} else {
|
2019-07-15 13:05:39 +00:00
|
|
|
return this._checkToken(token).catch((e) => {
|
|
|
|
if (e instanceof TermsNotSignedError) {
|
|
|
|
// retrying won't help this
|
|
|
|
throw e;
|
|
|
|
}
|
2019-07-11 15:00:24 +00:00
|
|
|
return this.registerForToken();
|
|
|
|
});
|
2017-12-17 03:04:32 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-07-09 17:51:56 +00:00
|
|
|
_getAccountName(token) {
|
2018-10-12 02:54:53 +00:00
|
|
|
const url = SdkConfig.get().integrations_rest_url + "/account";
|
2017-12-17 03:04:32 +00:00
|
|
|
|
2018-10-02 00:57:27 +00:00
|
|
|
return new Promise(function(resolve, reject) {
|
|
|
|
request({
|
|
|
|
method: "GET",
|
|
|
|
uri: url,
|
2019-03-13 10:16:44 +00:00
|
|
|
qs: {scalar_token: token, v: imApiVersion},
|
2018-10-02 00:57:27 +00:00
|
|
|
json: true,
|
|
|
|
}, (err, response, body) => {
|
|
|
|
if (err) {
|
|
|
|
reject(err);
|
2019-07-09 17:51:56 +00:00
|
|
|
} else if (body && body.errcode === 'M_TERMS_NOT_SIGNED') {
|
|
|
|
reject(new TermsNotSignedError());
|
2018-10-02 00:57:27 +00:00
|
|
|
} else if (response.statusCode / 100 !== 2) {
|
2019-07-09 17:51:56 +00:00
|
|
|
reject(body);
|
2018-10-02 00:57:27 +00:00
|
|
|
} else if (!body || !body.user_id) {
|
|
|
|
reject(new Error("Missing user_id in response"));
|
|
|
|
} else {
|
|
|
|
resolve(body.user_id);
|
|
|
|
}
|
|
|
|
});
|
2018-10-12 02:54:53 +00:00
|
|
|
});
|
2017-12-17 03:04:32 +00:00
|
|
|
}
|
2016-09-02 15:03:24 +00:00
|
|
|
|
2019-07-09 17:51:56 +00:00
|
|
|
_checkToken(token) {
|
|
|
|
return this._getAccountName(token).then(userId => {
|
|
|
|
const me = MatrixClientPeg.get().getUserId();
|
|
|
|
if (userId !== me) {
|
|
|
|
throw new Error("Scalar token is owned by someone else: " + me);
|
|
|
|
}
|
|
|
|
return token;
|
|
|
|
}).catch((e) => {
|
|
|
|
if (e instanceof TermsNotSignedError) {
|
|
|
|
console.log("Integrations manager requires new terms to be agreed to");
|
2019-07-22 17:54:04 +00:00
|
|
|
// The terms endpoints are new and so live on standard _matrix prefixes,
|
|
|
|
// but IM rest urls are currently configured with paths, so remove the
|
|
|
|
// path from the base URL before passing it to the js-sdk
|
2019-07-23 09:09:16 +00:00
|
|
|
|
|
|
|
// We continue to use the full URL for the calls done by
|
|
|
|
// matrix-react-sdk, but the standard terms API called
|
|
|
|
// by the js-sdk lives on the standard _matrix path. This means we
|
|
|
|
// don't support running IMs on a non-root path, but it's the only
|
|
|
|
// realistic way of transitioning to _matrix paths since configs in
|
|
|
|
// the wild contain bits of the API path.
|
|
|
|
|
|
|
|
// Once we've fully transitioned to _matrix URLs, we can give people
|
|
|
|
// a grace period to update their configs, then use the rest url as
|
|
|
|
// a regular base url.
|
2019-07-22 17:54:04 +00:00
|
|
|
const parsedImRestUrl = url.parse(SdkConfig.get().integrations_rest_url);
|
|
|
|
parsedImRestUrl.path = '';
|
|
|
|
parsedImRestUrl.pathname = '';
|
2019-07-23 14:11:38 +00:00
|
|
|
return startTermsFlow([new Service(
|
2019-07-10 11:08:26 +00:00
|
|
|
Matrix.SERVICE_TYPES.IM,
|
2019-07-22 17:54:04 +00:00
|
|
|
parsedImRestUrl.format(),
|
2019-07-09 17:51:56 +00:00
|
|
|
token,
|
2019-07-23 14:11:38 +00:00
|
|
|
)], this.termsInteractionCallback).then(() => {
|
2019-07-09 17:51:56 +00:00
|
|
|
return token;
|
|
|
|
});
|
2019-07-11 15:00:24 +00:00
|
|
|
} else {
|
|
|
|
throw e;
|
2019-07-09 17:51:56 +00:00
|
|
|
}
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
2017-12-17 03:04:32 +00:00
|
|
|
registerForToken() {
|
|
|
|
// Get openid bearer token from the HS as the first part of our dance
|
2019-07-04 11:59:20 +00:00
|
|
|
return MatrixClientPeg.get().getOpenIdToken().then((tokenObject) => {
|
2016-09-02 15:03:24 +00:00
|
|
|
// Now we can send that to scalar and exchange it for a scalar token
|
2019-07-04 11:59:20 +00:00
|
|
|
return this.exchangeForScalarToken(tokenObject);
|
2019-07-09 17:51:56 +00:00
|
|
|
}).then((tokenObject) => {
|
|
|
|
// Validate it (this mostly checks to see if the IM needs us to agree to some terms)
|
|
|
|
return this._checkToken(tokenObject);
|
2019-07-04 11:59:20 +00:00
|
|
|
}).then((tokenObject) => {
|
|
|
|
window.localStorage.setItem("mx_scalar_token", tokenObject);
|
|
|
|
return tokenObject;
|
2016-09-02 15:03:24 +00:00
|
|
|
});
|
|
|
|
}
|
|
|
|
|
2019-07-04 11:59:20 +00:00
|
|
|
exchangeForScalarToken(openidTokenObject) {
|
|
|
|
const scalarRestUrl = SdkConfig.get().integrations_rest_url;
|
2016-05-06 13:19:56 +00:00
|
|
|
|
2018-10-02 00:57:27 +00:00
|
|
|
return new Promise(function(resolve, reject) {
|
|
|
|
request({
|
|
|
|
method: 'POST',
|
2019-07-04 11:59:20 +00:00
|
|
|
uri: scalarRestUrl + '/register',
|
2019-03-13 10:16:44 +00:00
|
|
|
qs: {v: imApiVersion},
|
2019-07-04 11:59:20 +00:00
|
|
|
body: openidTokenObject,
|
2018-10-02 00:57:27 +00:00
|
|
|
json: true,
|
|
|
|
}, (err, response, body) => {
|
|
|
|
if (err) {
|
|
|
|
reject(err);
|
|
|
|
} else if (response.statusCode / 100 !== 2) {
|
|
|
|
reject({statusCode: response.statusCode});
|
|
|
|
} else if (!body || !body.scalar_token) {
|
|
|
|
reject(new Error("Missing scalar_token in response"));
|
|
|
|
} else {
|
|
|
|
resolve(body.scalar_token);
|
|
|
|
}
|
|
|
|
});
|
2018-10-12 02:54:53 +00:00
|
|
|
});
|
2016-05-06 13:19:56 +00:00
|
|
|
}
|
2016-09-02 15:03:24 +00:00
|
|
|
|
2017-12-05 21:41:44 +00:00
|
|
|
getScalarPageTitle(url) {
|
2017-12-08 18:47:00 +00:00
|
|
|
let scalarPageLookupUrl = SdkConfig.get().integrations_rest_url + '/widgets/title_lookup';
|
2017-12-05 21:41:44 +00:00
|
|
|
scalarPageLookupUrl = this.getStarterLink(scalarPageLookupUrl);
|
|
|
|
scalarPageLookupUrl += '&curl=' + encodeURIComponent(url);
|
|
|
|
|
2018-10-02 00:57:27 +00:00
|
|
|
return new Promise(function(resolve, reject) {
|
|
|
|
request({
|
|
|
|
method: 'GET',
|
|
|
|
uri: scalarPageLookupUrl,
|
|
|
|
json: true,
|
|
|
|
}, (err, response, body) => {
|
|
|
|
if (err) {
|
|
|
|
reject(err);
|
|
|
|
} else if (response.statusCode / 100 !== 2) {
|
|
|
|
reject({statusCode: response.statusCode});
|
|
|
|
} else if (!body) {
|
|
|
|
reject(new Error("Missing page title in response"));
|
|
|
|
} else {
|
|
|
|
let title = "";
|
|
|
|
if (body.page_title_cache_item && body.page_title_cache_item.cached_title) {
|
|
|
|
title = body.page_title_cache_item.cached_title;
|
|
|
|
}
|
|
|
|
resolve(title);
|
|
|
|
}
|
|
|
|
});
|
2018-10-12 02:54:53 +00:00
|
|
|
});
|
2017-12-05 21:41:44 +00:00
|
|
|
}
|
|
|
|
|
2018-02-25 23:00:46 +00:00
|
|
|
/**
|
|
|
|
* Mark all assets associated with the specified widget as "disabled" in the
|
|
|
|
* integration manager database.
|
|
|
|
* This can be useful to temporarily prevent purchased assets from being displayed.
|
|
|
|
* @param {string} widgetType [description]
|
|
|
|
* @param {string} widgetId [description]
|
|
|
|
* @return {Promise} Resolves on completion
|
|
|
|
*/
|
2018-02-24 00:10:28 +00:00
|
|
|
disableWidgetAssets(widgetType, widgetId) {
|
|
|
|
let url = SdkConfig.get().integrations_rest_url + '/widgets/set_assets_state';
|
|
|
|
url = this.getStarterLink(url);
|
|
|
|
return new Promise((resolve, reject) => {
|
|
|
|
request({
|
|
|
|
method: 'GET',
|
|
|
|
uri: url,
|
|
|
|
json: true,
|
|
|
|
qs: {
|
|
|
|
'widget_type': widgetType,
|
|
|
|
'widget_id': widgetId,
|
|
|
|
'state': 'disable',
|
|
|
|
},
|
|
|
|
}, (err, response, body) => {
|
|
|
|
if (err) {
|
|
|
|
reject(err);
|
|
|
|
} else if (response.statusCode / 100 !== 2) {
|
|
|
|
reject({statusCode: response.statusCode});
|
|
|
|
} else if (!body) {
|
|
|
|
reject(new Error("Failed to set widget assets state"));
|
|
|
|
} else {
|
|
|
|
resolve();
|
|
|
|
}
|
|
|
|
});
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
2018-02-09 11:44:27 +00:00
|
|
|
getScalarInterfaceUrlForRoom(room, screen, id) {
|
|
|
|
const roomId = room.roomId;
|
|
|
|
const roomName = room.name;
|
2017-10-11 16:56:17 +00:00
|
|
|
let url = SdkConfig.get().integrations_ui_url;
|
2016-09-02 15:03:24 +00:00
|
|
|
url += "?scalar_token=" + encodeURIComponent(this.scalarToken);
|
|
|
|
url += "&room_id=" + encodeURIComponent(roomId);
|
2018-02-09 11:44:27 +00:00
|
|
|
url += "&room_name=" + encodeURIComponent(roomName);
|
2017-12-17 03:13:27 +00:00
|
|
|
url += "&theme=" + encodeURIComponent(SettingsStore.getValue("theme"));
|
2017-08-18 17:40:00 +00:00
|
|
|
if (id) {
|
2017-08-22 08:59:27 +00:00
|
|
|
url += '&integ_id=' + encodeURIComponent(id);
|
2017-08-18 17:40:00 +00:00
|
|
|
}
|
2017-07-06 14:59:59 +00:00
|
|
|
if (screen) {
|
|
|
|
url += '&screen=' + encodeURIComponent(screen);
|
|
|
|
}
|
2016-09-02 15:03:24 +00:00
|
|
|
return url;
|
|
|
|
}
|
2016-09-02 15:36:43 +00:00
|
|
|
|
|
|
|
getStarterLink(starterLinkUrl) {
|
|
|
|
return starterLinkUrl + "?scalar_token=" + encodeURIComponent(this.scalarToken);
|
|
|
|
}
|
2016-05-06 13:19:56 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
module.exports = ScalarAuthClient;
|