From 5f4c15bebd2722e5ff4c76b78301bfa2f8184d3d Mon Sep 17 00:00:00 2001
From: Marcus Kok <47163063+Marcusk19@users.noreply.github.com>
Date: Fri, 9 Feb 2024 09:41:46 -0500
Subject: [PATCH] ui: add export button for usage logs (PROJQUAY-6420) (#2492)
Adds tab for logs in the organization and repository views. Adds modal
for exporting the logs to either an email or callback url.
---
.gitignore | 2 +
web/cypress/e2e/usage-logs.cy.ts | 34 +++++
web/src/hooks/UseExportLogs.ts | 36 +++++
.../Organization/Organization.tsx | 12 ++
.../Tabs/UsageLogs/UsageLogsTab.tsx | 3 -
.../RepositoryDetails/RepositoryDetails.tsx | 11 ++
web/src/routes/UsageLogs/UsageLogs.tsx | 99 ++++++++++++++
.../routes/UsageLogs/UsageLogsExportModal.tsx | 127 ++++++++++++++++++
web/src/routes/UsageLogs/css/UsageLogs.scss | 7 +
9 files changed, 328 insertions(+), 3 deletions(-)
create mode 100644 web/cypress/e2e/usage-logs.cy.ts
create mode 100644 web/src/hooks/UseExportLogs.ts
delete mode 100644 web/src/routes/OrganizationsList/Organization/Tabs/UsageLogs/UsageLogsTab.tsx
create mode 100644 web/src/routes/UsageLogs/UsageLogs.tsx
create mode 100644 web/src/routes/UsageLogs/UsageLogsExportModal.tsx
create mode 100644 web/src/routes/UsageLogs/css/UsageLogs.scss
diff --git a/.gitignore b/.gitignore
index 5f2127af81..199876f326 100644
--- a/.gitignore
+++ b/.gitignore
@@ -14,6 +14,7 @@ GIT_HEAD
.python-version
.pylintrc
.coverage
+.coverage.*
coverage
htmlcov
.tox
@@ -30,6 +31,7 @@ test/dockerclients/.*
tmp/*
/.build-image-quay-stamp
/requirements-build.in
+__pycache__/
# files generated by local dev,
# do not need to check in and can be deleted
diff --git a/web/cypress/e2e/usage-logs.cy.ts b/web/cypress/e2e/usage-logs.cy.ts
new file mode 100644
index 0000000000..33093e6aba
--- /dev/null
+++ b/web/cypress/e2e/usage-logs.cy.ts
@@ -0,0 +1,34 @@
+describe('Usage Logs Export', () => {
+ beforeEach(() => {
+ cy.exec('npm run quay:seed');
+ cy.request('GET', `${Cypress.env('REACT_QUAY_APP_API_URL')}/csrf_token`)
+ .then((response) => response.body.csrf_token)
+ .then((token) => {
+ cy.loginByCSRF(token);
+ });
+ });
+
+ it('exports repository logs', () => {
+ cy.intercept(
+ 'POST',
+ 'api/v1/repository/user1/hello-world/exportlogs?starttime=$endtime=',
+ ).as('exportRepositoryLogs');
+ cy.visit('/repository/user1/hello-world');
+ cy.contains('Logs').click();
+ cy.contains('Export').click();
+ cy.get('[id="export-logs-callback"]').type('example@example.com');
+ cy.contains('Confirm').click();
+ cy.contains('Logs exported with id').should('be.visible');
+ });
+ it('exports repository logs failure', () => {
+ cy.intercept(
+ 'POST',
+ 'api/v1/repository/user1/hello-world/exportlogs?starttime=$endtime=',
+ ).as('exportRepositoryLogs');
+ cy.visit('/repository/user1/hello-world');
+ cy.contains('Logs').click();
+ cy.contains('Export').click();
+ cy.get('[id="export-logs-callback"]').type('blahblah');
+ cy.contains('Confirm').should('be.disabled');
+ });
+});
diff --git a/web/src/hooks/UseExportLogs.ts b/web/src/hooks/UseExportLogs.ts
new file mode 100644
index 0000000000..51125c534f
--- /dev/null
+++ b/web/src/hooks/UseExportLogs.ts
@@ -0,0 +1,36 @@
+import axios from 'src/libs/axios';
+import {ResourceError} from 'src/resources/ErrorHandling';
+
+export async function exportLogs(
+ org: string,
+ repo: string = null,
+ starttime: string,
+ endtime: string,
+ callback: string,
+) {
+ const exportLogsCallback = {};
+ if (callback.includes('@')) {
+ exportLogsCallback['callback_email'] = callback;
+ } else {
+ exportLogsCallback['callback_url'] = callback;
+ }
+
+ const url = repo
+ ? `/api/v1/repository/${org}/${repo}/exportlogs?starttime=${starttime}&endtime=${endtime}`
+ : `/api/v1/organization/${org}/exportlogs?starttime=${starttime}&endtime=${endtime}`;
+
+ try {
+ const response = await axios.post(url, exportLogsCallback);
+ return response.data;
+ } catch (error) {
+ if (error.response) {
+ return error.response.data;
+ } else {
+ throw new ResourceError(
+ 'Unable to export logs',
+ error.response.status,
+ error.response.data,
+ );
+ }
+ }
+}
diff --git a/web/src/routes/OrganizationsList/Organization/Organization.tsx b/web/src/routes/OrganizationsList/Organization/Organization.tsx
index a818aeda00..d9b8658068 100644
--- a/web/src/routes/OrganizationsList/Organization/Organization.tsx
+++ b/web/src/routes/OrganizationsList/Organization/Organization.tsx
@@ -16,6 +16,7 @@ import {useOrganization} from 'src/hooks/UseOrganization';
import {useQuayConfig} from 'src/hooks/UseQuayConfig';
import RepositoriesList from 'src/routes/RepositoriesList/RepositoriesList';
import RobotAccountsList from 'src/routes/RepositoriesList/RobotAccountsList';
+import UsageLogs from 'src/routes/UsageLogs/UsageLogs';
import CreatePermissionDrawer from './Tabs/DefaultPermissions/createPermissionDrawer/CreatePermissionDrawer';
import DefaultPermissionsList from './Tabs/DefaultPermissions/DefaultPermissionsList';
import Settings from './Tabs/Settings/Settings';
@@ -133,6 +134,17 @@ export default function Organization() {
),
visible: !isUserOrganization && organization?.is_admin,
},
+ {
+ name: 'Logs',
+ component: (
+
+ ),
+ visible: organization?.is_admin || !isUserOrganization,
+ },
{
name: 'Settings',
component: (
diff --git a/web/src/routes/OrganizationsList/Organization/Tabs/UsageLogs/UsageLogsTab.tsx b/web/src/routes/OrganizationsList/Organization/Tabs/UsageLogs/UsageLogsTab.tsx
deleted file mode 100644
index a89176418b..0000000000
--- a/web/src/routes/OrganizationsList/Organization/Tabs/UsageLogs/UsageLogsTab.tsx
+++ /dev/null
@@ -1,3 +0,0 @@
-export default function UsageLogsTab() {
- return
UsageLogsTab
;
-}
diff --git a/web/src/routes/RepositoryDetails/RepositoryDetails.tsx b/web/src/routes/RepositoryDetails/RepositoryDetails.tsx
index fd92e6d7d4..e07d8375f4 100644
--- a/web/src/routes/RepositoryDetails/RepositoryDetails.tsx
+++ b/web/src/routes/RepositoryDetails/RepositoryDetails.tsx
@@ -41,6 +41,7 @@ import Settings from './Settings/Settings';
import TagHistory from './TagHistory/TagHistory';
import TagsList from './Tags/TagsList';
import {DrawerContentType} from './Types';
+import UsageLogs from '../UsageLogs/UsageLogs';
enum TabIndex {
Tags = 'tags',
@@ -266,6 +267,16 @@ export default function RepositoryDetails() {
repoDetails={repoDetails}
/>
+ Logs}
+ >
+
+
Builds}
diff --git a/web/src/routes/UsageLogs/UsageLogs.tsx b/web/src/routes/UsageLogs/UsageLogs.tsx
new file mode 100644
index 0000000000..e2472752d5
--- /dev/null
+++ b/web/src/routes/UsageLogs/UsageLogs.tsx
@@ -0,0 +1,99 @@
+import React from 'react';
+import {DatePicker, Split, SplitItem} from '@patternfly/react-core';
+import ExportLogsModal from './UsageLogsExportModal';
+import './css/UsageLogs.scss';
+
+interface UsageLogsProps {
+ organization: string;
+ repository: string;
+ type: string;
+}
+
+function defaultStartDate() {
+ // should be 1 month before current date
+ const currentDate = new Date();
+ currentDate.setMonth(currentDate.getMonth() - 1);
+
+ const year = currentDate.getFullYear();
+ const month = (currentDate.getMonth() + 1).toString().padStart(2, '0');
+ const day = currentDate.getDate().toString().padStart(2, '0');
+ const formattedDate = `${year}-${month}-${day}`;
+
+ return formattedDate;
+}
+
+function defaultEndDate() {
+ // should be current date
+ const date = new Date();
+
+ const year = date.getFullYear();
+ const month = (date.getMonth() + 1).toString().padStart(2, '0');
+ const day = (date.getDate() + 1).toString().padStart(2, '0');
+ const formattedDate = `${year}-${month}-${day}`;
+
+ return formattedDate;
+}
+
+function formatDate(date: string) {
+ /**
+ * change date string from y-m-d to m%d%y for api
+ */
+ const dates = date.split('-');
+ const year = dates[0];
+ const month = dates[1];
+ const day = dates[2];
+
+ return `${month}/${day}/${year}`;
+}
+
+export default function UsageLogs(props: UsageLogsProps) {
+ const [logStartDate, setLogStartDate] = React.useState(
+ formatDate(defaultStartDate()),
+ );
+ const [logEndDate, setLogEndDate] = React.useState(
+ formatDate(defaultEndDate()),
+ );
+ const minDate = new Date(defaultStartDate());
+ const maxDate = new Date(defaultEndDate());
+ const rangeValidator = (date: Date) => {
+ if (date < minDate) {
+ return 'Date is before the allowable range';
+ } else if (date > maxDate) {
+ return 'Date is after the allowable range';
+ }
+ return '';
+ };
+
+ return (
+
+
+
+ {
+ setLogStartDate(formatDate(str));
+ }}
+ validators={[rangeValidator]}
+ />
+
+
+ {
+ setLogEndDate(formatDate(str));
+ }}
+ validators={[rangeValidator]}
+ />
+
+
+
+
+
+ );
+}
diff --git a/web/src/routes/UsageLogs/UsageLogsExportModal.tsx b/web/src/routes/UsageLogs/UsageLogsExportModal.tsx
new file mode 100644
index 0000000000..d53af27484
--- /dev/null
+++ b/web/src/routes/UsageLogs/UsageLogsExportModal.tsx
@@ -0,0 +1,127 @@
+import React from 'react';
+import {
+ Button,
+ Text,
+ Modal,
+ ModalVariant,
+ TextInput,
+ HelperText,
+ HelperTextItem,
+} from '@patternfly/react-core';
+
+import {exportLogs} from 'src/hooks/UseExportLogs';
+import {useAlerts} from 'src/hooks/UseAlerts';
+import {AlertVariant} from 'src/atoms/AlertState';
+
+export default function ExportLogsModal(props: ExportLogsModalProps) {
+ const [isModalOpen, setIsModalOpen] = React.useState(false);
+ const [callbackEmailOrUrl, setCallbackEmailOrUrl] = React.useState('');
+ const [userInputValidated, setUserInputValidated] = React.useState<
+ 'success' | 'warning' | 'error' | 'default'
+ >('default');
+ const handleModalToggle = (_event: KeyboardEvent | React.MouseEvent) => {
+ setIsModalOpen(!isModalOpen);
+ };
+ const {addAlert} = useAlerts();
+
+ const exportLogsClick = (callback: string) => {
+ return exportLogs(
+ props.organization,
+ props.repository,
+ props.starttime,
+ props.endtime,
+ callback,
+ );
+ };
+
+ const validateUserInput = (userInput: string) => {
+ return /(http(s)?:.+)|.+@.+/.test(userInput);
+ };
+
+ const validateDate = () => {
+ if (new Date(props.endtime) < new Date(props.starttime)) return false;
+ return true;
+ };
+
+ return (
+
+
+
+ exportLogsClick(callbackEmailOrUrl).then((response) => {
+ if (response['export_id']) {
+ addAlert({
+ variant: AlertVariant.Success,
+ title: `Logs exported with id ${response['export_id']}`,
+ });
+ setIsModalOpen(false);
+ } else {
+ addAlert({
+ variant: AlertVariant.Failure,
+ title: `Problem exporting logs: ${response['error_message']}`,
+ });
+ setIsModalOpen(false);
+ }
+ })
+ }
+ isDisabled={userInputValidated === 'error'}
+ >
+ {' '}
+ Confirm
+ ,
+ ,
+ ]}
+ >
+
+ Enter an e-mail address or callback URL (must start with http:// or
+ https://) at which to receive the exported logs once they have been
+ fully processed:
+
+ {
+ if (validateUserInput(callbackEmailOrUrl))
+ setUserInputValidated('success');
+ else setUserInputValidated('error');
+ setCallbackEmailOrUrl(callbackEmailOrUrl);
+ }}
+ validated={userInputValidated}
+ />
+
+
+ {' '}
+ Note: The export process can take up to an hour to process if there
+ are many logs. As well, only a single export process can run at a
+ time for each namespace. Additional export requests will be queued.{' '}
+
+
+
+
+ );
+}
+
+interface ExportLogsModalProps {
+ organization: string;
+ repository: string;
+ starttime: string;
+ endtime: string;
+ type: string;
+}
diff --git a/web/src/routes/UsageLogs/css/UsageLogs.scss b/web/src/routes/UsageLogs/css/UsageLogs.scss
new file mode 100644
index 0000000000..63244b4dbc
--- /dev/null
+++ b/web/src/routes/UsageLogs/css/UsageLogs.scss
@@ -0,0 +1,7 @@
+.usage-logs-header {
+ margin: 20px;
+}
+
+.usage-logs {
+ margin: 20px;
+}