Integrate Set Retention Config on existing Bucket in UI (#515)
This commit is contained in:
@@ -0,0 +1,173 @@
|
|||||||
|
// This file is part of MinIO Console Server
|
||||||
|
// Copyright (c) 2020 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, { useEffect, useState } from "react";
|
||||||
|
import Grid from "@material-ui/core/Grid";
|
||||||
|
import { Button, LinearProgress } from "@material-ui/core";
|
||||||
|
import { createStyles, Theme, withStyles } from "@material-ui/core/styles";
|
||||||
|
import { modalBasic } from "../../Common/FormComponents/common/styleLibrary";
|
||||||
|
import api from "../../../../common/api";
|
||||||
|
import ModalWrapper from "../../Common/ModalWrapper/ModalWrapper";
|
||||||
|
import ErrorBlock from "../../../shared/ErrorBlock";
|
||||||
|
import RadioGroupSelector from "../../Common/FormComponents/RadioGroupSelector/RadioGroupSelector";
|
||||||
|
import InputBoxWrapper from "../../Common/FormComponents/InputBoxWrapper/InputBoxWrapper";
|
||||||
|
|
||||||
|
const styles = (theme: Theme) =>
|
||||||
|
createStyles({
|
||||||
|
...modalBasic,
|
||||||
|
});
|
||||||
|
|
||||||
|
interface ISetRetentionConfigProps {
|
||||||
|
classes: any;
|
||||||
|
open: boolean;
|
||||||
|
bucketName: string;
|
||||||
|
closeModalAndRefresh: () => void;
|
||||||
|
}
|
||||||
|
|
||||||
|
const SetRetentionConfig = ({
|
||||||
|
classes,
|
||||||
|
open,
|
||||||
|
bucketName,
|
||||||
|
closeModalAndRefresh,
|
||||||
|
}: ISetRetentionConfigProps) => {
|
||||||
|
const [addLoading, setAddLoading] = useState<boolean>(false);
|
||||||
|
const [error, setError] = useState<string>("");
|
||||||
|
const [retentionMode, setRetentionMode] = useState<string>("compliance");
|
||||||
|
const [retentionUnit, setRetentionUnit] = useState<string>("days");
|
||||||
|
const [retentionValidity, setRetentionValidity] = useState<number>(1);
|
||||||
|
const [valid, setValid] = useState<boolean>(false);
|
||||||
|
|
||||||
|
const setRetention = (event: React.FormEvent) => {
|
||||||
|
event.preventDefault();
|
||||||
|
if (addLoading) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
setAddLoading(true);
|
||||||
|
api
|
||||||
|
.invoke("PUT", `/api/v1/buckets/${bucketName}/retention`, {
|
||||||
|
mode: retentionMode,
|
||||||
|
unit: retentionUnit,
|
||||||
|
validity: retentionValidity,
|
||||||
|
})
|
||||||
|
.then((res) => {
|
||||||
|
setAddLoading(false);
|
||||||
|
setError("");
|
||||||
|
closeModalAndRefresh();
|
||||||
|
})
|
||||||
|
.catch((err) => {
|
||||||
|
setAddLoading(false);
|
||||||
|
setError(err);
|
||||||
|
});
|
||||||
|
};
|
||||||
|
|
||||||
|
useEffect(() => {
|
||||||
|
if (Number.isNaN(retentionValidity) || retentionValidity < 1) {
|
||||||
|
setValid(false);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
setValid(true);
|
||||||
|
}, [retentionValidity]);
|
||||||
|
|
||||||
|
return (
|
||||||
|
<ModalWrapper
|
||||||
|
title="Set Retention Configuration"
|
||||||
|
modalOpen={open}
|
||||||
|
onClose={() => {
|
||||||
|
setError("");
|
||||||
|
closeModalAndRefresh();
|
||||||
|
}}
|
||||||
|
>
|
||||||
|
<form
|
||||||
|
noValidate
|
||||||
|
autoComplete="off"
|
||||||
|
onSubmit={(e: React.FormEvent<HTMLFormElement>) => {
|
||||||
|
setRetention(e);
|
||||||
|
}}
|
||||||
|
>
|
||||||
|
<Grid container>
|
||||||
|
<Grid item xs={12} className={classes.formScrollable}>
|
||||||
|
{error !== "" && (
|
||||||
|
<Grid item xs={12}>
|
||||||
|
<ErrorBlock errorMessage={error} withBreak={false} />
|
||||||
|
</Grid>
|
||||||
|
)}
|
||||||
|
<Grid item xs={12}>
|
||||||
|
<RadioGroupSelector
|
||||||
|
currentSelection={retentionMode}
|
||||||
|
id="retention_mode"
|
||||||
|
name="retention_mode"
|
||||||
|
label="Retention Mode"
|
||||||
|
onChange={(e: React.ChangeEvent<{ value: unknown }>) => {
|
||||||
|
setRetentionMode(e.target.value as string);
|
||||||
|
}}
|
||||||
|
selectorOptions={[
|
||||||
|
{ value: "compliance", label: "Compliance" },
|
||||||
|
{ value: "governance", label: "Governance" },
|
||||||
|
]}
|
||||||
|
/>
|
||||||
|
</Grid>
|
||||||
|
<Grid item xs={12}>
|
||||||
|
<RadioGroupSelector
|
||||||
|
currentSelection={retentionUnit}
|
||||||
|
id="retention_unit"
|
||||||
|
name="retention_unit"
|
||||||
|
label="Retention Unit"
|
||||||
|
onChange={(e: React.ChangeEvent<{ value: unknown }>) => {
|
||||||
|
setRetentionUnit(e.target.value as string);
|
||||||
|
}}
|
||||||
|
selectorOptions={[
|
||||||
|
{ value: "days", label: "Days" },
|
||||||
|
{ value: "years", label: "Years" },
|
||||||
|
]}
|
||||||
|
/>
|
||||||
|
</Grid>
|
||||||
|
<Grid item xs={12}>
|
||||||
|
<InputBoxWrapper
|
||||||
|
type="number"
|
||||||
|
id="retention_validity"
|
||||||
|
name="retention_validity"
|
||||||
|
onChange={(e: React.ChangeEvent<HTMLInputElement>) => {
|
||||||
|
setRetentionValidity(e.target.valueAsNumber);
|
||||||
|
}}
|
||||||
|
label="Retention Validity"
|
||||||
|
value={String(retentionValidity)}
|
||||||
|
required
|
||||||
|
min="1"
|
||||||
|
/>
|
||||||
|
</Grid>
|
||||||
|
</Grid>
|
||||||
|
<Grid item xs={12}>
|
||||||
|
<Button
|
||||||
|
type="submit"
|
||||||
|
variant="contained"
|
||||||
|
color="primary"
|
||||||
|
fullWidth
|
||||||
|
disabled={addLoading || !valid}
|
||||||
|
>
|
||||||
|
Set
|
||||||
|
</Button>
|
||||||
|
</Grid>
|
||||||
|
{addLoading && (
|
||||||
|
<Grid item xs={12}>
|
||||||
|
<LinearProgress />
|
||||||
|
</Grid>
|
||||||
|
)}
|
||||||
|
</Grid>
|
||||||
|
</form>
|
||||||
|
</ModalWrapper>
|
||||||
|
);
|
||||||
|
};
|
||||||
|
|
||||||
|
export default withStyles(styles)(SetRetentionConfig);
|
||||||
@@ -37,6 +37,7 @@ import {
|
|||||||
} from "../types";
|
} from "../types";
|
||||||
import { Button } from "@material-ui/core";
|
import { Button } from "@material-ui/core";
|
||||||
import SetAccessPolicy from "./SetAccessPolicy";
|
import SetAccessPolicy from "./SetAccessPolicy";
|
||||||
|
import SetRetentionConfig from "./SetRetentionConfig";
|
||||||
import { CreateIcon } from "../../../../icons";
|
import { CreateIcon } from "../../../../icons";
|
||||||
import AddEvent from "./AddEvent";
|
import AddEvent from "./AddEvent";
|
||||||
import DeleteEvent from "./DeleteEvent";
|
import DeleteEvent from "./DeleteEvent";
|
||||||
@@ -215,10 +216,9 @@ const ViewBucket = ({ classes, match }: IViewBucketProps) => {
|
|||||||
);
|
);
|
||||||
const [curTab, setCurTab] = useState<number>(0);
|
const [curTab, setCurTab] = useState<number>(0);
|
||||||
const [addScreenOpen, setAddScreenOpen] = useState<boolean>(false);
|
const [addScreenOpen, setAddScreenOpen] = useState<boolean>(false);
|
||||||
const [
|
const [enableEncryptionScreenOpen, setEnableEncryptionScreenOpen] = useState<
|
||||||
enableEncryptionScreenOpen,
|
boolean
|
||||||
setEnableEncryptionScreenOpen,
|
>(false);
|
||||||
] = useState<boolean>(false);
|
|
||||||
const [deleteOpen, setDeleteOpen] = useState<boolean>(false);
|
const [deleteOpen, setDeleteOpen] = useState<boolean>(false);
|
||||||
const [selectedEvent, setSelectedEvent] = useState<BucketEvent | null>(null);
|
const [selectedEvent, setSelectedEvent] = useState<BucketEvent | null>(null);
|
||||||
const [bucketSize, setBucketSize] = useState<string>("0");
|
const [bucketSize, setBucketSize] = useState<string>("0");
|
||||||
@@ -226,6 +226,9 @@ const ViewBucket = ({ classes, match }: IViewBucketProps) => {
|
|||||||
const [openSetReplication, setOpenSetReplication] = useState<boolean>(false);
|
const [openSetReplication, setOpenSetReplication] = useState<boolean>(false);
|
||||||
const [isVersioned, setIsVersioned] = useState<boolean>(false);
|
const [isVersioned, setIsVersioned] = useState<boolean>(false);
|
||||||
const [encryptionEnabled, setEncryptionEnabled] = useState<boolean>(false);
|
const [encryptionEnabled, setEncryptionEnabled] = useState<boolean>(false);
|
||||||
|
const [retentionConfigOpen, setRetentionConfigOpen] = useState<boolean>(
|
||||||
|
false
|
||||||
|
);
|
||||||
|
|
||||||
const fetchEvents = useCallback(() => {
|
const fetchEvents = useCallback(() => {
|
||||||
setLoadingBucket(true);
|
setLoadingBucket(true);
|
||||||
@@ -319,6 +322,7 @@ const ViewBucket = ({ classes, match }: IViewBucketProps) => {
|
|||||||
|
|
||||||
const closeAddModalAndRefresh = () => {
|
const closeAddModalAndRefresh = () => {
|
||||||
setAccessPolicyScreenOpen(false);
|
setAccessPolicyScreenOpen(false);
|
||||||
|
setRetentionConfigOpen(false);
|
||||||
loadInfo();
|
loadInfo();
|
||||||
};
|
};
|
||||||
|
|
||||||
@@ -420,6 +424,15 @@ const ViewBucket = ({ classes, match }: IViewBucketProps) => {
|
|||||||
}}
|
}}
|
||||||
/>
|
/>
|
||||||
)}
|
)}
|
||||||
|
{retentionConfigOpen && (
|
||||||
|
<SetRetentionConfig
|
||||||
|
bucketName={bucketName}
|
||||||
|
open={retentionConfigOpen}
|
||||||
|
closeModalAndRefresh={() => {
|
||||||
|
closeAddModalAndRefresh();
|
||||||
|
}}
|
||||||
|
/>
|
||||||
|
)}
|
||||||
{openSetReplication && (
|
{openSetReplication && (
|
||||||
<AddReplicationModal
|
<AddReplicationModal
|
||||||
closeModalAndRefresh={() => {
|
closeModalAndRefresh={() => {
|
||||||
@@ -511,6 +524,19 @@ const ViewBucket = ({ classes, match }: IViewBucketProps) => {
|
|||||||
Change Access Policy
|
Change Access Policy
|
||||||
</Button>
|
</Button>
|
||||||
</div>
|
</div>
|
||||||
|
<div>
|
||||||
|
<Button
|
||||||
|
variant="contained"
|
||||||
|
color="primary"
|
||||||
|
fullWidth
|
||||||
|
size="medium"
|
||||||
|
onClick={() => {
|
||||||
|
setRetentionConfigOpen(true);
|
||||||
|
}}
|
||||||
|
>
|
||||||
|
Set Retention Configuration
|
||||||
|
</Button>
|
||||||
|
</div>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
</Grid>
|
</Grid>
|
||||||
|
|||||||
Reference in New Issue
Block a user