-
Notifications
You must be signed in to change notification settings - Fork 67
/
Test.provider.tsx
134 lines (120 loc) 路 3.81 KB
/
Test.provider.tsx
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
import {noop} from 'lodash';
import {createContext, useCallback, useContext, useMemo, useState} from 'react';
import {useGetTestByIdQuery, useGetTestVersionByIdQuery} from 'redux/apis/TraceTest.api';
import {TDraftTest} from 'types/Test.types';
import VersionMismatchModal from 'components/VersionMismatchModal';
import TestService from 'services/Test.service';
import Test from 'models/Test.model';
import useTestCrud, {TTestRunRequest} from './hooks/useTestCrud';
interface IContext {
onEdit(values: TDraftTest): void;
onRun(runRequest?: Partial<TTestRunRequest>): void;
isLoading: boolean;
isError: boolean;
test: Test;
latestTest: Test;
isLatestVersion: boolean;
isEditLoading: boolean;
}
export const Context = createContext<IContext>({
onEdit: noop,
onRun: noop,
test: {} as Test,
latestTest: {} as Test,
isLoading: false,
isError: false,
isLatestVersion: true,
isEditLoading: false,
});
interface IProps {
testId: string;
version?: number;
children: React.ReactNode;
}
export const useTest = () => useContext(Context);
const TestProvider = ({children, testId, version = 0}: IProps) => {
const [isVersionModalOpen, setIsVersionModalOpen] = useState(false);
const [draft, setDraft] = useState<TDraftTest>({});
const [action, setAction] = useState<'edit' | 'run'>();
const {runTest, edit, isEditLoading} = useTestCrud();
const {
data: test,
isLoading: isCurrentLoading,
isError: isCurrentError,
} = useGetTestVersionByIdQuery({testId, version}, {skip: !version});
const {data: latestTest, isLoading: isLatestLoading, isError: isLatestError} = useGetTestByIdQuery({testId});
const isLoading = isLatestLoading || isCurrentLoading;
const isError = isLatestError || isCurrentError;
const currentTest = (version ? test : latestTest)!;
const isLatestVersion = useMemo(
() => (Boolean(version) && version === latestTest?.version) || currentTest?.version === latestTest?.version,
[currentTest?.version, latestTest?.version, version]
);
const onEdit = useCallback(
(values: TDraftTest) => {
if (isLatestVersion) edit(test!, values);
else {
setAction('edit');
setDraft(values);
setIsVersionModalOpen(true);
}
},
[edit, isLatestVersion, test]
);
const onRun = useCallback(
(request: Partial<TTestRunRequest> = {}) => {
if (isLatestVersion)
runTest({
test: currentTest,
...request,
});
else {
setAction('run');
setIsVersionModalOpen(true);
}
},
[currentTest, isLatestVersion, runTest]
);
const onConfirm = useCallback(() => {
if (action === 'edit') edit(test!, draft);
else {
const initialValues = TestService.getInitialValues(test!);
edit(test!, initialValues);
}
setIsVersionModalOpen(false);
}, [action, draft, edit, test]);
const value = useMemo<IContext>(
() => ({
onEdit,
onRun,
isLoading,
isError,
test: currentTest,
latestTest: latestTest!,
isLatestVersion,
isEditLoading,
}),
[onEdit, onRun, isLoading, isError, currentTest, latestTest, isLatestVersion, isEditLoading]
);
return currentTest && latestTest ? (
<>
<Context.Provider value={value}>{children}</Context.Provider>
<VersionMismatchModal
description={
action === 'edit'
? 'Editing it will result in a new version that will become the latest.'
: 'Running the test will use the latest version of the test.'
}
currentVersion={currentTest.version}
isOpen={isVersionModalOpen}
latestVersion={latestTest.version}
okText="Run Test"
onCancel={() => setIsVersionModalOpen(false)}
onConfirm={onConfirm}
/>
</>
) : (
<div data-cy="loading-test" />
);
};
export default TestProvider;