Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Task/WP-724: Mutation Hooks: Compress Files #1009

Open
wants to merge 36 commits into
base: main
Choose a base branch
from
Open
Show file tree
Hide file tree
Changes from 16 commits
Commits
Show all changes
36 commits
Select commit Hold shift + click to select a range
c30c46c
Starting from scratch
Nov 6, 2024
b74ed53
Set up useCompress.ts
Nov 6, 2024
199f671
Merge branch 'main' of github.com:TACC/Core-Portal into task/WP-724
Nov 11, 2024
cae111b
Still working on useCompress()
Nov 12, 2024
cec7b4f
Progressing with useCompress mutation
Nov 12, 2024
c5028dc
Committing branch in its current state; not fully functioning
Nov 14, 2024
64db5d1
wip
rstijerina Nov 14, 2024
79957a4
Reworked useCompress.ts
Nov 14, 2024
00aaf76
wip
rstijerina Nov 14, 2024
82ab9cf
Still trying to fix it; much closer now
Nov 15, 2024
e96a7d3
handle undefined execSystemId
rstijerina Nov 15, 2024
b55e8e8
Compress mutation finally successful
Nov 15, 2024
07a213b
Toasts and modals work correctly
Nov 15, 2024
4f4b5b9
Linted client-side code
Nov 15, 2024
aa1a35d
Added an additional asynchronous call; updated Compress modal
Nov 26, 2024
757c637
Merge branch 'main' into task/WP-724
rstijerina Nov 27, 2024
6e13705
Refactored types into useSubmitJob.ts
Dec 6, 2024
9ff27eb
Corrected mutation hook to return archive in current directory instea…
Dec 6, 2024
7cbe3a6
Update client/src/hooks/datafiles/mutations/useCompress.ts
jmcmillenmusic Dec 6, 2024
3ef04df
Linted client-side code
Dec 6, 2024
d4ad0b1
Merge branch 'main' into task/WP-724
jmcmillenmusic Dec 6, 2024
b7f8389
Can't get this test to pass still
Dec 6, 2024
f5e6bb4
Merge branch 'task/WP-724' of github.com:TACC/Core-Portal into task/W…
Dec 6, 2024
688f9e1
Linted client-side code
Dec 6, 2024
ef118aa
Finally got the failing test to pass
Dec 6, 2024
d42eb23
Skipping tests temporarily, cleaned up code
Dec 9, 2024
d9e4bfb
Skipping saga tests
Dec 9, 2024
0c17dc3
Merge branch 'main' into task/WP-724
jmcmillenmusic Dec 16, 2024
4e5364d
Corrected defaultPrivateSystem and edited package-lock.json
Dec 17, 2024
99c2d88
Merge branch 'task/WP-724' of github.com:TACC/Core-Portal into task/W…
Dec 17, 2024
88420e5
Made corrections to files based on feedback
Dec 17, 2024
a82283b
Merge branch 'main' into task/WP-724
jmcmillenmusic Dec 18, 2024
b36c3f5
Added final change to account for empty strings or undefined in job_p…
Dec 20, 2024
86e16aa
Task/WP-725: Mutation Hooks: Extract Files (#1035)
jmcmillenmusic Dec 20, 2024
105f4a5
Merge branch 'main' into task/WP-724
rstijerina Dec 20, 2024
d28d502
update example; add comment
rstijerina Dec 20, 2024
File filter

Filter by extension

Filter by extension


Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
10 changes: 6 additions & 4 deletions client/package-lock.json

Some generated files are not rendered by default. Learn more about how customized files appear on GitHub.

1 change: 1 addition & 0 deletions client/package.json
Original file line number Diff line number Diff line change
Expand Up @@ -72,6 +72,7 @@
"@testing-library/jest-dom": "^5.0.2",
"@testing-library/react": "^16.0.1",
"@types/js-cookie": "^3.0.6",
"@types/node": "^22.9.0",
jmcmillenmusic marked this conversation as resolved.
Show resolved Hide resolved
"@types/react": "^18.3.10",
"@types/react-dom": "^18.3.0",
"@types/react-redux": "^7.1.18",
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -108,7 +108,7 @@ const DataFilesCompressModal = () => {
/>
<p>
A job to compress these files will be submitted. The
compressed file archive will appear in this directory.
compressed file archive will appear in the Root directory.
jmcmillenmusic marked this conversation as resolved.
Show resolved Hide resolved
</p>
</ModalBody>
<ModalFooter>
Expand Down
27 changes: 0 additions & 27 deletions client/src/hooks/datafiles/mutations/useCompress.js

This file was deleted.

209 changes: 209 additions & 0 deletions client/src/hooks/datafiles/mutations/useCompress.ts
Original file line number Diff line number Diff line change
@@ -0,0 +1,209 @@
import { useMutation } from '@tanstack/react-query';
import { useSelector, useDispatch, shallowEqual } from 'react-redux';
import { getCompressParams } from 'utils/getCompressParams';
import { apiClient } from 'utils/apiClient';
import {
TTapisSystem,
TAppFileInput,
TTapisJob,
TJobArgSpecs,
TJobKeyValuePair,
TTapisFile,
TPortalSystem,
} from 'utils/types';

export type TJobPostOperations = 'resubmitJob' | 'cancelJob' | 'submitJob';

export type TParameterSetSubmit = {
appArgs?: TJobArgSpecs;
containerArgs?: TJobArgSpecs;
schedulerOptions?: TJobArgSpecs;
envVariables?: TJobKeyValuePair[];
};

export type TConfigurationValues = {
execSystemId?: string;
execSystemLogicalQueue?: string;
maxMinutes?: number;
nodeCount?: number;
coresPerNode?: number;
allocation?: string;
memoryMB?: number;
};

export type TOutputValues = {
name: string;
archiveSystemId?: string;
archiveSystemDir?: string;
};

export interface TJobSubmit extends TConfigurationValues, TOutputValues {
archiveOnAppError?: boolean;
appId: string;
fileInputs?: TAppFileInput[];
parameterSet?: TParameterSetSubmit;
}

export type TJobBody = {
operation?: TJobPostOperations;
uuid?: string;
job: TJobSubmit;
licenseType?: string;
isInteractive?: boolean;
execSystemId?: string;
};

interface IJobPostResponse extends TTapisJob {
execSys?: TTapisSystem;
}

type TJobPostResponse = {
response: IJobPostResponse;
status: number;
};

async function submitJobUtil(body: TJobBody) {
const res = await apiClient.post<TJobPostResponse>(
`/api/workspace/jobs`,
body
);
return res.data.response;
}
jmcmillenmusic marked this conversation as resolved.
Show resolved Hide resolved

function useCompress() {
const dispatch = useDispatch();
const status = useSelector(
(state: any) => state.files.operationStatus.compress,
shallowEqual
);

const setStatus = (newStatus: any) => {
dispatch({
type: 'DATA_FILES_SET_OPERATION_STATUS',
payload: { status: newStatus, operation: 'compress' },
});
};

const compressErrorAction = (errorMessage: any) => {
return {
type: 'DATA_FILES_SET_OPERATION_STATUS',
payload: {
status: { type: 'ERROR', message: errorMessage },
operation: 'compress',
},
};
};

const compressApp = useSelector(
(state: any) => state.workbench.config.compressApp
);

const defaultAllocation = useSelector(
(state: any) =>
state.allocations.portal_alloc || state.allocations.active[0].projectName

Check failure on line 103 in client/src/hooks/datafiles/mutations/useCompress.ts

View workflow job for this annotation

GitHub Actions / Client_Side_Unit_Tests

src/components/DataFiles/tests/DataFiles.test.jsx > DataFiles > should render Data Files with multiple private systems

TypeError: Cannot read properties of undefined (reading 'portal_alloc') ❯ src/hooks/datafiles/mutations/useCompress.ts:103:25 ❯ useSelectorWithStoreAndSubscription node_modules/react-redux/lib/hooks/useSelector.js:39:30 ❯ Proxy.useSelector node_modules/react-redux/lib/hooks/useSelector.js:139:25 ❯ Module.useCompress src/hooks/datafiles/mutations/useCompress.ts:101:29 ❯ DataFilesCompressModal src/components/DataFiles/DataFilesModals/DataFilesCompressModal.jsx:17:43 ❯ renderWithHooks node_modules/react-dom/cjs/react-dom.development.js:15486:18 ❯ mountIndeterminateComponent node_modules/react-dom/cjs/react-dom.development.js:20103:13 ❯ beginWork node_modules/react-dom/cjs/react-dom.development.js:21626:16 ❯ beginWork$1 node_modules/react-dom/cjs/react-dom.development.js:27465:14 ❯ performUnitOfWork node_modules/react-dom/cjs/react-dom.development.js:26599:12
);

const systems = useSelector(
(state: any) => state.systems.storage.configuration
);

const { mutateAsync } = useMutation({ mutationFn: submitJobUtil });

const compress = ({
scheme,
files,
filename,
compressionType,
}: {
scheme: string;
files: TTapisFile[];
filename: string;
compressionType: string;
}) => {
dispatch({
type: 'DATA_FILES_SET_OPERATION_STATUS',
payload: { status: 'RUNNING', operation: 'compress' },
});

let defaultPrivateSystem: TPortalSystem | undefined;

if (scheme !== 'private' && scheme !== 'projects') {
defaultPrivateSystem = systems.find((s: any) => s.default);

if (!defaultPrivateSystem) {
throw new Error('Folder downloads are unavailable in this portal', {
cause: 'compressError',
});
}
}

const params = getCompressParams(
files,
filename,
compressionType,
compressApp,
defaultAllocation,
defaultPrivateSystem
jmcmillenmusic marked this conversation as resolved.
Show resolved Hide resolved
);

return mutateAsync(
{
job: params,
},
{
onSuccess: (response: any) => {
// If the execution system requires pushing keys, then
// bring up the modal and retry the compress action
if (response.execSys) {
dispatch({
type: 'SYSTEMS_TOGGLE_MODAL',
payload: {
operation: 'pushKeys',
props: {
system: response.execSys,
onCancel: compressErrorAction('An error has occurred'),
},
},
});
} else if (response.status === 'PENDING') {
dispatch({
type: 'DATA_FILES_SET_OPERATION_STATUS',
payload: { status: { type: 'SUCCESS' }, operation: 'compress' },
});
dispatch({
type: 'ADD_TOAST',
payload: {
message: 'Compressed ZIP file in Root directory shortly',
jmcmillenmusic marked this conversation as resolved.
Show resolved Hide resolved
},
});
dispatch({
type: 'DATA_FILES_SET_OPERATION_STATUS',
payload: { operation: 'compress', status: {} },
});
dispatch({
type: 'DATA_FILES_TOGGLE_MODAL',
payload: { operation: 'compress', props: {} },
});
rstijerina marked this conversation as resolved.
Show resolved Hide resolved
}
},
onError: (response) => {
const errorMessage =
response.cause === 'compressError'
? response.message
: 'An error has occurred.';
dispatch({
type: 'DATA_FILES_SET_OPERATION_STATUS',
payload: {
status: { type: 'ERROR', message: errorMessage },
operation: 'compress',
},
});
},
}
);
};

return { compress, status, setStatus };
}

export default useCompress;
71 changes: 3 additions & 68 deletions client/src/redux/sagas/datafiles.sagas.js
Original file line number Diff line number Diff line change
Expand Up @@ -14,6 +14,7 @@ import {
import { fetchUtil } from 'utils/fetchUtil';
import truncateMiddle from '../../utils/truncateMiddle';
import { fetchAppDefinitionUtil } from './apps.sagas';
import { getCompressParams } from 'utils/getCompressParams';

/**
* Utility function to replace instances of 2 or more slashes in a URL with
Expand Down Expand Up @@ -984,72 +985,6 @@ export function* watchExtract() {
yield takeLeading('DATA_FILES_EXTRACT', extractFiles);
}

/**
* Create JSON string of job params
* @async
* @param {Array<Object>} files
* @param {String} archiveFileName
* @returns {String}
*/
const getCompressParams = (
files,
archiveFileName,
compressionType,
defaultPrivateSystem,
latestCompress,
defaultAllocation
) => {
const fileInputs = files.map((file) => ({
sourceUrl: `tapis://${file.system}/${file.path}`,
}));

let archivePath, archiveSystem;

if (defaultPrivateSystem) {
archivePath = defaultPrivateSystem.homeDir;
archiveSystem = defaultPrivateSystem.system;
} else {
archivePath = `${files[0].path.slice(0, -files[0].name.length)}`;
archiveSystem = files[0].system;
}

return JSON.stringify({
job: {
fileInputs: fileInputs,
name: `${latestCompress.definition.id}-${
latestCompress.definition.version
}_${new Date().toISOString().split('.')[0]}`,
archiveSystemId: archiveSystem,
archiveSystemDir: archivePath,
archiveOnAppError: false,
appId: latestCompress.definition.id,
appVersion: latestCompress.definition.version,
parameterSet: {
appArgs: [
{
name: 'Archive File Name',
arg: archiveFileName,
},
{
name: 'Compression Type',
arg: compressionType,
},
],
schedulerOptions: [
{
name: 'TACC Allocation',
description:
'The TACC allocation associated with this job execution',
include: true,
arg: `-A ${defaultAllocation}`,
},
],
},
execSystemId: latestCompress.definition.jobAttributes.execSystemId,
},
});
};

export const compressAppSelector = (state) =>
state.workbench.config.compressApp;

Expand Down Expand Up @@ -1095,9 +1030,9 @@ export function* compressFiles(action) {
action.payload.files,
action.payload.filename,
action.payload.compressionType,
defaultPrivateSystem,
latestCompress,
defaultAllocation
defaultAllocation,
defaultPrivateSystem
);

const res = yield call(jobHelper, params);
Expand Down
Loading
Loading