139 lines
3.7 KiB
TypeScript
139 lines
3.7 KiB
TypeScript
import { settings as RocketChatSettings } from '@rocket.chat/sdk';
|
|
import { KJUR } from 'jsrsasign';
|
|
import moment from 'moment';
|
|
|
|
import { getSupportedVersionsCloud } from '../services/restApi';
|
|
import { TCloudInfo, IServerInfo, ISupportedVersions, ISupportedVersionsData, IApiServerInfo } from '../../definitions';
|
|
import { selectServerFailure } from '../../actions/server';
|
|
import { store } from '../store/auxStore';
|
|
import I18n from '../../i18n';
|
|
import { SIGNED_SUPPORTED_VERSIONS_PUBLIC_KEY } from '../constants';
|
|
import { getServerById } from '../database/services/Server';
|
|
import log from './helpers/log';
|
|
|
|
interface IServerInfoFailure {
|
|
success: false;
|
|
message: string;
|
|
}
|
|
|
|
interface IServerInfoSuccess extends IServerInfo {
|
|
success: true;
|
|
}
|
|
|
|
export type TServerInfoResult = IServerInfoSuccess | IServerInfoFailure;
|
|
|
|
const SV_CLOUD_UPDATE_INTERVAL = 12; // hours
|
|
|
|
// Verifies if JWT is valid and returns the payload
|
|
const verifyJWT = (jwt?: string): ISupportedVersionsData | null => {
|
|
try {
|
|
if (!jwt) {
|
|
return null;
|
|
}
|
|
const isValid = KJUR.jws.JWS.verify(jwt, SIGNED_SUPPORTED_VERSIONS_PUBLIC_KEY, ['RS256']);
|
|
if (!isValid) {
|
|
return null;
|
|
}
|
|
|
|
const { payloadObj } = KJUR.jws.JWS.parse(jwt);
|
|
return payloadObj as ISupportedVersions;
|
|
} catch {
|
|
return null;
|
|
}
|
|
};
|
|
|
|
export async function getServerInfo(server: string): Promise<TServerInfoResult> {
|
|
try {
|
|
const response = await fetch(`${server}/api/info`, {
|
|
...RocketChatSettings.customHeaders
|
|
});
|
|
try {
|
|
const jsonRes: IApiServerInfo = await response.json();
|
|
if (!jsonRes?.success) {
|
|
return {
|
|
success: false,
|
|
message: I18n.t('Not_RC_Server', { contact: I18n.t('Contact_your_server_admin') })
|
|
};
|
|
}
|
|
|
|
// Makes use of signed JWT to get supported versions
|
|
const supportedVersions = verifyJWT(jsonRes.supportedVersions?.signed);
|
|
|
|
// if backend doesn't have supported versions or JWT is invalid, request from cloud
|
|
if (!supportedVersions) {
|
|
// fetches from cloud only every 12h
|
|
const serverRecord = await getServerById(server);
|
|
if (
|
|
serverRecord?.supportedVersionsUpdatedAt &&
|
|
moment(new Date()).diff(serverRecord?.supportedVersionsUpdatedAt, 'hours') <= SV_CLOUD_UPDATE_INTERVAL
|
|
) {
|
|
return {
|
|
...jsonRes,
|
|
success: true
|
|
};
|
|
}
|
|
|
|
const cloudInfo = await getCloudInfo(server);
|
|
|
|
// Allows airgapped servers to use the app until enforcementStartDate
|
|
if (!cloudInfo) {
|
|
return {
|
|
...jsonRes,
|
|
success: true
|
|
};
|
|
}
|
|
|
|
// Makes use of signed JWT to get supported versions
|
|
const supportedVersionsCloud = verifyJWT(cloudInfo?.signed);
|
|
|
|
return {
|
|
...jsonRes,
|
|
success: true,
|
|
supportedVersions: supportedVersionsCloud
|
|
};
|
|
}
|
|
|
|
return {
|
|
...jsonRes,
|
|
success: true,
|
|
supportedVersions
|
|
};
|
|
} catch (error) {
|
|
// Request is successful, but response isn't a json
|
|
}
|
|
} catch (e: any) {
|
|
if (e?.message) {
|
|
if (e.message === 'Aborted') {
|
|
store.dispatch(selectServerFailure());
|
|
throw e;
|
|
}
|
|
return {
|
|
success: false,
|
|
message: e.message
|
|
};
|
|
}
|
|
}
|
|
|
|
return {
|
|
success: false,
|
|
message: I18n.t('Not_RC_Server', { contact: I18n.t('Contact_your_server_admin') })
|
|
};
|
|
}
|
|
|
|
const getUniqueId = async (server: string): Promise<string> => {
|
|
const response = await fetch(`${server}/api/v1/settings.public?query={"_id": "uniqueID"}`);
|
|
const result = await response.json();
|
|
return result?.settings?.[0]?.value;
|
|
};
|
|
|
|
export const getCloudInfo = async (domain: string): Promise<TCloudInfo | null> => {
|
|
try {
|
|
const uniqueId = await getUniqueId(domain);
|
|
const response = await getSupportedVersionsCloud(uniqueId, domain);
|
|
return response.json() as unknown as TCloudInfo;
|
|
} catch (e) {
|
|
log(e);
|
|
return null;
|
|
}
|
|
};
|