"official/legacy/bert/common_flags.py" did not exist on "2741cc5ffe61f5a6e43827e72e2939fa7baf3767"
App.tsx 8.47 KB
Newer Older
Deshui Yu's avatar
Deshui Yu committed
1
import * as React from 'react';
2
import { Stack } from '@fluentui/react';
3
4
import { COLUMN } from './static/const';
import { EXPERIMENT, TRIALS } from './static/datamodel';
5
import NavCon from './components/NavCon';
6
import MessageInfo from './components/modals/MessageInfo';
7
import { TrialConfigButton } from './components/public-child/config/TrialConfigButton';
8
import './App.scss';
Deshui Yu's avatar
Deshui Yu committed
9

10
interface AppState {
11
    interval: number;
12
    columnList: string[];
13
14
    experimentUpdateBroadcast: number;
    trialsUpdateBroadcast: number;
Lijiao's avatar
Lijiao committed
15
    metricGraphMode: 'max' | 'min'; // tuner's optimize_mode filed
16
17
    isillegalFinal: boolean;
    expWarningMessage: string;
Lijiaoa's avatar
Lijiaoa committed
18
    bestTrialEntries: string; // for overview page: best trial entreis
Lijiaoa's avatar
Lijiaoa committed
19
    isUpdate: boolean;
20
21
}

22
23
24
25
26
27
28
29
30
31
export const AppContext = React.createContext({
    interval: 10, // sendons
    columnList: COLUMN,
    experimentUpdateBroadcast: 0,
    trialsUpdateBroadcast: 0,
    metricGraphMode: 'max',
    bestTrialEntries: '10',
    // eslint-disable-next-line @typescript-eslint/no-empty-function, @typescript-eslint/no-unused-vars
    changeColumn: (val: string[]) => {},
    // eslint-disable-next-line @typescript-eslint/no-empty-function, @typescript-eslint/no-unused-vars
32
    changeMetricGraphMode: (val: 'max' | 'min') => {},
33
    // eslint-disable-next-line @typescript-eslint/no-empty-function, @typescript-eslint/no-unused-vars
34
35
36
    changeEntries: (val: string) => {},
    // eslint-disable-next-line @typescript-eslint/no-empty-function, @typescript-eslint/no-unused-vars
    updateOverviewPage: () => {}
37
38
});

39
class App extends React.Component<{}, AppState> {
40
41
    private timerId!: number | undefined;
    private firstLoad: boolean = false; // when click refresh selector options
42
43
44
45
46
47
48
    constructor(props: {}) {
        super(props);
        this.state = {
            interval: 10, // sendons
            columnList: COLUMN,
            experimentUpdateBroadcast: 0,
            trialsUpdateBroadcast: 0,
49
50
            metricGraphMode: 'max',
            isillegalFinal: false,
Lijiaoa's avatar
Lijiaoa committed
51
            expWarningMessage: '',
Lijiaoa's avatar
Lijiaoa committed
52
53
            bestTrialEntries: '10',
            isUpdate: true
54
        };
55
56
    }

Lijiao's avatar
Lijiao committed
57
    async componentDidMount(): Promise<void> {
58
        await Promise.all([EXPERIMENT.init(), TRIALS.init()]);
59
60
        this.setState(state => ({
            experimentUpdateBroadcast: state.experimentUpdateBroadcast + 1,
Lijiaoa's avatar
Lijiaoa committed
61
            trialsUpdateBroadcast: state.trialsUpdateBroadcast + 1,
62
            metricGraphMode: EXPERIMENT.optimizeMode === 'minimize' ? 'min' : 'max'
Lijiaoa's avatar
Lijiaoa committed
63
64
        }));
        this.timerId = window.setTimeout(this.refresh, this.state.interval * 100);
65
66
    }

Lijiao's avatar
Lijiao committed
67
    changeInterval = (interval: number): void => {
68
69
        window.clearTimeout(this.timerId);
        if (interval === 0) {
Lijiaoa's avatar
Lijiaoa committed
70
            return;
71
        }
72
73
        // setState will trigger page refresh at once.
        // setState is asyc, interval not update to (this.state.interval) at once.
Lijiaoa's avatar
Lijiaoa committed
74
        this.setState({ interval }, () => {
75
76
77
            this.firstLoad = true;
            this.refresh();
        });
78
    };
79

80
    // TODO: use local storage
81
    changeColumn = (columnList: string[]): void => {
82
        this.setState({ columnList: columnList });
83
    };
84

Lijiao's avatar
Lijiao committed
85
    changeMetricGraphMode = (val: 'max' | 'min'): void => {
Lijiao's avatar
Lijiao committed
86
        this.setState({ metricGraphMode: val });
87
    };
Lijiao's avatar
Lijiao committed
88

Lijiaoa's avatar
Lijiaoa committed
89
90
    // overview best trial module
    changeEntries = (entries: string): void => {
Lijiaoa's avatar
Lijiaoa committed
91
        this.setState({ bestTrialEntries: entries });
92
    };
Lijiaoa's avatar
Lijiaoa committed
93

94
95
96
97
98
99
    updateOverviewPage = (): void => {
        this.setState(state => ({
            experimentUpdateBroadcast: state.experimentUpdateBroadcast + 1
        }));
    };

Lijiaoa's avatar
Lijiaoa committed
100
    shouldComponentUpdate(nextProps: any, nextState: AppState): boolean {
101
        if (!(nextState.isUpdate || nextState.isUpdate === undefined)) {
Lijiaoa's avatar
Lijiaoa committed
102
103
104
105
106
107
            nextState.isUpdate = true;
            return false;
        }
        return true;
    }

108
    render(): React.ReactNode {
109
110
111
112
113
114
115
116
117
        const {
            interval,
            columnList,
            experimentUpdateBroadcast,
            trialsUpdateBroadcast,
            metricGraphMode,
            isillegalFinal,
            expWarningMessage,
            bestTrialEntries
118
        } = this.state;
119
        if (experimentUpdateBroadcast === 0 || trialsUpdateBroadcast === 0) {
120
            return null; // TODO: render a loading page
121
        }
Lijiaoa's avatar
Lijiaoa committed
122
123
124
125
126
127
128
129
        const errorList = [
            { errorWhere: TRIALS.jobListError(), errorMessage: TRIALS.getJobErrorMessage() },
            { errorWhere: EXPERIMENT.experimentError(), errorMessage: EXPERIMENT.getExperimentMessage() },
            { errorWhere: EXPERIMENT.statusError(), errorMessage: EXPERIMENT.getStatusMessage() },
            { errorWhere: TRIALS.MetricDataError(), errorMessage: TRIALS.getMetricDataErrorMessage() },
            { errorWhere: TRIALS.latestMetricDataError(), errorMessage: TRIALS.getLatestMetricDataErrorMessage() },
            { errorWhere: TRIALS.metricDataRangeError(), errorMessage: TRIALS.metricDataRangeErrorMessage() }
        ];
130
        return (
131
132
133
            <Stack className='nni' style={{ minHeight: window.innerHeight }}>
                <div className='header'>
                    <div className='headerCon'>
134
                        <NavCon changeInterval={this.changeInterval} refreshFunction={this.lastRefresh} />
135
136
                    </div>
                </div>
137
138
                <Stack className='contentBox'>
                    <Stack className='content'>
139
140
                        {/* search space & config */}
                        <TrialConfigButton />
Lijiaoa's avatar
Lijiaoa committed
141
                        {/* if api has error field, show error message */}
142
143
144
145
146
                        {errorList.map(
                            (item, key) =>
                                item.errorWhere && (
                                    <div key={key} className='warning'>
                                        <MessageInfo info={item.errorMessage} typeInfo='error' />
Lijiaoa's avatar
Lijiaoa committed
147
                                    </div>
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
                                )
                        )}
                        {isillegalFinal && (
                            <div className='warning'>
                                <MessageInfo info={expWarningMessage} typeInfo='warning' />
                            </div>
                        )}
                        <AppContext.Provider
                            value={{
                                interval,
                                columnList,
                                changeColumn: this.changeColumn,
                                experimentUpdateBroadcast,
                                trialsUpdateBroadcast,
                                metricGraphMode,
                                changeMetricGraphMode: this.changeMetricGraphMode,
                                bestTrialEntries,
165
166
                                changeEntries: this.changeEntries,
                                updateOverviewPage: this.updateOverviewPage
167
168
                            }}
                        >
169
170
                            {this.props.children}
                        </AppContext.Provider>
171
172
173
                    </Stack>
                </Stack>
            </Stack>
174
175
176
        );
    }

Lijiao's avatar
Lijiao committed
177
    private refresh = async (): Promise<void> => {
178
179
180
181
182
183
184
185
186
187
188
189
        // resolve this question: 10s -> 20s, page refresh twice.
        // only refresh this page after clicking the refresh options
        if (this.firstLoad !== true) {
            const [experimentUpdated, trialsUpdated] = await Promise.all([EXPERIMENT.update(), TRIALS.update()]);
            if (experimentUpdated) {
                this.setState(state => ({ experimentUpdateBroadcast: state.experimentUpdateBroadcast + 1 }));
            }
            if (trialsUpdated) {
                this.setState(state => ({ trialsUpdateBroadcast: state.trialsUpdateBroadcast + 1 }));
            }
        } else {
            this.firstLoad = false;
190
191
        }

Lijiaoa's avatar
Lijiaoa committed
192
193
        // experiment status and /trial-jobs api's status could decide website update
        if (['DONE', 'ERROR', 'STOPPED'].includes(EXPERIMENT.status) || TRIALS.jobListError()) {
194
            // experiment finished, refresh once more to ensure consistency
Lijiaoa's avatar
Lijiaoa committed
195
            this.setState(() => ({ interval: 0, isUpdate: false }));
196
            return;
197
        }
198

Lijiaoa's avatar
Lijiaoa committed
199
        this.timerId = window.setTimeout(this.refresh, this.state.interval * 1000);
200
    };
201

202
    public async lastRefresh(): Promise<void> {
203
204
        await EXPERIMENT.update();
        await TRIALS.update(true);
205
206
207
208
        this.setState(state => ({
            experimentUpdateBroadcast: state.experimentUpdateBroadcast + 1,
            trialsUpdateBroadcast: state.trialsUpdateBroadcast + 1
        }));
209
    }
Deshui Yu's avatar
Deshui Yu committed
210
211
212
}

export default App;