/* * Licensed to the Apache Software Foundation (ASF) under one * or more contributor license agreements. See the NOTICE file * distributed with this work for additional information * regarding copyright ownership. The ASF licenses this file * to you under the Apache License, Version 2.0 (the * "License"); you may not use this file except in compliance * with the License. You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. */ import { Intent } from '@blueprintjs/core'; import { IconNames } from '@blueprintjs/icons'; import axios from 'axios'; import * as classNames from 'classnames'; import * as React from 'react'; import { HashRouter, Route, Switch } from 'react-router-dom'; import { ExternalLink } from './components/external-link/external-link'; import { HeaderActiveTab, HeaderBar } from './components/header-bar/header-bar'; import { Loader } from './components/loader/loader'; import { AppToaster } from './singletons/toaster'; import { UrlBaser } from './singletons/url-baser'; import { QueryManager } from './utils'; import { DRUID_DOCS_API, DRUID_DOCS_SQL } from './variables'; import { DatasourcesView } from './views/datasource-view/datasource-view'; import { HomeView } from './views/home-view/home-view'; import { LoadDataView } from './views/load-data-view/load-data-view'; import { LookupsView } from './views/lookups-view/lookups-view'; import { SegmentsView } from './views/segments-view/segments-view'; import { ServersView } from './views/servers-view/servers-view'; import { SqlView } from './views/sql-view/sql-view'; import { TasksView } from './views/task-view/tasks-view'; import './console-application.scss'; export interface ConsoleApplicationProps extends React.Props { hideLegacy: boolean; baseURL?: string; customHeaderName?: string; customHeaderValue?: string; } export interface ConsoleApplicationState { aboutDialogOpen: boolean; noSqlMode: boolean; capabilitiesLoading: boolean; } export class ConsoleApplication extends React.Component { static MESSAGE_KEY = 'druid-console-message'; static MESSAGE_DISMISSED = 'dismissed'; private capabilitiesQueryManager: QueryManager; static async discoverCapabilities(): Promise<'working-with-sql' | 'working-without-sql' | 'broken'> { try { await axios.post('/druid/v2/sql', { query: 'SELECT 1337' }); } catch (e) { const { response } = e; if (response.status !== 405 || response.statusText !== 'Method Not Allowed') return 'working-with-sql'; // other failure try { await axios.get('/status'); } catch (e) { return 'broken'; // total failure } // Status works but SQL 405s => the SQL endpoint is disabled return 'working-without-sql'; } return 'working-with-sql'; } static shownNotifications(capabilities: string) { let message: JSX.Element = <>; /* tslint:disable:jsx-alignment */ if (capabilities === 'working-without-sql') { message = <> It appears that the SQL endpoint is disabled. The console will fall back to native Druid APIs and will be limited in functionality. Look at the SQL docs to enable the SQL endpoint. ; } else if (capabilities === 'broken') { message = <> It appears that the Druid is not responding. Data cannot be retrieved right now ; } /* tslint:enable:jsx-alignment */ AppToaster.show({ icon: IconNames.ERROR, intent: Intent.DANGER, timeout: 120000, message: message }); } private supervisorId: string | null; private taskId: string | null; private openDialog: string | null; private datasource: string | null; private onlyUnavailable: boolean | null; private initSql: string | null; private middleManager: string | null; constructor(props: ConsoleApplicationProps, context: any) { super(props, context); this.state = { aboutDialogOpen: false, noSqlMode: false, capabilitiesLoading: true }; if (props.baseURL) { axios.defaults.baseURL = props.baseURL; UrlBaser.baseURL = props.baseURL; } if (props.customHeaderName && props.customHeaderValue) { axios.defaults.headers.common[props.customHeaderName] = props.customHeaderValue; } this.capabilitiesQueryManager = new QueryManager({ processQuery: async (query: string) => { const capabilities = await ConsoleApplication.discoverCapabilities(); if (capabilities !== 'working-with-sql') { ConsoleApplication.shownNotifications(capabilities); } return capabilities; }, onStateChange: ({ result, loading, error }) => { this.setState({ noSqlMode: result === 'working-with-sql' ? false : true, capabilitiesLoading: loading }); } }); } componentDidMount(): void { this.capabilitiesQueryManager.runQuery('dummy'); } componentWillUnmount(): void { this.capabilitiesQueryManager.terminate(); } private resetInitialsWithDelay() { setTimeout(() => { this.taskId = null; this.supervisorId = null; this.openDialog = null; this.datasource = null; this.onlyUnavailable = null; this.initSql = null; this.middleManager = null; }, 50); } private goToLoadDataView = (supervisorId?: string, taskId?: string ) => { if (taskId) this.taskId = taskId; if (supervisorId) this.supervisorId = supervisorId; window.location.hash = 'load-data'; this.resetInitialsWithDelay(); } private goToTask = (taskId: string | null, openDialog?: string) => { this.taskId = taskId; if (openDialog) this.openDialog = openDialog; window.location.hash = 'tasks'; this.resetInitialsWithDelay(); } private goToSegments = (datasource: string, onlyUnavailable = false) => { this.datasource = `"${datasource}"`; this.onlyUnavailable = onlyUnavailable; window.location.hash = 'segments'; this.resetInitialsWithDelay(); } private goToMiddleManager = (middleManager: string) => { this.middleManager = middleManager; window.location.hash = 'servers'; this.resetInitialsWithDelay(); } private goToSql = (initSql: string) => { this.initSql = initSql; window.location.hash = 'query'; this.resetInitialsWithDelay(); } private wrapInViewContainer = (active: HeaderActiveTab, el: JSX.Element, classType: 'normal' | 'narrow-pad' = 'normal') => { const { hideLegacy } = this.props; return <>
{el}
; } private wrappedHomeView = () => { const { noSqlMode } = this.state; return this.wrapInViewContainer(null, ); } private wrappedLoadDataView = () => { return this.wrapInViewContainer('load-data', , 'narrow-pad'); } private wrappedSqlView = () => { return this.wrapInViewContainer('query', ); } private wrappedDatasourcesView = () => { const { noSqlMode } = this.state; return this.wrapInViewContainer('datasources', ); } private wrappedSegmentsView = () => { const { noSqlMode } = this.state; return this.wrapInViewContainer('segments', ); } private wrappedTasksView = () => { const { noSqlMode } = this.state; return this.wrapInViewContainer('tasks', ); } private wrappedServersView = () => { const { noSqlMode } = this.state; return this.wrapInViewContainer('servers', ); } private wrappedLookupsView = () => { return this.wrapInViewContainer('lookups', ); } render() { const { capabilitiesLoading } = this.state; if (capabilitiesLoading) { return
; } return
; } }