Files
object-browser/portal-ui/src/screens/Console/Buckets/BucketDetails/BucketDetails.tsx
Daniel Valdivia b218cbf503 Anonymous Access (#2600)
Signed-off-by: Daniel Valdivia <18384552+dvaldivia@users.noreply.github.com>
2023-01-27 12:23:30 -08:00

434 lines
13 KiB
TypeScript

// This file is part of MinIO Console Server
// Copyright (c) 2021 MinIO, Inc.
//
// This program is free software: you can redistribute it and/or modify
// it under the terms of the GNU Affero General Public License as published by
// the Free Software Foundation, either version 3 of the License, or
// (at your option) any later version.
//
// This program is distributed in the hope that it will be useful,
// but WITHOUT ANY WARRANTY; without even the implied warranty of
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
// GNU Affero General Public License for more details.
//
// You should have received a copy of the GNU Affero General Public License
// along with this program. If not, see <http://www.gnu.org/licenses/>.
import React, { Fragment, useEffect, useState } from "react";
import {
Link,
Navigate,
Route,
Routes,
useNavigate,
useParams,
} from "react-router-dom";
import { Button } from "mds";
import { useSelector } from "react-redux";
import { Theme } from "@mui/material/styles";
import createStyles from "@mui/styles/createStyles";
import withStyles from "@mui/styles/withStyles";
import Grid from "@mui/material/Grid";
import api from "../../../../common/api";
import { BucketInfo } from "../types";
import {
buttonsStyles,
containerForHeader,
hrClass,
pageContentStyles,
searchField,
} from "../../Common/FormComponents/common/styleLibrary";
import { ErrorResponseHandler } from "../../../../common/types";
import PageHeader from "../../Common/PageHeader/PageHeader";
import ScreenTitle from "../../Common/ScreenTitle/ScreenTitle";
import { Box } from "@mui/material";
import { RefreshIcon, BucketsIcon, FolderIcon } from "mds";
import {
IAM_SCOPES,
IAM_PERMISSIONS,
IAM_ROLES,
permissionTooltipHelper,
deleteBucketPermissions,
browseBucketPermissions,
} from "../../../../common/SecureComponent/permissions";
import PageLayout from "../../Common/Layout/PageLayout";
import VerticalTabs from "../../Common/VerticalTabs/VerticalTabs";
import BackLink from "../../../../common/BackLink";
import {
hasPermission,
SecureComponent,
} from "../../../../common/SecureComponent";
import withSuspense from "../../Common/Components/withSuspense";
import { TrashIcon } from "mds";
import {
selDistSet,
selSiteRep,
setErrorSnackMessage,
} from "../../../../systemSlice";
import {
selBucketDetailsInfo,
selBucketDetailsLoading,
setBucketDetailsLoad,
setBucketInfo,
} from "./bucketDetailsSlice";
import { useAppDispatch } from "../../../../store";
import TooltipWrapper from "../../Common/TooltipWrapper/TooltipWrapper";
const DeleteBucket = withSuspense(
React.lazy(() => import("../ListBuckets/DeleteBucket"))
);
const AccessRulePanel = withSuspense(
React.lazy(() => import("./AccessRulePanel"))
);
const AccessDetailsPanel = withSuspense(
React.lazy(() => import("./AccessDetailsPanel"))
);
const BucketSummaryPanel = withSuspense(
React.lazy(() => import("./BucketSummaryPanel"))
);
const BucketEventsPanel = withSuspense(
React.lazy(() => import("./BucketEventsPanel"))
);
const BucketReplicationPanel = withSuspense(
React.lazy(() => import("./BucketReplicationPanel"))
);
const BucketLifecyclePanel = withSuspense(
React.lazy(() => import("./BucketLifecyclePanel"))
);
const styles = (theme: Theme) =>
createStyles({
pageContainer: {
height: "100%",
},
screenTitle: {
border: 0,
paddingTop: 0,
},
...pageContentStyles,
...searchField,
capitalize: {
textTransform: "capitalize",
},
...hrClass,
...buttonsStyles,
...containerForHeader(theme.spacing(4)),
});
interface IBucketDetailsProps {
classes: any;
}
const BucketDetails = ({ classes }: IBucketDetailsProps) => {
const dispatch = useAppDispatch();
const navigate = useNavigate();
const params = useParams();
const distributedSetup = useSelector(selDistSet);
const loadingBucket = useSelector(selBucketDetailsLoading);
const bucketInfo = useSelector(selBucketDetailsInfo);
const siteReplicationInfo = useSelector(selSiteRep);
const [iniLoad, setIniLoad] = useState<boolean>(false);
const [deleteOpen, setDeleteOpen] = useState<boolean>(false);
const bucketName = params.bucketName || "";
let selTab = params["0"] || "";
selTab = selTab ? selTab : "summary";
const [activeTab, setActiveTab] = useState(selTab);
const canDelete = hasPermission(bucketName, deleteBucketPermissions);
const canBrowse = hasPermission(bucketName, browseBucketPermissions);
useEffect(() => {
setActiveTab(selTab);
}, [selTab]);
useEffect(() => {
if (!iniLoad) {
dispatch(setBucketDetailsLoad(true));
setIniLoad(true);
}
}, [iniLoad, dispatch, setIniLoad]);
useEffect(() => {
if (loadingBucket) {
api
.invoke("GET", `/api/v1/buckets/${bucketName}`)
.then((res: BucketInfo) => {
dispatch(setBucketDetailsLoad(false));
dispatch(setBucketInfo(res));
})
.catch((err: ErrorResponseHandler) => {
dispatch(setBucketDetailsLoad(false));
dispatch(setErrorSnackMessage(err));
});
}
}, [bucketName, loadingBucket, dispatch]);
let topLevelRoute = `/buckets/${bucketName}`;
const defaultRoute = "/admin/summary";
const manageBucketRoutes: Record<string, any> = {
events: "/admin/events",
replication: "/admin/replication",
lifecycle: "/admin/lifecycle",
access: "/admin/access",
prefix: "/admin/prefix",
};
const getRoutePath = (routeKey: string) => {
let path = manageBucketRoutes[routeKey];
if (!path) {
path = `${topLevelRoute}${defaultRoute}`;
} else {
path = `${topLevelRoute}${path}`;
}
return path;
};
const closeDeleteModalAndRefresh = (refresh: boolean) => {
setDeleteOpen(false);
if (refresh) {
navigate("/buckets");
}
};
return (
<Fragment>
{deleteOpen && (
<DeleteBucket
deleteOpen={deleteOpen}
selectedBucket={bucketName}
closeDeleteModalAndRefresh={(refresh: boolean) => {
closeDeleteModalAndRefresh(refresh);
}}
/>
)}
<PageHeader
label={<BackLink to={"/buckets"} label={"Buckets"} />}
actions={
<TooltipWrapper
tooltip={
canBrowse
? "Browse Bucket"
: permissionTooltipHelper(
IAM_PERMISSIONS[IAM_ROLES.BUCKET_VIEWER],
"browsing this bucket"
)
}
>
<Button
id={"switch-browse-view"}
aria-label="Browse Bucket"
onClick={() => {
navigate(`/browser/${bucketName}`);
}}
icon={
<FolderIcon style={{ width: 20, height: 20, marginTop: -3 }} />
}
style={{
padding: "0 10px",
}}
disabled={!canBrowse}
/>
</TooltipWrapper>
}
/>
<PageLayout className={classes.pageContainer}>
<Grid item xs={12}>
<ScreenTitle
icon={
<Fragment>
<BucketsIcon width={40} />
</Fragment>
}
title={bucketName}
subTitle={
<SecureComponent
scopes={[
IAM_SCOPES.S3_GET_BUCKET_POLICY,
IAM_SCOPES.S3_GET_ACTIONS,
]}
resource={bucketName}
>
<span style={{ fontSize: 15 }}>Access: </span>
<span
className={classes.capitalize}
style={{ fontWeight: 600, fontSize: 15 }}
>
{bucketInfo?.access.toLowerCase()}
</span>
</SecureComponent>
}
actions={
<Fragment>
<SecureComponent
scopes={deleteBucketPermissions}
resource={bucketName}
errorProps={{ disabled: true }}
>
<TooltipWrapper
tooltip={
canDelete
? ""
: permissionTooltipHelper(
[
IAM_SCOPES.S3_DELETE_BUCKET,
IAM_SCOPES.S3_FORCE_DELETE_BUCKET,
],
"deleting this bucket"
)
}
>
<Button
id={"delete-bucket-button"}
onClick={() => {
setDeleteOpen(true);
}}
label={"Delete Bucket"}
icon={<TrashIcon />}
variant={"secondary"}
disabled={!canDelete}
/>
</TooltipWrapper>
</SecureComponent>
<Button
id={"refresh-bucket-info"}
onClick={() => {
dispatch(setBucketDetailsLoad(true));
}}
label={"Refresh"}
icon={<RefreshIcon />}
/>
</Fragment>
}
/>
</Grid>
<Box sx={{ border: "1px solid #eaeaea" }}>
<VerticalTabs
selectedTab={activeTab}
isRouteTabs
routes={
<div className={classes.contentSpacer}>
<Routes>
<Route path="summary" element={<BucketSummaryPanel />} />
<Route path="events" element={<BucketEventsPanel />} />
{distributedSetup && (
<Route
path="replication"
element={<BucketReplicationPanel />}
/>
)}
{distributedSetup && (
<Route
path="lifecycle"
element={<BucketLifecyclePanel />}
/>
)}
<Route path="access" element={<AccessDetailsPanel />} />
<Route path="prefix" element={<AccessRulePanel />} />
<Route
path="*"
element={
<Navigate to={`/buckets/${bucketName}/admin/summary`} />
}
/>
</Routes>
</div>
}
>
{{
tabConfig: {
label: "Summary",
value: "summary",
component: Link,
to: getRoutePath("summary"),
},
}}
{{
tabConfig: {
label: "Events",
value: "events",
component: Link,
disabled: !hasPermission(bucketName, [
IAM_SCOPES.S3_GET_BUCKET_NOTIFICATIONS,
IAM_SCOPES.S3_PUT_BUCKET_NOTIFICATIONS,
IAM_SCOPES.S3_GET_ACTIONS,
IAM_SCOPES.S3_PUT_ACTIONS,
]),
to: getRoutePath("events"),
},
}}
{{
tabConfig: {
label: "Replication",
value: "replication",
component: Link,
disabled:
!distributedSetup ||
(siteReplicationInfo.enabled &&
siteReplicationInfo.curSite) ||
!hasPermission(bucketName, [
IAM_SCOPES.S3_GET_REPLICATION_CONFIGURATION,
IAM_SCOPES.S3_PUT_REPLICATION_CONFIGURATION,
IAM_SCOPES.S3_GET_ACTIONS,
IAM_SCOPES.S3_PUT_ACTIONS,
]),
to: getRoutePath("replication"),
},
}}
{{
tabConfig: {
label: "Lifecycle",
value: "lifecycle",
component: Link,
disabled:
!distributedSetup ||
!hasPermission(bucketName, [
IAM_SCOPES.S3_GET_LIFECYCLE_CONFIGURATION,
IAM_SCOPES.S3_PUT_LIFECYCLE_CONFIGURATION,
IAM_SCOPES.S3_GET_ACTIONS,
IAM_SCOPES.S3_PUT_ACTIONS,
]),
to: getRoutePath("lifecycle"),
},
}}
{{
tabConfig: {
label: "Access",
value: "access",
component: Link,
disabled: !hasPermission(bucketName, [
IAM_SCOPES.ADMIN_GET_POLICY,
IAM_SCOPES.ADMIN_LIST_USER_POLICIES,
IAM_SCOPES.ADMIN_LIST_USERS,
]),
to: getRoutePath("access"),
},
}}
{{
tabConfig: {
label: "Anonymous",
value: "prefix",
component: Link,
disabled: !hasPermission(bucketName, [
IAM_SCOPES.S3_GET_BUCKET_POLICY,
IAM_SCOPES.S3_GET_ACTIONS,
]),
to: getRoutePath("prefix"),
},
}}
</VerticalTabs>
</Box>
</PageLayout>
</Fragment>
);
};
export default withStyles(styles)(BucketDetails);