mirror of
https://github.com/apache/superset.git
synced 2026-05-01 14:04:21 +00:00
Compare commits
1 Commits
docs/testi
...
db-upload-
| Author | SHA1 | Date | |
|---|---|---|---|
|
|
5b51ae8bdc |
@@ -50,6 +50,7 @@ const ExtraOptions = ({
|
|||||||
onExtraInputChange,
|
onExtraInputChange,
|
||||||
onExtraEditorChange,
|
onExtraEditorChange,
|
||||||
extraExtension,
|
extraExtension,
|
||||||
|
testedEngineInfo,
|
||||||
}: {
|
}: {
|
||||||
db: DatabaseObject | null;
|
db: DatabaseObject | null;
|
||||||
onInputChange: (
|
onInputChange: (
|
||||||
@@ -62,13 +63,14 @@ const ExtraOptions = ({
|
|||||||
) => void;
|
) => void;
|
||||||
onExtraEditorChange: Function;
|
onExtraEditorChange: Function;
|
||||||
extraExtension: DatabaseConnectionExtension | undefined;
|
extraExtension: DatabaseConnectionExtension | undefined;
|
||||||
|
testedEngineInfo?: any;
|
||||||
}) => {
|
}) => {
|
||||||
const expandableModalIsOpen = !!db?.expose_in_sqllab;
|
const expandableModalIsOpen = !!db?.expose_in_sqllab;
|
||||||
const createAsOpen = !!(db?.allow_ctas || db?.allow_cvas);
|
const createAsOpen = !!(db?.allow_ctas || db?.allow_cvas);
|
||||||
const isFileUploadSupportedByEngine =
|
// Use tested engine info if available, otherwise fall back to initial engine info
|
||||||
db?.engine_information?.supports_file_upload;
|
const engineInfo = testedEngineInfo || db?.engine_information;
|
||||||
const supportsDynamicCatalog =
|
const isFileUploadSupportedByEngine = engineInfo?.supports_file_upload;
|
||||||
db?.engine_information?.supports_dynamic_catalog;
|
const supportsDynamicCatalog = engineInfo?.supports_dynamic_catalog;
|
||||||
|
|
||||||
// JSON.parse will deep parse engine_params
|
// JSON.parse will deep parse engine_params
|
||||||
// if it's an object, and we want to keep it a string
|
// if it's an object, and we want to keep it a string
|
||||||
@@ -528,23 +530,29 @@ const ExtraOptions = ({
|
|||||||
/>
|
/>
|
||||||
</div>
|
</div>
|
||||||
</StyledInputContainer>
|
</StyledInputContainer>
|
||||||
{isFileUploadSupportedByEngine && (
|
<StyledInputContainer
|
||||||
<StyledInputContainer
|
css={!db?.allow_file_upload ? no_margin_bottom : {}}
|
||||||
css={!db?.allow_file_upload ? no_margin_bottom : {}}
|
>
|
||||||
>
|
<div className="input-container">
|
||||||
<div className="input-container">
|
<Checkbox
|
||||||
<Checkbox
|
id="allow_file_upload"
|
||||||
id="allow_file_upload"
|
name="allow_file_upload"
|
||||||
name="allow_file_upload"
|
indeterminate={false}
|
||||||
indeterminate={false}
|
checked={!!db?.allow_file_upload}
|
||||||
checked={!!db?.allow_file_upload}
|
disabled={!isFileUploadSupportedByEngine}
|
||||||
onChange={onInputChange}
|
onChange={onInputChange}
|
||||||
>
|
>
|
||||||
{t('Allow file uploads to database')}
|
{t('Allow file uploads to database')}
|
||||||
</Checkbox>
|
</Checkbox>
|
||||||
</div>
|
{!isFileUploadSupportedByEngine && (
|
||||||
</StyledInputContainer>
|
<InfoTooltip
|
||||||
)}
|
tooltip={t(
|
||||||
|
'File upload is not supported for this database engine'
|
||||||
|
)}
|
||||||
|
/>
|
||||||
|
)}
|
||||||
|
</div>
|
||||||
|
</StyledInputContainer>
|
||||||
{isFileUploadSupportedByEngine && !!db?.allow_file_upload && (
|
{isFileUploadSupportedByEngine && !!db?.allow_file_upload && (
|
||||||
<StyledInputContainer css={no_margin_bottom}>
|
<StyledInputContainer css={no_margin_bottom}>
|
||||||
<div className="control-label">
|
<div className="control-label">
|
||||||
|
|||||||
@@ -607,6 +607,7 @@ const DatabaseModal: FunctionComponent<DatabaseModalProps> = ({
|
|||||||
const [editNewDb, setEditNewDb] = useState<boolean>(false);
|
const [editNewDb, setEditNewDb] = useState<boolean>(false);
|
||||||
const [isLoading, setLoading] = useState<boolean>(false);
|
const [isLoading, setLoading] = useState<boolean>(false);
|
||||||
const [testInProgress, setTestInProgress] = useState<boolean>(false);
|
const [testInProgress, setTestInProgress] = useState<boolean>(false);
|
||||||
|
const [testedEngineInfo, setTestedEngineInfo] = useState<any>(null);
|
||||||
const [passwords, setPasswords] = useState<Record<string, string>>({});
|
const [passwords, setPasswords] = useState<Record<string, string>>({});
|
||||||
const [sshTunnelPasswords, setSSHTunnelPasswords] = useState<
|
const [sshTunnelPasswords, setSSHTunnelPasswords] = useState<
|
||||||
Record<string, string>
|
Record<string, string>
|
||||||
@@ -735,6 +736,9 @@ const DatabaseModal: FunctionComponent<DatabaseModalProps> = ({
|
|||||||
addSuccessToast(errorMsg);
|
addSuccessToast(errorMsg);
|
||||||
setHasValidated(true);
|
setHasValidated(true);
|
||||||
},
|
},
|
||||||
|
(engineInfo: any) => {
|
||||||
|
setTestedEngineInfo(engineInfo);
|
||||||
|
},
|
||||||
);
|
);
|
||||||
};
|
};
|
||||||
|
|
||||||
@@ -797,6 +801,7 @@ const DatabaseModal: FunctionComponent<DatabaseModalProps> = ({
|
|||||||
setSSHTunnelPrivateKeyPasswords({});
|
setSSHTunnelPrivateKeyPasswords({});
|
||||||
setConfirmedOverwrite(false);
|
setConfirmedOverwrite(false);
|
||||||
setUseSSHTunneling(undefined);
|
setUseSSHTunneling(undefined);
|
||||||
|
setTestedEngineInfo(null);
|
||||||
onHide();
|
onHide();
|
||||||
};
|
};
|
||||||
|
|
||||||
@@ -1771,6 +1776,7 @@ const DatabaseModal: FunctionComponent<DatabaseModalProps> = ({
|
|||||||
<ExtraOptions
|
<ExtraOptions
|
||||||
extraExtension={dbConfigExtraExtension}
|
extraExtension={dbConfigExtraExtension}
|
||||||
db={db as DatabaseObject}
|
db={db as DatabaseObject}
|
||||||
|
testedEngineInfo={testedEngineInfo}
|
||||||
onInputChange={(
|
onInputChange={(
|
||||||
e: CheckboxChangeEvent | React.ChangeEvent<HTMLInputElement>,
|
e: CheckboxChangeEvent | React.ChangeEvent<HTMLInputElement>,
|
||||||
) => {
|
) => {
|
||||||
@@ -2020,6 +2026,7 @@ const DatabaseModal: FunctionComponent<DatabaseModalProps> = ({
|
|||||||
<ExtraOptions
|
<ExtraOptions
|
||||||
extraExtension={dbConfigExtraExtension}
|
extraExtension={dbConfigExtraExtension}
|
||||||
db={db as DatabaseObject}
|
db={db as DatabaseObject}
|
||||||
|
testedEngineInfo={testedEngineInfo}
|
||||||
onInputChange={(e: CheckboxChangeEvent) => {
|
onInputChange={(e: CheckboxChangeEvent) => {
|
||||||
const { target } = e;
|
const { target } = e;
|
||||||
onChange(ActionType.InputChange, {
|
onChange(ActionType.InputChange, {
|
||||||
|
|||||||
@@ -712,14 +712,18 @@ export const testDatabaseConnection = (
|
|||||||
connection: Partial<DatabaseObject>,
|
connection: Partial<DatabaseObject>,
|
||||||
handleErrorMsg: (errorMsg: string) => void,
|
handleErrorMsg: (errorMsg: string) => void,
|
||||||
addSuccessToast: (arg0: string) => void,
|
addSuccessToast: (arg0: string) => void,
|
||||||
|
onEngineInfo?: (engineInfo: any) => void,
|
||||||
) => {
|
) => {
|
||||||
SupersetClient.post({
|
SupersetClient.post({
|
||||||
endpoint: 'api/v1/database/test_connection/',
|
endpoint: 'api/v1/database/test_connection/',
|
||||||
body: JSON.stringify(connection),
|
body: JSON.stringify(connection),
|
||||||
headers: { 'Content-Type': 'application/json' },
|
headers: { 'Content-Type': 'application/json' },
|
||||||
}).then(
|
}).then(
|
||||||
() => {
|
(response) => {
|
||||||
addSuccessToast(t('Connection looks good!'));
|
addSuccessToast(t('Connection looks good!'));
|
||||||
|
if (onEngineInfo && response?.json?.engine_information) {
|
||||||
|
onEngineInfo(response.json.engine_information);
|
||||||
|
}
|
||||||
},
|
},
|
||||||
createErrorHandler((errMsg: Record<string, string[] | string> | string) => {
|
createErrorHandler((errMsg: Record<string, string[] | string> | string) => {
|
||||||
handleErrorMsg(t('ERROR: %s', parsedErrorMessage(errMsg)));
|
handleErrorMsg(t('ERROR: %s', parsedErrorMessage(errMsg)));
|
||||||
|
|||||||
@@ -91,7 +91,7 @@ class TestConnectionDatabaseCommand(BaseCommand):
|
|||||||
|
|
||||||
def run( # noqa: C901
|
def run( # noqa: C901
|
||||||
self,
|
self,
|
||||||
) -> None: # pylint: disable=too-many-statements,too-many-branches
|
) -> Database: # pylint: disable=too-many-statements,too-many-branches
|
||||||
self.validate()
|
self.validate()
|
||||||
ex_str = ""
|
ex_str = ""
|
||||||
ssh_tunnel = self._properties.get("ssh_tunnel")
|
ssh_tunnel = self._properties.get("ssh_tunnel")
|
||||||
@@ -168,6 +168,8 @@ class TestConnectionDatabaseCommand(BaseCommand):
|
|||||||
action=get_log_connection_action("test_connection_success", ssh_tunnel),
|
action=get_log_connection_action("test_connection_success", ssh_tunnel),
|
||||||
engine=database.db_engine_spec.__name__,
|
engine=database.db_engine_spec.__name__,
|
||||||
)
|
)
|
||||||
|
|
||||||
|
return database
|
||||||
|
|
||||||
except (NoSuchModuleError, ModuleNotFoundError) as ex:
|
except (NoSuchModuleError, ModuleNotFoundError) as ex:
|
||||||
event_logger.log_with_context(
|
event_logger.log_with_context(
|
||||||
|
|||||||
@@ -1258,6 +1258,17 @@ class DatabaseRestApi(BaseSupersetModelRestApi):
|
|||||||
properties:
|
properties:
|
||||||
message:
|
message:
|
||||||
type: string
|
type: string
|
||||||
|
engine_information:
|
||||||
|
type: object
|
||||||
|
properties:
|
||||||
|
supports_file_upload:
|
||||||
|
type: boolean
|
||||||
|
disable_ssh_tunneling:
|
||||||
|
type: boolean
|
||||||
|
supports_dynamic_catalog:
|
||||||
|
type: boolean
|
||||||
|
supports_oauth2:
|
||||||
|
type: boolean
|
||||||
400:
|
400:
|
||||||
$ref: '#/components/responses/400'
|
$ref: '#/components/responses/400'
|
||||||
422:
|
422:
|
||||||
@@ -1271,8 +1282,9 @@ class DatabaseRestApi(BaseSupersetModelRestApi):
|
|||||||
except ValidationError as error:
|
except ValidationError as error:
|
||||||
return self.response_400(message=error.messages)
|
return self.response_400(message=error.messages)
|
||||||
try:
|
try:
|
||||||
TestConnectionDatabaseCommand(item).run()
|
database = TestConnectionDatabaseCommand(item).run()
|
||||||
return self.response(200, message="OK")
|
engine_information = database.db_engine_spec.get_public_information()
|
||||||
|
return self.response(200, message="OK", engine_information=engine_information)
|
||||||
except (SSHTunnelingNotEnabledError, SSHTunnelDatabasePortError) as ex:
|
except (SSHTunnelingNotEnabledError, SSHTunnelDatabasePortError) as ex:
|
||||||
return self.response_400(message=str(ex))
|
return self.response_400(message=str(ex))
|
||||||
|
|
||||||
|
|||||||
Reference in New Issue
Block a user