import React from "react"; import { observer } from "mobx-react"; import { Redirect, Route, Router, Switch } from "react-router"; import { I18nProvider } from "@lingui/react"; import { _i18n } from "../i18n"; import { history } from "../navigation"; import { Notifications } from "./notifications"; import { NotFound } from "./+404"; import { UserManagement } from "./+user-management/user-management"; import { ConfirmDialog } from "./confirm-dialog"; import { usersManagementRoute } from "./+user-management/user-management.route"; import { clusterRoute, clusterURL } from "./+cluster"; import { KubeConfigDialog } from "./kubeconfig-dialog/kubeconfig-dialog"; import { Nodes, nodesRoute } from "./+nodes"; import { Workloads, workloadsRoute, workloadsURL } from "./+workloads"; import { Namespaces, namespacesRoute } from "./+namespaces"; import { Network, networkRoute } from "./+network"; import { Storage, storageRoute } from "./+storage"; import { Cluster } from "./+cluster/cluster"; import { Config, configRoute } from "./+config"; import { Events } from "./+events/events"; import { eventRoute } from "./+events"; import { Apps, appsRoute } from "./+apps"; import { KubeObjectDetails } from "./kube-object/kube-object-details"; import { AddRoleBindingDialog } from "./+user-management-roles-bindings"; import { DeploymentScaleDialog } from "./+workloads-deployments/deployment-scale-dialog"; import { CronJobTriggerDialog } from "./+workloads-cronjobs/cronjob-trigger-dialog"; import { CustomResources } from "./+custom-resources/custom-resources"; import { crdRoute } from "./+custom-resources"; import { isAllowedResource } from "../../common/rbac"; import { MainLayout } from "./layout/main-layout"; import { ErrorBoundary } from "./error-boundary"; import { Terminal } from "./dock/terminal"; import { getHostedCluster, getHostedClusterId } from "../../common/cluster-store"; import logger from "../../main/logger"; import { webFrame } from "electron"; import { clusterPageRegistry, getExtensionPageUrl } from "../../extensions/registries/page-registry"; import { extensionLoader } from "../../extensions/extension-loader"; import { appEventBus } from "../../common/event-bus"; import { broadcastMessage, requestMain } from "../../common/ipc"; import whatInput from "what-input"; import { clusterSetFrameIdHandler } from "../../common/cluster-ipc"; import { ClusterPageMenuRegistration, clusterPageMenuRegistry } from "../../extensions/registries"; import { TabLayoutRoute, TabLayout } from "./layout/tab-layout"; import { StatefulSetScaleDialog } from "./+workloads-statefulsets/statefulset-scale-dialog"; import { eventStore } from "./+events/event.store"; import { reaction, computed } from "mobx"; import { nodesStore } from "./+nodes/nodes.store"; import { podsStore } from "./+workloads-pods/pods.store"; import { sum } from "lodash"; @observer export class App extends React.Component { static async init() { const frameId = webFrame.routingId; const clusterId = getHostedClusterId(); logger.info(`[APP]: Init dashboard, clusterId=${clusterId}, frameId=${frameId}`); await Terminal.preloadFonts(); await requestMain(clusterSetFrameIdHandler, clusterId, frameId); await getHostedCluster().whenReady; // cluster.activate() is done at this point extensionLoader.loadOnClusterRenderer(); setTimeout(() => { appEventBus.emit({ name: "cluster", action: "open", params: { clusterId } }); }); window.addEventListener("online", () => { window.location.reload(); }); whatInput.ask(); // Start to monitor user input device } async componentDidMount() { const cluster = getHostedCluster(); const promises: Promise[] = []; if (isAllowedResource("events") && isAllowedResource("pods")) { promises.push(eventStore.loadAll()); promises.push(podsStore.loadAll()); } if (isAllowedResource("nodes")) { promises.push(nodesStore.loadAll()); } await Promise.all(promises); if (eventStore.isLoaded && podsStore.isLoaded) { eventStore.subscribe(); podsStore.subscribe(); } if (nodesStore.isLoaded) { nodesStore.subscribe(); } reaction(() => this.warningsCount, (count) => { broadcastMessage(`cluster-warning-event-count:${cluster.id}`, count); }); } @computed get warningsCount() { let warnings = sum(nodesStore.items .map(node => node.getWarningConditions().length)); warnings = warnings + eventStore.getWarnings().length; return warnings; } get startURL() { if (isAllowedResource(["events", "nodes", "pods"])) { return clusterURL(); } return workloadsURL(); } getTabLayoutRoutes(menuItem: ClusterPageMenuRegistration) { const routes: TabLayoutRoute[] = []; if (!menuItem.id) { return routes; } clusterPageMenuRegistry.getSubItems(menuItem).forEach((item) => { const page = clusterPageRegistry.getByPageMenuTarget(item.target); if (page) { routes.push({ routePath: page.routePath, url: getExtensionPageUrl({ extensionId: page.extensionId, pageId: page.id, params: item.target.params }), title: item.title, component: page.components.Page, exact: page.exact }); } }); return routes; } renderExtensionTabLayoutRoutes() { return clusterPageMenuRegistry.getRootItems().map((menu, index) => { const tabRoutes = this.getTabLayoutRoutes(menu); if (tabRoutes.length > 0) { const pageComponent = () => ; return tab.routePath)} />; } else { const page = clusterPageRegistry.getByPageMenuTarget(menu.target); if (page) { const pageComponent = () => ; return ; } } }); } renderExtensionRoutes() { return clusterPageRegistry.getItems().map((page, index) => { const menu = clusterPageMenuRegistry.getByPage(page); if (!menu) { return ; } }); } render() { return ( {this.renderExtensionTabLayoutRoutes()} {this.renderExtensionRoutes()} ); } }