From 8ad495a5723f72e048a8d11aa229eab46a294cc3 Mon Sep 17 00:00:00 2001 From: Kamil Gabryjelski Date: Thu, 26 Aug 2021 21:53:44 +0200 Subject: [PATCH 1/8] perf(dashboard): reduce number of rerenders of Charts (#16444) * perf(dashboard): reduce number of rerenders of Charts * Remove static property * Cleanup --- .../src/dashboard/actions/hydrate.js | 4 ++-- .../components/gridComponents/Chart.jsx | 3 ++- .../src/dashboard/containers/Chart.jsx | 6 +++--- .../charts/getFormDataWithExtraFilters.ts | 19 +++++++++++-------- 4 files changed, 18 insertions(+), 14 deletions(-) diff --git a/superset-frontend/src/dashboard/actions/hydrate.js b/superset-frontend/src/dashboard/actions/hydrate.js index 9c524e2d05be4..7b0e0ae144b1e 100644 --- a/superset-frontend/src/dashboard/actions/hydrate.js +++ b/superset-frontend/src/dashboard/actions/hydrate.js @@ -371,8 +371,8 @@ export const hydrateDashboard = (dashboardData, chartData) => ( // only persistent refreshFrequency will be saved to backend shouldPersistRefreshFrequency: false, css: dashboardData.css || '', - colorNamespace: metadata?.color_namespace || null, - colorScheme: metadata?.color_scheme || null, + colorNamespace: metadata?.color_namespace, + colorScheme: metadata?.color_scheme, editMode: canEdit && editMode, isPublished: dashboardData.published, hasUnsavedChanges: false, diff --git a/superset-frontend/src/dashboard/components/gridComponents/Chart.jsx b/superset-frontend/src/dashboard/components/gridComponents/Chart.jsx index ae40888fcc4dc..60cbab7295303 100644 --- a/superset-frontend/src/dashboard/components/gridComponents/Chart.jsx +++ b/superset-frontend/src/dashboard/components/gridComponents/Chart.jsx @@ -87,7 +87,8 @@ const defaultProps = { // resizing across all slices on a dashboard on every update const RESIZE_TIMEOUT = 350; const SHOULD_UPDATE_ON_PROP_CHANGES = Object.keys(propTypes).filter( - prop => prop !== 'width' && prop !== 'height', + prop => + prop !== 'width' && prop !== 'height' && prop !== 'isComponentVisible', ); const OVERFLOWABLE_VIZ_TYPES = new Set(['filter_box']); const DEFAULT_HEADER_HEIGHT = 22; diff --git a/superset-frontend/src/dashboard/containers/Chart.jsx b/superset-frontend/src/dashboard/containers/Chart.jsx index 4fdfcfdb952cd..c629209d42541 100644 --- a/superset-frontend/src/dashboard/containers/Chart.jsx +++ b/superset-frontend/src/dashboard/containers/Chart.jsx @@ -37,7 +37,7 @@ import getFormDataWithExtraFilters from '../util/charts/getFormDataWithExtraFilt import Chart from '../components/gridComponents/Chart'; import { PLACEHOLDER_DATASOURCE } from '../constants'; -const EMPTY_FILTERS = {}; +const EMPTY_OBJECT = {}; function mapStateToProps( { @@ -54,7 +54,7 @@ function mapStateToProps( ownProps, ) { const { id } = ownProps; - const chart = chartQueries[id] || {}; + const chart = chartQueries[id] || EMPTY_OBJECT; const datasource = (chart && chart.form_data && datasources[chart.form_data.datasource]) || PLACEHOLDER_DATASOURCE; @@ -82,7 +82,7 @@ function mapStateToProps( datasource, slice: sliceEntities.slices[id], timeout: dashboardInfo.common.conf.SUPERSET_WEBSERVER_TIMEOUT, - filters: getActiveFilters() || EMPTY_FILTERS, + filters: getActiveFilters() || EMPTY_OBJECT, formData, editMode: dashboardState.editMode, isExpanded: !!dashboardState.expandedSlices[id], diff --git a/superset-frontend/src/dashboard/util/charts/getFormDataWithExtraFilters.ts b/superset-frontend/src/dashboard/util/charts/getFormDataWithExtraFilters.ts index 48cfd8ffcddc8..2587588eac0c3 100644 --- a/superset-frontend/src/dashboard/util/charts/getFormDataWithExtraFilters.ts +++ b/superset-frontend/src/dashboard/util/charts/getFormDataWithExtraFilters.ts @@ -25,6 +25,7 @@ import { import { ChartQueryPayload, Charts, LayoutItem } from 'src/dashboard/types'; import { getExtraFormData } from 'src/dashboard/components/nativeFilters/utils'; import { DataMaskStateWithId } from 'src/dataMask/types'; +import { areObjectsEqual } from 'src/reduxUtils'; import getEffectiveExtraFilters from './getEffectiveExtraFilters'; import { ChartConfiguration, NativeFiltersState } from '../../reducers/types'; import { getAllActiveFilters } from '../activeAllDashboardFilters'; @@ -67,16 +68,18 @@ export default function getFormDataWithExtraFilters({ const labelColors = scale.getColorMap(); // if dashboard metadata + filters have not changed, use cache if possible + const cachedFormData = cachedFormdataByChart[sliceId]; if ( - (cachedFiltersByChart[sliceId] || {}) === filters && - (colorScheme == null || - cachedFormdataByChart[sliceId].color_scheme === colorScheme) && - cachedFormdataByChart[sliceId].color_namespace === colorNamespace && - isEqual(cachedFormdataByChart[sliceId].label_colors, labelColors) && - !!cachedFormdataByChart[sliceId] && - dataMask === undefined + cachedFiltersByChart[sliceId] === filters && + cachedFormData?.color_scheme === colorScheme && + cachedFormData?.color_namespace === colorNamespace && + isEqual(cachedFormData?.label_colors, labelColors) && + !!cachedFormData && + areObjectsEqual(cachedFormData?.dataMask, dataMask, { + ignoreUndefined: true, + }) ) { - return cachedFormdataByChart[sliceId]; + return cachedFormData; } let extraData: { extra_form_data?: JsonObject } = {}; From f422f1ea49310ad7a059a82c70ed385dbacd17d4 Mon Sep 17 00:00:00 2001 From: Kamil Gabryjelski Date: Thu, 26 Aug 2021 21:54:11 +0200 Subject: [PATCH 2/8] perf(dashboard): decouple redux props from dashboard components (#16421) * perf(dashboard): decouple redux props from dashboard components * Lint fix * Dont make copy of filters object * Remove unnecessary exports --- .../components/gridComponents/ChartHolder_spec.jsx | 4 ++-- .../components/gridComponents/Markdown_spec.jsx | 4 ++-- .../components/gridComponents/ChartHolder.jsx | 10 ++++++++-- .../dashboard/components/gridComponents/Markdown.jsx | 11 +++++++++-- .../src/dashboard/components/gridComponents/Tabs.jsx | 6 +++++- .../src/dashboard/containers/DashboardComponent.jsx | 8 -------- .../src/dashboard/util/activeDashboardFilters.js | 4 +--- 7 files changed, 27 insertions(+), 20 deletions(-) diff --git a/superset-frontend/spec/javascripts/dashboard/components/gridComponents/ChartHolder_spec.jsx b/superset-frontend/spec/javascripts/dashboard/components/gridComponents/ChartHolder_spec.jsx index 3f7d0bfaa2469..35b3cf3a42bc6 100644 --- a/superset-frontend/spec/javascripts/dashboard/components/gridComponents/ChartHolder_spec.jsx +++ b/superset-frontend/spec/javascripts/dashboard/components/gridComponents/ChartHolder_spec.jsx @@ -25,7 +25,7 @@ import { DndProvider } from 'react-dnd'; import { HTML5Backend } from 'react-dnd-html5-backend'; import Chart from 'src/dashboard/containers/Chart'; -import ChartHolder from 'src/dashboard/components/gridComponents/ChartHolder'; +import ChartHolderConnected from 'src/dashboard/components/gridComponents/ChartHolder'; import DeleteComponentButton from 'src/dashboard/components/DeleteComponentButton'; import DragDroppable from 'src/dashboard/components/dnd/DragDroppable'; import HoverMenu from 'src/dashboard/components/menu/HoverMenu'; @@ -71,7 +71,7 @@ describe('ChartHolder', () => { const wrapper = mount( - + , { diff --git a/superset-frontend/spec/javascripts/dashboard/components/gridComponents/Markdown_spec.jsx b/superset-frontend/spec/javascripts/dashboard/components/gridComponents/Markdown_spec.jsx index dc3f1b18b96a0..fa5ca73568009 100644 --- a/superset-frontend/spec/javascripts/dashboard/components/gridComponents/Markdown_spec.jsx +++ b/superset-frontend/spec/javascripts/dashboard/components/gridComponents/Markdown_spec.jsx @@ -26,7 +26,7 @@ import { HTML5Backend } from 'react-dnd-html5-backend'; import { act } from 'react-dom/test-utils'; import { MarkdownEditor } from 'src/components/AsyncAceEditor'; -import Markdown from 'src/dashboard/components/gridComponents/Markdown'; +import MarkdownConnected from 'src/dashboard/components/gridComponents/Markdown'; import MarkdownModeDropdown from 'src/dashboard/components/menu/MarkdownModeDropdown'; import DeleteComponentButton from 'src/dashboard/components/DeleteComponentButton'; import waitForComponentToPaint from 'spec/helpers/waitForComponentToPaint'; @@ -66,7 +66,7 @@ describe('Markdown', () => { const wrapper = mount( - + , ); diff --git a/superset-frontend/src/dashboard/components/gridComponents/ChartHolder.jsx b/superset-frontend/src/dashboard/components/gridComponents/ChartHolder.jsx index 23f46f292d9dd..87f1cbc78de40 100644 --- a/superset-frontend/src/dashboard/components/gridComponents/ChartHolder.jsx +++ b/superset-frontend/src/dashboard/components/gridComponents/ChartHolder.jsx @@ -20,7 +20,7 @@ import React from 'react'; import PropTypes from 'prop-types'; import cx from 'classnames'; import { useTheme } from '@superset-ui/core'; -import { useSelector } from 'react-redux'; +import { useSelector, connect } from 'react-redux'; import { getChartIdsInFilterScope } from 'src/dashboard/util/activeDashboardFilters'; import Chart from '../../containers/Chart'; @@ -381,4 +381,10 @@ class ChartHolder extends React.Component { ChartHolder.propTypes = propTypes; ChartHolder.defaultProps = defaultProps; -export default ChartHolder; +function mapStateToProps(state) { + return { + directPathToChild: state.dashboardState.directPathToChild, + directPathLastUpdated: state.dashboardState.directPathLastUpdated, + }; +} +export default connect(mapStateToProps)(ChartHolder); diff --git a/superset-frontend/src/dashboard/components/gridComponents/Markdown.jsx b/superset-frontend/src/dashboard/components/gridComponents/Markdown.jsx index 1465b92db1ee9..a21402c3ab24d 100644 --- a/superset-frontend/src/dashboard/components/gridComponents/Markdown.jsx +++ b/superset-frontend/src/dashboard/components/gridComponents/Markdown.jsx @@ -18,8 +18,9 @@ */ import React from 'react'; import PropTypes from 'prop-types'; - +import { connect } from 'react-redux'; import cx from 'classnames'; + import { t, SafeMarkdown } from '@superset-ui/core'; import { Logger, LOG_ACTIONS_RENDER_CHART } from 'src/logger/LogUtils'; import { MarkdownEditor } from 'src/components/AsyncAceEditor'; @@ -366,4 +367,10 @@ class Markdown extends React.PureComponent { Markdown.propTypes = propTypes; Markdown.defaultProps = defaultProps; -export default Markdown; +function mapStateToProps(state) { + return { + undoLength: state.dashboardLayout.past.length, + redoLength: state.dashboardLayout.future.length, + }; +} +export default connect(mapStateToProps)(Markdown); diff --git a/superset-frontend/src/dashboard/components/gridComponents/Tabs.jsx b/superset-frontend/src/dashboard/components/gridComponents/Tabs.jsx index a2a74829d0679..9916520c76116 100644 --- a/superset-frontend/src/dashboard/components/gridComponents/Tabs.jsx +++ b/superset-frontend/src/dashboard/components/gridComponents/Tabs.jsx @@ -405,6 +405,10 @@ Tabs.propTypes = propTypes; Tabs.defaultProps = defaultProps; function mapStateToProps(state) { - return { nativeFilters: state.nativeFilters }; + return { + nativeFilters: state.nativeFilters, + directPathToChild: state.dashboardState.directPathToChild, + activeTabs: state.dashboardState.activeTabs, + }; } export default connect(mapStateToProps)(Tabs); diff --git a/superset-frontend/src/dashboard/containers/DashboardComponent.jsx b/superset-frontend/src/dashboard/containers/DashboardComponent.jsx index c7ead005701cb..50497152f8509 100644 --- a/superset-frontend/src/dashboard/containers/DashboardComponent.jsx +++ b/superset-frontend/src/dashboard/containers/DashboardComponent.jsx @@ -63,8 +63,6 @@ const propTypes = { }; const defaultProps = { - directPathToChild: [], - directPathLastUpdated: 0, isComponentVisible: true, }; @@ -77,15 +75,9 @@ function mapStateToProps( const component = dashboardLayout[id]; const props = { component, - dashboardLayout, parentComponent: dashboardLayout[parentId], editMode: dashboardState.editMode, - undoLength: undoableLayout.past.length, - redoLength: undoableLayout.future.length, filters: getActiveFilters(), - directPathToChild: dashboardState.directPathToChild, - activeTabs: dashboardState.activeTabs, - directPathLastUpdated: dashboardState.directPathLastUpdated, dashboardId: dashboardInfo.id, fullSizeChartId: dashboardState.fullSizeChartId, }; diff --git a/superset-frontend/src/dashboard/util/activeDashboardFilters.js b/superset-frontend/src/dashboard/util/activeDashboardFilters.js index 96db2c3129411..30bdc2540aa4b 100644 --- a/superset-frontend/src/dashboard/util/activeDashboardFilters.js +++ b/superset-frontend/src/dashboard/util/activeDashboardFilters.js @@ -33,9 +33,7 @@ let allComponents = {}; // output: { [id_column]: { values, scope } } export function getActiveFilters() { - return { - ...activeFilters, - }; + return activeFilters; } // currently filter_box is a chart, From fd6456186df07a14fad2007b39a4085fe43dc626 Mon Sep 17 00:00:00 2001 From: Ke Zhu Date: Thu, 26 Aug 2021 18:16:30 -0400 Subject: [PATCH 3/8] docs: make code snippet usable with required imports (#16473) --- docs/src/pages/docs/installation/configuring.mdx | 6 +++++- 1 file changed, 5 insertions(+), 1 deletion(-) diff --git a/docs/src/pages/docs/installation/configuring.mdx b/docs/src/pages/docs/installation/configuring.mdx index d7437f270a3b0..922f97d7a2d4b 100644 --- a/docs/src/pages/docs/installation/configuring.mdx +++ b/docs/src/pages/docs/installation/configuring.mdx @@ -207,7 +207,11 @@ CUSTOM_SECURITY_MANAGER = CustomSsoSecurityManager the app object and can alter it in any way. For example, add `FLASK_APP_MUTATOR` into your `superset_config.py` to setup session cookie expiration time to 24 hours: -``` +```python +from flask import session +from flask import Flask + + def make_session_permanent(): ''' Enable maxAge for the cookie 'session' From ee2eccdb674c87347d60e80c8d43d315743a6d93 Mon Sep 17 00:00:00 2001 From: AAfghahi <48933336+AAfghahi@users.noreply.github.com> Date: Thu, 26 Aug 2021 18:20:02 -0400 Subject: [PATCH 4/8] fix: queryEditor bug (#16452) * queryEditor bug * update tests Co-authored-by: Elizabeth Thompson --- .../javascripts/sqllab/actions/sqlLab_spec.js | 40 ++++++++++++++----- .../src/SqlLab/actions/sqlLab.js | 17 ++++---- 2 files changed, 37 insertions(+), 20 deletions(-) diff --git a/superset-frontend/spec/javascripts/sqllab/actions/sqlLab_spec.js b/superset-frontend/spec/javascripts/sqllab/actions/sqlLab_spec.js index 44e468cf57e81..a3f7e5b8df435 100644 --- a/superset-frontend/spec/javascripts/sqllab/actions/sqlLab_spec.js +++ b/superset-frontend/spec/javascripts/sqllab/actions/sqlLab_spec.js @@ -399,7 +399,7 @@ describe('async actions', () => { let isFeatureEnabledMock; - beforeAll(() => { + beforeEach(() => { isFeatureEnabledMock = jest .spyOn(featureFlags, 'isFeatureEnabled') .mockImplementation( @@ -407,7 +407,7 @@ describe('async actions', () => { ); }); - afterAll(() => { + afterEach(() => { isFeatureEnabledMock.mockRestore(); }); @@ -612,9 +612,29 @@ describe('async actions', () => { }); describe('queryEditorSetSql', () => { + describe('with backend persistence flag on', () => { + it('does not update the tab state in the backend', () => { + expect.assertions(2); + + const sql = 'SELECT * '; + const store = mockStore({}); + + return store + .dispatch(actions.queryEditorSetSql(queryEditor, sql)) + .then(() => { + expect(store.getActions()).toHaveLength(0); + expect(fetchMock.calls(updateTabStateEndpoint)).toHaveLength(1); + }); + }); + }); + }); + describe('with backend persistence flag off', () => { it('updates the tab state in the backend', () => { - expect.assertions(2); - + const backendPersistenceOffMock = jest + .spyOn(featureFlags, 'isFeatureEnabled') + .mockImplementation( + feature => !(feature === 'SQLLAB_BACKEND_PERSISTENCE'), + ); const sql = 'SELECT * '; const store = mockStore({}); const expectedActions = [ @@ -624,12 +644,12 @@ describe('async actions', () => { sql, }, ]; - return store - .dispatch(actions.queryEditorSetSql(queryEditor, sql)) - .then(() => { - expect(store.getActions()).toEqual(expectedActions); - expect(fetchMock.calls(updateTabStateEndpoint)).toHaveLength(1); - }); + + store.dispatch(actions.queryEditorSetSql(queryEditor, sql)); + + expect(store.getActions()).toEqual(expectedActions); + expect(fetchMock.calls(updateTabStateEndpoint)).toHaveLength(0); + backendPersistenceOffMock.mockRestore(); }); }); diff --git a/superset-frontend/src/SqlLab/actions/sqlLab.js b/superset-frontend/src/SqlLab/actions/sqlLab.js index 94391e0ac8a3c..56353ce175702 100644 --- a/superset-frontend/src/SqlLab/actions/sqlLab.js +++ b/superset-frontend/src/SqlLab/actions/sqlLab.js @@ -898,16 +898,11 @@ export function updateSavedQuery(query) { export function queryEditorSetSql(queryEditor, sql) { return function (dispatch) { - const sync = isFeatureEnabled(FeatureFlag.SQLLAB_BACKEND_PERSISTENCE) - ? SupersetClient.put({ - endpoint: encodeURI(`/tabstateview/${queryEditor.id}`), - postPayload: { sql, latest_query_id: queryEditor.latestQueryId }, - }) - : Promise.resolve(); - - return sync - .then(() => dispatch({ type: QUERY_EDITOR_SET_SQL, queryEditor, sql })) - .catch(() => + if (isFeatureEnabled(FeatureFlag.SQLLAB_BACKEND_PERSISTENCE)) { + return SupersetClient.put({ + endpoint: encodeURI(`/tabstateview/${queryEditor.id}`), + postPayload: { sql, latest_query_id: queryEditor.latestQueryId }, + }).catch(() => dispatch( addDangerToast( t( @@ -918,6 +913,8 @@ export function queryEditorSetSql(queryEditor, sql) { ), ), ); + } + return dispatch({ type: QUERY_EDITOR_SET_SQL, queryEditor, sql }); }; } From a413f796a68b0cdb656b3a35a552d407cd3cdfbb Mon Sep 17 00:00:00 2001 From: Jesse Yang Date: Thu, 26 Aug 2021 17:04:45 -0700 Subject: [PATCH 5/8] fix(explore): JS error for creating new metrics from columns (#16477) --- .../controls/DndColumnSelectControl/DndMetricSelect.tsx | 4 +--- 1 file changed, 1 insertion(+), 3 deletions(-) diff --git a/superset-frontend/src/explore/components/controls/DndColumnSelectControl/DndMetricSelect.tsx b/superset-frontend/src/explore/components/controls/DndColumnSelectControl/DndMetricSelect.tsx index 0e71f18eacefd..2fcf53f8a87bf 100644 --- a/superset-frontend/src/explore/components/controls/DndColumnSelectControl/DndMetricSelect.tsx +++ b/superset-frontend/src/explore/components/controls/DndColumnSelectControl/DndMetricSelect.tsx @@ -186,9 +186,7 @@ export const DndMetricSelect = (props: any) => { const onNewMetric = useCallback( (newMetric: Metric) => { - const newValue = props.multi - ? [...value, newMetric.metric_name] - : [newMetric.metric_name]; + const newValue = props.multi ? [...value, newMetric] : [newMetric]; setValue(newValue); handleChange(newValue); }, From 8adc31d14c3b711cad6eb08a5223e06ef5fb723b Mon Sep 17 00:00:00 2001 From: Erik Ritter Date: Thu, 26 Aug 2021 18:28:04 -0700 Subject: [PATCH 6/8] Revert "chore: Changes the DatabaseSelector to use the new Select component (#16334)" (#16478) This reverts commit c768941f2f662e1a0dfa1e1731319d22ec9ca886. --- .../sqllab/SqlEditorLeftBar_spec.jsx | 10 +- .../src/components/CertifiedIcon/index.tsx | 9 +- .../DatabaseSelector.test.tsx | 62 +-- .../src/components/DatabaseSelector/index.tsx | 310 +++++------ .../src/components/Icons/Icon.tsx | 8 +- .../src/components/Select/Select.tsx | 76 ++- .../TableSelector/TableSelector.test.jsx | 291 +++++++++++ .../TableSelector/TableSelector.test.tsx | 91 ---- .../src/components/TableSelector/index.tsx | 484 +++++++++--------- .../WarningIconWithTooltip/index.tsx | 6 +- .../src/datasource/DatasourceEditor.jsx | 132 +++-- .../controls/DatasourceControl/index.jsx | 5 +- .../views/CRUD/data/dataset/DatasetList.tsx | 2 - superset/datasets/api.py | 2 +- superset/views/core.py | 21 +- tests/integration_tests/datasets/api_tests.py | 14 +- 16 files changed, 831 insertions(+), 692 deletions(-) create mode 100644 superset-frontend/src/components/TableSelector/TableSelector.test.jsx delete mode 100644 superset-frontend/src/components/TableSelector/TableSelector.test.tsx diff --git a/superset-frontend/spec/javascripts/sqllab/SqlEditorLeftBar_spec.jsx b/superset-frontend/spec/javascripts/sqllab/SqlEditorLeftBar_spec.jsx index b153c148394ce..1ba1ac821fb1e 100644 --- a/superset-frontend/spec/javascripts/sqllab/SqlEditorLeftBar_spec.jsx +++ b/superset-frontend/spec/javascripts/sqllab/SqlEditorLeftBar_spec.jsx @@ -81,13 +81,9 @@ describe('Left Panel Expansion', () => { , ); - const dbSelect = screen.getByRole('combobox', { - name: 'Select a database', - }); - const schemaSelect = screen.getByRole('combobox', { - name: 'Select a schema', - }); - const dropdown = screen.getByText(/Select a table/i); + const dbSelect = screen.getByText(/select a database/i); + const schemaSelect = screen.getByText(/select a schema \(0\)/i); + const dropdown = screen.getByText(/Select table/i); const abUser = screen.getByText(/ab_user/i); expect(dbSelect).toBeInTheDocument(); expect(schemaSelect).toBeInTheDocument(); diff --git a/superset-frontend/src/components/CertifiedIcon/index.tsx b/superset-frontend/src/components/CertifiedIcon/index.tsx index 4aa0dad236b12..f08e9bf6047ce 100644 --- a/superset-frontend/src/components/CertifiedIcon/index.tsx +++ b/superset-frontend/src/components/CertifiedIcon/index.tsx @@ -18,19 +18,19 @@ */ import React from 'react'; import { t, supersetTheme } from '@superset-ui/core'; -import Icons, { IconType } from 'src/components/Icons'; +import Icons from 'src/components/Icons'; import { Tooltip } from 'src/components/Tooltip'; export interface CertifiedIconProps { certifiedBy?: string; details?: string; - size?: IconType['iconSize']; + size?: number; } function CertifiedIcon({ certifiedBy, details, - size = 'l', + size = 24, }: CertifiedIconProps) { return ( ); diff --git a/superset-frontend/src/components/DatabaseSelector/DatabaseSelector.test.tsx b/superset-frontend/src/components/DatabaseSelector/DatabaseSelector.test.tsx index 6d4abb3fd9634..0d812824d1cf8 100644 --- a/superset-frontend/src/components/DatabaseSelector/DatabaseSelector.test.tsx +++ b/superset-frontend/src/components/DatabaseSelector/DatabaseSelector.test.tsx @@ -26,11 +26,11 @@ import DatabaseSelector from '.'; const SupersetClientGet = jest.spyOn(SupersetClient, 'get'); const createProps = () => ({ - db: { id: 1, database_name: 'test', backend: 'postgresql' }, + dbId: 1, formMode: false, isDatabaseSelectEnabled: true, readOnly: false, - schema: undefined, + schema: 'public', sqlLabMode: true, getDbList: jest.fn(), getTableList: jest.fn(), @@ -129,7 +129,7 @@ beforeEach(() => { changed_on: '2021-03-09T19:02:07.141095', changed_on_delta_humanized: 'a day ago', created_by: null, - database_name: 'test', + database_name: 'examples', explore_database_id: 1, expose_in_sqllab: true, force_ctas_schema: null, @@ -153,62 +153,50 @@ test('Refresh should work', async () => { render(); - const select = screen.getByRole('combobox', { - name: 'Select a schema', - }); - - userEvent.click(select); - await waitFor(() => { - expect(SupersetClientGet).toBeCalledTimes(1); - expect(props.getDbList).toBeCalledTimes(0); + expect(SupersetClientGet).toBeCalledTimes(2); + expect(props.getDbList).toBeCalledTimes(1); expect(props.getTableList).toBeCalledTimes(0); expect(props.handleError).toBeCalledTimes(0); expect(props.onDbChange).toBeCalledTimes(0); expect(props.onSchemaChange).toBeCalledTimes(0); - expect(props.onSchemasLoad).toBeCalledTimes(0); + expect(props.onSchemasLoad).toBeCalledTimes(1); expect(props.onUpdate).toBeCalledTimes(0); }); - userEvent.click(screen.getByRole('button', { name: 'refresh' })); + userEvent.click(screen.getByRole('button')); await waitFor(() => { - expect(SupersetClientGet).toBeCalledTimes(2); - expect(props.getDbList).toBeCalledTimes(0); + expect(SupersetClientGet).toBeCalledTimes(3); + expect(props.getDbList).toBeCalledTimes(1); expect(props.getTableList).toBeCalledTimes(0); expect(props.handleError).toBeCalledTimes(0); - expect(props.onDbChange).toBeCalledTimes(0); - expect(props.onSchemaChange).toBeCalledTimes(0); + expect(props.onDbChange).toBeCalledTimes(1); + expect(props.onSchemaChange).toBeCalledTimes(1); expect(props.onSchemasLoad).toBeCalledTimes(2); - expect(props.onUpdate).toBeCalledTimes(0); + expect(props.onUpdate).toBeCalledTimes(1); }); }); test('Should database select display options', async () => { const props = createProps(); render(); - const select = screen.getByRole('combobox', { - name: 'Select a database', - }); - expect(select).toBeInTheDocument(); - userEvent.click(select); - expect( - await screen.findByRole('option', { name: 'postgresql: test' }), - ).toBeInTheDocument(); + const selector = await screen.findByText('Database:'); + expect(selector).toBeInTheDocument(); + expect(selector.parentElement).toHaveTextContent( + 'Database:postgresql examples', + ); }); test('Should schema select display options', async () => { const props = createProps(); render(); - const select = screen.getByRole('combobox', { - name: 'Select a schema', - }); - expect(select).toBeInTheDocument(); - userEvent.click(select); - expect( - await screen.findByRole('option', { name: 'public' }), - ).toBeInTheDocument(); - expect( - await screen.findByRole('option', { name: 'information_schema' }), - ).toBeInTheDocument(); + + const selector = await screen.findByText('Schema:'); + expect(selector).toBeInTheDocument(); + expect(selector.parentElement).toHaveTextContent('Schema: public'); + + userEvent.click(screen.getByRole('button')); + + expect(await screen.findByText('Select a schema (2)')).toBeInTheDocument(); }); diff --git a/superset-frontend/src/components/DatabaseSelector/index.tsx b/superset-frontend/src/components/DatabaseSelector/index.tsx index c96fba7c81e3c..0282e4a4a4c08 100644 --- a/superset-frontend/src/components/DatabaseSelector/index.tsx +++ b/superset-frontend/src/components/DatabaseSelector/index.tsx @@ -16,51 +16,58 @@ * specific language governing permissions and limitations * under the License. */ -import React, { ReactNode, useState, useMemo } from 'react'; +import React, { ReactNode, useEffect, useState } from 'react'; import { styled, SupersetClient, t } from '@superset-ui/core'; import rison from 'rison'; -import { Select } from 'src/components'; -import { FormLabel } from 'src/components/Form'; +import { Select } from 'src/components/Select'; +import Label from 'src/components/Label'; import RefreshLabel from 'src/components/RefreshLabel'; +import SupersetAsyncSelect from 'src/components/AsyncSelect'; + +const FieldTitle = styled.p` + color: ${({ theme }) => theme.colors.secondary.light2}; + font-size: ${({ theme }) => theme.typography.sizes.s}px; + margin: 20px 0 10px 0; + text-transform: uppercase; +`; const DatabaseSelectorWrapper = styled.div` - ${({ theme }) => ` - .refresh { - display: flex; - align-items: center; - width: 30px; - margin-left: ${theme.gridUnit}px; - margin-top: ${theme.gridUnit * 5}px; - } + .fa-refresh { + padding-left: 9px; + } - .section { - display: flex; - flex-direction: row; - align-items: center; - } + .refresh-col { + display: flex; + align-items: center; + width: 30px; + margin-left: ${({ theme }) => theme.gridUnit}px; + } - .select { - flex: 1; - } + .section { + padding-bottom: 5px; + display: flex; + flex-direction: row; + } - & > div { - margin-bottom: ${theme.gridUnit * 4}px; - } - `} + .select { + flex-grow: 1; + } `; -type DatabaseValue = { label: string; value: number }; - -type SchemaValue = { label: string; value: string }; +const DatabaseOption = styled.span` + display: inline-flex; + align-items: center; +`; interface DatabaseSelectorProps { - db?: { id: number; database_name: string; backend: string }; + dbId: number; formMode?: boolean; getDbList?: (arg0: any) => {}; + getTableList?: (dbId: number, schema: string, force: boolean) => {}; handleError: (msg: string) => void; isDatabaseSelectEnabled?: boolean; onDbChange?: (db: any) => void; - onSchemaChange?: (schema?: string) => void; + onSchemaChange?: (arg0?: any) => {}; onSchemasLoad?: (schemas: Array) => void; readOnly?: boolean; schema?: string; @@ -76,9 +83,10 @@ interface DatabaseSelectorProps { } export default function DatabaseSelector({ - db, + dbId, formMode = false, getDbList, + getTableList, handleError, isDatabaseSelectEnabled = true, onUpdate, @@ -89,189 +97,193 @@ export default function DatabaseSelector({ schema, sqlLabMode = false, }: DatabaseSelectorProps) { - const [currentDb, setCurrentDb] = useState( - db - ? { label: `${db.backend}: ${db.database_name}`, value: db.id } - : undefined, - ); - const [currentSchema, setCurrentSchema] = useState( - schema ? { label: schema, value: schema } : undefined, + const [currentDbId, setCurrentDbId] = useState(dbId); + const [currentSchema, setCurrentSchema] = useState( + schema, ); - const [refresh, setRefresh] = useState(0); + const [schemaLoading, setSchemaLoading] = useState(false); + const [schemaOptions, setSchemaOptions] = useState([]); - const loadSchemas = useMemo( - () => async (): Promise<{ - data: SchemaValue[]; - totalCount: number; - }> => { - if (currentDb) { - const queryParams = rison.encode({ force: refresh > 0 }); - const endpoint = `/api/v1/database/${currentDb.value}/schemas/?q=${queryParams}`; - - // TODO: Would be nice to add pagination in a follow-up. Needs endpoint changes. - return SupersetClient.get({ endpoint }).then(({ json }) => { + function fetchSchemas(databaseId: number, forceRefresh = false) { + const actualDbId = databaseId || dbId; + if (actualDbId) { + setSchemaLoading(true); + const queryParams = rison.encode({ + force: Boolean(forceRefresh), + }); + const endpoint = `/api/v1/database/${actualDbId}/schemas/?q=${queryParams}`; + return SupersetClient.get({ endpoint }) + .then(({ json }) => { const options = json.result.map((s: string) => ({ value: s, label: s, title: s, })); + setSchemaOptions(options); + setSchemaLoading(false); if (onSchemasLoad) { onSchemasLoad(options); } - return { - data: options, - totalCount: options.length, - }; + }) + .catch(() => { + setSchemaOptions([]); + setSchemaLoading(false); + handleError(t('Error while fetching schema list')); }); - } - return { - data: [], - totalCount: 0, - }; - }, - [currentDb, refresh, onSchemasLoad], - ); + } + return Promise.resolve(); + } - function onSelectChange({ - db, - schema, - }: { - db: DatabaseValue; - schema?: SchemaValue; - }) { - setCurrentDb(db); + useEffect(() => { + if (currentDbId) { + fetchSchemas(currentDbId); + } + }, [currentDbId]); + + function onSelectChange({ dbId, schema }: { dbId: number; schema?: string }) { + setCurrentDbId(dbId); setCurrentSchema(schema); if (onUpdate) { - onUpdate({ - dbId: db.value, - schema: schema?.value, - tableName: undefined, - }); + onUpdate({ dbId, schema, tableName: undefined }); } } - function changeDataBase(selectedValue: DatabaseValue) { - const actualDb = selectedValue || db; + function dbMutator(data: any) { + if (getDbList) { + getDbList(data.result); + } + if (data.result.length === 0) { + handleError(t("It seems you don't have access to any database")); + } + return data.result.map((row: any) => ({ + ...row, + // label is used for the typeahead + label: `${row.backend} ${row.database_name}`, + })); + } + + function changeDataBase(db: any, force = false) { + const dbId = db ? db.id : null; + setSchemaOptions([]); if (onSchemaChange) { - onSchemaChange(undefined); + onSchemaChange(null); } if (onDbChange) { onDbChange(db); } - onSelectChange({ db: actualDb, schema: undefined }); + fetchSchemas(dbId, force); + onSelectChange({ dbId, schema: undefined }); } - function changeSchema(schema: SchemaValue) { + function changeSchema(schemaOpt: any, force = false) { + const schema = schemaOpt ? schemaOpt.value : null; if (onSchemaChange) { - onSchemaChange(schema.value); + onSchemaChange(schema); } - if (currentDb) { - onSelectChange({ db: currentDb, schema }); + setCurrentSchema(schema); + onSelectChange({ dbId: currentDbId, schema }); + if (getTableList) { + getTableList(currentDbId, schema, force); } } + function renderDatabaseOption(db: any) { + return ( + + {db.database_name} + + ); + } + function renderSelectRow(select: ReactNode, refreshBtn: ReactNode) { return (
{select} - {refreshBtn} + {refreshBtn}
); } - const loadDatabases = useMemo( - () => async ( - search: string, - page: number, - pageSize: number, - ): Promise<{ - data: DatabaseValue[]; - totalCount: number; - }> => { - const queryParams = rison.encode({ - order_columns: 'database_name', - order_direction: 'asc', - page, - page_size: pageSize, - ...(formMode || !sqlLabMode - ? { filters: [{ col: 'database_name', opr: 'ct', value: search }] } - : { - filters: [ - { col: 'database_name', opr: 'ct', value: search }, - { - col: 'expose_in_sqllab', - opr: 'eq', - value: true, - }, - ], - }), - }); - const endpoint = `/api/v1/database/?q=${queryParams}`; - return SupersetClient.get({ endpoint }).then(({ json }) => { - const { result } = json; - if (getDbList) { - getDbList(result); - } - if (result.length === 0) { - handleError(t("It seems you don't have access to any database")); - } - const options = result.map( - (row: { backend: string; database_name: string; id: number }) => ({ - label: `${row.backend}: ${row.database_name}`, - value: row.id, + function renderDatabaseSelect() { + const queryParams = rison.encode({ + order_columns: 'database_name', + order_direction: 'asc', + page: 0, + page_size: -1, + ...(formMode || !sqlLabMode + ? {} + : { + filters: [ + { + col: 'expose_in_sqllab', + opr: 'eq', + value: true, + }, + ], }), - ); - return { - data: options, - totalCount: options.length, - }; - }); - }, - [formMode, getDbList, handleError, sqlLabMode], - ); + }); - function renderDatabaseSelect() { return renderSelectRow( - {t('Schema')}} name="select-schema" - placeholder={t('Select a schema')} - onChange={item => changeSchema(item as SchemaValue)} - options={loadSchemas} - value={currentSchema} + placeholder={t('Select a schema (%s)', schemaOptions.length)} + options={schemaOptions} + value={value} + valueRenderer={o => ( +
+ {t('Schema:')} {o.label} +
+ )} + isLoading={schemaLoading} + autosize={false} + onChange={item => changeSchema(item)} + isDisabled={readOnly} />, - refreshIcon, + refresh, ); } return ( + {formMode && {t('datasource')}} {renderDatabaseSelect()} + {formMode && {t('schema')}} {renderSchemaSelect()} ); diff --git a/superset-frontend/src/components/Icons/Icon.tsx b/superset-frontend/src/components/Icons/Icon.tsx index efb78dc65cc14..9e3d0e1878ec3 100644 --- a/superset-frontend/src/components/Icons/Icon.tsx +++ b/superset-frontend/src/components/Icons/Icon.tsx @@ -53,21 +53,15 @@ export const Icon = (props: IconProps) => { const name = fileName.replace('_', '-'); useEffect(() => { - let cancelled = false; async function importIcon(): Promise { ImportedSVG.current = ( await import( `!!@svgr/webpack?-svgo,+titleProp,+ref!images/icons/${fileName}.svg` ) ).default; - if (!cancelled) { - setLoaded(true); - } + setLoaded(true); } importIcon(); - return () => { - cancelled = true; - }; }, [fileName, ImportedSVG]); return ( diff --git a/superset-frontend/src/components/Select/Select.tsx b/superset-frontend/src/components/Select/Select.tsx index 96eb79b7d8d6b..596722eaee50c 100644 --- a/superset-frontend/src/components/Select/Select.tsx +++ b/superset-frontend/src/components/Select/Select.tsx @@ -86,9 +86,12 @@ const StyledContainer = styled.div` flex-direction: column; `; -const StyledSelect = styled(AntdSelect)` - ${({ theme }) => ` +const StyledSelect = styled(AntdSelect, { + shouldForwardProp: prop => prop !== 'hasHeader', +})<{ hasHeader: boolean }>` + ${({ theme, hasHeader }) => ` width: 100%; + margin-top: ${hasHeader ? theme.gridUnit : 0}px; && .ant-select-selector { border-radius: ${theme.gridUnit}px; @@ -186,7 +189,6 @@ const Select = ({ : 'multiple'; useEffect(() => { - fetchedQueries.current.clear(); setSelectOptions( options && Array.isArray(options) ? options : EMPTY_OPTIONS, ); @@ -364,45 +366,34 @@ const Select = ({ [options], ); - const handleOnSearch = useMemo( - () => - debounce((search: string) => { - const searchValue = search.trim(); - // enables option creation - if (allowNewOptions && isSingleMode) { - const firstOption = - selectOptions.length > 0 && selectOptions[0].value; - // replaces the last search value entered with the new one - // only when the value wasn't part of the original options - if ( - searchValue && - firstOption === searchedValue && - !initialOptions.find(o => o.value === searchedValue) - ) { - selectOptions.shift(); - setSelectOptions(selectOptions); - } - if (searchValue && !hasOption(searchValue, selectOptions)) { - const newOption = { - label: searchValue, - value: searchValue, - }; - // adds a custom option - const newOptions = [...selectOptions, newOption]; - setSelectOptions(newOptions); - setSelectValue(searchValue); - } - } - setSearchedValue(searchValue); - }, DEBOUNCE_TIMEOUT), - [ - allowNewOptions, - initialOptions, - isSingleMode, - searchedValue, - selectOptions, - ], - ); + const handleOnSearch = debounce((search: string) => { + const searchValue = search.trim(); + // enables option creation + if (allowNewOptions && isSingleMode) { + const firstOption = selectOptions.length > 0 && selectOptions[0].value; + // replaces the last search value entered with the new one + // only when the value wasn't part of the original options + if ( + searchValue && + firstOption === searchedValue && + !initialOptions.find(o => o.value === searchedValue) + ) { + selectOptions.shift(); + setSelectOptions(selectOptions); + } + if (searchValue && !hasOption(searchValue, selectOptions)) { + const newOption = { + label: searchValue, + value: searchValue, + }; + // adds a custom option + const newOptions = [...selectOptions, newOption]; + setSelectOptions(newOptions); + setSelectValue(searchValue); + } + } + setSearchedValue(searchValue); + }, DEBOUNCE_TIMEOUT); const handlePagination = (e: UIEvent) => { const vScroll = e.currentTarget; @@ -495,6 +486,7 @@ const Select = ({ {header} , { + context: { store }, + wrappingComponent: ThemeProvider, + wrappingComponentProps: { theme: supersetTheme }, + }); + await waitForComponentToPaint(mounted); + + return mounted; +} + +describe('TableSelector', () => { + let wrapper; + + beforeEach(async () => { + fetchMock.reset(); + wrapper = await mountAndWait(); + }); + + it('renders', () => { + expect(wrapper.find(TableSelector)).toExist(); + expect(wrapper.find(DatabaseSelector)).toExist(); + }); + + describe('change database', () => { + afterEach(fetchMock.resetHistory); + afterAll(fetchMock.reset); + + it('should fetch schemas', async () => { + fetchMock.get(FETCH_SCHEMAS_ENDPOINT, { overwriteRoutes: true }); + act(() => { + wrapper.find('[data-test="select-database"]').first().props().onChange({ + id: 1, + database_name: 'main', + }); + }); + await waitForComponentToPaint(wrapper); + expect(fetchMock.calls(FETCH_SCHEMAS_ENDPOINT)).toHaveLength(1); + }); + + it('should fetch schema options', async () => { + fetchMock.get(FETCH_SCHEMAS_ENDPOINT, schemaOptions, { + overwriteRoutes: true, + }); + act(() => { + wrapper.find('[data-test="select-database"]').first().props().onChange({ + id: 1, + database_name: 'main', + }); + }); + await waitForComponentToPaint(wrapper); + wrapper.update(); + expect(fetchMock.calls(FETCH_SCHEMAS_ENDPOINT)).toHaveLength(1); + + expect( + wrapper.find('[name="select-schema"]').first().props().options, + ).toEqual([ + { value: 'main', label: 'main', title: 'main' }, + { value: 'erf', label: 'erf', title: 'erf' }, + { value: 'superset', label: 'superset', title: 'superset' }, + ]); + }); + + it('should clear table options', async () => { + act(() => { + wrapper.find('[data-test="select-database"]').first().props().onChange({ + id: 1, + database_name: 'main', + }); + }); + await waitForComponentToPaint(wrapper); + const props = wrapper.find('[name="async-select-table"]').first().props(); + expect(props.isDisabled).toBe(true); + expect(props.value).toEqual(undefined); + }); + }); + + describe('change schema', () => { + beforeEach(async () => { + fetchMock.get(FETCH_SCHEMAS_ENDPOINT, schemaOptions, { + overwriteRoutes: true, + }); + }); + + afterEach(fetchMock.resetHistory); + afterAll(fetchMock.reset); + + it('should fetch table', async () => { + fetchMock.get(GET_TABLE_NAMES_ENDPOINT, { overwriteRoutes: true }); + act(() => { + wrapper.find('[data-test="select-database"]').first().props().onChange({ + id: 1, + database_name: 'main', + }); + }); + await waitForComponentToPaint(wrapper); + act(() => { + wrapper + .find('[name="select-schema"]') + .first() + .props() + .onChange(selectedSchema); + }); + await waitForComponentToPaint(wrapper); + expect(fetchMock.calls(GET_TABLE_NAMES_ENDPOINT)).toHaveLength(1); + }); + + it('should fetch table options', async () => { + fetchMock.get(GET_TABLE_NAMES_ENDPOINT, tables, { + overwriteRoutes: true, + }); + act(() => { + wrapper.find('[data-test="select-database"]').first().props().onChange({ + id: 1, + database_name: 'main', + }); + }); + await waitForComponentToPaint(wrapper); + act(() => { + wrapper + .find('[name="select-schema"]') + .first() + .props() + .onChange(selectedSchema); + }); + await waitForComponentToPaint(wrapper); + expect( + wrapper.find('[name="select-schema"]').first().props().value[0], + ).toEqual(selectedSchema); + expect(fetchMock.calls(GET_TABLE_NAMES_ENDPOINT)).toHaveLength(1); + const { options } = wrapper.find('[name="select-table"]').first().props(); + expect({ options }).toEqual(tables); + }); + }); + + describe('change table', () => { + beforeEach(async () => { + fetchMock.get(GET_TABLE_NAMES_ENDPOINT, tables, { + overwriteRoutes: true, + }); + }); + + it('should change table value', async () => { + act(() => { + wrapper + .find('[name="select-schema"]') + .first() + .props() + .onChange(selectedSchema); + }); + await waitForComponentToPaint(wrapper); + act(() => { + wrapper + .find('[name="select-table"]') + .first() + .props() + .onChange(selectedTable); + }); + await waitForComponentToPaint(wrapper); + expect( + wrapper.find('[name="select-table"]').first().props().value, + ).toEqual('birth_names'); + }); + + it('should call onTableChange with schema from table object', async () => { + act(() => { + wrapper + .find('[name="select-schema"]') + .first() + .props() + .onChange(selectedSchema); + }); + await waitForComponentToPaint(wrapper); + act(() => { + wrapper + .find('[name="select-table"]') + .first() + .props() + .onChange(selectedTable); + }); + await waitForComponentToPaint(wrapper); + expect(mockedProps.onTableChange.getCall(0).args[0]).toBe('birth_names'); + expect(mockedProps.onTableChange.getCall(0).args[1]).toBe('main'); + }); + }); + + describe('getTableNamesBySubStr', () => { + afterEach(fetchMock.resetHistory); + afterAll(fetchMock.reset); + + it('should handle empty', async () => { + act(() => { + wrapper + .find('[name="async-select-table"]') + .first() + .props() + .loadOptions(); + }); + await waitForComponentToPaint(wrapper); + const props = wrapper.find('[name="async-select-table"]').first().props(); + expect(props.isDisabled).toBe(true); + expect(props.value).toEqual(''); + }); + + it('should handle table name', async () => { + wrapper.setProps({ schema: 'main' }); + fetchMock.get(GET_TABLE_ENDPOINT, tables, { + overwriteRoutes: true, + }); + act(() => { + wrapper + .find('[name="async-select-table"]') + .first() + .props() + .loadOptions(); + }); + await waitForComponentToPaint(wrapper); + expect(fetchMock.calls(GET_TABLE_ENDPOINT)).toHaveLength(1); + }); + }); +}); diff --git a/superset-frontend/src/components/TableSelector/TableSelector.test.tsx b/superset-frontend/src/components/TableSelector/TableSelector.test.tsx deleted file mode 100644 index 3b8b61715399d..0000000000000 --- a/superset-frontend/src/components/TableSelector/TableSelector.test.tsx +++ /dev/null @@ -1,91 +0,0 @@ -/** - * 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 React from 'react'; -import { render, screen, waitFor } from 'spec/helpers/testing-library'; -import { SupersetClient } from '@superset-ui/core'; -import userEvent from '@testing-library/user-event'; -import TableSelector from '.'; - -const SupersetClientGet = jest.spyOn(SupersetClient, 'get'); - -const createProps = () => ({ - dbId: 1, - schema: 'test_schema', - handleError: jest.fn(), -}); - -beforeAll(() => { - SupersetClientGet.mockImplementation( - async () => - ({ - json: { - options: [ - { label: 'table_a', value: 'table_a' }, - { label: 'table_b', value: 'table_b' }, - ], - }, - } as any), - ); -}); - -test('renders with default props', async () => { - const props = createProps(); - render(); - const databaseSelect = screen.getByRole('combobox', { - name: 'Select a database', - }); - const schemaSelect = screen.getByRole('combobox', { - name: 'Select a database', - }); - const tableSelect = screen.getByRole('combobox', { - name: 'Select a table', - }); - await waitFor(() => { - expect(databaseSelect).toBeInTheDocument(); - expect(schemaSelect).toBeInTheDocument(); - expect(tableSelect).toBeInTheDocument(); - }); -}); - -test('renders table options', async () => { - const props = createProps(); - render(); - const tableSelect = screen.getByRole('combobox', { - name: 'Select a table', - }); - userEvent.click(tableSelect); - expect( - await screen.findByRole('option', { name: 'table_a' }), - ).toBeInTheDocument(); - expect( - await screen.findByRole('option', { name: 'table_b' }), - ).toBeInTheDocument(); -}); - -test('renders disabled without schema', async () => { - const props = createProps(); - render(); - const tableSelect = screen.getByRole('combobox', { - name: 'Select a table', - }); - await waitFor(() => { - expect(tableSelect).toBeDisabled(); - }); -}); diff --git a/superset-frontend/src/components/TableSelector/index.tsx b/superset-frontend/src/components/TableSelector/index.tsx index 5f68a94dfe56b..c437b1e59a2c0 100644 --- a/superset-frontend/src/components/TableSelector/index.tsx +++ b/superset-frontend/src/components/TableSelector/index.tsx @@ -18,49 +18,57 @@ */ import React, { FunctionComponent, + useEffect, useState, ReactNode, - useMemo, - useEffect, } from 'react'; import { styled, SupersetClient, t } from '@superset-ui/core'; -import { Select } from 'src/components'; +import { AsyncSelect, CreatableSelect, Select } from 'src/components/Select'; + import { FormLabel } from 'src/components/Form'; -import Icons from 'src/components/Icons'; + import DatabaseSelector from 'src/components/DatabaseSelector'; import RefreshLabel from 'src/components/RefreshLabel'; import CertifiedIcon from 'src/components/CertifiedIcon'; import WarningIconWithTooltip from 'src/components/WarningIconWithTooltip'; +const FieldTitle = styled.p` + color: ${({ theme }) => theme.colors.secondary.light2}; + font-size: ${({ theme }) => theme.typography.sizes.s}px; + margin: 20px 0 10px 0; + text-transform: uppercase; +`; + const TableSelectorWrapper = styled.div` - ${({ theme }) => ` - .refresh { - display: flex; - align-items: center; - width: 30px; - margin-left: ${theme.gridUnit}px; - margin-top: ${theme.gridUnit * 5}px; - } + .fa-refresh { + padding-left: 9px; + } - .section { - display: flex; - flex-direction: row; - align-items: center; - } + .refresh-col { + display: flex; + align-items: center; + width: 30px; + margin-left: ${({ theme }) => theme.gridUnit}px; + } - .divider { - border-bottom: 1px solid ${theme.colors.secondary.light5}; - margin: 15px 0; - } + .section { + padding-bottom: 5px; + display: flex; + flex-direction: row; + } - .table-length { - color: ${theme.colors.grayscale.light1}; - } + .select { + flex-grow: 1; + } - .select { - flex: 1; - } - `} + .divider { + border-bottom: 1px solid ${({ theme }) => theme.colors.secondary.light5}; + margin: 15px 0; + } + + .table-length { + color: ${({ theme }) => theme.colors.grayscale.light1}; + } `; const TableLabel = styled.span` @@ -90,15 +98,7 @@ interface TableSelectorProps { schema?: string; tableName?: string; }) => void; - onDbChange?: ( - db: - | { - id: number; - database_name: string; - backend: string; - } - | undefined, - ) => void; + onDbChange?: (db: any) => void; onSchemaChange?: (arg0?: any) => {}; onSchemasLoad?: () => void; onTableChange?: (tableName: string, schema: string) => void; @@ -110,52 +110,6 @@ interface TableSelectorProps { tableNameSticky?: boolean; } -interface Table { - label: string; - value: string; - type: string; - extra?: { - certification?: { - certified_by: string; - details: string; - }; - warning_markdown?: string; - }; -} - -interface TableOption { - label: JSX.Element; - text: string; - value: string; -} - -const TableOption = ({ table }: { table: Table }) => { - const { label, type, extra } = table; - return ( - - {type === 'view' ? ( - - ) : ( - - )} - {extra?.certification && ( - - )} - {extra?.warning_markdown && ( - - )} - {label} - - ); -}; - const TableSelector: FunctionComponent = ({ database, dbId, @@ -175,187 +129,179 @@ const TableSelector: FunctionComponent = ({ tableName, tableNameSticky = true, }) => { - const [currentDbId, setCurrentDbId] = useState(dbId); const [currentSchema, setCurrentSchema] = useState( schema, ); - const [currentTable, setCurrentTable] = useState(); - const [refresh, setRefresh] = useState(0); - const [previousRefresh, setPreviousRefresh] = useState(0); - - const loadTable = useMemo( - () => async (dbId: number, schema: string, tableName: string) => { + const [currentTableName, setCurrentTableName] = useState( + tableName, + ); + const [tableLoading, setTableLoading] = useState(false); + const [tableOptions, setTableOptions] = useState([]); + + function fetchTables( + databaseId?: number, + schema?: string, + forceRefresh = false, + substr = 'undefined', + ) { + const dbSchema = schema || currentSchema; + const actualDbId = databaseId || dbId; + if (actualDbId && dbSchema) { + const encodedSchema = encodeURIComponent(dbSchema); + const encodedSubstr = encodeURIComponent(substr); + setTableLoading(true); + setTableOptions([]); const endpoint = encodeURI( - `/superset/tables/${dbId}/${schema}/${encodeURIComponent( - tableName, - )}/false/true`, + `/superset/tables/${actualDbId}/${encodedSchema}/${encodedSubstr}/${!!forceRefresh}/`, ); - - if (previousRefresh !== refresh) { - setPreviousRefresh(refresh); - } - - return SupersetClient.get({ endpoint }).then(({ json }) => { - const options = json.options as Table[]; - if (options && options.length > 0) { - return options[0]; - } - return null; - }); - }, - [], // eslint-disable-line react-hooks/exhaustive-deps - ); - - const loadTables = useMemo( - () => async (search: string) => { - const dbSchema = schema || currentSchema; - if (currentDbId && dbSchema) { - const encodedSchema = encodeURIComponent(dbSchema); - const encodedSubstr = encodeURIComponent(search || 'undefined'); - const forceRefresh = refresh !== previousRefresh; - const endpoint = encodeURI( - `/superset/tables/${currentDbId}/${encodedSchema}/${encodedSubstr}/${forceRefresh}/`, - ); - - if (previousRefresh !== refresh) { - setPreviousRefresh(refresh); - } - - return SupersetClient.get({ endpoint }).then(({ json }) => { - const options = json.options - .map((table: Table) => ({ - value: table.value, - label: , - text: table.label, - })) - .sort((a: { text: string }, b: { text: string }) => - a.text.localeCompare(b.text), - ); - + return SupersetClient.get({ endpoint }) + .then(({ json }) => { + const options = json.options.map((o: any) => ({ + value: o.value, + schema: o.schema, + label: o.label, + title: o.title, + type: o.type, + extra: o?.extra, + })); + setTableLoading(false); + setTableOptions(options); if (onTablesLoad) { onTablesLoad(json.options); } - - return { - data: options, - totalCount: options.length, - }; + }) + .catch(() => { + setTableLoading(false); + setTableOptions([]); + handleError(t('Error while fetching table list')); }); - } - return { data: [], totalCount: 0 }; - }, - // We are using the refresh state to re-trigger the query - // previousRefresh should be out of dependencies array - // eslint-disable-next-line react-hooks/exhaustive-deps - [currentDbId, currentSchema, onTablesLoad, schema, refresh], - ); + } + setTableLoading(false); + setTableOptions([]); + return Promise.resolve(); + } useEffect(() => { - async function fetchTable() { - if (schema && tableName) { - const table = await loadTable(dbId, schema, tableName); - if (table) { - setCurrentTable({ - label: , - text: table.label, - value: table.value, - }); - } - } + if (dbId && schema) { + fetchTables(); } - fetchTable(); - }, []); // eslint-disable-line react-hooks/exhaustive-deps + }, [dbId, schema]); function onSelectionChange({ dbId, schema, - table, + tableName, }: { dbId: number; schema?: string; - table?: TableOption; + tableName?: string; }) { - setCurrentTable(table); - setCurrentDbId(dbId); + setCurrentTableName(tableName); setCurrentSchema(schema); if (onUpdate) { - onUpdate({ dbId, schema, tableName: table?.value }); + onUpdate({ dbId, schema, tableName }); + } + } + + function getTableNamesBySubStr(substr = 'undefined') { + if (!dbId || !substr) { + const options: any[] = []; + return Promise.resolve({ options }); } + const encodedSchema = encodeURIComponent(schema || ''); + const encodedSubstr = encodeURIComponent(substr); + return SupersetClient.get({ + endpoint: encodeURI( + `/superset/tables/${dbId}/${encodedSchema}/${encodedSubstr}`, + ), + }).then(({ json }) => { + const options = json.options.map((o: any) => ({ + value: o.value, + schema: o.schema, + label: o.label, + title: o.title, + type: o.type, + })); + return { options }; + }); } - function changeTable(table: TableOption) { - if (!table) { - setCurrentTable(undefined); + function changeTable(tableOpt: any) { + if (!tableOpt) { + setCurrentTableName(''); return; } - const tableOptTableName = table.value; - if (currentDbId && tableNameSticky) { + const schemaName = tableOpt.schema; + const tableOptTableName = tableOpt.value; + if (tableNameSticky) { onSelectionChange({ - dbId: currentDbId, - schema: currentSchema, - table, + dbId, + schema: schemaName, + tableName: tableOptTableName, }); } - if (onTableChange && currentSchema) { - onTableChange(tableOptTableName, currentSchema); + if (onTableChange) { + onTableChange(tableOptTableName, schemaName); } } - function onRefresh() { + function changeSchema(schemaOpt: any, force = false) { + const value = schemaOpt ? schemaOpt.value : null; if (onSchemaChange) { - onSchemaChange(currentSchema); + onSchemaChange(value); } - if (currentDbId && currentSchema) { - onSelectionChange({ - dbId: currentDbId, - schema: currentSchema, - table: currentTable, - }); - } - setRefresh(refresh + 1); + onSelectionChange({ + dbId, + schema: value, + tableName: undefined, + }); + fetchTables(dbId, currentSchema, force); + } + + function renderTableOption(option: any) { + return ( + + + + + {option.extra?.certification && ( + + )} + {option.extra?.warning_markdown && ( + + )} + {option.label} + + ); } function renderSelectRow(select: ReactNode, refreshBtn: ReactNode) { return (
{select} - {refreshBtn} + {refreshBtn}
); } - const internalDbChange = ( - db: - | { - id: number; - database_name: string; - backend: string; - } - | undefined, - ) => { - setCurrentDbId(db?.id); - if (onDbChange) { - onDbChange(db); - } - }; - - const internalSchemaChange = (schema?: string) => { - setCurrentSchema(schema); - if (onSchemaChange) { - onSchemaChange(schema); - } - }; - function renderDatabaseSelector() { return ( = ({ ); } - const handleFilterOption = useMemo( - () => (search: string, option: TableOption) => { - const searchValue = search.trim().toLowerCase(); - const { text } = option; - return text.toLowerCase().includes(searchValue); - }, - [], - ); - function renderTableSelect() { - const disabled = - (currentSchema && !formMode && readOnly) || - (!currentSchema && !database?.allow_multi_schema_metadata_fetch); - - const header = sqlLabMode ? ( - {t('See table schema')} - ) : ( - {t('Table')} - ); - - const select = ( - + ); + } else if (formMode) { + select = ( + + ); + } else { + // sql lab + let tableSelectPlaceholder; + let tableSelectDisabled = false; + if (database && database.allow_multi_schema_metadata_fetch) { + tableSelectPlaceholder = t('Type to search ...'); + } else { + tableSelectPlaceholder = t('Select table '); + tableSelectDisabled = true; + } + select = ( + + ); + } const refresh = !formMode && !readOnly && ( changeSchema({ value: schema }, true)} tooltipContent={t('Force refresh table list')} /> ); - return renderSelectRow(select, refresh); } + function renderSeeTableLabel() { + return ( +
+ + {t('See table schema')}{' '} + {schema && ( + + {tableOptions.length} in {schema} + + )} + +
+ ); + } + return ( {renderDatabaseSelector()} - {sqlLabMode && !formMode &&
} + {!formMode &&
} + {sqlLabMode && renderSeeTableLabel()} + {formMode && {t('Table')}} {renderTableSelect()} ); diff --git a/superset-frontend/src/components/WarningIconWithTooltip/index.tsx b/superset-frontend/src/components/WarningIconWithTooltip/index.tsx index f732554e15aaa..f160ade50ec72 100644 --- a/superset-frontend/src/components/WarningIconWithTooltip/index.tsx +++ b/superset-frontend/src/components/WarningIconWithTooltip/index.tsx @@ -18,17 +18,16 @@ */ import React from 'react'; import { useTheme, SafeMarkdown } from '@superset-ui/core'; -import Icons, { IconType } from 'src/components/Icons'; +import Icons from 'src/components/Icons'; import { Tooltip } from 'src/components/Tooltip'; export interface WarningIconWithTooltipProps { warningMarkdown: string; - size?: IconType['iconSize']; + size?: number; } function WarningIconWithTooltip({ warningMarkdown, - size, }: WarningIconWithTooltipProps) { const theme = useTheme(); return ( @@ -38,7 +37,6 @@ function WarningIconWithTooltip({ > diff --git a/superset-frontend/src/datasource/DatasourceEditor.jsx b/superset-frontend/src/datasource/DatasourceEditor.jsx index d8a0a3425f244..e11b8310bb75c 100644 --- a/superset-frontend/src/datasource/DatasourceEditor.jsx +++ b/superset-frontend/src/datasource/DatasourceEditor.jsx @@ -775,47 +775,41 @@ class DatasourceEditor extends React.PureComponent {
{this.state.isSqla && ( <> - - - - this.state.isEditMode && - this.onDatasourcePropChange('schema', schema) - } - onDbChange={database => - this.state.isEditMode && - this.onDatasourcePropChange('database', database) - } - formMode={false} - handleError={this.props.addDangerToast} - readOnly={!this.state.isEditMode} - /> -
- } - /> -
- { - this.onDatasourcePropChange('table_name', table); - }} - placeholder={t('Dataset name')} - disabled={!this.state.isEditMode} - /> + + this.state.isEditMode && + this.onDatasourcePropChange('schema', schema) + } + onDbChange={database => + this.state.isEditMode && + this.onDatasourcePropChange('database', database) } + formMode={false} + handleError={this.props.addDangerToast} + readOnly={!this.state.isEditMode} /> -
- + } + /> + { + this.onDatasourcePropChange('table_name', table); + }} + placeholder={t('Dataset name')} + disabled={!this.state.isEditMode} + /> + } + /> - - this.onDatasourcePropChange('schema', schema) - : undefined - } - onDbChange={ - this.state.isEditMode - ? database => - this.onDatasourcePropChange( - 'database', - database, - ) - : undefined - } - onTableChange={ - this.state.isEditMode - ? table => - this.onDatasourcePropChange('table_name', table) - : undefined - } - readOnly={!this.state.isEditMode} - /> -
+ + this.onDatasourcePropChange('schema', schema) + : undefined + } + onDbChange={ + this.state.isEditMode + ? database => + this.onDatasourcePropChange('database', database) + : undefined + } + onTableChange={ + this.state.isEditMode + ? table => + this.onDatasourcePropChange('table_name', table) + : undefined + } + readOnly={!this.state.isEditMode} + /> } description={t( 'The pointer to a physical table (or view). Keep in mind that the chart is ' + diff --git a/superset-frontend/src/explore/components/controls/DatasourceControl/index.jsx b/superset-frontend/src/explore/components/controls/DatasourceControl/index.jsx index 3df55323d11d7..9278c29e79c09 100644 --- a/superset-frontend/src/explore/components/controls/DatasourceControl/index.jsx +++ b/superset-frontend/src/explore/components/controls/DatasourceControl/index.jsx @@ -227,7 +227,10 @@ class DatasourceControl extends React.PureComponent { )} {extra?.warning_markdown && ( - + )} = ({ )} {parsedExtra?.warning_markdown && ( )} {titleLink} diff --git a/superset/datasets/api.py b/superset/datasets/api.py index 261a7d749e40e..8d8eb6efee875 100644 --- a/superset/datasets/api.py +++ b/superset/datasets/api.py @@ -160,7 +160,7 @@ class DatasetRestApi(BaseSupersetModelRestApi): "url", "extra", ] - show_columns = show_select_columns + ["columns.type_generic", "database.backend"] + show_columns = show_select_columns + ["columns.type_generic"] add_model_schema = DatasetPostSchema() edit_model_schema = DatasetPutSchema() add_columns = ["database", "schema", "table_name", "owners"] diff --git a/superset/views/core.py b/superset/views/core.py index 29109f13c3b53..d7e626e261b64 100755 --- a/superset/views/core.py +++ b/superset/views/core.py @@ -1058,14 +1058,8 @@ def schemas( # pylint: disable=no-self-use @event_logger.log_this @expose("/tables////") @expose("/tables/////") - @expose("/tables/////") - def tables( # pylint: disable=too-many-locals,no-self-use,too-many-arguments - self, - db_id: int, - schema: str, - substr: str, - force_refresh: str = "false", - exact_match: str = "false", + def tables( # pylint: disable=too-many-locals,no-self-use + self, db_id: int, schema: str, substr: str, force_refresh: str = "false" ) -> FlaskResponse: """Endpoint to fetch the list of tables for given database""" # Guarantees database filtering by security access @@ -1078,7 +1072,6 @@ def tables( # pylint: disable=too-many-locals,no-self-use,too-many-arguments return json_error_response("Not found", 404) force_refresh_parsed = force_refresh.lower() == "true" - exact_match_parsed = exact_match.lower() == "true" schema_parsed = utils.parse_js_uri_path_item(schema, eval_undefined=True) substr_parsed = utils.parse_js_uri_path_item(substr, eval_undefined=True) @@ -1120,15 +1113,9 @@ def get_datasource_label(ds_name: utils.DatasourceName) -> str: ds_name.table if schema_parsed else f"{ds_name.schema}.{ds_name.table}" ) - def is_match(src: str, target: utils.DatasourceName) -> bool: - target_label = get_datasource_label(target) - if exact_match_parsed: - return src == target_label - return src in target_label - if substr_parsed: - tables = [tn for tn in tables if is_match(substr_parsed, tn)] - views = [vn for vn in views if is_match(substr_parsed, vn)] + tables = [tn for tn in tables if substr_parsed in get_datasource_label(tn)] + views = [vn for vn in views if substr_parsed in get_datasource_label(vn)] if not schema_parsed and database.default_schemas: user_schemas = ( diff --git a/tests/integration_tests/datasets/api_tests.py b/tests/integration_tests/datasets/api_tests.py index c9701534b2664..385025e3835cc 100644 --- a/tests/integration_tests/datasets/api_tests.py +++ b/tests/integration_tests/datasets/api_tests.py @@ -222,7 +222,6 @@ def test_get_dataset_item(self): Dataset API: Test get dataset item """ table = self.get_energy_usage_dataset() - main_db = get_main_database() self.login(username="admin") uri = f"api/v1/dataset/{table.id}" rv = self.get_assert_metric(uri, "get") @@ -230,11 +229,7 @@ def test_get_dataset_item(self): response = json.loads(rv.data.decode("utf-8")) expected_result = { "cache_timeout": None, - "database": { - "backend": main_db.backend, - "database_name": "examples", - "id": 1, - }, + "database": {"database_name": "examples", "id": 1}, "default_endpoint": None, "description": "Energy consumption", "extra": None, @@ -249,10 +244,9 @@ def test_get_dataset_item(self): "table_name": "energy_usage", "template_params": None, } - if response["result"]["database"]["backend"] not in ("presto", "hive"): - assert { - k: v for k, v in response["result"].items() if k in expected_result - } == expected_result + assert { + k: v for k, v in response["result"].items() if k in expected_result + } == expected_result assert len(response["result"]["columns"]) == 3 assert len(response["result"]["metrics"]) == 2 From 147637a02d259888008e03167ae69bd978f56a17 Mon Sep 17 00:00:00 2001 From: Ville Brofeldt <33317356+villebro@users.noreply.github.com> Date: Fri, 27 Aug 2021 14:03:05 +0300 Subject: [PATCH 7/8] fix(native-filters): add handle undefined control value gracefully (#16468) --- .../src/dashboard/components/nativeFilters/FilterBar/utils.ts | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/superset-frontend/src/dashboard/components/nativeFilters/FilterBar/utils.ts b/superset-frontend/src/dashboard/components/nativeFilters/FilterBar/utils.ts index 84bd6cbf369ee..af358a09fe214 100644 --- a/superset-frontend/src/dashboard/components/nativeFilters/FilterBar/utils.ts +++ b/superset-frontend/src/dashboard/components/nativeFilters/FilterBar/utils.ts @@ -56,7 +56,7 @@ export const checkIsMissingRequiredValue = ( const value = filterState?.value; // TODO: this property should be unhardcoded return ( - filter.controlValues.enableEmptyFilter && + filter.controlValues?.enableEmptyFilter && (value === null || value === undefined) ); }; From 62d8ab7f9c044e2887de794484c30c2108213e0b Mon Sep 17 00:00:00 2001 From: Beto Dealmeida Date: Fri, 27 Aug 2021 07:19:37 -0700 Subject: [PATCH 8/8] fix: create example DB if needed (#16451) * fix: create example DB if needed * fix lint --- superset/commands/importers/v1/examples.py | 6 ++---- superset/models/slice.py | 2 +- 2 files changed, 3 insertions(+), 5 deletions(-) diff --git a/superset/commands/importers/v1/examples.py b/superset/commands/importers/v1/examples.py index 66563f856b620..b7431358dd9f3 100644 --- a/superset/commands/importers/v1/examples.py +++ b/superset/commands/importers/v1/examples.py @@ -43,8 +43,8 @@ from superset.datasets.commands.importers.v1 import ImportDatasetsCommand from superset.datasets.commands.importers.v1.utils import import_dataset from superset.datasets.schemas import ImportV1DatasetSchema -from superset.models.core import Database from superset.models.dashboard import dashboard_slices +from superset.utils.core import get_example_database class ImportExamplesCommand(ImportModelsCommand): @@ -104,9 +104,7 @@ def _import( # If database_uuid is not in the list of UUIDs it means that the examples # database was created before its UUID was frozen, so it has a random UUID. # We need to determine its ID so we can point the dataset to it. - examples_db = ( - db.session.query(Database).filter_by(database_name="examples").first() - ) + examples_db = get_example_database() dataset_info: Dict[str, Dict[str, Any]] = {} for file_name, config in configs.items(): if file_name.startswith("datasets/"): diff --git a/superset/models/slice.py b/superset/models/slice.py index b4c7f6604f9a0..9093cfa43acc7 100644 --- a/superset/models/slice.py +++ b/superset/models/slice.py @@ -53,7 +53,7 @@ logger = logging.getLogger(__name__) -class Slice( # pylint: disable=too-many-public-methods +class Slice( # pylint: disable=too-many-public-methods, too-many-instance-attributes Model, AuditMixinNullable, ImportExportMixin ): """A slice is essentially a report or a view on data"""