mirror of
https://github.com/elastic/kibana.git
synced 2025-06-27 18:51:07 -04:00
## Summary This PR removes all imports of Route from react-router-dom and '@kbn/kibana-react-plugin/public' and instead imports Route from @kbn/shared-ux-router. ### Context Based on https://github.com/elastic/kibana/issues/132629#issue-1243243678 This PR executes steps 2 - 4: > 2. To make the transition easier, we want to re-export other react-router-dom exports alongside the modified' Route'. > 3. Solutions should start using that Route component in place of the one from react-router-dom. I.e. replace all occurrences of import { ... } from 'react-router-dom' with import { ... } from '@kbn/shared-ux-router'. > 4. All manual calls to useExecutionContext are not needed anymore and should be removed. ### Future PR Looks like this might be getting worked on in: https://github.com/elastic/kibana/pull/145863 (thanks!) > Introduce an ESlint rule that ensures that react-router-dom is not used directly in Kibana and that imports go through the new @kbn/shared-ux-router package. This is tangentially accomplished through https://github.com/elastic/kibana/pull/150340 but only addresses using Route through @kbn/kibana-react-plugin/public' ### Checklist Delete any items that are not applicable to this PR. - [x] [Unit or functional tests](https://www.elastic.co/guide/en/kibana/master/development-tests.html) were updated or added to match the most common scenarios --------- Co-authored-by: kibanamachine <42973632+kibanamachine@users.noreply.github.com> Co-authored-by: Tiago Costa <tiagoffcc@hotmail.com>
183 lines
5.4 KiB
TypeScript
183 lines
5.4 KiB
TypeScript
/*
|
|
* Copyright Elasticsearch B.V. and/or licensed to Elasticsearch B.V. under one
|
|
* or more contributor license agreements. Licensed under the Elastic License
|
|
* 2.0; you may not use this file except in compliance with the Elastic License
|
|
* 2.0.
|
|
*/
|
|
|
|
import React, { Component } from 'react';
|
|
import { Switch, Router, Redirect } from 'react-router-dom';
|
|
import { Route } from '@kbn/shared-ux-router';
|
|
|
|
import { ScopedHistory, ApplicationStart } from '@kbn/core/public';
|
|
import { i18n } from '@kbn/i18n';
|
|
import { FormattedMessage } from '@kbn/i18n-react';
|
|
|
|
import { EuiEmptyPrompt, EuiPageContent_Deprecated as EuiPageContent } from '@elastic/eui';
|
|
|
|
import { getFatalErrors } from './services/notifications';
|
|
import { routing } from './services/routing';
|
|
// @ts-ignore
|
|
import { loadPermissions } from './services/api';
|
|
import { SectionLoading, PageError } from '../shared_imports';
|
|
|
|
// @ts-ignore
|
|
import {
|
|
CrossClusterReplicationHome,
|
|
AutoFollowPatternAdd,
|
|
AutoFollowPatternEdit,
|
|
FollowerIndexAdd,
|
|
FollowerIndexEdit,
|
|
} from './sections';
|
|
|
|
interface AppProps {
|
|
history: ScopedHistory;
|
|
getUrlForApp: ApplicationStart['getUrlForApp'];
|
|
}
|
|
|
|
interface AppState {
|
|
isFetchingPermissions: boolean;
|
|
fetchPermissionError: any;
|
|
hasPermission: boolean;
|
|
missingClusterPrivileges: any[];
|
|
}
|
|
|
|
class AppComponent extends Component<AppProps, AppState> {
|
|
constructor(props: any) {
|
|
super(props);
|
|
this.registerRouter();
|
|
|
|
this.state = {
|
|
isFetchingPermissions: false,
|
|
fetchPermissionError: undefined,
|
|
hasPermission: false,
|
|
missingClusterPrivileges: [],
|
|
};
|
|
}
|
|
|
|
componentDidMount() {
|
|
this.checkPermissions();
|
|
}
|
|
|
|
async checkPermissions() {
|
|
this.setState({
|
|
isFetchingPermissions: true,
|
|
});
|
|
|
|
try {
|
|
const { hasPermission, missingClusterPrivileges } = await loadPermissions();
|
|
|
|
this.setState({
|
|
isFetchingPermissions: false,
|
|
hasPermission,
|
|
missingClusterPrivileges,
|
|
});
|
|
} catch (error) {
|
|
// Expect an error in the shape provided by Angular's $http service.
|
|
if (error && error.body) {
|
|
return this.setState({
|
|
isFetchingPermissions: false,
|
|
fetchPermissionError: error,
|
|
});
|
|
}
|
|
|
|
// This error isn't an HTTP error, so let the fatal error screen tell the user something
|
|
// unexpected happened.
|
|
getFatalErrors().add(
|
|
error,
|
|
i18n.translate('xpack.crossClusterReplication.app.checkPermissionsFatalErrorTitle', {
|
|
defaultMessage: 'Cross-Cluster Replication app',
|
|
})
|
|
);
|
|
}
|
|
}
|
|
|
|
registerRouter() {
|
|
const { history, getUrlForApp } = this.props;
|
|
routing.reactRouter = {
|
|
history,
|
|
route: {
|
|
location: history.location,
|
|
},
|
|
getUrlForApp,
|
|
};
|
|
}
|
|
|
|
render() {
|
|
const { isFetchingPermissions, fetchPermissionError, hasPermission, missingClusterPrivileges } =
|
|
this.state;
|
|
|
|
if (isFetchingPermissions) {
|
|
return (
|
|
<EuiPageContent verticalPosition="center" horizontalPosition="center" color="subdued">
|
|
<SectionLoading>
|
|
<FormattedMessage
|
|
id="xpack.crossClusterReplication.app.permissionCheckTitle"
|
|
defaultMessage="Checking permissions…"
|
|
/>
|
|
</SectionLoading>
|
|
</EuiPageContent>
|
|
);
|
|
}
|
|
|
|
if (fetchPermissionError) {
|
|
return (
|
|
<PageError
|
|
title={
|
|
<FormattedMessage
|
|
id="xpack.crossClusterReplication.app.permissionCheckErrorTitle"
|
|
defaultMessage="Error checking permissions"
|
|
/>
|
|
}
|
|
error={fetchPermissionError}
|
|
/>
|
|
);
|
|
}
|
|
|
|
if (!hasPermission) {
|
|
return (
|
|
<EuiPageContent verticalPosition="center" horizontalPosition="center" color="subdued">
|
|
<EuiEmptyPrompt
|
|
iconType="securityApp"
|
|
title={
|
|
<h2>
|
|
<FormattedMessage
|
|
id="xpack.crossClusterReplication.app.deniedPermissionTitle"
|
|
defaultMessage="You're missing cluster privileges"
|
|
/>
|
|
</h2>
|
|
}
|
|
body={
|
|
<p>
|
|
<FormattedMessage
|
|
id="xpack.crossClusterReplication.app.deniedPermissionDescription"
|
|
defaultMessage="To use Cross-Cluster Replication, you must have {clusterPrivilegesCount,
|
|
plural, one {this cluster privilege} other {these cluster privileges}}: {clusterPrivileges}."
|
|
values={{
|
|
clusterPrivileges: missingClusterPrivileges.join(', '),
|
|
clusterPrivilegesCount: missingClusterPrivileges.length,
|
|
}}
|
|
/>
|
|
</p>
|
|
}
|
|
/>
|
|
</EuiPageContent>
|
|
);
|
|
}
|
|
|
|
return (
|
|
<Router history={this.props.history}>
|
|
<Switch>
|
|
<Redirect exact from="/" to="/follower_indices" />
|
|
<Route exact path="/auto_follow_patterns/add" component={AutoFollowPatternAdd} />
|
|
<Route exact path="/auto_follow_patterns/edit/:id" component={AutoFollowPatternEdit} />
|
|
<Route exact path="/follower_indices/add" component={FollowerIndexAdd} />
|
|
<Route exact path="/follower_indices/edit/:id" component={FollowerIndexEdit} />
|
|
<Route exact path={['/:section']} component={CrossClusterReplicationHome} />
|
|
</Switch>
|
|
</Router>
|
|
);
|
|
}
|
|
}
|
|
|
|
export const App = AppComponent;
|