2022-11-11 11:24:56 +01:00
|
|
|
|
import { useMemo, useState } from 'react';
|
2022-05-02 12:52:33 +02:00
|
|
|
|
import { ConditionallyRender } from 'component/common/ConditionallyRender/ConditionallyRender';
|
2022-05-09 14:38:12 +02:00
|
|
|
|
import { PageContent } from 'component/common/PageContent/PageContent';
|
|
|
|
|
import { PageHeader } from 'component/common/PageHeader/PageHeader';
|
2022-03-28 10:49:59 +02:00
|
|
|
|
import { UPDATE_PROJECT } from 'component/providers/AccessProvider/permissions';
|
|
|
|
|
import ApiError from 'component/common/ApiError/ApiError';
|
|
|
|
|
import useToast from 'hooks/useToast';
|
|
|
|
|
import useUiConfig from 'hooks/api/getters/useUiConfig/useUiConfig';
|
2022-11-11 11:24:56 +01:00
|
|
|
|
import { Alert, styled, TableBody, TableRow } from '@mui/material';
|
2022-03-28 10:49:59 +02:00
|
|
|
|
import useProjectApi from 'hooks/api/actions/useProjectApi/useProjectApi';
|
2021-10-07 12:44:46 +02:00
|
|
|
|
import { Link } from 'react-router-dom';
|
2022-03-28 10:49:59 +02:00
|
|
|
|
import PermissionSwitch from 'component/common/PermissionSwitch/PermissionSwitch';
|
2024-03-18 13:58:05 +01:00
|
|
|
|
import type { IProjectEnvironment } from 'interfaces/environments';
|
2021-11-04 14:24:36 +01:00
|
|
|
|
import { getEnabledEnvs } from './helpers';
|
2022-06-21 09:08:37 +02:00
|
|
|
|
import { usePageTitle } from 'hooks/usePageTitle';
|
2022-08-04 13:57:25 +02:00
|
|
|
|
import { useRequiredPathParam } from 'hooks/useRequiredPathParam';
|
2024-05-20 14:20:13 +02:00
|
|
|
|
import { useGlobalFilter, useTable } from 'react-table';
|
2022-11-11 11:24:56 +01:00
|
|
|
|
import {
|
|
|
|
|
SortableTableHeader,
|
|
|
|
|
Table,
|
|
|
|
|
TableCell,
|
|
|
|
|
TablePlaceholder,
|
|
|
|
|
} from 'component/common/Table';
|
|
|
|
|
import { SearchHighlightProvider } from 'component/common/Table/SearchHighlightContext/SearchHighlightContext';
|
|
|
|
|
import { Search } from 'component/common/Search/Search';
|
|
|
|
|
import { EnvironmentNameCell } from 'component/environments/EnvironmentTable/EnvironmentNameCell/EnvironmentNameCell';
|
|
|
|
|
import { HighlightCell } from 'component/common/Table/cells/HighlightCell/HighlightCell';
|
|
|
|
|
import { ActionCell } from 'component/common/Table/cells/ActionCell/ActionCell';
|
|
|
|
|
import { EnvironmentHideDialog } from './EnvironmentHideDialog/EnvironmentHideDialog';
|
|
|
|
|
import { useProjectEnvironments } from 'hooks/api/getters/useProjectEnvironments/useProjectEnvironments';
|
|
|
|
|
import { TextCell } from 'component/common/Table/cells/TextCell/TextCell';
|
2024-05-20 15:15:24 +02:00
|
|
|
|
import useProjectOverview, {
|
|
|
|
|
useProjectOverviewNameOrId,
|
|
|
|
|
} from 'hooks/api/getters/useProjectOverview/useProjectOverview';
|
2022-11-11 11:24:56 +01:00
|
|
|
|
|
|
|
|
|
const StyledAlert = styled(Alert)(({ theme }) => ({
|
|
|
|
|
marginBottom: theme.spacing(4),
|
|
|
|
|
}));
|
2021-09-30 10:24:16 +02:00
|
|
|
|
|
2023-01-05 09:45:17 +01:00
|
|
|
|
const StyledDivContainer = styled('div')(({ theme }) => ({
|
|
|
|
|
display: 'flex',
|
|
|
|
|
flexWrap: 'wrap',
|
|
|
|
|
[theme.breakpoints.down('sm')]: {
|
|
|
|
|
justifyContent: 'center',
|
|
|
|
|
},
|
|
|
|
|
}));
|
|
|
|
|
|
|
|
|
|
const StyledApiError = styled(ApiError)(({ theme }) => ({
|
|
|
|
|
maxWidth: '400px',
|
|
|
|
|
marginBottom: theme.spacing(2),
|
|
|
|
|
}));
|
|
|
|
|
|
2022-08-04 13:57:25 +02:00
|
|
|
|
const ProjectEnvironmentList = () => {
|
|
|
|
|
const projectId = useRequiredPathParam('projectId');
|
2024-05-20 14:20:13 +02:00
|
|
|
|
const projectName = useProjectOverviewNameOrId(projectId);
|
2022-06-21 09:08:37 +02:00
|
|
|
|
usePageTitle(`Project environments – ${projectName}`);
|
2022-08-04 13:57:25 +02:00
|
|
|
|
|
2021-09-30 10:24:16 +02:00
|
|
|
|
// api state
|
2022-01-14 15:50:02 +01:00
|
|
|
|
const { setToastData, setToastApiError } = useToast();
|
2022-03-23 12:55:00 +01:00
|
|
|
|
const { environments, loading, error, refetchEnvironments } =
|
2022-11-11 11:24:56 +01:00
|
|
|
|
useProjectEnvironments(projectId);
|
2024-05-20 15:15:24 +02:00
|
|
|
|
const { project, refetch: refetchProject } = useProjectOverview(projectId);
|
2021-10-08 11:23:29 +02:00
|
|
|
|
const { removeEnvironmentFromProject, addEnvironmentToProject } =
|
|
|
|
|
useProjectApi();
|
|
|
|
|
|
2021-09-30 10:24:16 +02:00
|
|
|
|
// local state
|
2022-11-11 11:24:56 +01:00
|
|
|
|
const [selectedEnvironment, setSelectedEnvironment] =
|
|
|
|
|
useState<IProjectEnvironment>();
|
|
|
|
|
const [hideDialog, setHideDialog] = useState(false);
|
2022-05-12 09:41:36 +02:00
|
|
|
|
const { isOss } = useUiConfig();
|
2021-09-30 10:24:16 +02:00
|
|
|
|
|
2022-11-11 11:24:56 +01:00
|
|
|
|
const projectEnvironments = useMemo<IProjectEnvironment[]>(
|
|
|
|
|
() =>
|
2023-10-02 14:25:46 +02:00
|
|
|
|
environments.map((environment) => ({
|
2022-11-11 11:24:56 +01:00
|
|
|
|
...environment,
|
2023-05-02 09:17:05 +02:00
|
|
|
|
projectVisible: project?.environments
|
2023-10-02 14:25:46 +02:00
|
|
|
|
.map((projectEnvironment) => projectEnvironment.environment)
|
2023-05-02 09:17:05 +02:00
|
|
|
|
.includes(environment.name),
|
2022-11-11 11:24:56 +01:00
|
|
|
|
})),
|
2023-10-02 14:25:46 +02:00
|
|
|
|
[environments, project?.environments],
|
2022-11-11 11:24:56 +01:00
|
|
|
|
);
|
2021-11-04 14:24:36 +01:00
|
|
|
|
|
2021-09-30 10:24:16 +02:00
|
|
|
|
const refetch = () => {
|
2022-03-23 12:55:00 +01:00
|
|
|
|
refetchEnvironments();
|
2021-09-30 10:24:16 +02:00
|
|
|
|
refetchProject();
|
2021-10-08 11:23:29 +02:00
|
|
|
|
};
|
2021-09-30 10:24:16 +02:00
|
|
|
|
|
|
|
|
|
const renderError = () => {
|
|
|
|
|
return (
|
2023-01-05 09:45:17 +01:00
|
|
|
|
<StyledApiError
|
2021-09-30 10:24:16 +02:00
|
|
|
|
onClick={refetch}
|
2023-10-02 14:25:46 +02:00
|
|
|
|
text='Error fetching environments'
|
2021-09-30 10:24:16 +02:00
|
|
|
|
/>
|
|
|
|
|
);
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
const errorMsg = (enable: boolean): string => {
|
2022-11-11 11:24:56 +01:00
|
|
|
|
return `Got an API error when trying to set the environment as ${
|
|
|
|
|
enable ? 'visible' : 'hidden'
|
|
|
|
|
}`;
|
2021-10-08 11:23:29 +02:00
|
|
|
|
};
|
2021-09-30 10:24:16 +02:00
|
|
|
|
|
2022-02-25 10:55:39 +01:00
|
|
|
|
const toggleEnv = async (env: IProjectEnvironment) => {
|
2022-11-11 11:24:56 +01:00
|
|
|
|
if (env.projectVisible) {
|
|
|
|
|
const enabledEnvs = getEnabledEnvs(projectEnvironments);
|
2021-11-04 14:24:36 +01:00
|
|
|
|
|
|
|
|
|
if (enabledEnvs > 1) {
|
2022-11-11 11:24:56 +01:00
|
|
|
|
setSelectedEnvironment(env);
|
|
|
|
|
setHideDialog(true);
|
2021-11-04 14:24:36 +01:00
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
setToastData({
|
2022-11-11 11:24:56 +01:00
|
|
|
|
title: 'One environment must be visible',
|
|
|
|
|
text: 'You must always have at least one visible environment per project',
|
2021-11-04 14:24:36 +01:00
|
|
|
|
type: 'error',
|
|
|
|
|
});
|
2021-09-30 10:24:16 +02:00
|
|
|
|
} else {
|
|
|
|
|
try {
|
2021-10-01 12:15:02 +02:00
|
|
|
|
await addEnvironmentToProject(projectId, env.name);
|
2022-11-11 11:24:56 +01:00
|
|
|
|
refetch();
|
2021-10-08 11:23:29 +02:00
|
|
|
|
setToastData({
|
2022-11-11 11:24:56 +01:00
|
|
|
|
title: 'Environment set as visible',
|
|
|
|
|
text: 'Environment successfully set as visible.',
|
2021-10-08 11:23:29 +02:00
|
|
|
|
type: 'success',
|
|
|
|
|
});
|
2021-09-30 10:24:16 +02:00
|
|
|
|
} catch (error) {
|
2022-01-14 15:50:02 +01:00
|
|
|
|
setToastApiError(errorMsg(true));
|
2021-09-30 10:24:16 +02:00
|
|
|
|
}
|
|
|
|
|
}
|
2021-10-08 11:23:29 +02:00
|
|
|
|
};
|
2021-09-30 10:24:16 +02:00
|
|
|
|
|
2022-11-11 11:24:56 +01:00
|
|
|
|
const onHideConfirm = async () => {
|
|
|
|
|
if (selectedEnvironment) {
|
2021-09-30 10:24:16 +02:00
|
|
|
|
try {
|
2022-11-11 11:24:56 +01:00
|
|
|
|
await removeEnvironmentFromProject(
|
|
|
|
|
projectId,
|
2023-10-02 14:25:46 +02:00
|
|
|
|
selectedEnvironment.name,
|
2022-11-11 11:24:56 +01:00
|
|
|
|
);
|
|
|
|
|
refetch();
|
2021-10-08 11:23:29 +02:00
|
|
|
|
setToastData({
|
2022-11-11 11:24:56 +01:00
|
|
|
|
title: 'Environment set as hidden',
|
|
|
|
|
text: 'Environment successfully set as hidden.',
|
2021-10-08 11:23:29 +02:00
|
|
|
|
type: 'success',
|
|
|
|
|
});
|
2021-09-30 10:24:16 +02:00
|
|
|
|
} catch (e) {
|
2022-01-14 15:50:02 +01:00
|
|
|
|
setToastApiError(errorMsg(false));
|
2022-11-11 11:24:56 +01:00
|
|
|
|
} finally {
|
|
|
|
|
setHideDialog(false);
|
2021-09-30 10:24:16 +02:00
|
|
|
|
}
|
2021-10-08 11:23:29 +02:00
|
|
|
|
}
|
|
|
|
|
};
|
2021-09-30 10:24:16 +02:00
|
|
|
|
|
2022-05-12 09:41:36 +02:00
|
|
|
|
const envIsDisabled = (projectName: string) => {
|
|
|
|
|
return isOss() && projectName === 'default';
|
|
|
|
|
};
|
|
|
|
|
|
2022-11-11 11:24:56 +01:00
|
|
|
|
const COLUMNS = useMemo(
|
|
|
|
|
() => [
|
|
|
|
|
{
|
|
|
|
|
Header: 'Name',
|
|
|
|
|
accessor: 'name',
|
|
|
|
|
Cell: ({ row: { original } }: any) => (
|
|
|
|
|
<EnvironmentNameCell environment={original} />
|
|
|
|
|
),
|
|
|
|
|
},
|
|
|
|
|
{
|
|
|
|
|
Header: 'Type',
|
|
|
|
|
accessor: 'type',
|
|
|
|
|
Cell: HighlightCell,
|
|
|
|
|
},
|
|
|
|
|
{
|
|
|
|
|
Header: 'Project API tokens',
|
|
|
|
|
accessor: (row: IProjectEnvironment) =>
|
|
|
|
|
row.projectApiTokenCount === 1
|
|
|
|
|
? '1 token'
|
|
|
|
|
: `${row.projectApiTokenCount} tokens`,
|
|
|
|
|
Cell: TextCell,
|
|
|
|
|
},
|
|
|
|
|
{
|
|
|
|
|
Header: 'Visible in project',
|
|
|
|
|
accessor: 'enabled',
|
|
|
|
|
align: 'center',
|
|
|
|
|
width: 1,
|
|
|
|
|
Cell: ({ row: { original } }: any) => (
|
|
|
|
|
<ActionCell>
|
|
|
|
|
<PermissionSwitch
|
|
|
|
|
tooltip={
|
|
|
|
|
original.projectVisible
|
|
|
|
|
? 'Hide environment and disable feature toggles'
|
|
|
|
|
: 'Make it visible'
|
|
|
|
|
}
|
2023-10-02 14:25:46 +02:00
|
|
|
|
size='medium'
|
2022-11-11 11:24:56 +01:00
|
|
|
|
disabled={envIsDisabled(original.name)}
|
|
|
|
|
projectId={projectId}
|
|
|
|
|
permission={UPDATE_PROJECT}
|
|
|
|
|
checked={original.projectVisible}
|
|
|
|
|
onChange={() => toggleEnv(original)}
|
|
|
|
|
/>
|
|
|
|
|
</ActionCell>
|
|
|
|
|
),
|
|
|
|
|
disableGlobalFilter: true,
|
|
|
|
|
},
|
|
|
|
|
],
|
2023-10-02 14:25:46 +02:00
|
|
|
|
[projectEnvironments],
|
2022-11-11 11:24:56 +01:00
|
|
|
|
);
|
2021-09-30 10:24:16 +02:00
|
|
|
|
|
2022-11-11 11:24:56 +01:00
|
|
|
|
const {
|
|
|
|
|
getTableProps,
|
|
|
|
|
getTableBodyProps,
|
|
|
|
|
headerGroups,
|
|
|
|
|
rows,
|
|
|
|
|
prepareRow,
|
|
|
|
|
state: { globalFilter },
|
|
|
|
|
setGlobalFilter,
|
|
|
|
|
} = useTable(
|
|
|
|
|
{
|
|
|
|
|
columns: COLUMNS as any,
|
|
|
|
|
data: projectEnvironments,
|
|
|
|
|
disableSortBy: true,
|
|
|
|
|
},
|
2023-10-02 14:25:46 +02:00
|
|
|
|
useGlobalFilter,
|
2022-11-11 11:24:56 +01:00
|
|
|
|
);
|
|
|
|
|
|
|
|
|
|
const header = (
|
|
|
|
|
<PageHeader
|
|
|
|
|
title={`Environments (${rows.length})`}
|
|
|
|
|
actions={
|
|
|
|
|
<Search
|
|
|
|
|
initialValue={globalFilter}
|
|
|
|
|
onChange={setGlobalFilter}
|
2021-10-19 13:08:25 +02:00
|
|
|
|
/>
|
2022-05-18 11:26:38 +02:00
|
|
|
|
}
|
2022-11-11 11:24:56 +01:00
|
|
|
|
/>
|
|
|
|
|
);
|
|
|
|
|
|
|
|
|
|
return (
|
|
|
|
|
<PageContent header={header} isLoading={loading}>
|
2023-01-11 08:00:26 +01:00
|
|
|
|
<StyledDivContainer>
|
|
|
|
|
<ConditionallyRender
|
|
|
|
|
condition={Boolean(error)}
|
|
|
|
|
show={renderError()}
|
|
|
|
|
/>
|
2023-10-02 14:25:46 +02:00
|
|
|
|
<StyledAlert severity='info'>
|
2023-01-11 08:00:26 +01:00
|
|
|
|
<strong>Important!</strong> In order for your application to
|
|
|
|
|
retrieve configured activation strategies for a specific
|
|
|
|
|
environment, the application must use an environment
|
|
|
|
|
specific API token. You can look up the environment-specific{' '}
|
2023-10-02 14:25:46 +02:00
|
|
|
|
<Link to='/admin/api'>API tokens here</Link>.
|
2023-01-11 08:00:26 +01:00
|
|
|
|
<br />
|
|
|
|
|
<br />
|
|
|
|
|
Your administrator can configure an environment-specific API
|
|
|
|
|
token to be used in the SDK. If you are an administrator you
|
2023-10-02 14:25:46 +02:00
|
|
|
|
can <Link to='/admin/api'>create a new API token here</Link>
|
2023-01-11 08:00:26 +01:00
|
|
|
|
.
|
|
|
|
|
</StyledAlert>
|
|
|
|
|
<SearchHighlightProvider value={globalFilter}>
|
2023-10-02 14:25:46 +02:00
|
|
|
|
<Table {...getTableProps()} rowHeight='compact'>
|
2023-01-11 08:00:26 +01:00
|
|
|
|
<SortableTableHeader
|
|
|
|
|
headerGroups={headerGroups as any}
|
2022-05-18 11:26:38 +02:00
|
|
|
|
/>
|
2023-01-11 08:00:26 +01:00
|
|
|
|
<TableBody {...getTableBodyProps()}>
|
2023-10-02 14:25:46 +02:00
|
|
|
|
{rows.map((row) => {
|
2023-01-11 08:00:26 +01:00
|
|
|
|
prepareRow(row);
|
|
|
|
|
return (
|
|
|
|
|
<TableRow hover {...row.getRowProps()}>
|
2023-10-02 14:25:46 +02:00
|
|
|
|
{row.cells.map((cell) => (
|
2023-01-11 08:00:26 +01:00
|
|
|
|
<TableCell {...cell.getCellProps()}>
|
|
|
|
|
{cell.render('Cell')}
|
|
|
|
|
</TableCell>
|
|
|
|
|
))}
|
|
|
|
|
</TableRow>
|
|
|
|
|
);
|
|
|
|
|
})}
|
|
|
|
|
</TableBody>
|
|
|
|
|
</Table>
|
|
|
|
|
</SearchHighlightProvider>
|
|
|
|
|
<ConditionallyRender
|
|
|
|
|
condition={rows.length === 0}
|
|
|
|
|
show={
|
2022-05-18 11:26:38 +02:00
|
|
|
|
<ConditionallyRender
|
2023-01-11 08:00:26 +01:00
|
|
|
|
condition={globalFilter?.length > 0}
|
2022-11-11 11:24:56 +01:00
|
|
|
|
show={
|
2023-01-11 08:00:26 +01:00
|
|
|
|
<TablePlaceholder>
|
|
|
|
|
No environments found matching “
|
|
|
|
|
{globalFilter}
|
|
|
|
|
”
|
|
|
|
|
</TablePlaceholder>
|
|
|
|
|
}
|
|
|
|
|
elseShow={
|
|
|
|
|
<TablePlaceholder>
|
|
|
|
|
No environments available. Get started by
|
|
|
|
|
adding one.
|
|
|
|
|
</TablePlaceholder>
|
2022-05-18 11:26:38 +02:00
|
|
|
|
}
|
2022-11-11 11:24:56 +01:00
|
|
|
|
/>
|
2023-01-11 08:00:26 +01:00
|
|
|
|
}
|
|
|
|
|
/>
|
|
|
|
|
<EnvironmentHideDialog
|
|
|
|
|
environment={selectedEnvironment}
|
|
|
|
|
open={hideDialog}
|
|
|
|
|
setOpen={setHideDialog}
|
|
|
|
|
onConfirm={onHideConfirm}
|
|
|
|
|
/>
|
|
|
|
|
</StyledDivContainer>
|
2022-05-18 11:26:38 +02:00
|
|
|
|
</PageContent>
|
2021-09-30 10:24:16 +02:00
|
|
|
|
);
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
export default ProjectEnvironmentList;
|