ydb-embedded-ui 1.13.2 → 1.14.1

Sign up to get free protection for your applications and to get access to all the features.
Files changed (60) hide show
  1. package/CHANGELOG.md +40 -0
  2. package/dist/assets/icons/flask.svg +3 -0
  3. package/dist/components/InfoViewer/formatters/common.ts +15 -0
  4. package/dist/components/InfoViewer/formatters/index.ts +2 -0
  5. package/dist/components/InfoViewer/formatters/schema.ts +43 -0
  6. package/dist/components/InfoViewer/schemaInfo/CDCStreamInfo.tsx +44 -0
  7. package/dist/components/InfoViewer/schemaInfo/PersQueueGroupInfo.tsx +34 -0
  8. package/dist/components/{IndexInfoViewer/IndexInfoViewer.tsx → InfoViewer/schemaInfo/TableIndexInfo.tsx} +7 -18
  9. package/dist/components/InfoViewer/schemaInfo/index.ts +3 -0
  10. package/dist/components/InfoViewer/schemaOverview/CDCStreamOverview.tsx +44 -0
  11. package/dist/components/InfoViewer/schemaOverview/PersQueueGroupOverview.tsx +35 -0
  12. package/dist/components/InfoViewer/schemaOverview/index.ts +2 -0
  13. package/dist/components/QueryResultTable/Cell/Cell.tsx +33 -0
  14. package/dist/components/QueryResultTable/Cell/index.ts +1 -0
  15. package/dist/components/QueryResultTable/QueryResultTable.scss +11 -0
  16. package/dist/components/QueryResultTable/QueryResultTable.tsx +115 -0
  17. package/dist/components/QueryResultTable/i18n/en.json +3 -0
  18. package/dist/components/QueryResultTable/i18n/index.ts +11 -0
  19. package/dist/components/QueryResultTable/i18n/ru.json +3 -0
  20. package/dist/components/QueryResultTable/index.ts +1 -0
  21. package/dist/containers/App/App.scss +1 -0
  22. package/dist/containers/Storage/DiskStateProgressBar/DiskStateProgressBar.scss +39 -14
  23. package/dist/containers/Storage/DiskStateProgressBar/DiskStateProgressBar.tsx +18 -7
  24. package/dist/containers/Storage/Pdisk/__tests__/colors.tsx +4 -3
  25. package/dist/containers/Storage/Vdisk/__tests__/colors.tsx +7 -7
  26. package/dist/containers/Tenant/Diagnostics/DiagnosticsPages.ts +6 -2
  27. package/dist/containers/Tenant/Diagnostics/HotKeys/HotKeys.js +1 -1
  28. package/dist/containers/Tenant/Diagnostics/Overview/Overview.tsx +8 -3
  29. package/dist/containers/Tenant/Diagnostics/TopQueries/TopQueries.js +1 -1
  30. package/dist/containers/Tenant/Diagnostics/TopShards/TopShards.js +1 -1
  31. package/dist/containers/Tenant/ObjectSummary/ObjectSummary.tsx +36 -10
  32. package/dist/containers/Tenant/Preview/Preview.js +15 -57
  33. package/dist/containers/Tenant/Preview/Preview.scss +4 -8
  34. package/dist/containers/Tenant/QueryEditor/QueryEditor.js +12 -41
  35. package/dist/containers/Tenant/QueryEditor/QueryEditor.scss +0 -4
  36. package/dist/containers/Tenant/QueryEditor/QueryExplain/QueryExplain.scss +1 -2
  37. package/dist/containers/Tenant/QueryEditor/QueryResult/QueryResult.scss +2 -2
  38. package/dist/containers/Tenant/Schema/SchemaTree/SchemaTree.tsx +1 -1
  39. package/dist/containers/Tenant/utils/schema.ts +3 -0
  40. package/dist/containers/Tenant/utils/schemaActions.ts +1 -2
  41. package/dist/containers/Tenants/Tenants.js +12 -2
  42. package/dist/containers/UserSettings/UserSettings.tsx +26 -3
  43. package/dist/services/api.d.ts +19 -2
  44. package/dist/services/api.js +2 -2
  45. package/dist/setupTests.js +4 -0
  46. package/dist/store/reducers/executeQuery.js +4 -9
  47. package/dist/store/reducers/{preview.js → preview.ts} +22 -18
  48. package/dist/store/reducers/settings.js +3 -1
  49. package/dist/store/utils.ts +88 -0
  50. package/dist/types/api/query.ts +147 -0
  51. package/dist/types/api/schema.ts +235 -2
  52. package/dist/types/index.ts +33 -0
  53. package/dist/types/store/query.ts +9 -0
  54. package/dist/utils/{constants.js → constants.ts} +11 -6
  55. package/dist/utils/index.js +0 -24
  56. package/dist/utils/query.test.ts +189 -0
  57. package/dist/utils/query.ts +156 -0
  58. package/dist/utils/tests/providers.tsx +29 -0
  59. package/package.json +2 -2
  60. package/dist/store/utils.js +0 -51
@@ -20,9 +20,8 @@ const alterTableTemplate = (path: string) => {
20
20
  ADD COLUMN is_deleted Bool;`;
21
21
  };
22
22
  const selectQueryTemplate = (path: string) => {
23
- return `SELECT \`id\`, \`name\`
23
+ return `SELECT *
24
24
  FROM \`${path}\`
25
- ORDER BY \`id\`
26
25
  LIMIT 10;`;
27
26
  };
28
27
  const upsertQueryTemplate = (path: string) => {
@@ -14,7 +14,7 @@ import ProblemFilter, {problemFilterType} from '../../components/ProblemFilter/P
14
14
  import {AutoFetcher} from '../../utils/autofetcher';
15
15
 
16
16
  import routes, {CLUSTER_PAGES, createHref} from '../../routes';
17
- import {formatCPU, formatBytesToGigabyte} from '../../utils';
17
+ import {formatCPU, formatBytesToGigabyte, formatNumber} from '../../utils';
18
18
  import {hideTooltip, showTooltip} from '../../store/reducers/tooltip';
19
19
  import {withSearch} from '../../HOCS';
20
20
  import {ALL, DEFAULT_TABLE_SETTINGS, TENANT_INITIAL_TAB_KEY} from '../../utils/constants';
@@ -257,13 +257,23 @@ class Tenants extends React.Component {
257
257
  align: DataTable.RIGHT,
258
258
  defaultOrder: DataTable.DESCENDING,
259
259
  },
260
+ {
261
+ name: 'NodeIds',
262
+ header: 'Nodes',
263
+ width: 100,
264
+ accessor: ({NodeIds}) => NodeIds?.length || 0,
265
+ sortAccessor: ({NodeIds}) => NodeIds?.length || 0,
266
+ render: ({value}) => formatNumber(value),
267
+ align: DataTable.RIGHT,
268
+ defaultOrder: DataTable.DESCENDING,
269
+ },
260
270
  {
261
271
  name: 'StorageGroups',
262
272
  header: 'Groups',
263
273
  width: 100,
264
274
  sortAccessor: ({StorageGroups}) =>
265
275
  isNaN(Number(StorageGroups)) ? 0 : Number(StorageGroups),
266
- render: ({value}) => value ?? '—',
276
+ render: ({value}) => formatNumber(value) || '—',
267
277
  align: DataTable.RIGHT,
268
278
  defaultOrder: DataTable.DESCENDING,
269
279
  },
@@ -1,10 +1,11 @@
1
1
  import {connect} from 'react-redux';
2
2
 
3
- import {RadioButton} from '@yandex-cloud/uikit';
3
+ import {RadioButton, Switch} from '@yandex-cloud/uikit';
4
4
  import {Settings} from '../../components/AsideNavigation/Settings';
5
5
  import favoriteFilledIcon from '../../assets/icons/star.svg';
6
+ import flaskIcon from '../../assets/icons/flask.svg';
6
7
  //@ts-ignore
7
- import {THEME_KEY} from '../../utils/constants';
8
+ import {INVERTED_DISKS_KEY, THEME_KEY} from '../../utils/constants';
8
9
  //@ts-ignore
9
10
  import {setSettingValue} from '../../store/reducers/settings';
10
11
 
@@ -19,6 +20,10 @@ function UserSettings(props: any) {
19
20
  props.setSettingValue(THEME_KEY, value);
20
21
  };
21
22
 
23
+ const _onInvertedDisksChangeHandler = (value: boolean) => {
24
+ props.setSettingValue(INVERTED_DISKS_KEY, value);
25
+ };
26
+
22
27
  return (
23
28
  <Settings>
24
29
  <Settings.Page
@@ -36,15 +41,33 @@ function UserSettings(props: any) {
36
41
  </Settings.Item>
37
42
  </Settings.Section>
38
43
  </Settings.Page>
44
+ <Settings.Page
45
+ id="experiments"
46
+ title="Experiments"
47
+ icon={{data: flaskIcon}}
48
+ >
49
+ <Settings.Section title="Experiments">
50
+ <Settings.Item title="Inverted disks space indicators">
51
+ <Switch
52
+ checked={props.invertedDisks}
53
+ onUpdate={_onInvertedDisksChangeHandler}
54
+ />
55
+ </Settings.Item>
56
+ </Settings.Section>
57
+ </Settings.Page>
39
58
  </Settings>
40
59
  );
41
60
  }
42
61
 
43
62
  const mapStateToProps = (state: any) => {
44
- const {theme} = state.settings.userSettings;
63
+ const {
64
+ theme,
65
+ invertedDisks,
66
+ } = state.settings.userSettings;
45
67
 
46
68
  return {
47
69
  theme,
70
+ invertedDisks,
48
71
  };
49
72
  };
50
73
 
@@ -1,8 +1,12 @@
1
+ type AxiosOptions = {
2
+ concurrentId?: string;
3
+ };
4
+
1
5
  interface Window {
2
6
  api: {
3
7
  getSchema: (
4
8
  params: {path: string},
5
- axiosOptions?: {concurrentId?: string},
9
+ axiosOptions?: AxiosOptions,
6
10
  ) => Promise<import('../types/api/schema').TEvDescribeSchemeResult>;
7
11
  getStorageInfo: (
8
12
  params: {
@@ -11,8 +15,21 @@ interface Window {
11
15
  nodeId: string,
12
16
  type: 'Groups' | 'Nodes',
13
17
  },
14
- axiosOptions?: {concurrentId?: string},
18
+ axiosOptions?: AxiosOptions,
15
19
  ) => Promise<import('../types/api/storage').TStorageInfo>;
20
+ sendQuery: <
21
+ Action extends import('../types/api/query').Actions,
22
+ Schema extends import('../types/api/query').Schemas = undefined
23
+ >(
24
+ params: {
25
+ query?: string,
26
+ database?: string,
27
+ action?: Action,
28
+ stats?: string,
29
+ schema?: Schema,
30
+ },
31
+ axiosOptions?: AxiosOptions,
32
+ ) => Promise<import('../types/api/query').QueryAPIResponse<Action, Schema>>;
16
33
  [method: string]: Function;
17
34
  };
18
35
  }
@@ -146,9 +146,9 @@ export class YdbEmbeddedAPI extends AxiosWrapper {
146
146
  state: 0,
147
147
  });
148
148
  }
149
- sendQuery({query, database, action, stats}, {concurrentId} = {}) {
149
+ sendQuery({query, database, action, stats, schema}, {concurrentId} = {}) {
150
150
  return this.post(
151
- this.getPath('/viewer/json/query'),
151
+ this.getPath(`/viewer/json/query${schema ? `?schema=${schema}` : ''}`),
152
152
  {
153
153
  query,
154
154
  database,
@@ -11,3 +11,7 @@ import {i18n, Lang} from '../src/utils/i18n';
11
11
  i18n.setLang(Lang.En);
12
12
  configureYdbUiComponents({lang: Lang.En});
13
13
  configureUiKit({lang: Lang.En});
14
+
15
+ // only to prevent warnings from history lib
16
+ // all api calls in tests should be mocked
17
+ window.custom_backend = '/';
@@ -2,6 +2,7 @@ import {createRequestActionTypes, createApiRequest} from '../utils';
2
2
  import '../../services/api';
3
3
  import {getValueFromLS, parseJson} from '../../utils/utils';
4
4
  import {QUERIES_HISTORY_KEY, QUERY_INITIAL_RUN_ACTION_KEY} from '../../utils/constants';
5
+ import {parseQueryAPIExecuteResponse} from '../../utils/query';
5
6
  import {readSavedSettingsValue} from './settings';
6
7
 
7
8
  const MAXIMUM_QUERIES_IN_HISTORY = 20;
@@ -57,7 +58,7 @@ const executeQuery = (state = initialState, action) => {
57
58
  case SEND_QUERY.SUCCESS: {
58
59
  return {
59
60
  ...state,
60
- data: action.data.result ?? action.data,
61
+ data: action.data,
61
62
  stats: action.data.stats,
62
63
  loading: false,
63
64
  error: undefined,
@@ -142,15 +143,9 @@ const executeQuery = (state = initialState, action) => {
142
143
 
143
144
  export const sendQuery = ({query, database, action}) => {
144
145
  return createApiRequest({
145
- request: window.api.sendQuery({query, database, action, stats: 'profile'}),
146
+ request: window.api.sendQuery({schema: 'modern', query, database, action, stats: 'profile'}),
146
147
  actions: SEND_QUERY,
147
- dataHandler: (result) => {
148
- const resultData = result.result ?? result;
149
- if (resultData && typeof resultData === 'string') {
150
- throw 'Unexpected token in JSON.';
151
- }
152
- return result;
153
- },
148
+ dataHandler: parseQueryAPIExecuteResponse,
154
149
  });
155
150
  };
156
151
 
@@ -1,15 +1,23 @@
1
- import {createRequestActionTypes, createApiRequest} from '../utils';
2
1
  import '../../services/api';
3
2
 
3
+ import type {ErrorRepsonse, ExecuteActions} from '../../types/api/query';
4
+ import type {IQueryResult} from '../../types/store/query';
5
+ import {parseQueryAPIExecuteResponse} from '../../utils/query';
6
+
7
+ import {createRequestActionTypes, createApiRequest, ApiRequestAction} from '../utils';
8
+
4
9
  const SEND_QUERY = createRequestActionTypes('preview', 'SEND_QUERY');
5
- const SET_QUERY_OPTIONS = createRequestActionTypes('preview', 'SET_QUERY_OPTIONS');
10
+ const SET_QUERY_OPTIONS = 'preview/SET_QUERY_OPTIONS';
6
11
 
7
12
  const initialState = {
8
13
  loading: false,
9
14
  wasLoaded: false,
10
15
  };
11
16
 
12
- const preview = (state = initialState, action) => {
17
+ const preview = (
18
+ state = initialState,
19
+ action: ApiRequestAction<typeof SEND_QUERY, IQueryResult, ErrorRepsonse> | ReturnType<typeof setQueryOptions>,
20
+ ) => {
13
21
  switch (action.type) {
14
22
  case SEND_QUERY.REQUEST: {
15
23
  return {
@@ -45,29 +53,25 @@ const preview = (state = initialState, action) => {
45
53
  }
46
54
  };
47
55
 
48
- export const sendQuery = ({query, database, action}) => {
56
+ interface SendQueryParams {
57
+ query?: string;
58
+ database?: string;
59
+ action?: ExecuteActions;
60
+ };
61
+
62
+ export const sendQuery = ({query, database, action}: SendQueryParams) => {
49
63
  return createApiRequest({
50
- request: window.api.sendQuery({query, database, action}),
64
+ request: window.api.sendQuery({schema: 'modern', query, database, action}),
51
65
  actions: SEND_QUERY,
52
- dataHandler: (data) => {
53
- if (!Array.isArray(data)) {
54
- try {
55
- return JSON.parse(data);
56
- } catch (e) {
57
- return [];
58
- }
59
- }
60
-
61
- return data;
62
- },
66
+ dataHandler: parseQueryAPIExecuteResponse,
63
67
  });
64
68
  };
65
69
 
66
- export function setQueryOptions(options) {
70
+ export function setQueryOptions(options: any) {
67
71
  return {
68
72
  type: SET_QUERY_OPTIONS,
69
73
  data: options,
70
- };
74
+ } as const;
71
75
  }
72
76
 
73
77
  export default preview;
@@ -5,6 +5,7 @@ import {
5
5
  THEME_KEY,
6
6
  TENANT_INITIAL_TAB_KEY,
7
7
  QUERY_INITIAL_RUN_ACTION_KEY,
8
+ INVERTED_DISKS_KEY,
8
9
  } from '../../utils/constants';
9
10
  import '../../services/api';
10
11
  import {getValueFromLS} from '../../utils/utils';
@@ -28,7 +29,8 @@ export const initialState = {
28
29
  userSettings: {
29
30
  ...defaultUserSettings,
30
31
  ...userSettings,
31
- theme: readSavedSettingsValue(THEME_KEY, 'light'),
32
+ [THEME_KEY]: readSavedSettingsValue(THEME_KEY, 'light'),
33
+ [INVERTED_DISKS_KEY]: readSavedSettingsValue(INVERTED_DISKS_KEY) === 'true',
32
34
  [SAVED_QUERIES_KEY]: readSavedSettingsValue(SAVED_QUERIES_KEY, '[]'),
33
35
  [TENANT_INITIAL_TAB_KEY]: readSavedSettingsValue(TENANT_INITIAL_TAB_KEY),
34
36
  [QUERY_INITIAL_RUN_ACTION_KEY]: readSavedSettingsValue(QUERY_INITIAL_RUN_ACTION_KEY),
@@ -0,0 +1,88 @@
1
+ import type {Dispatch} from 'redux';
2
+ import {AxiosResponse} from 'axios';
3
+
4
+ import createToast from '../utils/createToast';
5
+
6
+ import {SET_UNAUTHENTICATED} from './reducers/authentication';
7
+
8
+ export const nop = (result: any) => result;
9
+
10
+ export function createRequestActionTypes<Prefix extends string, Type extends string>(prefix: Prefix, type: Type) {
11
+ return {
12
+ REQUEST: `${prefix}/${type}_REQUEST`,
13
+ SUCCESS: `${prefix}/${type}_SUCCESS`,
14
+ FAILURE: `${prefix}/${type}_FAILURE`,
15
+ } as const;
16
+ }
17
+
18
+ const isAxiosResponse = (response: any): response is AxiosResponse => response && 'status' in response;
19
+
20
+ type CreateApiRequestParams<Actions, Response, HandledResponse> = {
21
+ actions: Actions;
22
+ request: Promise<Response>;
23
+ dataHandler: (data: Response, getState?: () => any) => HandledResponse;
24
+ };
25
+
26
+ export function createApiRequest<
27
+ Actions extends ReturnType<typeof createRequestActionTypes>,
28
+ Response,
29
+ HandledResponse,
30
+ >({actions, request, dataHandler = nop}: CreateApiRequestParams<Actions, Response, HandledResponse>) {
31
+ const doRequest = async function (dispatch: Dispatch, getState: () => any) {
32
+ dispatch({
33
+ type: actions.REQUEST,
34
+ });
35
+
36
+ try {
37
+ const result = await request;
38
+ const data = dataHandler(result, getState);
39
+
40
+ dispatch({
41
+ type: actions.SUCCESS,
42
+ data,
43
+ });
44
+
45
+ return data;
46
+ } catch (error) {
47
+ if (isAxiosResponse(error) && error.status === 401) {
48
+ dispatch({
49
+ type: SET_UNAUTHENTICATED.SUCCESS,
50
+ });
51
+ } else if (isAxiosResponse(error) && error.status >= 500 && error.statusText) {
52
+ createToast({
53
+ name: 'Request failure',
54
+ title: 'Request failure',
55
+ type: 'error',
56
+ content: `${error.status} ${error.statusText}`,
57
+ });
58
+ }
59
+
60
+ dispatch({
61
+ type: actions.FAILURE,
62
+ error,
63
+ });
64
+
65
+ // TODO should probably throw the received error here, but this change requires a thorough revision of all api calls
66
+ return undefined;
67
+ }
68
+ };
69
+
70
+ return doRequest;
71
+ }
72
+
73
+ export type ApiRequestAction<
74
+ Actions extends ReturnType<typeof createRequestActionTypes>,
75
+ SuccessResponse = unknown,
76
+ ErrorResponse = unknown
77
+ > =
78
+ | {
79
+ type: Actions['REQUEST'],
80
+ }
81
+ | {
82
+ type: Actions['SUCCESS'],
83
+ data: SuccessResponse,
84
+ }
85
+ | {
86
+ type: Actions['FAILURE'],
87
+ error: ErrorResponse,
88
+ };
@@ -0,0 +1,147 @@
1
+ // common
2
+
3
+ type Plan = Record<string, any>;
4
+ type AST = string;
5
+ type Stats = Record<string, any>;
6
+
7
+ export interface CommonFields {
8
+ ast?: AST;
9
+ plan?: Plan;
10
+ stats?: Stats;
11
+ };
12
+
13
+ interface DeprecatedCommonFields {
14
+ stats?: Stats;
15
+ }
16
+
17
+ export interface ErrorRepsonse {
18
+ error?: any;
19
+ issues?: any;
20
+ };
21
+
22
+ export type ExecuteActions = 'execute-script' | 'execute' | 'execute-scan' | undefined;
23
+ export type ExplainActions = 'explain' | 'explain-ast';
24
+ export type Actions = ExecuteActions | ExplainActions;
25
+
26
+ // undefined == 'classic'
27
+ export type Schemas = 'classic' | 'modern' | 'ydb' | undefined;
28
+
29
+ // ==== EXECUTE ====
30
+
31
+ // common types
32
+
33
+ type CellValue = string | number | null | undefined;
34
+
35
+ export type KeyValueRow<T = CellValue> = {
36
+ [key: string]: T;
37
+ }
38
+
39
+ export type ArrayRow<T = CellValue> = Array<T>;
40
+
41
+ export interface ColumnType {
42
+ name: string;
43
+ type: string;
44
+ }
45
+
46
+ // modern response
47
+
48
+ export type ExecuteModernResponse = {
49
+ // either both fields exist, or neither one does
50
+ // they can be undefined for queries like `insert into`
51
+ result?: ArrayRow[];
52
+ columns?: ColumnType[];
53
+ } & CommonFields;
54
+
55
+ export type ExecuteClassicResponseDeep = {
56
+ // can be undefined for queries like `insert into`
57
+ result?: KeyValueRow[];
58
+ } & CommonFields;
59
+
60
+ // can be undefined for queries like `insert into`
61
+ export type ExecuteClassicResponsePlain = KeyValueRow[] | undefined;
62
+
63
+ export type ExecuteClassicResponse = ExecuteClassicResponseDeep | ExecuteClassicResponsePlain;
64
+
65
+ export type ExecuteYdbResponse = {
66
+ // can be undefined for queries like `insert into`
67
+ result?: KeyValueRow[];
68
+ } & CommonFields;
69
+
70
+ type ExecuteResponse<Schema extends Schemas> = (
71
+ Schema extends 'modern'
72
+ ? ExecuteModernResponse
73
+ : Schema extends 'ydb'
74
+ ? ExecuteYdbResponse
75
+ : Schema extends 'classic' | undefined
76
+ ? ExecuteClassicResponse
77
+ : unknown
78
+ );
79
+
80
+ // deprecated response from older versions, backward compatibility
81
+
82
+ type DeprecatedExecuteResponseValue =
83
+ | KeyValueRow[]
84
+ | string
85
+ // can be here because of a bug in the previous backend version
86
+ // should be ignored in parsing
87
+ | Plan;
88
+
89
+ export type DeprecatedExecuteResponseDeep = {
90
+ // can be undefined for queries like `insert into`
91
+ result?: DeprecatedExecuteResponseValue;
92
+ } & DeprecatedCommonFields;
93
+
94
+ // can be undefined for queries like `insert into`
95
+ export type DeprecatedExecuteResponsePlain = DeprecatedExecuteResponseValue | undefined;
96
+
97
+ export type DeprecatedExecuteResponse = DeprecatedExecuteResponseDeep | DeprecatedExecuteResponsePlain;
98
+
99
+
100
+ // ==== EXPLAIN ====
101
+
102
+ // modern response
103
+
104
+ type ExplainResponse = CommonFields;
105
+
106
+ // deprecated response from older versions, backward compatibility
107
+
108
+ type DeprecatedExplainResponse<Action extends ExplainActions> = (
109
+ Action extends 'explain-ast'
110
+ ? {ast: AST} & DeprecatedCommonFields
111
+ : Actions extends 'explain'
112
+ ? ({result: Plan} & DeprecatedCommonFields) | Plan
113
+ : unknown
114
+ );
115
+
116
+
117
+ // ==== COMBINED API RESPONSE ====
118
+
119
+ export type QueryAPIExecuteResponse<Schema extends Schemas = undefined> =
120
+ | ExecuteResponse<Schema>
121
+ | DeprecatedExecuteResponse
122
+ | null;
123
+
124
+ export type QueryAPIExplainResponse<Action extends ExplainActions> =
125
+ | ExplainResponse
126
+ | DeprecatedExplainResponse<Action>
127
+ | null;
128
+
129
+ export type QueryAPIResponse<Action extends Actions, Schema extends Schemas = undefined> = (
130
+ Action extends ExecuteActions
131
+ ? QueryAPIExecuteResponse<Schema>
132
+ : Action extends ExplainActions
133
+ ? QueryAPIExplainResponse<Action>
134
+ : unknown
135
+ );
136
+
137
+ export type AnyExecuteResponse =
138
+ | ExecuteModernResponse
139
+ | ExecuteClassicResponse
140
+ | ExecuteYdbResponse
141
+ | DeprecatedExecuteResponse;
142
+
143
+ export type DeepExecuteResponse =
144
+ | ExecuteModernResponse
145
+ | ExecuteClassicResponseDeep
146
+ | ExecuteYdbResponse
147
+ | DeprecatedExecuteResponseDeep;