diff --git a/superset-frontend/src/components/DatabaseSelector/DatabaseSelector.test.tsx b/superset-frontend/src/components/DatabaseSelector/DatabaseSelector.test.tsx index b5ebd044ee0aa..af3fa96109f66 100644 --- a/superset-frontend/src/components/DatabaseSelector/DatabaseSelector.test.tsx +++ b/superset-frontend/src/components/DatabaseSelector/DatabaseSelector.test.tsx @@ -26,7 +26,12 @@ import DatabaseSelector from '.'; const SupersetClientGet = jest.spyOn(SupersetClient, 'get'); const createProps = () => ({ - db: { id: 1, database_name: 'test', backend: 'test-postgresql' }, + db: { + id: 1, + database_name: 'test', + backend: 'test-postgresql', + allow_multi_schema_metadata_fetch: false, + }, formMode: false, isDatabaseSelectEnabled: true, readOnly: false, @@ -246,6 +251,7 @@ test('Sends the correct db when changing the database', async () => { id: 2, database_name: 'test-mysql', backend: 'mysql', + allow_multi_schema_metadata_fetch: false, }), ), ); diff --git a/superset-frontend/src/components/DatabaseSelector/index.tsx b/superset-frontend/src/components/DatabaseSelector/index.tsx index 0c912fd83f30d..1a8e99bba892a 100644 --- a/superset-frontend/src/components/DatabaseSelector/index.tsx +++ b/superset-frontend/src/components/DatabaseSelector/index.tsx @@ -63,21 +63,25 @@ type DatabaseValue = { id: number; database_name: string; backend: string; + allow_multi_schema_metadata_fetch: boolean; +}; + +export type DatabaseObject = { + id: number; + database_name: string; + backend: string; + allow_multi_schema_metadata_fetch: boolean; }; type SchemaValue = { label: string; value: string }; interface DatabaseSelectorProps { - db?: { id: number; database_name: string; backend: string }; + db?: DatabaseObject; formMode?: boolean; getDbList?: (arg0: any) => {}; handleError: (msg: string) => void; isDatabaseSelectEnabled?: boolean; - onDbChange?: (db: { - id: number; - database_name: string; - backend: string; - }) => void; + onDbChange?: (db: DatabaseObject) => void; onSchemaChange?: (schema?: string) => void; onSchemasLoad?: (schemas: Array) => void; readOnly?: boolean; @@ -165,20 +169,20 @@ export default function DatabaseSelector({ if (result.length === 0) { handleError(t("It seems you don't have access to any database")); } - const options = result.map( - (row: { id: number; database_name: string; backend: string }) => ({ - label: ( - - ), - value: row.id, - id: row.id, - database_name: row.database_name, - backend: row.backend, - }), - ); + const options = result.map((row: DatabaseObject) => ({ + label: ( + + ), + value: row.id, + id: row.id, + database_name: row.database_name, + backend: row.backend, + allow_multi_schema_metadata_fetch: + row.allow_multi_schema_metadata_fetch, + })); return { data: options, totalCount: options.length, diff --git a/superset-frontend/src/components/TableSelector/TableSelector.test.tsx b/superset-frontend/src/components/TableSelector/TableSelector.test.tsx index 0c5dbcdcca0ab..51e40b511d339 100644 --- a/superset-frontend/src/components/TableSelector/TableSelector.test.tsx +++ b/superset-frontend/src/components/TableSelector/TableSelector.test.tsx @@ -26,7 +26,12 @@ import TableSelector from '.'; const SupersetClientGet = jest.spyOn(SupersetClient, 'get'); const createProps = () => ({ - dbId: 1, + database: { + id: 1, + database_name: 'main', + backend: 'sqlite', + allow_multi_schema_metadata_fetch: false, + }, schema: 'test_schema', handleError: jest.fn(), }); diff --git a/superset-frontend/src/components/TableSelector/index.tsx b/superset-frontend/src/components/TableSelector/index.tsx index 06dad72153722..8ee92ad2d0897 100644 --- a/superset-frontend/src/components/TableSelector/index.tsx +++ b/superset-frontend/src/components/TableSelector/index.tsx @@ -27,7 +27,9 @@ import { styled, SupersetClient, t } from '@superset-ui/core'; import { Select } from 'src/components'; import { FormLabel } from 'src/components/Form'; import Icons from 'src/components/Icons'; -import DatabaseSelector from 'src/components/DatabaseSelector'; +import DatabaseSelector, { + DatabaseObject, +} from 'src/components/DatabaseSelector'; import RefreshLabel from 'src/components/RefreshLabel'; import CertifiedIcon from 'src/components/CertifiedIcon'; import WarningIconWithTooltip from 'src/components/WarningIconWithTooltip'; @@ -76,22 +78,12 @@ const TableLabel = styled.span` interface TableSelectorProps { clearable?: boolean; - database?: { - id: number; - database_name: string; - backend: string; - allow_multi_schema_metadata_fetch: boolean; - }; - dbId: number; + database?: DatabaseObject; formMode?: boolean; getDbList?: (arg0: any) => {}; handleError: (msg: string) => void; isDatabaseSelectEnabled?: boolean; - onDbChange?: (db: { - id: number; - database_name: string; - backend: string; - }) => void; + onDbChange?: (db: DatabaseObject) => void; onSchemaChange?: (schema?: string) => void; onSchemasLoad?: () => void; onTableChange?: (tableName?: string, schema?: string) => void; @@ -150,7 +142,6 @@ const TableOption = ({ table }: { table: Table }) => { const TableSelector: FunctionComponent = ({ database, - dbId, formMode = false, getDbList, handleError, @@ -165,7 +156,9 @@ const TableSelector: FunctionComponent = ({ sqlLabMode = true, tableName, }) => { - const [currentDbId, setCurrentDbId] = useState(dbId); + const [currentDatabase, setCurrentDatabase] = useState< + DatabaseObject | undefined + >(database); const [currentSchema, setCurrentSchema] = useState( schema, ); @@ -176,13 +169,22 @@ const TableSelector: FunctionComponent = ({ const [tableOptions, setTableOptions] = useState([]); useEffect(() => { - if (currentDbId && currentSchema) { + // reset selections + if (database === undefined) { + setCurrentDatabase(undefined); + setCurrentSchema(undefined); + setCurrentTable(undefined); + } + }, [database]); + + useEffect(() => { + if (currentDatabase && currentSchema) { setLoadingTables(true); const encodedSchema = encodeURIComponent(currentSchema); const forceRefresh = refresh !== previousRefresh; // TODO: Would be nice to add pagination in a follow-up. Needs endpoint changes. const endpoint = encodeURI( - `/superset/tables/${currentDbId}/${encodedSchema}/undefined/${forceRefresh}/`, + `/superset/tables/${currentDatabase.id}/${encodedSchema}/undefined/${forceRefresh}/`, ); if (previousRefresh !== refresh) { @@ -221,7 +223,7 @@ const TableSelector: FunctionComponent = ({ // 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, refresh]); + }, [currentDatabase, currentSchema, onTablesLoad, refresh]); function renderSelectRow(select: ReactNode, refreshBtn: ReactNode) { return ( @@ -239,12 +241,8 @@ const TableSelector: FunctionComponent = ({ } }; - const internalDbChange = (db: { - id: number; - database_name: string; - backend: string; - }) => { - setCurrentDbId(db?.id); + const internalDbChange = (db: DatabaseObject) => { + setCurrentDatabase(db); if (onDbChange) { onDbChange(db); } @@ -261,7 +259,8 @@ const TableSelector: FunctionComponent = ({ function renderDatabaseSelector() { return ( = ({ onHide, show, }) => { + const [currentDatabase, setCurrentDatabase] = useState< + DatabaseObject | undefined + >(); const [currentSchema, setSchema] = useState(''); const [currentTableName, setTableName] = useState(''); - const [datasourceId, setDatasourceId] = useState(0); const [disableSave, setDisableSave] = useState(true); const { createResource } = useSingleViewResource>( 'dataset', @@ -61,15 +63,11 @@ const DatasetModal: FunctionComponent = ({ ); useEffect(() => { - setDisableSave(isNil(datasourceId) || isEmpty(currentTableName)); - }, [currentTableName, datasourceId]); + setDisableSave(currentDatabase === undefined || currentTableName === ''); + }, [currentTableName, currentDatabase]); - const onDbChange = (db: { - id: number; - database_name: string; - backend: string; - }) => { - setDatasourceId(db.id); + const onDbChange = (db: DatabaseObject) => { + setCurrentDatabase(db); }; const onSchemaChange = (schema?: string) => { @@ -80,9 +78,24 @@ const DatasetModal: FunctionComponent = ({ setTableName(tableName); }; + const clearModal = () => { + setSchema(''); + setTableName(''); + setCurrentDatabase(undefined); + setDisableSave(true); + }; + + const hide = () => { + clearModal(); + onHide(); + }; + const onSave = () => { + if (currentDatabase === undefined) { + return; + } const data = { - database: datasourceId, + database: currentDatabase.id, ...(currentSchema ? { schema: currentSchema } : {}), table_name: currentTableName, }; @@ -94,7 +107,7 @@ const DatasetModal: FunctionComponent = ({ onDatasetAdd({ id: response.id, ...response }); } addSuccessToast(t('The dataset has been saved')); - onHide(); + hide(); }); }; @@ -102,7 +115,7 @@ const DatasetModal: FunctionComponent = ({ = ({