Rocket.Chat.ReactNative/app/lib/methods/getServerInfo.ts

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;
}
};