1
0
mirror of https://github.com/lensapp/lens.git synced 2025-05-20 05:10:56 +00:00
lens/src/renderer/components/cluster-manager/cluster-status.tsx
Sebastian Malton 853573afcb Fix cluster connect statuses being intertwined
- Broadcast on a signle channel and add clusterId as data argument

- Make everything else computed based on the current props.cluster value

- Fix view staying on cluster connect screen after disconnect

- Initially clicking on a disconnected cluster should not immediately
  nativate to the Catalog if currently viewing a connected cluster

- Make route-path-parameters more type correct by not using keyedSingleton

Signed-off-by: Sebastian Malton <sebastian@malton.name>
2022-08-18 10:41:52 -04:00

146 lines
4.3 KiB
TypeScript

/**
* Copyright (c) OpenLens Authors. All rights reserved.
* Licensed under MIT License. See LICENSE in root directory for more information.
*/
import styles from "./cluster-status.module.scss";
import { runInAction } from "mobx";
import { observer } from "mobx-react";
import React from "react";
import type { Cluster } from "../../../common/cluster/cluster";
import type { IClassName } from "../../utils";
import { cssNames } from "../../utils";
import { Button } from "../button";
import { Icon } from "../icon";
import { Spinner } from "../spinner";
import type { CatalogEntityRegistry } from "../../api/catalog/entity/registry";
import { requestClusterActivation } from "../../ipc";
import type { NavigateToEntitySettings } from "../../../common/front-end-routing/routes/entity-settings/navigate-to-entity-settings.injectable";
import { withInjectables } from "@ogre-tools/injectable-react";
import navigateToEntitySettingsInjectable from "../../../common/front-end-routing/routes/entity-settings/navigate-to-entity-settings.injectable";
import catalogEntityRegistryInjectable from "../../api/catalog/entity/registry.injectable";
import type { ClusterConnectionStatus } from "./cluster-status.state.injectable";
import clusterConnectionStatusStateInjectable from "./cluster-status.state.injectable";
export interface ClusterStatusProps {
className?: IClassName;
cluster: Cluster;
}
interface Dependencies {
navigateToEntitySettings: NavigateToEntitySettings;
entityRegistry: CatalogEntityRegistry;
state: ClusterConnectionStatus;
}
const NonInjectedClusterStatus = observer((props: ClusterStatusProps & Dependencies) => {
const {
cluster,
navigateToEntitySettings,
state,
className,
entityRegistry,
} = props;
const entity = entityRegistry.getById(cluster.id);
const clusterName = entity?.getName() ?? cluster.name;
const reconnect = async () => {
runInAction(() => {
state.resetAuthOutput();
state.setAsReconnecting();
});
try {
await requestClusterActivation(cluster.id, true);
} catch (error) {
state.appendAuthUpdate({
message: String(error),
isError: true,
});
} finally {
state.clearReconnectingState();
}
};
const manageProxySettings = () => navigateToEntitySettings(cluster.id, "proxy");
const renderAuthenticationOutput = () => {
return (
<pre>
{
state.authOutput
.get()
.map(({ message, isError }, index) => (
<p key={index} className={cssNames({ error: isError })}>
{message.trim()}
</p>
))
}
</pre>
);
};
const renderStatusIcon = () => {
if (state.hasErrorOutput.get()) {
return <Icon material="cloud_off" className={styles.icon} />;
}
return (
<>
<Spinner singleColor={false} className={styles.spinner} />
<pre className="kube-auth-out">
<p>
{state.isReconnecting.get() ? "Reconnecting" : "Connecting"}
&hellip;
</p>
</pre>
</>
);
};
const renderReconnectionHelp = () => {
if (state.hasErrorOutput.get() && !state.isReconnecting.get()) {
return (
<>
<Button
primary
label="Reconnect"
className="box center"
onClick={reconnect}
waiting={state.isReconnecting.get()}
/>
<a
className="box center interactive"
onClick={manageProxySettings}
>
Manage Proxy Settings
</a>
</>
);
}
return undefined;
};
return (
<div className={cssNames(styles.status, "flex column box center align-center justify-center", className)}>
<div className="flex items-center column gaps">
<h2>{clusterName}</h2>
{renderStatusIcon()}
{renderAuthenticationOutput()}
{renderReconnectionHelp()}
</div>
</div>
);
});
export const ClusterStatus = withInjectables<Dependencies, ClusterStatusProps>(NonInjectedClusterStatus, {
getProps: (di, props) => ({
...props,
navigateToEntitySettings: di.inject(navigateToEntitySettingsInjectable),
entityRegistry: di.inject(catalogEntityRegistryInjectable),
state: di.inject(clusterConnectionStatusStateInjectable).forCluster(props.cluster.id),
}),
});