-
Notifications
You must be signed in to change notification settings - Fork 2.1k
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
UI Support for Store filtering #3166
Changes from 34 commits
c435466
cab4ad4
4630a6f
85024a6
4b067f3
94d1ab2
782efe8
dccb006
d8a4caa
fde8af4
7b756d8
bc9b4af
7b7556b
0c18949
d741a7d
3658e84
1d4c9c3
8cbdd85
20a4b5d
08121ef
0d89425
df62297
471db93
e4e9894
063eff8
9fa8532
5853968
62a5931
126bac5
f5125c8
bbf2969
1d12439
9e92d7e
ba86337
06bfef5
3adef70
02534b9
File filter
Filter by extension
Conversations
Jump to
Diff view
Diff view
There are no files selected for viewing
Large diffs are not rendered by default.
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -1,6 +1,7 @@ | ||
import React, { Component } from 'react'; | ||
|
||
import { UncontrolledAlert, Button, Col, Nav, NavItem, NavLink, Row, TabContent, TabPane } from 'reactstrap'; | ||
import Select from 'react-select'; | ||
|
||
import moment from 'moment-timezone'; | ||
|
||
|
@@ -11,6 +12,7 @@ import { GraphTabContent } from './GraphTabContent'; | |
import DataTable from './DataTable'; | ||
import TimeInput from './TimeInput'; | ||
import QueryStatsView, { QueryStats } from './QueryStatsView'; | ||
import { Store } from '../../thanos/pages/stores/store'; | ||
import PathPrefixProps from '../../types/PathPrefixProps'; | ||
import { QueryParams } from '../../types/types'; | ||
|
||
|
@@ -23,6 +25,7 @@ interface PanelProps { | |
metricNames: string[]; | ||
removePanel: () => void; | ||
onExecuteQuery: (query: string) => void; | ||
stores: Store[]; | ||
} | ||
|
||
interface PanelState { | ||
|
@@ -44,6 +47,7 @@ export interface PanelOptions { | |
maxSourceResolution: string; | ||
useDeduplication: boolean; | ||
usePartialResponse: boolean; | ||
storeMatches: Store[]; | ||
} | ||
|
||
export enum PanelType { | ||
|
@@ -61,6 +65,7 @@ export const PanelDefaultOptions: PanelOptions = { | |
maxSourceResolution: '0s', | ||
useDeduplication: true, | ||
usePartialResponse: false, | ||
storeMatches: [], | ||
}; | ||
|
||
class Panel extends Component<PanelProps & PathPrefixProps, PanelState> { | ||
|
@@ -80,6 +85,7 @@ class Panel extends Component<PanelProps & PathPrefixProps, PanelState> { | |
|
||
this.handleChangeDeduplication = this.handleChangeDeduplication.bind(this); | ||
this.handleChangePartialResponse = this.handleChangePartialResponse.bind(this); | ||
this.handleStoreMatchChange = this.handleStoreMatchChange.bind(this); | ||
} | ||
|
||
componentDidUpdate({ options: prevOpts }: PanelProps) { | ||
|
@@ -91,6 +97,7 @@ class Panel extends Component<PanelProps & PathPrefixProps, PanelState> { | |
maxSourceResolution, | ||
useDeduplication, | ||
usePartialResponse, | ||
// TODO: Add support for Store Matches | ||
} = this.props.options; | ||
if ( | ||
prevOpts.endTime !== endTime || | ||
|
@@ -100,6 +107,7 @@ class Panel extends Component<PanelProps & PathPrefixProps, PanelState> { | |
prevOpts.maxSourceResolution !== maxSourceResolution || | ||
prevOpts.useDeduplication !== useDeduplication || | ||
prevOpts.usePartialResponse !== usePartialResponse | ||
// Check store matches | ||
) { | ||
this.executeQuery(); | ||
} | ||
|
@@ -138,6 +146,11 @@ class Panel extends Component<PanelProps & PathPrefixProps, PanelState> { | |
partial_response: this.props.options.usePartialResponse.toString(), | ||
}); | ||
|
||
// Add storeMatches to query params. | ||
this.props.options.storeMatches?.forEach((store: Store) => | ||
params.append('storeMatch[]', `{__address__="${store.name}"}`) | ||
); | ||
|
||
let path: string; | ||
switch (this.props.options.type) { | ||
case 'graph': | ||
|
@@ -255,8 +268,12 @@ class Panel extends Component<PanelProps & PathPrefixProps, PanelState> { | |
this.setOptions({ usePartialResponse: event.target.checked }); | ||
}; | ||
|
||
handleStoreMatchChange = (selectedStores: any): void => { | ||
this.setOptions({ storeMatches: selectedStores || [] }); | ||
}; | ||
|
||
render() { | ||
const { pastQueries, metricNames, options, id } = this.props; | ||
const { pastQueries, metricNames, options, id, stores } = this.props; | ||
return ( | ||
<div className="panel"> | ||
<Row> | ||
|
@@ -296,6 +313,23 @@ class Panel extends Component<PanelProps & PathPrefixProps, PanelState> { | |
</Checkbox> | ||
</Col> | ||
</Row> | ||
{stores.length > 0 ? (<Row> | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. Cases like this are called conditional rendering in React and it's better to use So instead of Also, if |
||
<Col> | ||
<div className="store-filter-wrapper"> | ||
<label className="store-filter-label">Store Filter:</label> | ||
<Select | ||
defaultValue={options.storeMatches} | ||
options={stores} | ||
isMulti | ||
getOptionLabel={(option: Store) => option.name} | ||
getOptionValue={(option: Store) => option.name} | ||
closeMenuOnSelect={false} | ||
styles={{ container: (provided, state) => ({ ...provided, marginBottom: 20, zIndex: 3, width: '100%' }) }} | ||
onChange={this.handleStoreMatchChange} | ||
/> | ||
</div> | ||
</Col> | ||
</Row>) : ""} | ||
<Row> | ||
<Col> | ||
<Nav tabs> | ||
|
Original file line number | Diff line number | Diff line change |
---|---|---|
|
@@ -134,6 +134,26 @@ describe('Utils', () => { | |
}); | ||
|
||
describe('URL Params', () => { | ||
const stores: any = [ | ||
{ | ||
name: 'thanos_sidecar_one:10901', | ||
lastCheck: '2020-09-20T11:35:18.250713478Z', | ||
lastError: null, | ||
labelSets: [ | ||
{ | ||
labels: [ | ||
{ | ||
name: 'monitor', | ||
value: 'prometheus_one', | ||
}, | ||
], | ||
}, | ||
], | ||
minTime: 1600598100000, | ||
maxTime: 9223372036854776000, | ||
}, | ||
]; | ||
|
||
const panels: any = [ | ||
{ | ||
key: '0', | ||
|
@@ -147,6 +167,7 @@ describe('Utils', () => { | |
useDeduplication: true, | ||
usePartialResponse: false, | ||
type: PanelType.Graph, | ||
storeMatches: [], | ||
}, | ||
}, | ||
{ | ||
|
@@ -161,11 +182,12 @@ describe('Utils', () => { | |
useDeduplication: false, | ||
usePartialResponse: true, | ||
type: PanelType.Table, | ||
storeMatches: stores, | ||
}, | ||
}, | ||
]; | ||
const query = | ||
'?g0.expr=rate(node_cpu_seconds_total%7Bmode%3D%22system%22%7D%5B1m%5D)&g0.tab=0&g0.stacked=0&g0.range_input=1h&g0.max_source_resolution=raw&g0.deduplicate=1&g0.partial_response=0&g0.end_input=2019-10-25%2023%3A37%3A00&g0.moment_input=2019-10-25%2023%3A37%3A00&g1.expr=node_filesystem_avail_bytes&g1.tab=1&g1.stacked=0&g1.range_input=1h&g1.max_source_resolution=auto&g1.deduplicate=0&g1.partial_response=1'; | ||
'?g0.expr=rate(node_cpu_seconds_total%7Bmode%3D%22system%22%7D%5B1m%5D)&g0.tab=0&g0.stacked=0&g0.range_input=1h&g0.max_source_resolution=raw&g0.deduplicate=1&g0.partial_response=0&g0.store_matches=%5B%5D&g0.end_input=2019-10-25%2023%3A37%3A00&g0.moment_input=2019-10-25%2023%3A37%3A00&g1.expr=node_filesystem_avail_bytes&g1.tab=1&g1.stacked=0&g1.range_input=1h&g1.max_source_resolution=auto&g1.deduplicate=0&g1.partial_response=1&g1.store_matches=%5B%7B%22name%22%3A%22thanos_sidecar_one%3A10901%22%2C%22lastCheck%22%3A%222020-09-20T11%3A35%3A18.250713478Z%22%2C%22lastError%22%3Anull%2C%22labelSets%22%3A%5B%7B%22labels%22%3A%5B%7B%22name%22%3A%22monitor%22%2C%22value%22%3A%22prometheus_one%22%7D%5D%7D%5D%2C%22minTime%22%3A1600598100000%2C%22maxTime%22%3A9223372036854776000%7D%5D'; | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. I think we should only store the There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. I too thought of passing only the store name, but the Select component that is being used takes in only objects as options. So passing a string array was not an option. I can create an interface with only store name as the key though. Would like your opinion on this. |
||
|
||
describe('decodePanelOptionsFromQueryString', () => { | ||
it('returns [] when query is empty', () => { | ||
|
@@ -204,6 +226,13 @@ describe('Utils', () => { | |
it('should parse partial_response', () => { | ||
expect(parseOption('partial_response=1')).toEqual({ usePartialResponse: true }); | ||
}); | ||
it('it should parse store_matches', () => { | ||
expect( | ||
parseOption( | ||
'store_matches=%5B%7B%22name%22%3A%22thanos_sidecar_one%3A10901%22%2C%22lastCheck%22%3A%222020-09-20T11%3A35%3A18.250713478Z%22%2C%22lastError%22%3Anull%2C%22labelSets%22%3A%5B%7B%22labels%22%3A%5B%7B%22name%22%3A%22monitor%22%2C%22value%22%3A%22prometheus_one%22%7D%5D%7D%5D%2C%22minTime%22%3A1600598100000%2C%22maxTime%22%3A9223372036854776000%7D%5D' | ||
) | ||
).toEqual({ storeMatches: stores }); | ||
}); | ||
|
||
describe('step_input', () => { | ||
it('should return step_input parsed if > 0', () => { | ||
|
@@ -249,10 +278,11 @@ describe('Utils', () => { | |
maxSourceResolution: 'raw', | ||
useDeduplication: true, | ||
usePartialResponse: false, | ||
storeMatches: [], | ||
}, | ||
}) | ||
).toEqual( | ||
'g0.expr=foo&g0.tab=0&g0.stacked=1&g0.range_input=0y&g0.max_source_resolution=raw&g0.deduplicate=1&g0.partial_response=0&g0.step_input=1' | ||
'g0.expr=foo&g0.tab=0&g0.stacked=1&g0.range_input=0y&g0.max_source_resolution=raw&g0.deduplicate=1&g0.partial_response=0&g0.store_matches=%5B%5D&g0.step_input=1' | ||
); | ||
}); | ||
}); | ||
|
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
It'd nice to see this get done in this PR, but it's not a blocker :)