mirror of
https://github.com/Azure/cosmos-explorer.git
synced 2025-12-24 03:11:32 +00:00
179 lines
6.4 KiB
TypeScript
179 lines
6.4 KiB
TypeScript
import promiseRetry, { AbortError } from "p-retry";
|
|
import { ContainerStatusType, HttpHeaders, HttpStatusCodes, Notebook } from "../Common/Constants";
|
|
import { getErrorMessage } from "../Common/ErrorHandlingUtils";
|
|
import * as Logger from "../Common/Logger";
|
|
import { configContext } from "../ConfigContext";
|
|
import {
|
|
ContainerInfo,
|
|
IAccountData,
|
|
IContainerData,
|
|
IPhoenixConnectionInfoResult,
|
|
IProvisionData,
|
|
IResponse,
|
|
} from "../Contracts/DataModels";
|
|
import { useNotebook } from "../Explorer/Notebook/useNotebook";
|
|
import { userContext } from "../UserContext";
|
|
import { getAuthorizationHeader } from "../Utils/AuthorizationUtils";
|
|
|
|
export class PhoenixClient {
|
|
private containerHealthHandler: NodeJS.Timeout;
|
|
private retryOptions: promiseRetry.Options = {
|
|
retries: Notebook.retryAttempts,
|
|
maxTimeout: Notebook.retryAttemptDelayMs,
|
|
minTimeout: Notebook.retryAttemptDelayMs,
|
|
};
|
|
|
|
public async allocateContainer(
|
|
provisionData: IProvisionData,
|
|
accountData: IAccountData
|
|
): Promise<IResponse<IPhoenixConnectionInfoResult>> {
|
|
return this.executeContainerAssignmentOperation(provisionData, accountData, "allocate");
|
|
}
|
|
|
|
public async resetContainer(
|
|
provisionData: IProvisionData,
|
|
accountData: IAccountData
|
|
): Promise<IResponse<IPhoenixConnectionInfoResult>> {
|
|
return this.executeContainerAssignmentOperation(provisionData, accountData, "reset");
|
|
}
|
|
|
|
private async executeContainerAssignmentOperation(
|
|
provisionData: IProvisionData,
|
|
accountData: IAccountData,
|
|
operation: string
|
|
): Promise<IResponse<IPhoenixConnectionInfoResult>> {
|
|
try {
|
|
const response = await fetch(
|
|
`${this.getPhoenixContainerPoolingEndPoint()}/subscriptions/${accountData.subscriptionId}/resourceGroups/${
|
|
accountData.resourceGroup
|
|
}/providers/Microsoft.DocumentDB/databaseAccounts/${accountData.dbAccountName}/containerconnections`,
|
|
{
|
|
method: operation === "allocate" ? "POST" : "PATCH",
|
|
headers: PhoenixClient.getHeaders(),
|
|
body: JSON.stringify(provisionData),
|
|
}
|
|
);
|
|
|
|
let data: IPhoenixConnectionInfoResult;
|
|
if (response.status === HttpStatusCodes.OK) {
|
|
data = await response.json();
|
|
}
|
|
return {
|
|
status: response.status,
|
|
data,
|
|
};
|
|
} catch (error) {
|
|
console.error(error);
|
|
throw error;
|
|
}
|
|
}
|
|
|
|
public async initiateContainerHeartBeat(containerData: IContainerData) {
|
|
if (this.containerHealthHandler) {
|
|
clearTimeout(this.containerHealthHandler);
|
|
}
|
|
await this.getContainerHealth(Notebook.containerStatusHeartbeatDelayMs, containerData);
|
|
}
|
|
|
|
private scheduleContainerHeartbeat(delayMs: number, containerData: IContainerData): void {
|
|
this.containerHealthHandler = setTimeout(async () => {
|
|
await this.getContainerHealth(delayMs, containerData);
|
|
}, delayMs);
|
|
}
|
|
|
|
private async getContainerStatusAsync(containerData: IContainerData): Promise<ContainerInfo> {
|
|
try {
|
|
const runContainerStatusAsync = async () => {
|
|
const response = await window.fetch(
|
|
`${this.getPhoenixContainerPoolingEndPoint()}/subscriptions/${containerData.subscriptionId}/resourceGroups/${
|
|
containerData.resourceGroup
|
|
}/providers/Microsoft.DocumentDB/databaseAccounts/${containerData.dbAccountName}/${
|
|
containerData.forwardingId
|
|
}`,
|
|
{
|
|
method: "GET",
|
|
headers: PhoenixClient.getHeaders(),
|
|
}
|
|
);
|
|
if (response.status === HttpStatusCodes.OK) {
|
|
const containerStatus = await response.json();
|
|
return {
|
|
durationLeftInMinutes: containerStatus?.durationLeftInMinutes,
|
|
notebookServerInfo: containerStatus?.notebookServerInfo,
|
|
status: ContainerStatusType.Active,
|
|
};
|
|
} else if (response.status === HttpStatusCodes.NotFound) {
|
|
throw new AbortError(response.statusText);
|
|
}
|
|
throw new Error(response.statusText);
|
|
};
|
|
return await promiseRetry(runContainerStatusAsync, this.retryOptions);
|
|
} catch (error) {
|
|
Logger.logError(getErrorMessage(error), "PhoenixClient/getContainerStatus");
|
|
return {
|
|
durationLeftInMinutes: undefined,
|
|
notebookServerInfo: undefined,
|
|
status: ContainerStatusType.Disconnected,
|
|
};
|
|
}
|
|
}
|
|
|
|
private async getContainerHealth(delayMs: number, containerData: IContainerData) {
|
|
try {
|
|
const containerInfo = await this.getContainerStatusAsync(containerData);
|
|
useNotebook.getState().setContainerStatus(containerInfo);
|
|
if (useNotebook.getState().containerStatus?.status === ContainerStatusType.Active) {
|
|
this.scheduleContainerHeartbeat(delayMs, containerData);
|
|
}
|
|
} catch (exception) {
|
|
useNotebook.getState().setContainerStatus({
|
|
durationLeftInMinutes: undefined,
|
|
notebookServerInfo: undefined,
|
|
status: ContainerStatusType.Disconnected,
|
|
});
|
|
}
|
|
}
|
|
|
|
public async IsDbAcountWhitelisted(accountData: IAccountData) {
|
|
try {
|
|
const response = await window.fetch(
|
|
`${this.getPhoenixContainerPoolingEndPoint()}/subscriptions/${accountData.subscriptionId}/resourceGroups/${
|
|
accountData.resourceGroup
|
|
}/providers/Microsoft.DocumentDB/databaseAccounts/${accountData.dbAccountName}`,
|
|
{
|
|
method: "GET",
|
|
headers: PhoenixClient.getHeaders(),
|
|
}
|
|
);
|
|
return response.status === HttpStatusCodes.OK;
|
|
} catch (error) {
|
|
Logger.logError(getErrorMessage(error), "PhoenixClient/IsDbAcountWhitelisted");
|
|
return false;
|
|
}
|
|
}
|
|
|
|
public static getPhoenixEndpoint(): string {
|
|
const phoenixEndpoint =
|
|
userContext.features.phoenixEndpoint ?? userContext.features.junoEndpoint ?? configContext.JUNO_ENDPOINT;
|
|
if (configContext.allowedJunoOrigins.indexOf(new URL(phoenixEndpoint).origin) === -1) {
|
|
const error = `${phoenixEndpoint} not allowed as juno endpoint`;
|
|
console.error(error);
|
|
throw new Error(error);
|
|
}
|
|
|
|
return phoenixEndpoint;
|
|
}
|
|
|
|
public getPhoenixContainerPoolingEndPoint(): string {
|
|
return `${PhoenixClient.getPhoenixEndpoint()}/api/controlplane/toolscontainer/cosmosaccounts`;
|
|
}
|
|
|
|
private static getHeaders(): HeadersInit {
|
|
const authorizationHeader = getAuthorizationHeader();
|
|
return {
|
|
[authorizationHeader.header]: authorizationHeader.token,
|
|
[HttpHeaders.contentType]: "application/json",
|
|
};
|
|
}
|
|
}
|