ydb-embedded-ui 1.14.1 → 2.0.0
Sign up to get free protection for your applications and to get access to all the features.
- package/CHANGELOG.md +32 -0
- package/dist/assets/illustrations/dark/thumbsUp.svg +15 -0
- package/dist/assets/illustrations/light/thumbsUp.svg +15 -0
- package/dist/components/AsideNavigation/AsideHeader.tsx +1 -1
- package/dist/components/AsideNavigation/AsideHeaderFooterItem/AsideHeaderFooterItem.tsx +1 -1
- package/dist/components/AsideNavigation/AsideHeaderTooltip/AsideHeaderTooltip.tsx +1 -1
- package/dist/components/AsideNavigation/CompositeBar/CompositeBar.tsx +1 -1
- package/dist/components/AsideNavigation/Logo/Logo.tsx +1 -1
- package/dist/components/AsideNavigation/Settings/Settings.tsx +1 -1
- package/dist/components/AsideNavigation/Settings/SettingsMenu/SettingsMenu.tsx +1 -1
- package/dist/components/AsideNavigation/Settings/SettingsSearch/SettingsSearch.tsx +1 -1
- package/dist/components/AsideNavigation/Settings/collect-settings.ts +1 -1
- package/dist/components/Breadcrumbs/Breadcrumbs.js +1 -1
- package/dist/components/ClusterInfo/ClusterInfo.tsx +1 -1
- package/dist/components/Collapse/Collapse.js +1 -1
- package/dist/components/CopyToClipboard/CopyToClipboard.tsx +1 -1
- package/dist/components/CriticalActionDialog/CriticalActionDialog.js +1 -1
- package/dist/components/EmptyState/EmptyState.scss +0 -1
- package/dist/components/EnableFullscreenButton/EnableFullscreenButton.tsx +1 -1
- package/dist/components/EntityStatus/EntityStatus.js +1 -1
- package/dist/components/Fullscreen/Fullscreen.tsx +1 -1
- package/dist/components/Icon/Icon.js +1 -1
- package/dist/components/Illustration/Illustration.tsx +3 -1
- package/dist/components/Loader/Loader.tsx +1 -1
- package/dist/components/NodesViewer/NodesViewer.js +4 -2
- package/dist/components/Pagination/Pagination.js +1 -1
- package/dist/components/ProblemFilter/ProblemFilter.js +1 -1
- package/dist/components/ShortyString/ShortyString.tsx +1 -1
- package/dist/components/TableSkeleton/TableSkeleton.tsx +1 -1
- package/dist/components/TabletsOverall/TabletsOverall.tsx +1 -1
- package/dist/containers/App/App.js +1 -1
- package/dist/containers/App/App.scss +0 -10
- package/dist/containers/App/Content.js +1 -1
- package/dist/containers/App/TipPopup/TipPopup.js +1 -1
- package/dist/containers/AsideNavigation/AsideNavigation.tsx +1 -1
- package/dist/containers/Authentication/Authentication.tsx +1 -1
- package/dist/containers/Group/Group.js +1 -1
- package/dist/containers/Header/Header.tsx +1 -1
- package/dist/containers/Header/Host/Host.js +1 -1
- package/dist/containers/Heatmap/Heatmap.js +1 -1
- package/dist/containers/Node/Node.tsx +1 -1
- package/dist/containers/Node/NodeStructure/Pdisk.tsx +1 -1
- package/dist/containers/Nodes/Nodes.js +3 -2
- package/dist/containers/Pdisk/Pdisk.js +1 -1
- package/dist/containers/Pool/Pool.js +3 -2
- package/dist/containers/ReduxTooltip/ReduxTooltip.js +1 -1
- package/dist/containers/Storage/EmptyFilter/EmptyFilter.tsx +34 -0
- package/dist/containers/Storage/EmptyFilter/i18n/en.json +4 -0
- package/dist/containers/Storage/EmptyFilter/i18n/index.ts +11 -0
- package/dist/containers/Storage/EmptyFilter/i18n/ru.json +4 -0
- package/dist/containers/Storage/Pdisk/Pdisk.tsx +1 -1
- package/dist/containers/Storage/Storage.js +3 -1
- package/dist/containers/Storage/StorageFilter/StorageFilter.tsx +1 -1
- package/dist/containers/Storage/StorageGroups/StorageGroups.tsx +27 -6
- package/dist/containers/Storage/StorageGroups/i18n/en.json +6 -0
- package/dist/containers/Storage/StorageGroups/i18n/index.ts +11 -0
- package/dist/containers/Storage/StorageGroups/i18n/ru.json +6 -0
- package/dist/containers/Storage/StorageNodes/StorageNodes.tsx +24 -7
- package/dist/containers/Storage/StorageNodes/i18n/en.json +6 -0
- package/dist/containers/Storage/StorageNodes/i18n/index.ts +11 -0
- package/dist/containers/Storage/StorageNodes/i18n/ru.json +6 -0
- package/dist/containers/Storage/UsageFilter/UsageFilter.tsx +1 -1
- package/dist/containers/Storage/Vdisk/Vdisk.js +1 -1
- package/dist/containers/Tablet/Tablet.js +1 -1
- package/dist/containers/Tablets/Tablets.js +1 -1
- package/dist/containers/TabletsFilters/TabletsFilters.js +1 -1
- package/dist/containers/Tenant/Acl/Acl.js +1 -1
- package/dist/containers/Tenant/Diagnostics/Compute/Compute.js +1 -1
- package/dist/containers/Tenant/Diagnostics/Describe/Describe.js +1 -1
- package/dist/containers/Tenant/Diagnostics/DetailedOverview/DetailedOverview.tsx +1 -1
- package/dist/containers/Tenant/Diagnostics/Diagnostics.tsx +1 -1
- package/dist/containers/Tenant/Diagnostics/Healthcheck/Healthcheck.js +1 -1
- package/dist/containers/Tenant/Diagnostics/HotKeys/HotKeys.js +1 -1
- package/dist/containers/Tenant/Diagnostics/Network/Network.js +3 -2
- package/dist/containers/Tenant/Diagnostics/Network/Network.scss +1 -1
- package/dist/containers/Tenant/Diagnostics/Overview/Overview.tsx +23 -23
- package/dist/containers/Tenant/Diagnostics/TenantOverview/TenantOverview.js +1 -1
- package/dist/containers/Tenant/Diagnostics/TopQueries/TopQueries.js +3 -3
- package/dist/containers/Tenant/Diagnostics/TopShards/TopShards.js +3 -3
- package/dist/containers/Tenant/ObjectGeneral/ObjectGeneral.tsx +1 -1
- package/dist/containers/Tenant/ObjectGeneralTabs/ObjectGeneralTabs.tsx +1 -1
- package/dist/containers/Tenant/ObjectSummary/ObjectSummary.tsx +1 -1
- package/dist/containers/Tenant/Preview/Preview.js +1 -1
- package/dist/containers/Tenant/QueryEditor/Issues/Issues.scss +2 -3
- package/dist/containers/Tenant/QueryEditor/Issues/Issues.tsx +1 -1
- package/dist/containers/Tenant/QueryEditor/QueriesHistory/QueriesHistory.tsx +1 -1
- package/dist/containers/Tenant/QueryEditor/QueryEditor.js +2 -2
- package/dist/containers/Tenant/QueryEditor/QueryExplain/QueryExplain.js +61 -42
- package/dist/containers/Tenant/QueryEditor/QueryExplain/QueryExplain.scss +8 -4
- package/dist/containers/Tenant/QueryEditor/QueryResult/QueryResult.js +1 -1
- package/dist/containers/Tenant/QueryEditor/SaveQuery/SaveQuery.js +1 -1
- package/dist/containers/Tenant/QueryEditor/SavedQueries/SavedQueries.js +1 -1
- package/dist/containers/Tenant/Schema/SchemaInfoViewer/SchemaInfoViewer.scss +2 -2
- package/dist/containers/Tenant/utils/paneVisibilityToggleHelpers.tsx +1 -1
- package/dist/containers/Tenants/Tenants.js +3 -2
- package/dist/containers/UserSettings/UserSettings.tsx +2 -2
- package/dist/containers/Vdisk/Vdisk.js +1 -1
- package/dist/containers/VdiskPdiskNode/VdiskPdiskNode.js +1 -1
- package/dist/index.js +1 -1
- package/dist/setupTests.js +1 -1
- package/dist/store/reducers/executeTopQueries.js +6 -9
- package/dist/store/reducers/explainQuery.js +27 -32
- package/dist/store/reducers/olapStats.js +6 -8
- package/dist/store/reducers/shardsWorkload.js +6 -8
- package/dist/types/api/query.ts +32 -21
- package/dist/types/assets.d.ts +1 -1
- package/dist/types/store/query.ts +1 -1
- package/dist/utils/createToast.tsx +1 -1
- package/dist/utils/getNodesColumns.js +1 -1
- package/dist/utils/i18n/i18n.ts +1 -1
- package/dist/utils/prepareQueryExplain.ts +0 -4
- package/dist/utils/query.test.ts +59 -1
- package/dist/utils/query.ts +38 -4
- package/package.json +9 -11
@@ -4,7 +4,7 @@ import {useLocation} from 'react-router';
|
|
4
4
|
import {Link} from 'react-router-dom';
|
5
5
|
import cn from 'bem-cn-lite';
|
6
6
|
|
7
|
-
import {Tabs} from '@
|
7
|
+
import {Tabs} from '@gravity-ui/uikit';
|
8
8
|
|
9
9
|
import routes, {createHref} from '../../../routes';
|
10
10
|
import {TENANT_INITIAL_TAB_KEY} from '../../../utils/constants';
|
@@ -6,7 +6,7 @@ import {useHistory, useLocation} from 'react-router';
|
|
6
6
|
import qs from 'qs';
|
7
7
|
import _ from 'lodash';
|
8
8
|
|
9
|
-
import {Button, HelpPopover, Loader, Tabs} from '@
|
9
|
+
import {Button, HelpPopover, Loader, Tabs} from '@gravity-ui/uikit';
|
10
10
|
|
11
11
|
import SplitPane from '../../../components/SplitPane';
|
12
12
|
import {SchemaTree} from '../Schema/SchemaTree/SchemaTree';
|
@@ -3,7 +3,7 @@ import PropTypes from 'prop-types';
|
|
3
3
|
import {connect} from 'react-redux';
|
4
4
|
import cn from 'bem-cn-lite';
|
5
5
|
|
6
|
-
import {Loader, Button} from '@
|
6
|
+
import {Loader, Button} from '@gravity-ui/uikit';
|
7
7
|
|
8
8
|
import Icon from '../../../components/Icon/Icon';
|
9
9
|
import Fullscreen from '../../../components/Fullscreen/Fullscreen';
|
@@ -1,8 +1,6 @@
|
|
1
1
|
.kv-result-issues {
|
2
|
-
overflow: auto;
|
3
|
-
|
4
|
-
height: 100%;
|
5
2
|
padding: 0 10px;
|
3
|
+
|
6
4
|
&__error-message {
|
7
5
|
position: sticky;
|
8
6
|
z-index: 2;
|
@@ -16,6 +14,7 @@
|
|
16
14
|
|
17
15
|
background-color: var(--yc-color-base-background);
|
18
16
|
}
|
17
|
+
|
19
18
|
&__error-message-text {
|
20
19
|
margin: 0 10px;
|
21
20
|
}
|
@@ -1,7 +1,7 @@
|
|
1
1
|
import * as React from 'react';
|
2
2
|
import cn from 'bem-cn-lite';
|
3
3
|
|
4
|
-
import {Button, Icon, ArrowToggle} from '@
|
4
|
+
import {Button, Icon, ArrowToggle} from '@gravity-ui/uikit';
|
5
5
|
import ShortyString from '../../../../components/ShortyString/ShortyString';
|
6
6
|
|
7
7
|
import {IssueType, SEVERITY, getSeverity} from './models';
|
@@ -1,7 +1,7 @@
|
|
1
1
|
import React, {useRef, useState} from 'react';
|
2
2
|
import cn from 'bem-cn-lite';
|
3
3
|
import _ from 'lodash';
|
4
|
-
import {Button, Popup} from '@
|
4
|
+
import {Button, Popup} from '@gravity-ui/uikit';
|
5
5
|
import TruncatedQuery from '../../../../components/TruncatedQuery/TruncatedQuery';
|
6
6
|
import {useSelector} from 'react-redux';
|
7
7
|
|
@@ -4,7 +4,7 @@ import {connect} from 'react-redux';
|
|
4
4
|
import cn from 'bem-cn-lite';
|
5
5
|
import _ from 'lodash';
|
6
6
|
import MonacoEditor from 'react-monaco-editor';
|
7
|
-
import {Button, DropdownMenu} from '@
|
7
|
+
import {Button, DropdownMenu} from '@gravity-ui/uikit';
|
8
8
|
|
9
9
|
import SplitPane from '../../../components/SplitPane';
|
10
10
|
import {QueryResultTable} from '../../../components/QueryResultTable';
|
@@ -341,7 +341,7 @@ function QueryEditor(props) {
|
|
341
341
|
error={error}
|
342
342
|
explain={data}
|
343
343
|
astQuery={handleAstQuery}
|
344
|
-
ast={dataAst
|
344
|
+
ast={dataAst}
|
345
345
|
loading={loading}
|
346
346
|
loadingAst={loadingAst}
|
347
347
|
theme={theme}
|
@@ -1,7 +1,7 @@
|
|
1
|
-
import React, {useEffect, useState} from 'react';
|
1
|
+
import React, {useEffect, useRef, useState} from 'react';
|
2
2
|
import cn from 'bem-cn-lite';
|
3
3
|
import MonacoEditor from 'react-monaco-editor';
|
4
|
-
import {Loader, RadioButton} from '@
|
4
|
+
import {Loader, RadioButton} from '@gravity-ui/uikit';
|
5
5
|
import JSONTree from 'react-json-inspector';
|
6
6
|
import {LANGUAGE_S_EXPRESSION_ID} from '../../../../utils/monaco';
|
7
7
|
import {
|
@@ -62,18 +62,25 @@ const explainOptions = [
|
|
62
62
|
];
|
63
63
|
|
64
64
|
function GraphRoot(props) {
|
65
|
-
|
66
|
-
|
65
|
+
const paranoid = useRef();
|
66
|
+
|
67
|
+
const render = () => {
|
67
68
|
const {data, opts, shapes, version} = props;
|
69
|
+
|
68
70
|
if (version === explainVersions.v2) {
|
69
|
-
paranoid = getTopology('graphRoot',
|
70
|
-
paranoid.render();
|
71
|
+
paranoid.current = getTopology('graphRoot', data, opts, shapes);
|
72
|
+
paranoid.current.render();
|
71
73
|
} else if (version === explainVersions.v1) {
|
72
|
-
paranoid = getCompactTopology('graphRoot', data, opts);
|
73
|
-
paranoid.renderCompactTopology();
|
74
|
+
paranoid.current = getCompactTopology('graphRoot', data, opts);
|
75
|
+
paranoid.current.renderCompactTopology();
|
74
76
|
}
|
77
|
+
}
|
78
|
+
|
79
|
+
useEffect(() => {
|
80
|
+
render();
|
81
|
+
|
75
82
|
return () => {
|
76
|
-
paranoid = undefined;
|
83
|
+
paranoid.current = undefined;
|
77
84
|
};
|
78
85
|
}, []);
|
79
86
|
|
@@ -86,13 +93,11 @@ function GraphRoot(props) {
|
|
86
93
|
|
87
94
|
graphRoot.innerHTML = '';
|
88
95
|
|
89
|
-
|
90
|
-
paranoid = getCompactTopology('graphRoot', data, opts);
|
91
|
-
paranoid.renderCompactTopology();
|
96
|
+
render();
|
92
97
|
}, [props.opts.colors]);
|
93
98
|
|
94
99
|
useEffect(() => {
|
95
|
-
paranoid?.updateData(props.data);
|
100
|
+
paranoid.current?.updateData?.(props.data);
|
96
101
|
}, [props.data]);
|
97
102
|
|
98
103
|
return <div id="graphRoot" style={{height: '100vh'}} />;
|
@@ -110,9 +115,6 @@ function QueryExplain(props) {
|
|
110
115
|
};
|
111
116
|
}, []);
|
112
117
|
|
113
|
-
const {explain = {}, theme} = props;
|
114
|
-
const {links, nodes, version, graphDepth} = explain;
|
115
|
-
|
116
118
|
useEffect(() => {
|
117
119
|
if (!props.ast && activeOption === ExplainOptionIds.ast) {
|
118
120
|
props.astQuery();
|
@@ -132,20 +134,23 @@ function QueryExplain(props) {
|
|
132
134
|
};
|
133
135
|
|
134
136
|
const renderStub = () => {
|
137
|
+
return (
|
138
|
+
<div className={b('text-message')}>
|
139
|
+
There is no explanation for the request
|
140
|
+
</div>
|
141
|
+
);
|
142
|
+
};
|
143
|
+
|
144
|
+
const renderTextExplain = () => {
|
135
145
|
const {explain} = props;
|
146
|
+
|
136
147
|
if (!explain) {
|
137
|
-
return
|
138
|
-
}
|
139
|
-
if (!explain.nodes.length) {
|
140
|
-
return 'There is no explanation for the request';
|
148
|
+
return renderStub();
|
141
149
|
}
|
142
|
-
return null;
|
143
|
-
};
|
144
150
|
|
145
|
-
const renderTextExplain = () => {
|
146
151
|
const content = (
|
147
152
|
<JSONTree
|
148
|
-
data={
|
153
|
+
data={explain.pristine}
|
149
154
|
isExpanded={() => true}
|
150
155
|
className={b('inspector')}
|
151
156
|
searchOptions={{
|
@@ -153,22 +158,22 @@ function QueryExplain(props) {
|
|
153
158
|
}}
|
154
159
|
/>
|
155
160
|
);
|
161
|
+
|
156
162
|
return (
|
157
163
|
<React.Fragment>
|
158
164
|
{content}
|
159
|
-
{
|
160
|
-
{isFullscreen && (
|
161
|
-
<Fullscreen>
|
162
|
-
<div className={b('inspector', {fullscreen: true})}>{content}</div>
|
163
|
-
</Fullscreen>
|
164
|
-
)}
|
165
|
+
{isFullscreen && <Fullscreen>{content}</Fullscreen>}
|
165
166
|
</React.Fragment>
|
166
167
|
);
|
167
168
|
};
|
168
169
|
|
169
170
|
const renderAstExplain = () => {
|
170
171
|
if (!props.ast) {
|
171
|
-
return
|
172
|
+
return (
|
173
|
+
<div className={b('text-message')}>
|
174
|
+
There is no AST explanation for the request
|
175
|
+
</div>
|
176
|
+
);
|
172
177
|
}
|
173
178
|
const content = (
|
174
179
|
<div className={b('ast')}>
|
@@ -189,7 +194,12 @@ function QueryExplain(props) {
|
|
189
194
|
};
|
190
195
|
|
191
196
|
const renderGraph = () => {
|
192
|
-
const
|
197
|
+
const {explain = {}, theme} = props;
|
198
|
+
const {links, nodes, version} = explain;
|
199
|
+
|
200
|
+
if (!explain.nodes?.length) {
|
201
|
+
return renderStub();
|
202
|
+
}
|
193
203
|
|
194
204
|
const content =
|
195
205
|
links && nodes && nodes.length ? (
|
@@ -197,11 +207,6 @@ function QueryExplain(props) {
|
|
197
207
|
className={b('explain-canvas-container', {
|
198
208
|
hidden: activeOption !== ExplainOptionIds.schema,
|
199
209
|
})}
|
200
|
-
style={{
|
201
|
-
height: isFullscreen ? '100%' : graphHeight,
|
202
|
-
minHeight: graphHeight,
|
203
|
-
width: '100%',
|
204
|
-
}}
|
205
210
|
>
|
206
211
|
<GraphRoot
|
207
212
|
version={version}
|
@@ -222,11 +227,28 @@ function QueryExplain(props) {
|
|
222
227
|
<React.Fragment>
|
223
228
|
{!isFullscreen && content}
|
224
229
|
{isFullscreen && <Fullscreen>{content}</Fullscreen>}
|
225
|
-
{renderStub()}
|
226
230
|
</React.Fragment>
|
227
231
|
);
|
228
232
|
};
|
229
233
|
|
234
|
+
const renderError = () => {
|
235
|
+
const {error} = props;
|
236
|
+
|
237
|
+
let message;
|
238
|
+
|
239
|
+
if (error.data) {
|
240
|
+
message = typeof error.data === 'string' ? error.data : error.data.error?.message;
|
241
|
+
} else {
|
242
|
+
message = error;
|
243
|
+
}
|
244
|
+
|
245
|
+
return (
|
246
|
+
<div className={b('text-message')}>
|
247
|
+
{message}
|
248
|
+
</div>
|
249
|
+
);
|
250
|
+
};
|
251
|
+
|
230
252
|
const renderContent = () => {
|
231
253
|
const {error, loading, loadingAst} = props;
|
232
254
|
if (loading || loadingAst) {
|
@@ -234,10 +256,7 @@ function QueryExplain(props) {
|
|
234
256
|
}
|
235
257
|
|
236
258
|
if (error) {
|
237
|
-
|
238
|
-
return typeof error.data === 'string' ? error.data : error.data.error?.message;
|
239
|
-
}
|
240
|
-
return error;
|
259
|
+
return renderError();
|
241
260
|
}
|
242
261
|
|
243
262
|
switch (activeOption) {
|
@@ -6,8 +6,9 @@
|
|
6
6
|
overflow: auto;
|
7
7
|
flex-grow: 1;
|
8
8
|
flex-direction: column;
|
9
|
-
|
10
|
-
|
9
|
+
}
|
10
|
+
&__text-message {
|
11
|
+
padding: 15px 20px;
|
11
12
|
}
|
12
13
|
&__controls {
|
13
14
|
position: sticky;
|
@@ -40,7 +41,8 @@
|
|
40
41
|
&__explain-canvas-container {
|
41
42
|
overflow-y: auto;
|
42
43
|
|
43
|
-
|
44
|
+
width: 100%;
|
45
|
+
height: 100%;
|
44
46
|
&_hidden {
|
45
47
|
display: none;
|
46
48
|
}
|
@@ -49,6 +51,7 @@
|
|
49
51
|
overflow-y: auto;
|
50
52
|
|
51
53
|
width: 100%;
|
54
|
+
padding: 15px 20px;
|
52
55
|
@include json-tree-styles();
|
53
56
|
&
|
54
57
|
.json-inspector__leaf.json-inspector__leaf_root.json-inspector__leaf_expanded.json-inspector__leaf_composite {
|
@@ -61,9 +64,10 @@
|
|
61
64
|
}
|
62
65
|
|
63
66
|
&__ast {
|
67
|
+
overflow: hidden;
|
68
|
+
|
64
69
|
width: 100%;
|
65
70
|
height: 100%;
|
66
|
-
margin: 20px 0;
|
67
71
|
|
68
72
|
white-space: pre-wrap;
|
69
73
|
}
|
@@ -1,7 +1,7 @@
|
|
1
1
|
import React, {useEffect, useState} from 'react';
|
2
2
|
import {useDispatch, useSelector} from 'react-redux';
|
3
3
|
import cn from 'bem-cn-lite';
|
4
|
-
import {RadioButton} from '@
|
4
|
+
import {RadioButton} from '@gravity-ui/uikit';
|
5
5
|
import JSONTree from 'react-json-inspector';
|
6
6
|
|
7
7
|
import CopyToClipboard from '../../../../components/CopyToClipboard/CopyToClipboard';
|
@@ -2,7 +2,7 @@ import React, {useState} from 'react';
|
|
2
2
|
import _ from 'lodash';
|
3
3
|
import cn from 'bem-cn-lite';
|
4
4
|
import {useDispatch, useSelector} from 'react-redux';
|
5
|
-
import {Dialog, DropdownMenu, TextInput, Button} from '@
|
5
|
+
import {Dialog, DropdownMenu, TextInput, Button} from '@gravity-ui/uikit';
|
6
6
|
|
7
7
|
import {setQueryNameToEdit} from '../../../../store/reducers/saveQuery';
|
8
8
|
|
@@ -1,7 +1,7 @@
|
|
1
1
|
import React, {useRef, useState} from 'react';
|
2
2
|
import _ from 'lodash';
|
3
3
|
import cn from 'bem-cn-lite';
|
4
|
-
import {Dialog, Popup, Button} from '@
|
4
|
+
import {Dialog, Popup, Button} from '@gravity-ui/uikit';
|
5
5
|
|
6
6
|
import TruncatedQuery from '../../../../components/TruncatedQuery/TruncatedQuery';
|
7
7
|
import Icon from '../../../../components/Icon/Icon';
|
@@ -5,12 +5,13 @@ import {connect} from 'react-redux';
|
|
5
5
|
import _ from 'lodash';
|
6
6
|
|
7
7
|
import DataTable from '@yandex-cloud/react-data-table';
|
8
|
-
import {Loader, TextInput, Button} from '@
|
8
|
+
import {Loader, TextInput, Button} from '@gravity-ui/uikit';
|
9
9
|
|
10
10
|
import EntityStatus from '../../components/EntityStatus/EntityStatus';
|
11
11
|
import PoolsGraph from '../../components/PoolsGraph/PoolsGraph';
|
12
12
|
import TabletsStatistic from '../../components/TabletsStatistic/TabletsStatistic';
|
13
13
|
import ProblemFilter, {problemFilterType} from '../../components/ProblemFilter/ProblemFilter';
|
14
|
+
import {Illustration} from '../../components/Illustration';
|
14
15
|
import {AutoFetcher} from '../../utils/autofetcher';
|
15
16
|
|
16
17
|
import routes, {CLUSTER_PAGES, createHref} from '../../routes';
|
@@ -309,7 +310,7 @@ class Tenants extends React.Component {
|
|
309
310
|
];
|
310
311
|
|
311
312
|
if (filteredTenants.length === 0 && filter !== ALL) {
|
312
|
-
return <
|
313
|
+
return <Illustration name="thumbsUp" width="200" />;
|
313
314
|
}
|
314
315
|
|
315
316
|
return (
|
@@ -1,6 +1,6 @@
|
|
1
1
|
import {connect} from 'react-redux';
|
2
2
|
|
3
|
-
import {RadioButton, Switch} from '@
|
3
|
+
import {RadioButton, Switch} from '@gravity-ui/uikit';
|
4
4
|
import {Settings} from '../../components/AsideNavigation/Settings';
|
5
5
|
import favoriteFilledIcon from '../../assets/icons/star.svg';
|
6
6
|
import flaskIcon from '../../assets/icons/flask.svg';
|
@@ -21,7 +21,7 @@ function UserSettings(props: any) {
|
|
21
21
|
};
|
22
22
|
|
23
23
|
const _onInvertedDisksChangeHandler = (value: boolean) => {
|
24
|
-
props.setSettingValue(INVERTED_DISKS_KEY, value);
|
24
|
+
props.setSettingValue(INVERTED_DISKS_KEY, String(value));
|
25
25
|
};
|
26
26
|
|
27
27
|
return (
|
@@ -4,7 +4,7 @@ import cn from 'bem-cn-lite';
|
|
4
4
|
import {connect} from 'react-redux';
|
5
5
|
|
6
6
|
import {Link} from 'react-router-dom';
|
7
|
-
import {Loader} from '@
|
7
|
+
import {Loader} from '@gravity-ui/uikit';
|
8
8
|
|
9
9
|
import InfoViewer from '../../components/InfoViewer/InfoViewer';
|
10
10
|
import EntityStatus from '../../components/EntityStatus/EntityStatus';
|
@@ -5,7 +5,7 @@ import {connect} from 'react-redux';
|
|
5
5
|
import _ from 'lodash';
|
6
6
|
|
7
7
|
import {Link} from 'react-router-dom';
|
8
|
-
import {Loader} from '@
|
8
|
+
import {Loader} from '@gravity-ui/uikit';
|
9
9
|
|
10
10
|
import InfoViewer from '../../components/InfoViewer/InfoViewer';
|
11
11
|
import EntityStatus from '../../components/EntityStatus/EntityStatus';
|
package/dist/index.js
CHANGED
@@ -5,7 +5,7 @@ import App from './containers/App/App';
|
|
5
5
|
import {Provider} from 'react-redux';
|
6
6
|
import configureStore from './store';
|
7
7
|
import reportWebVitals from './reportWebVitals';
|
8
|
-
import '@
|
8
|
+
import '@gravity-ui/uikit/styles/styles.scss';
|
9
9
|
import HistoryContext from './contexts/HistoryContext';
|
10
10
|
|
11
11
|
const {store, history} = configureStore();
|
package/dist/setupTests.js
CHANGED
@@ -4,7 +4,7 @@
|
|
4
4
|
// learn more: https://github.com/testing-library/jest-dom
|
5
5
|
import '@testing-library/jest-dom';
|
6
6
|
|
7
|
-
import {configure as configureUiKit} from '@
|
7
|
+
import {configure as configureUiKit} from '@gravity-ui/uikit';
|
8
8
|
import {configure as configureYdbUiComponents} from 'ydb-ui-components';
|
9
9
|
import {i18n, Lang} from '../src/utils/i18n';
|
10
10
|
|
@@ -1,6 +1,9 @@
|
|
1
|
-
import {createRequestActionTypes, createApiRequest} from '../utils';
|
2
1
|
import '../../services/api';
|
3
2
|
|
3
|
+
import {parseQueryAPIExecuteResponse} from '../../utils/query';
|
4
|
+
|
5
|
+
import {createRequestActionTypes, createApiRequest} from '../utils';
|
6
|
+
|
4
7
|
const SEND_QUERY = createRequestActionTypes('top-queries', 'SEND_QUERY');
|
5
8
|
const SET_QUERY_OPTIONS = createRequestActionTypes('top-queries', 'SET_QUERY_OPTIONS');
|
6
9
|
|
@@ -47,15 +50,9 @@ const executeTopQueries = (state = initialState, action) => {
|
|
47
50
|
|
48
51
|
export const sendQuery = ({query, database, action}) => {
|
49
52
|
return createApiRequest({
|
50
|
-
request: window.api.sendQuery({query, database, action}),
|
53
|
+
request: window.api.sendQuery({schema: 'modern', query, database, action}),
|
51
54
|
actions: SEND_QUERY,
|
52
|
-
dataHandler:
|
53
|
-
if (result && typeof result === 'string') {
|
54
|
-
throw 'Unexpected token in JSON.';
|
55
|
-
}
|
56
|
-
|
57
|
-
return result;
|
58
|
-
},
|
55
|
+
dataHandler: parseQueryAPIExecuteResponse,
|
59
56
|
});
|
60
57
|
};
|
61
58
|
|
@@ -1,8 +1,12 @@
|
|
1
|
-
import {createRequestActionTypes, createApiRequest} from '../utils';
|
2
|
-
import '../../services/api';
|
3
1
|
import _ from 'lodash';
|
2
|
+
|
3
|
+
import '../../services/api';
|
4
|
+
|
4
5
|
import {getExplainNodeId, getMetaForExplainNode} from '../../utils';
|
5
6
|
import {preparePlan} from '../../utils/prepareQueryExplain';
|
7
|
+
import {parseQueryAPIExplainResponse} from '../../utils/query';
|
8
|
+
|
9
|
+
import {createRequestActionTypes, createApiRequest} from '../utils';
|
6
10
|
|
7
11
|
const GET_EXPLAIN_QUERY = createRequestActionTypes('query', 'GET_EXPLAIN_QUERY');
|
8
12
|
const GET_EXPLAIN_QUERY_AST = createRequestActionTypes('query', 'GET_EXPLAIN_QUERY_AST');
|
@@ -26,7 +30,8 @@ const explainQuery = (state = initialState, action) => {
|
|
26
30
|
case GET_EXPLAIN_QUERY.SUCCESS: {
|
27
31
|
return {
|
28
32
|
...state,
|
29
|
-
data: action.data,
|
33
|
+
data: action.data.plan,
|
34
|
+
dataAst: action.data.ast,
|
30
35
|
loading: false,
|
31
36
|
error: undefined,
|
32
37
|
};
|
@@ -50,7 +55,7 @@ const explainQuery = (state = initialState, action) => {
|
|
50
55
|
case GET_EXPLAIN_QUERY_AST.SUCCESS: {
|
51
56
|
return {
|
52
57
|
...state,
|
53
|
-
dataAst: action.data,
|
58
|
+
dataAst: action.data.ast,
|
54
59
|
loadingAst: false,
|
55
60
|
error: undefined,
|
56
61
|
};
|
@@ -72,6 +77,7 @@ export const getExplainQueryAst = ({query, database}) => {
|
|
72
77
|
return createApiRequest({
|
73
78
|
request: window.api.getExplainQueryAst(query, database),
|
74
79
|
actions: GET_EXPLAIN_QUERY_AST,
|
80
|
+
dataHandler: parseQueryAPIExplainResponse,
|
75
81
|
});
|
76
82
|
};
|
77
83
|
|
@@ -86,10 +92,15 @@ export const getExplainQuery = ({query, database}) => {
|
|
86
92
|
return createApiRequest({
|
87
93
|
request: window.api.getExplainQuery(query, database),
|
88
94
|
actions: GET_EXPLAIN_QUERY,
|
89
|
-
dataHandler: (
|
95
|
+
dataHandler: (response) => {
|
96
|
+
const {plan: result, ast} = parseQueryAPIExplainResponse(response);
|
97
|
+
|
98
|
+
if (!result) {
|
99
|
+
return {ast};
|
100
|
+
}
|
101
|
+
|
90
102
|
let links = [];
|
91
103
|
let nodes = [];
|
92
|
-
let graphDepth;
|
93
104
|
const {tables, meta, Plan} = result;
|
94
105
|
|
95
106
|
if (supportedExplainQueryVersions.indexOf(meta.version) === -1) {
|
@@ -102,9 +113,7 @@ export const getExplainQuery = ({query, database}) => {
|
|
102
113
|
const preparedPlan = preparePlan(Plan);
|
103
114
|
links = preparedPlan.links;
|
104
115
|
nodes = preparedPlan.nodes;
|
105
|
-
graphDepth = preparedPlan.graphDepth;
|
106
116
|
} else {
|
107
|
-
graphDepth = tables.length;
|
108
117
|
_.forEach(tables, (table) => {
|
109
118
|
nodes.push({
|
110
119
|
name: table.name,
|
@@ -113,7 +122,7 @@ export const getExplainQuery = ({query, database}) => {
|
|
113
122
|
const tableTypes = {};
|
114
123
|
|
115
124
|
const {reads = [], writes = []} = table;
|
116
|
-
let
|
125
|
+
let prevNodeId = table.name;
|
117
126
|
|
118
127
|
_.forEach([...reads, ...writes], (node) => {
|
119
128
|
if (tableTypes[node.type]) {
|
@@ -128,22 +137,6 @@ export const getExplainQuery = ({query, database}) => {
|
|
128
137
|
tableTypes[node.type],
|
129
138
|
);
|
130
139
|
|
131
|
-
let prevNodeId = table.name;
|
132
|
-
if (prevEl) {
|
133
|
-
prevNodeId =
|
134
|
-
node.type === prevEl.type
|
135
|
-
? getExplainNodeId(
|
136
|
-
table.name,
|
137
|
-
prevEl.type,
|
138
|
-
tableTypes[prevEl.type] - 1,
|
139
|
-
)
|
140
|
-
: getExplainNodeId(
|
141
|
-
table.name,
|
142
|
-
prevEl.type,
|
143
|
-
tableTypes[prevEl.type],
|
144
|
-
);
|
145
|
-
}
|
146
|
-
|
147
140
|
links.push({
|
148
141
|
from: prevNodeId,
|
149
142
|
to: nodeId,
|
@@ -154,18 +147,20 @@ export const getExplainQuery = ({query, database}) => {
|
|
154
147
|
id: nodeId,
|
155
148
|
});
|
156
149
|
|
157
|
-
|
150
|
+
prevNodeId = nodeId;
|
158
151
|
});
|
159
152
|
});
|
160
153
|
}
|
161
154
|
|
162
155
|
return {
|
163
|
-
|
164
|
-
|
165
|
-
|
166
|
-
|
167
|
-
|
168
|
-
|
156
|
+
plan: {
|
157
|
+
links,
|
158
|
+
nodes,
|
159
|
+
tables,
|
160
|
+
version: meta.version,
|
161
|
+
pristine: result,
|
162
|
+
},
|
163
|
+
ast,
|
169
164
|
};
|
170
165
|
},
|
171
166
|
});
|
@@ -1,6 +1,9 @@
|
|
1
|
-
import {createRequestActionTypes, createApiRequest} from '../utils';
|
2
1
|
import '../../services/api';
|
3
2
|
|
3
|
+
import {parseQueryAPIExecuteResponse} from '../../utils/query';
|
4
|
+
|
5
|
+
import {createRequestActionTypes, createApiRequest} from '../utils';
|
6
|
+
|
4
7
|
const FETCH_OLAP_STATS = createRequestActionTypes('query', 'SEND_OLAP_STATS_QUERY');
|
5
8
|
const SET_OLAP_STATS_OPTIONS = createRequestActionTypes('query', 'SET_OLAP_STATS_OPTIONS');
|
6
9
|
|
@@ -53,18 +56,13 @@ const olapStats = (state = initialState, action) => {
|
|
53
56
|
export const getOlapStats = ({path = ''}) => {
|
54
57
|
return createApiRequest({
|
55
58
|
request: window.api.sendQuery({
|
59
|
+
schema: 'modern',
|
56
60
|
query: createOlatStatsQuery(path),
|
57
61
|
database: path,
|
58
62
|
action: queryAction,
|
59
63
|
}),
|
60
64
|
actions: FETCH_OLAP_STATS,
|
61
|
-
dataHandler:
|
62
|
-
if (result && typeof result === 'string') {
|
63
|
-
throw 'Unexpected token in JSON.';
|
64
|
-
}
|
65
|
-
|
66
|
-
return result;
|
67
|
-
},
|
65
|
+
dataHandler: parseQueryAPIExecuteResponse,
|
68
66
|
});
|
69
67
|
};
|
70
68
|
|