element-web/src/Terms.ts

201 lines
7.2 KiB
TypeScript
Raw Normal View History

2019-07-10 09:50:10 +00:00
/*
Copyright 2024 New Vector Ltd.
Copyright 2019-2021 The Matrix.org Foundation C.I.C.
2019-07-10 09:50:10 +00:00
SPDX-License-Identifier: AGPL-3.0-only OR GPL-3.0-only
Please see LICENSE files in the repository root for full details.
2019-07-10 09:50:10 +00:00
*/
2022-12-12 11:24:14 +00:00
import classNames from "classnames";
import { SERVICE_TYPES, MatrixClient } from "matrix-js-sdk/src/matrix";
import { logger } from "matrix-js-sdk/src/logger";
2019-07-10 09:50:10 +00:00
2022-12-12 11:24:14 +00:00
import Modal from "./Modal";
import TermsDialog from "./components/views/dialogs/TermsDialog";
2019-07-10 09:50:10 +00:00
export class TermsNotSignedError extends Error {}
/**
* Class representing a service that may have terms & conditions that
* require agreement from the user before the user can use that service.
2019-07-10 09:50:10 +00:00
*/
export class Service {
/**
2019-07-10 11:08:26 +00:00
* @param {MatrixClient.SERVICE_TYPES} serviceType The type of service
2019-07-10 09:50:10 +00:00
* @param {string} baseUrl The Base URL of the service (ie. before '/_matrix')
* @param {string} accessToken The user's access token for the service
*/
public constructor(
public serviceType: SERVICE_TYPES,
public baseUrl: string,
public accessToken: string,
) {}
2019-07-10 09:50:10 +00:00
}
export interface LocalisedPolicy {
name: string;
url: string;
}
export interface Policy {
2021-04-06 11:26:50 +00:00
// @ts-ignore: No great way to express indexed types together with other keys
version: string;
[lang: string]: LocalisedPolicy;
2021-04-06 11:26:50 +00:00
}
export type Policies = {
[policy: string]: Policy;
2021-04-06 11:26:50 +00:00
};
export type ServicePolicyPair = {
policies: Policies;
service: Service;
};
2021-04-06 11:26:50 +00:00
export type TermsInteractionCallback = (
policiesAndServicePairs: ServicePolicyPair[],
2021-04-06 11:26:50 +00:00
agreedUrls: string[],
extraClassNames?: string,
) => Promise<string[]>;
/**
2019-07-10 13:22:50 +00:00
* Start a flow where the user is presented with terms & conditions for some services
*
* @param client The Matrix Client instance of the logged-in user
* @param {Service[]} services Object with keys 'serviceType', 'baseUrl', 'accessToken'
* @param {function} interactionCallback Function called with:
* * an array of { service: {Service}, policies: {terms response from API} }
* * an array of URLs the user has already agreed to
2019-07-10 13:22:50 +00:00
* Must return a Promise which resolves with a list of URLs of documents agreed to
* @returns {Promise} resolves when the user agreed to all necessary terms or rejects
* if they cancel.
*/
export async function startTermsFlow(
client: MatrixClient,
2021-04-06 11:26:50 +00:00
services: Service[],
interactionCallback: TermsInteractionCallback = dialogTermsInteractionCallback,
): Promise<void> {
const termsPromises = services.map((s) => client.getTerms(s.serviceType, s.baseUrl));
2019-07-10 09:50:10 +00:00
2019-07-10 13:25:30 +00:00
/*
* a /terms response looks like:
* {
* "policies": {
* "terms_of_service": {
* "version": "2.0",
* "en": {
* "name": "Terms of Service",
* "url": "https://example.org/somewhere/terms-2.0-en.html"
* },
* "fr": {
* "name": "Conditions d'utilisation",
* "url": "https://example.org/somewhere/terms-2.0-fr.html"
* }
* }
* }
* }
*/
2021-04-06 11:26:50 +00:00
const terms: { policies: Policies }[] = await Promise.all(termsPromises);
2022-12-12 11:24:14 +00:00
const policiesAndServicePairs = terms.map((t, i) => {
return { service: services[i], policies: t.policies };
});
2019-07-10 09:50:10 +00:00
// fetch the set of agreed policy URLs from account data
const currentAcceptedTerms = client.getAccountData("m.accepted_terms")?.getContent();
const agreedUrlSet = new Set<string>(currentAcceptedTerms?.accepted || []);
// remove any policies the user has already agreed to and any services where
// they've already agreed to all the policies
// NB. it could be nicer to show the user stuff they've already agreed to,
// but then they'd assume they can un-check the boxes to un-agree to a policy,
// but that is not a thing the API supports, so probably best to just show
// things they've not agreed to yet.
const unagreedPoliciesAndServicePairs: ServicePolicyPair[] = [];
2021-06-29 12:11:58 +00:00
for (const { service, policies } of policiesAndServicePairs) {
const unagreedPolicies: Policies = {};
for (const [policyName, policy] of Object.entries(policies)) {
let policyAgreed = false;
for (const lang of Object.keys(policy)) {
2022-12-12 11:24:14 +00:00
if (lang === "version") continue;
if (agreedUrlSet.has(policy[lang].url)) {
policyAgreed = true;
break;
}
}
if (!policyAgreed) unagreedPolicies[policyName] = policy;
}
if (Object.keys(unagreedPolicies).length > 0) {
2021-06-29 12:11:58 +00:00
unagreedPoliciesAndServicePairs.push({ service, policies: unagreedPolicies });
}
}
// if there's anything left to agree to, prompt the user
const numAcceptedBeforeAgreement = agreedUrlSet.size;
if (unagreedPoliciesAndServicePairs.length > 0) {
const newlyAgreedUrls = await interactionCallback(unagreedPoliciesAndServicePairs, [...agreedUrlSet]);
logger.log("User has agreed to URLs", newlyAgreedUrls);
// Merge with previously agreed URLs
2022-12-12 11:24:14 +00:00
newlyAgreedUrls.forEach((url) => agreedUrlSet.add(url));
} else {
logger.log("User has already agreed to all required policies");
}
// We only ever add to the set of URLs, so if anything has changed then we'd see a different length
if (agreedUrlSet.size !== numAcceptedBeforeAgreement) {
2021-06-29 12:11:58 +00:00
const newAcceptedTerms = { accepted: Array.from(agreedUrlSet) };
await client.setAccountData("m.accepted_terms", newAcceptedTerms);
}
2019-07-10 09:50:10 +00:00
2019-07-10 14:12:05 +00:00
const agreePromises = policiesAndServicePairs.map((policiesAndService) => {
2019-07-10 09:50:10 +00:00
// filter the agreed URL list for ones that are actually for this service
// (one URL may be used for multiple services)
// Not a particularly efficient loop but probably fine given the numbers involved
const urlsForService = Array.from(agreedUrlSet).filter((url) => {
2019-07-10 14:12:05 +00:00
for (const policy of Object.values(policiesAndService.policies)) {
for (const lang of Object.keys(policy)) {
2022-12-12 11:24:14 +00:00
if (lang === "version") continue;
2019-07-10 14:12:05 +00:00
if (policy[lang].url === url) return true;
2019-07-10 09:50:10 +00:00
}
}
return false;
});
if (urlsForService.length === 0) return Promise.resolve();
return client.agreeToTerms(
2019-07-10 14:12:05 +00:00
policiesAndService.service.serviceType,
policiesAndService.service.baseUrl,
policiesAndService.service.accessToken,
2019-07-10 09:50:10 +00:00
urlsForService,
);
});
await Promise.all(agreePromises);
2019-07-10 09:50:10 +00:00
}
export async function dialogTermsInteractionCallback(
2021-04-06 11:26:50 +00:00
policiesAndServicePairs: {
service: Service;
policies: { [policy: string]: Policy };
2021-04-06 11:26:50 +00:00
}[],
agreedUrls: string[],
extraClassNames?: string,
): Promise<string[]> {
logger.log("Terms that need agreement", policiesAndServicePairs);
const { finished } = Modal.createDialog(
2022-12-12 11:24:14 +00:00
TermsDialog,
{
policiesAndServicePairs,
agreedUrls,
},
classNames("mx_TermsDialog", extraClassNames),
);
const [done, _agreedUrls] = await finished;
if (!done || !_agreedUrls) {
throw new TermsNotSignedError();
}
return _agreedUrls;
2019-07-10 09:50:10 +00:00
}