Skip to content

Commit bf00714

Browse files
authored
sql editor (#50)
1 parent 4c46d1f commit bf00714

Some content is hidden

Large Commits have some content hidden by default. Use the searchbox below for content that may be hidden.

59 files changed

+3869
-6
lines changed

package.json

+6-2
Original file line numberDiff line numberDiff line change
@@ -15,19 +15,23 @@
1515
},
1616
"dependencies": {
1717
"@grafana/data": "^8.4.7",
18+
"@grafana/experimental": "^0.0.2-canary.39",
1819
"@grafana/runtime": "^8.4.7",
1920
"@grafana/toolkit": "^8.4.7",
20-
"@grafana/ui": "^8.4.7",
2121
"@grafana/tsconfig": "1.2.0-rc1",
22+
"@grafana/ui": "^8.4.7",
2223
"@types/chance": "^1.1.0",
2324
"@types/memoize-one": "^5.1.2",
2425
"@types/react-calendar": "^3.1.2",
25-
"react-use": "17.3.1",
2626
"chance": "^1.1.7",
2727
"copyfiles": "^2.4.1",
2828
"memoize-one": "^5.1.1",
2929
"rc-cascader": "1.0.1",
30+
"react-awesome-query-builder": "^5.3.1",
31+
"react-use": "17.3.1",
32+
"react-virtualized-auto-sizer": "^1.0.6",
3033
"semver": "^7.3.5",
34+
"sql-formatter-plus": "^1.3.6",
3135
"typescript": "^4.7.4"
3236
},
3337
"devDependencies": {
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,21 @@
1+
import { css, cx } from '@emotion/css';
2+
import React from 'react';
3+
4+
import { GrafanaTheme2 } from '@grafana/data';
5+
import { Button, ButtonProps, stylesFactory, useTheme2 } from '@grafana/ui';
6+
7+
interface AccessoryButtonProps extends ButtonProps {}
8+
9+
export const AccessoryButton: React.FC<AccessoryButtonProps> = ({ className, ...props }) => {
10+
const theme = useTheme2();
11+
const styles = getButtonStyles(theme);
12+
13+
return <Button {...props} className={cx(className, styles.button)} />;
14+
};
15+
16+
const getButtonStyles = stylesFactory((theme: GrafanaTheme2) => ({
17+
button: css({
18+
paddingLeft: theme.spacing(3 / 2),
19+
paddingRight: theme.spacing(3 / 2),
20+
}),
21+
}));
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,51 @@
1+
import React, { useRef, useEffect } from 'react';
2+
3+
import { Button, Icon, Modal } from '@grafana/ui';
4+
5+
type ConfirmModalProps = {
6+
isOpen: boolean;
7+
onCancel?: () => void;
8+
onDiscard?: () => void;
9+
onCopy?: () => void;
10+
};
11+
export function ConfirmModal({ isOpen, onCancel, onDiscard, onCopy }: ConfirmModalProps) {
12+
const buttonRef = useRef<HTMLButtonElement>(null);
13+
14+
// Moved from grafana/ui
15+
useEffect(() => {
16+
// for some reason autoFocus property did no work on this button, but this does
17+
if (isOpen) {
18+
buttonRef.current?.focus();
19+
}
20+
}, [isOpen]);
21+
22+
return (
23+
<Modal
24+
title={
25+
<div className="modal-header-title">
26+
<Icon name="exclamation-triangle" size="lg" />
27+
<span className="p-l-1">Warning</span>
28+
</div>
29+
}
30+
onDismiss={onCancel}
31+
isOpen={isOpen}
32+
>
33+
<p>
34+
Builder mode does not display changes made in code. The query builder will display the last changes you made in
35+
builder mode.
36+
</p>
37+
<p>Do you want to copy your code to the clipboard?</p>
38+
<Modal.ButtonRow>
39+
<Button type="button" variant="secondary" onClick={onCancel} fill="outline">
40+
Cancel
41+
</Button>
42+
<Button variant="destructive" type="button" onClick={onDiscard} ref={buttonRef}>
43+
Discard code and switch
44+
</Button>
45+
<Button variant="primary" onClick={onCopy}>
46+
Copy code and switch
47+
</Button>
48+
</Modal.ButtonRow>
49+
</Modal>
50+
);
51+
}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,74 @@
1+
import React from 'react';
2+
3+
import { FieldSet, InlineField } from '@grafana/ui';
4+
import { SQLConnectionLimits } from './types';
5+
import { NumberInput } from './NumberInput';
6+
7+
interface Props<T> {
8+
onPropertyChanged: (property: keyof T, value?: number) => void;
9+
labelWidth: number;
10+
jsonData: SQLConnectionLimits;
11+
}
12+
13+
export const ConnectionLimits = <T extends SQLConnectionLimits>(props: Props<T>) => {
14+
const { onPropertyChanged, labelWidth, jsonData } = props;
15+
16+
const onJSONDataNumberChanged = (property: keyof SQLConnectionLimits) => {
17+
return (number?: number) => {
18+
if (onPropertyChanged) {
19+
onPropertyChanged(property, number);
20+
}
21+
};
22+
};
23+
24+
return (
25+
<FieldSet label="Connection limits">
26+
<InlineField
27+
tooltip={
28+
<span>
29+
The maximum number of open connections to the database.If <i>Max idle connections</i> is greater than 0 and
30+
the <i>Max open connections</i> is less than <i>Max idle connections</i>, then
31+
<i>Max idle connections</i> will be reduced to match the <i>Max open connections</i> limit. If set to 0,
32+
there is no limit on the number of open connections.
33+
</span>
34+
}
35+
labelWidth={labelWidth}
36+
label="Max open"
37+
>
38+
<NumberInput
39+
placeholder="unlimited"
40+
value={jsonData.maxOpenConns}
41+
onChange={onJSONDataNumberChanged('maxOpenConns')}
42+
></NumberInput>
43+
</InlineField>
44+
<InlineField
45+
tooltip={
46+
<span>
47+
The maximum number of connections in the idle connection pool.If <i>Max open connections</i> is greater than
48+
0 but less than the <i>Max idle connections</i>, then the <i>Max idle connections</i> will be reduced to
49+
match the <i>Max open connections</i> limit. If set to 0, no idle connections are retained.
50+
</span>
51+
}
52+
labelWidth={labelWidth}
53+
label="Max idle"
54+
>
55+
<NumberInput
56+
placeholder="2"
57+
value={jsonData.maxIdleConns}
58+
onChange={onJSONDataNumberChanged('maxIdleConns')}
59+
></NumberInput>
60+
</InlineField>
61+
<InlineField
62+
tooltip="The maximum amount of time in seconds a connection may be reused. If set to 0, connections are reused forever."
63+
labelWidth={labelWidth}
64+
label="Max lifetime"
65+
>
66+
<NumberInput
67+
placeholder="14400"
68+
value={jsonData.connMaxLifetime}
69+
onChange={onJSONDataNumberChanged('connMaxLifetime')}
70+
></NumberInput>
71+
</InlineField>
72+
</FieldSet>
73+
);
74+
};
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,61 @@
1+
import React, { useEffect } from 'react';
2+
import { useAsync } from 'react-use';
3+
4+
import { SelectableValue } from '@grafana/data';
5+
import { Select } from '@grafana/ui';
6+
7+
import { DB, ResourceSelectorProps, toOption } from './types';
8+
9+
interface DatasetSelectorProps extends ResourceSelectorProps {
10+
db: DB;
11+
value: string | null;
12+
applyDefault?: boolean;
13+
disabled?: boolean;
14+
onChange: (v: SelectableValue) => void;
15+
}
16+
17+
export const DatasetSelector: React.FC<DatasetSelectorProps> = ({
18+
db,
19+
value,
20+
onChange,
21+
disabled,
22+
className,
23+
applyDefault,
24+
}) => {
25+
const state = useAsync(async () => {
26+
const datasets = await db.datasets();
27+
return datasets.map(toOption);
28+
}, []);
29+
30+
useEffect(() => {
31+
if (!applyDefault) {
32+
return;
33+
}
34+
// Set default dataset when values are fetched
35+
if (!value) {
36+
if (state.value && state.value[0]) {
37+
onChange(state.value[0]);
38+
}
39+
} else {
40+
if (state.value && state.value.find((v) => v.value === value) === undefined) {
41+
// if value is set and newly fetched values does not contain selected value
42+
if (state.value.length > 0) {
43+
onChange(state.value[0]);
44+
}
45+
}
46+
}
47+
}, [state.value, value, applyDefault, onChange]);
48+
49+
return (
50+
<Select
51+
className={className}
52+
aria-label="Dataset selector"
53+
value={value}
54+
options={state.value}
55+
onChange={onChange}
56+
disabled={disabled}
57+
isLoading={state.loading}
58+
menuShouldPortal={true}
59+
/>
60+
);
61+
};
+73
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,73 @@
1+
import { css } from '@emotion/css';
2+
import React, { ComponentProps } from 'react';
3+
4+
import { GrafanaTheme2 } from '@grafana/data';
5+
import { Space } from './Space';
6+
import { Field, Icon, PopoverContent, ReactUtils, stylesFactory, Tooltip, useTheme2 } from '@grafana/ui';
7+
8+
interface EditorFieldProps extends ComponentProps<typeof Field> {
9+
label: string;
10+
children: React.ReactElement;
11+
width?: number | string;
12+
optional?: boolean;
13+
tooltip?: PopoverContent;
14+
}
15+
16+
export const EditorField: React.FC<EditorFieldProps> = (props) => {
17+
const { label, optional, tooltip, children, width, ...fieldProps } = props;
18+
19+
const theme = useTheme2();
20+
const styles = getStyles(theme, width);
21+
22+
// Null check for backward compatibility
23+
const childInputId = fieldProps?.htmlFor || ReactUtils?.getChildId(children);
24+
25+
const labelEl = (
26+
<>
27+
<label className={styles.label} htmlFor={childInputId}>
28+
{label}
29+
{optional && <span className={styles.optional}> - optional</span>}
30+
{tooltip && (
31+
<Tooltip placement="top" content={tooltip} theme="info">
32+
<Icon name="info-circle" size="sm" className={styles.icon} />
33+
</Tooltip>
34+
)}
35+
</label>
36+
<Space v={0.5} />
37+
</>
38+
);
39+
40+
return (
41+
<div className={styles.root}>
42+
<Field className={styles.field} label={labelEl} {...fieldProps}>
43+
{children}
44+
</Field>
45+
</div>
46+
);
47+
};
48+
49+
const getStyles = stylesFactory((theme: GrafanaTheme2, width?: number | string) => {
50+
return {
51+
root: css({
52+
minWidth: theme.spacing(width ?? 0),
53+
}),
54+
label: css({
55+
fontSize: 12,
56+
fontWeight: theme.typography.fontWeightMedium,
57+
}),
58+
optional: css({
59+
fontStyle: 'italic',
60+
color: theme.colors.text.secondary,
61+
}),
62+
field: css({
63+
marginBottom: 0, // GrafanaUI/Field has a bottom margin which we must remove
64+
}),
65+
icon: css({
66+
color: theme.colors.text.secondary,
67+
marginLeft: theme.spacing(1),
68+
':hover': {
69+
color: theme.colors.text.primary,
70+
},
71+
}),
72+
};
73+
});
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,9 @@
1+
import React from 'react';
2+
3+
import { Stack } from './Stack';
4+
5+
interface EditorFieldGroupProps {}
6+
7+
export const EditorFieldGroup: React.FC<EditorFieldGroupProps> = ({ children }) => {
8+
return <Stack gap={1}>{children}</Stack>;
9+
};
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,24 @@
1+
import { css } from '@emotion/css';
2+
import React from 'react';
3+
4+
import { GrafanaTheme2 } from '@grafana/data';
5+
import { useTheme2, stylesFactory } from '@grafana/ui';
6+
7+
interface EditorHeaderProps {}
8+
9+
export const EditorHeader: React.FC<EditorHeaderProps> = ({ children }) => {
10+
const theme = useTheme2();
11+
const styles = getStyles(theme);
12+
13+
return <div className={styles.root}>{children}</div>;
14+
};
15+
16+
const getStyles = stylesFactory((theme: GrafanaTheme2) => ({
17+
root: css({
18+
display: 'flex',
19+
flexWrap: 'wrap',
20+
alignItems: 'center',
21+
gap: theme.spacing(3),
22+
minHeight: theme.spacing(4),
23+
}),
24+
}));
+48
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,48 @@
1+
import { Button } from '@grafana/ui';
2+
import React from 'react';
3+
4+
import { Stack } from './Stack';
5+
6+
interface EditorListProps<T> {
7+
items: Array<Partial<T>>;
8+
renderItem: (
9+
item: Partial<T>,
10+
onChangeItem: (item: Partial<T>) => void,
11+
onDeleteItem: () => void
12+
) => React.ReactElement;
13+
onChange: (items: Array<Partial<T>>) => void;
14+
}
15+
16+
export function EditorList<T>({ items, renderItem, onChange }: EditorListProps<T>) {
17+
const onAddItem = () => {
18+
const newItems = [...items, {}];
19+
20+
onChange(newItems);
21+
};
22+
23+
const onChangeItem = (itemIndex: number, newItem: Partial<T>) => {
24+
const newItems = [...items];
25+
newItems[itemIndex] = newItem;
26+
onChange(newItems);
27+
};
28+
29+
const onDeleteItem = (itemIndex: number) => {
30+
const newItems = [...items];
31+
newItems.splice(itemIndex, 1);
32+
onChange(newItems);
33+
};
34+
return (
35+
<Stack>
36+
{items.map((item, index) => (
37+
<div key={index}>
38+
{renderItem(
39+
item,
40+
(newItem) => onChangeItem(index, newItem),
41+
() => onDeleteItem(index)
42+
)}
43+
</div>
44+
))}
45+
<Button onClick={onAddItem} variant="secondary" size="md" icon="plus" aria-label="Add" type="button" />
46+
</Stack>
47+
);
48+
}

0 commit comments

Comments
 (0)