Web console: improve compaction status display (#13523)

* improve compaction status display

* even more accurate

* fix snapshot
This commit is contained in:
Vadim Ogievetsky 2022-12-07 21:03:59 -08:00 committed by GitHub
parent fbf76ad8f5
commit d85fb8cc4e
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
7 changed files with 181 additions and 53 deletions

View File

@ -23,6 +23,11 @@
height: 80vh; height: 80vh;
} }
.legacy-callout {
width: auto;
margin: 10px 15px 0;
}
.form-json-selector { .form-json-selector {
margin: 15px; margin: 15px;
} }

View File

@ -16,11 +16,16 @@
* limitations under the License. * limitations under the License.
*/ */
import { Button, Classes, Dialog, Intent } from '@blueprintjs/core'; import { Button, Callout, Classes, Code, Dialog, Intent } from '@blueprintjs/core';
import React, { useState } from 'react'; import React, { useState } from 'react';
import { AutoForm, FormJsonSelector, FormJsonTabs, JsonInput } from '../../components'; import { AutoForm, FormJsonSelector, FormJsonTabs, JsonInput } from '../../components';
import { COMPACTION_CONFIG_FIELDS, CompactionConfig } from '../../druid-models'; import {
COMPACTION_CONFIG_FIELDS,
CompactionConfig,
compactionConfigHasLegacyInputSegmentSizeBytesSet,
} from '../../druid-models';
import { deepDelete, formatBytesCompact } from '../../utils';
import './compaction-dialog.scss'; import './compaction-dialog.scss';
@ -55,13 +60,29 @@ export const CompactionDialog = React.memo(function CompactionDialog(props: Comp
canOutsideClickClose={false} canOutsideClickClose={false}
title={`Compaction config: ${datasource}`} title={`Compaction config: ${datasource}`}
> >
{compactionConfigHasLegacyInputSegmentSizeBytesSet(currentConfig) && (
<Callout className="legacy-callout" intent={Intent.WARNING}>
<p>
You current config sets the legacy <Code>inputSegmentSizeBytes</Code> to{' '}
<Code>{formatBytesCompact(currentConfig.inputSegmentSizeBytes!)}</Code> it is
recommended to unset this property.
</p>
<p>
<Button
intent={Intent.WARNING}
text="Remove legacy setting"
onClick={() => setCurrentConfig(deepDelete(currentConfig, 'inputSegmentSizeBytes'))}
/>
</p>
</Callout>
)}
<FormJsonSelector tab={currentTab} onChange={setCurrentTab} /> <FormJsonSelector tab={currentTab} onChange={setCurrentTab} />
<div className="content"> <div className="content">
{currentTab === 'form' ? ( {currentTab === 'form' ? (
<AutoForm <AutoForm
fields={COMPACTION_CONFIG_FIELDS} fields={COMPACTION_CONFIG_FIELDS}
model={currentConfig} model={currentConfig}
onChange={m => setCurrentConfig(m)} onChange={m => setCurrentConfig(m as CompactionConfig)}
/> />
) : ( ) : (
<JsonInput <JsonInput

View File

@ -22,7 +22,26 @@ import React from 'react';
import { Field } from '../../components'; import { Field } from '../../components';
import { deepGet, deepSet, oneOf } from '../../utils'; import { deepGet, deepSet, oneOf } from '../../utils';
export type CompactionConfig = Record<string, any>; export interface CompactionConfig {
dataSource: string;
skipOffsetFromLatest?: string;
tuningConfig?: any;
[key: string]: any;
// Deprecated:
inputSegmentSizeBytes?: number;
}
export const NOOP_INPUT_SEGMENT_SIZE_BYTES = 100000000000000;
export function compactionConfigHasLegacyInputSegmentSizeBytesSet(
config: CompactionConfig,
): boolean {
return (
typeof config.inputSegmentSizeBytes === 'number' &&
config.inputSegmentSizeBytes < NOOP_INPUT_SEGMENT_SIZE_BYTES
);
}
export const COMPACTION_CONFIG_FIELDS: Field<CompactionConfig>[] = [ export const COMPACTION_CONFIG_FIELDS: Field<CompactionConfig>[] = [
{ {
@ -182,7 +201,7 @@ export const COMPACTION_CONFIG_FIELDS: Field<CompactionConfig>[] = [
defined: t => defined: t =>
oneOf(deepGet(t, 'tuningConfig.partitionsSpec.type'), 'single_dim', 'range') && oneOf(deepGet(t, 'tuningConfig.partitionsSpec.type'), 'single_dim', 'range') &&
!deepGet(t, 'tuningConfig.partitionsSpec.maxRowsPerSegment'), !deepGet(t, 'tuningConfig.partitionsSpec.maxRowsPerSegment'),
required: (t: CompactionConfig) => required: t =>
!deepGet(t, 'tuningConfig.partitionsSpec.targetRowsPerSegment') && !deepGet(t, 'tuningConfig.partitionsSpec.targetRowsPerSegment') &&
!deepGet(t, 'tuningConfig.partitionsSpec.maxRowsPerSegment'), !deepGet(t, 'tuningConfig.partitionsSpec.maxRowsPerSegment'),
info: ( info: (
@ -205,7 +224,7 @@ export const COMPACTION_CONFIG_FIELDS: Field<CompactionConfig>[] = [
defined: t => defined: t =>
oneOf(deepGet(t, 'tuningConfig.partitionsSpec.type'), 'single_dim', 'range') && oneOf(deepGet(t, 'tuningConfig.partitionsSpec.type'), 'single_dim', 'range') &&
!deepGet(t, 'tuningConfig.partitionsSpec.targetRowsPerSegment'), !deepGet(t, 'tuningConfig.partitionsSpec.targetRowsPerSegment'),
required: (t: CompactionConfig) => required: t =>
!deepGet(t, 'tuningConfig.partitionsSpec.targetRowsPerSegment') && !deepGet(t, 'tuningConfig.partitionsSpec.targetRowsPerSegment') &&
!deepGet(t, 'tuningConfig.partitionsSpec.maxRowsPerSegment'), !deepGet(t, 'tuningConfig.partitionsSpec.maxRowsPerSegment'),
info: ( info: (
@ -277,7 +296,7 @@ export const COMPACTION_CONFIG_FIELDS: Field<CompactionConfig>[] = [
defaultValue: 1073741824, defaultValue: 1073741824,
min: 1000000, min: 1000000,
hideInMore: true, hideInMore: true,
adjustment: (t: CompactionConfig) => deepSet(t, 'tuningConfig.splitHintSpec.type', 'maxSize'), adjustment: t => deepSet(t, 'tuningConfig.splitHintSpec.type', 'maxSize'),
info: ( info: (
<> <>
Maximum number of bytes of input segments to process in a single task. If a single segment Maximum number of bytes of input segments to process in a single task. If a single segment
@ -293,7 +312,7 @@ export const COMPACTION_CONFIG_FIELDS: Field<CompactionConfig>[] = [
defaultValue: 1000, defaultValue: 1000,
min: 1, min: 1,
hideInMore: true, hideInMore: true,
adjustment: (t: CompactionConfig) => deepSet(t, 'tuningConfig.splitHintSpec.type', 'maxSize'), adjustment: t => deepSet(t, 'tuningConfig.splitHintSpec.type', 'maxSize'),
info: ( info: (
<> <>
Maximum number of input segments to process in a single subtask. This limit is to avoid task Maximum number of input segments to process in a single subtask. This limit is to avoid task

View File

@ -21,7 +21,13 @@ import { CompactionConfig } from '../compaction-config/compaction-config';
import { CompactionStatus, formatCompactionInfo, zeroCompactionStatus } from './compaction-status'; import { CompactionStatus, formatCompactionInfo, zeroCompactionStatus } from './compaction-status';
describe('compaction status', () => { describe('compaction status', () => {
const BASIC_CONFIG: CompactionConfig = {}; const BASIC_CONFIG: CompactionConfig = {
dataSource: 'tbl',
};
const LEGACY_CONFIG: CompactionConfig = {
dataSource: 'tbl',
inputSegmentSizeBytes: 1e6,
};
const ZERO_STATUS: CompactionStatus = { const ZERO_STATUS: CompactionStatus = {
dataSource: 'tbl', dataSource: 'tbl',
scheduleStatus: 'RUNNING', scheduleStatus: 'RUNNING',
@ -36,9 +42,12 @@ describe('compaction status', () => {
intervalCountSkipped: 0, intervalCountSkipped: 0,
}; };
it('zeroCompactionStatus', () => { describe('zeroCompactionStatus', () => {
it('works with zero', () => {
expect(zeroCompactionStatus(ZERO_STATUS)).toEqual(true); expect(zeroCompactionStatus(ZERO_STATUS)).toEqual(true);
});
it('works with non-zero', () => {
expect( expect(
zeroCompactionStatus({ zeroCompactionStatus({
dataSource: 'tbl', dataSource: 'tbl',
@ -55,16 +64,28 @@ describe('compaction status', () => {
}), }),
).toEqual(false); ).toEqual(false);
}); });
});
it('formatCompactionConfigAndStatus', () => { describe('formatCompactionConfigAndStatus', () => {
it('works with nothing', () => {
expect(formatCompactionInfo({})).toEqual('Not enabled'); expect(formatCompactionInfo({})).toEqual('Not enabled');
});
it('works when there is no status', () => {
expect(formatCompactionInfo({ config: BASIC_CONFIG })).toEqual('Awaiting first run'); expect(formatCompactionInfo({ config: BASIC_CONFIG })).toEqual('Awaiting first run');
});
it('works when here is no config', () => {
expect(formatCompactionInfo({ status: ZERO_STATUS })).toEqual('Not enabled'); expect(formatCompactionInfo({ status: ZERO_STATUS })).toEqual('Not enabled');
});
expect(formatCompactionInfo({ config: BASIC_CONFIG, status: ZERO_STATUS })).toEqual('Running'); it('works with config and zero status', () => {
expect(formatCompactionInfo({ config: BASIC_CONFIG, status: ZERO_STATUS })).toEqual(
'Running',
);
});
it('works when fully compacted', () => {
expect( expect(
formatCompactionInfo({ formatCompactionInfo({
config: BASIC_CONFIG, config: BASIC_CONFIG,
@ -84,4 +105,49 @@ describe('compaction status', () => {
}), }),
).toEqual('Fully compacted'); ).toEqual('Fully compacted');
}); });
it('works when fully compacted and some segments skipped', () => {
expect(
formatCompactionInfo({
config: BASIC_CONFIG,
status: {
dataSource: 'tbl',
scheduleStatus: 'RUNNING',
bytesAwaitingCompaction: 0,
bytesCompacted: 0,
bytesSkipped: 3776979,
segmentCountAwaitingCompaction: 0,
segmentCountCompacted: 0,
segmentCountSkipped: 24,
intervalCountAwaitingCompaction: 0,
intervalCountCompacted: 0,
intervalCountSkipped: 24,
},
}),
).toEqual('Fully compacted (except the last P1D of data, 24 segments skipped)');
});
it('works when fully compacted and some segments skipped (with legacy config)', () => {
expect(
formatCompactionInfo({
config: LEGACY_CONFIG,
status: {
dataSource: 'tbl',
scheduleStatus: 'RUNNING',
bytesAwaitingCompaction: 0,
bytesCompacted: 0,
bytesSkipped: 3776979,
segmentCountAwaitingCompaction: 0,
segmentCountCompacted: 0,
segmentCountSkipped: 24,
intervalCountAwaitingCompaction: 0,
intervalCountCompacted: 0,
intervalCountSkipped: 24,
},
}),
).toEqual(
'Fully compacted (except the last P1D of data and segments larger than 1.00MB, 24 segments skipped)',
);
});
});
}); });

View File

@ -16,7 +16,11 @@
* limitations under the License. * limitations under the License.
*/ */
import { CompactionConfig } from '../compaction-config/compaction-config'; import { formatBytesCompact, pluralIfNeeded } from '../../utils';
import {
CompactionConfig,
compactionConfigHasLegacyInputSegmentSizeBytesSet,
} from '../compaction-config/compaction-config';
function capitalizeFirst(str: string): string { function capitalizeFirst(str: string): string {
return str.slice(0, 1).toUpperCase() + str.slice(1).toLowerCase(); return str.slice(0, 1).toUpperCase() + str.slice(1).toLowerCase();
@ -59,8 +63,21 @@ export function formatCompactionInfo(compaction: CompactionInfo) {
const { config, status } = compaction; const { config, status } = compaction;
if (config) { if (config) {
if (status) { if (status) {
if (status.bytesAwaitingCompaction === 0 && !zeroCompactionStatus(status)) { if (
status.bytesAwaitingCompaction === 0 &&
status.segmentCountAwaitingCompaction === 0 &&
status.intervalCountAwaitingCompaction === 0 &&
!zeroCompactionStatus(status)
) {
if (status.segmentCountSkipped) {
return `Fully compacted (except the last ${config.skipOffsetFromLatest || 'P1D'} of data${
compactionConfigHasLegacyInputSegmentSizeBytesSet(config)
? ` and segments larger than ${formatBytesCompact(config.inputSegmentSizeBytes!)}`
: ''
}, ${pluralIfNeeded(status.segmentCountSkipped, 'segment')} skipped)`;
} else {
return 'Fully compacted'; return 'Fully compacted';
}
} else { } else {
return capitalizeFirst(status.scheduleStatus); return capitalizeFirst(status.scheduleStatus);
} }

View File

@ -283,7 +283,7 @@ exports[`DatasourcesView matches snapshot 1`] = `
"filterable": false, "filterable": false,
"id": "compactionStatus", "id": "compactionStatus",
"show": true, "show": true,
"width": 150, "width": 180,
}, },
Object { Object {
"Cell": [Function], "Cell": [Function],

View File

@ -1324,7 +1324,7 @@ ORDER BY 1`;
id: 'compactionStatus', id: 'compactionStatus',
accessor: row => Boolean(row.compaction?.status), accessor: row => Boolean(row.compaction?.status),
filterable: false, filterable: false,
width: 150, width: 180,
Cell: ({ original }) => { Cell: ({ original }) => {
const { datasource, compaction } = original as Datasource; const { datasource, compaction } = original as Datasource;
return ( return (