import { ActionIcon, Box, Button, Group, InputBase, Loader, Menu, Modal, rem, Skeleton, useComputedColorScheme, } from "@mantine/core"; import { CompleteStrategy, PromQLExtension, newCompleteStrategy, } from "@prometheus-io/codemirror-promql"; import { FC, Suspense, useEffect, useMemo, useRef, useState } from "react"; import CodeMirror, { EditorState, EditorView, Prec, ReactCodeMirrorRef, highlightSpecialChars, keymap, placeholder, } from "@uiw/react-codemirror"; import { baseTheme, darkPromqlHighlighter, darkTheme, lightTheme, promqlHighlighter, } from "../../codemirror/theme"; import { bracketMatching, indentOnInput, syntaxHighlighting, syntaxTree, } from "@codemirror/language"; import classes from "./ExpressionInput.module.css"; import { CompletionContext, CompletionResult, autocompletion, closeBrackets, closeBracketsKeymap, completionKeymap, } from "@codemirror/autocomplete"; import { defaultKeymap, history, historyKeymap, insertNewlineAndIndent, } from "@codemirror/commands"; import { highlightSelectionMatches } from "@codemirror/search"; import { lintKeymap } from "@codemirror/lint"; import { IconAlignJustified, IconDotsVertical, IconSearch, IconTerminal, IconTrash, } from "@tabler/icons-react"; import { useAPIQuery } from "../../api/api"; import { notifications } from "@mantine/notifications"; import { useSettings } from "../../state/settingsSlice"; import MetricsExplorer from "./MetricsExplorer/MetricsExplorer"; import ErrorBoundary from "../../components/ErrorBoundary"; const promqlExtension = new PromQLExtension(); // Autocompletion strategy that wraps the main one and enriches // it with past query items. export class HistoryCompleteStrategy implements CompleteStrategy { private complete: CompleteStrategy; private queryHistory: string[]; constructor(complete: CompleteStrategy, queryHistory: string[]) { this.complete = complete; this.queryHistory = queryHistory; } promQL( context: CompletionContext ): Promise | CompletionResult | null { return Promise.resolve(this.complete.promQL(context)).then((res) => { const { state, pos } = context; const tree = syntaxTree(state).resolve(pos, -1); const start = res != null ? res.from : tree.from; if (start !== 0) { return res; } const historyItems: CompletionResult = { from: start, to: pos, options: this.queryHistory.map((q) => ({ label: q.length < 80 ? q : q.slice(0, 76).concat("..."), detail: "past query", apply: q, info: q.length < 80 ? undefined : q, })), validFor: /^[a-zA-Z0-9_:]+$/, }; if (res !== null) { historyItems.options = historyItems.options.concat(res.options); } return historyItems; }); } } interface ExpressionInputProps { initialExpr: string; metricNames: string[]; executeQuery: (expr: string) => void; removePanel: () => void; } const ExpressionInput: FC = ({ initialExpr, metricNames, executeQuery, removePanel, }) => { const theme = useComputedColorScheme(); const { pathPrefix, enableAutocomplete, enableSyntaxHighlighting, enableLinter, } = useSettings(); const [expr, setExpr] = useState(initialExpr); useEffect(() => { setExpr(initialExpr); }, [initialExpr]); const { data: formatResult, error: formatError, isFetching: isFormatting, refetch: formatQuery, } = useAPIQuery({ key: expr, path: "/format_query", params: { query: expr, }, enabled: false, }); useEffect(() => { if (formatError) { notifications.show({ color: "red", title: "Error formatting query", message: formatError.message, }); return; } if (formatResult) { setExpr(formatResult.data); notifications.show({ title: "Expression formatted", message: "Expression formatted successfully!", }); } }, [formatResult, formatError]); const cmRef = useRef(null); const [showMetricsExplorer, setShowMetricsExplorer] = useState(false); // TODO: Implement query history. // This is just a placeholder until query history is implemented, so disable the linter warning. const queryHistory = useMemo(() => [], []); // (Re)initialize editor based on settings / setting changes. useEffect(() => { // Build the dynamic part of the config. promqlExtension .activateCompletion(enableAutocomplete) .activateLinter(enableLinter) .setComplete({ completeStrategy: new HistoryCompleteStrategy( newCompleteStrategy({ remote: { url: pathPrefix, cache: { initialMetricList: metricNames }, }, }), queryHistory ), }); }, [pathPrefix, metricNames, enableAutocomplete, enableLinter, queryHistory]); // TODO: Make this depend on external settings changes, maybe use dynamic config compartment again. return ( {/* eslint-disable-next-line @typescript-eslint/no-explicit-any */} leftSection={ isFormatting ? : } rightSection={ Query options } onClick={() => setShowMetricsExplorer(true)} > Explore metrics } onClick={() => formatQuery()} disabled={ isFormatting || expr === "" || expr === formatResult?.data } > Format expression } onClick={removePanel} > Remove query } component={CodeMirror} className={classes.input} basicSetup={false} value={expr} onChange={setExpr} autoFocus ref={cmRef} extensions={[ baseTheme, highlightSpecialChars(), history(), EditorState.allowMultipleSelections.of(true), indentOnInput(), bracketMatching(), closeBrackets(), autocompletion(), highlightSelectionMatches(), EditorView.lineWrapping, keymap.of([ ...closeBracketsKeymap, ...defaultKeymap, ...historyKeymap, ...completionKeymap, ...lintKeymap, ]), placeholder("Enter expression (press Shift+Enter for newlines)"), enableSyntaxHighlighting ? syntaxHighlighting( theme === "light" ? promqlHighlighter : darkPromqlHighlighter ) : [], promqlExtension.asExtension(), theme === "light" ? lightTheme : darkTheme, keymap.of([ { key: "Escape", run: (v: EditorView): boolean => { v.contentDOM.blur(); return false; }, }, ]), Prec.highest( keymap.of([ { key: "Enter", run: (): boolean => { executeQuery(expr); return true; }, }, { key: "Shift-Enter", run: insertNewlineAndIndent, }, ]) ), ]} multiline /> setShowMetricsExplorer(false)} title="Explore metrics" > {Array.from(Array(20), (_, i) => ( ))} } > { if (cmRef.current && cmRef.current.view) { const view = cmRef.current.view; view.dispatch( view.state.update({ changes: { from: view.state.selection.ranges[0].from, to: view.state.selection.ranges[0].to, insert: text, }, }) ); } }} close={() => setShowMetricsExplorer(false)} /> ); }; export default ExpressionInput;