mirror of https://github.com/apache/druid.git
Allow list for JDBC connection properties to address CVE-2021-26919 (#11047)
* Allow list for JDBC connection properties to address CVE-2021-26919 * fix tests for java 11
This commit is contained in:
parent
d7f5293364
commit
cfcebc40f6
|
@ -0,0 +1,63 @@
|
||||||
|
/*
|
||||||
|
* 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.
|
||||||
|
*/
|
||||||
|
|
||||||
|
package org.apache.druid.utils;
|
||||||
|
|
||||||
|
import com.google.common.base.Preconditions;
|
||||||
|
|
||||||
|
import java.util.Set;
|
||||||
|
|
||||||
|
public final class ConnectionUriUtils
|
||||||
|
{
|
||||||
|
// Note: MySQL JDBC connector 8 supports 7 other protocols than just `jdbc:mysql:`
|
||||||
|
// (https://dev.mysql.com/doc/connector-j/8.0/en/connector-j-reference-jdbc-url-format.html).
|
||||||
|
// We should consider either expanding recognized mysql protocols or restricting allowed protocols to
|
||||||
|
// just a basic one.
|
||||||
|
public static final String MYSQL_PREFIX = "jdbc:mysql:";
|
||||||
|
public static final String POSTGRES_PREFIX = "jdbc:postgresql:";
|
||||||
|
|
||||||
|
/**
|
||||||
|
* This method checks {@param actualProperties} against {@param allowedProperties} if they are not system properties.
|
||||||
|
* A property is regarded as a system property if its name starts with a prefix in {@param systemPropertyPrefixes}.
|
||||||
|
* See org.apache.druid.server.initialization.JDBCAccessSecurityConfig for more details.
|
||||||
|
*
|
||||||
|
* If a non-system property that is not allowed is found, this method throws an {@link IllegalArgumentException}.
|
||||||
|
*/
|
||||||
|
public static void throwIfPropertiesAreNotAllowed(
|
||||||
|
Set<String> actualProperties,
|
||||||
|
Set<String> systemPropertyPrefixes,
|
||||||
|
Set<String> allowedProperties
|
||||||
|
)
|
||||||
|
{
|
||||||
|
for (String property : actualProperties) {
|
||||||
|
if (systemPropertyPrefixes.stream().noneMatch(property::startsWith)) {
|
||||||
|
Preconditions.checkArgument(
|
||||||
|
allowedProperties.contains(property),
|
||||||
|
"The property [%s] is not in the allowed list %s",
|
||||||
|
property,
|
||||||
|
allowedProperties
|
||||||
|
);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private ConnectionUriUtils()
|
||||||
|
{
|
||||||
|
}
|
||||||
|
}
|
|
@ -0,0 +1,40 @@
|
||||||
|
/*
|
||||||
|
* 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.
|
||||||
|
*/
|
||||||
|
|
||||||
|
package org.apache.druid.utils;
|
||||||
|
|
||||||
|
public final class Throwables
|
||||||
|
{
|
||||||
|
public static boolean isThrowable(Throwable t, Class<? extends Throwable> searchFor)
|
||||||
|
{
|
||||||
|
if (t.getClass().isAssignableFrom(searchFor)) {
|
||||||
|
return true;
|
||||||
|
} else {
|
||||||
|
if (t.getCause() != null) {
|
||||||
|
return isThrowable(t.getCause(), searchFor);
|
||||||
|
} else {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private Throwables()
|
||||||
|
{
|
||||||
|
}
|
||||||
|
}
|
|
@ -0,0 +1,90 @@
|
||||||
|
/*
|
||||||
|
* 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.
|
||||||
|
*/
|
||||||
|
|
||||||
|
package org.apache.druid.utils;
|
||||||
|
|
||||||
|
import com.google.common.collect.ImmutableSet;
|
||||||
|
import org.junit.Rule;
|
||||||
|
import org.junit.Test;
|
||||||
|
import org.junit.experimental.runners.Enclosed;
|
||||||
|
import org.junit.rules.ExpectedException;
|
||||||
|
import org.junit.runner.RunWith;
|
||||||
|
|
||||||
|
@RunWith(Enclosed.class)
|
||||||
|
public class ConnectionUriUtilsTest
|
||||||
|
{
|
||||||
|
public static class ThrowIfURLHasNotAllowedPropertiesTest
|
||||||
|
{
|
||||||
|
@Rule
|
||||||
|
public ExpectedException expectedException = ExpectedException.none();
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void testEmptyActualProperties()
|
||||||
|
{
|
||||||
|
ConnectionUriUtils.throwIfPropertiesAreNotAllowed(
|
||||||
|
ImmutableSet.of(),
|
||||||
|
ImmutableSet.of("valid_key1", "valid_key2"),
|
||||||
|
ImmutableSet.of("system_key1", "system_key2")
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void testThrowForNonAllowedProperties()
|
||||||
|
{
|
||||||
|
expectedException.expect(IllegalArgumentException.class);
|
||||||
|
expectedException.expectMessage("The property [invalid_key] is not in the allowed list [valid_key1, valid_key2]");
|
||||||
|
|
||||||
|
ConnectionUriUtils.throwIfPropertiesAreNotAllowed(
|
||||||
|
ImmutableSet.of("valid_key1", "invalid_key"),
|
||||||
|
ImmutableSet.of("system_key1", "system_key2"),
|
||||||
|
ImmutableSet.of("valid_key1", "valid_key2")
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void testAllowedProperties()
|
||||||
|
{
|
||||||
|
ConnectionUriUtils.throwIfPropertiesAreNotAllowed(
|
||||||
|
ImmutableSet.of("valid_key2"),
|
||||||
|
ImmutableSet.of("system_key1", "system_key2"),
|
||||||
|
ImmutableSet.of("valid_key1", "valid_key2")
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void testAllowSystemProperties()
|
||||||
|
{
|
||||||
|
ConnectionUriUtils.throwIfPropertiesAreNotAllowed(
|
||||||
|
ImmutableSet.of("system_key1", "valid_key2"),
|
||||||
|
ImmutableSet.of("system_key1", "system_key2"),
|
||||||
|
ImmutableSet.of("valid_key1", "valid_key2")
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void testMatchSystemProperties()
|
||||||
|
{
|
||||||
|
ConnectionUriUtils.throwIfPropertiesAreNotAllowed(
|
||||||
|
ImmutableSet.of("system_key1.1", "system_key1.5", "system_key11.11", "valid_key2"),
|
||||||
|
ImmutableSet.of("system_key1", "system_key2"),
|
||||||
|
ImmutableSet.of("valid_key1", "valid_key2")
|
||||||
|
);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
|
@ -0,0 +1,48 @@
|
||||||
|
/*
|
||||||
|
* 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.
|
||||||
|
*/
|
||||||
|
|
||||||
|
package org.apache.druid.utils;
|
||||||
|
|
||||||
|
import org.junit.Assert;
|
||||||
|
import org.junit.Test;
|
||||||
|
|
||||||
|
public class ThrowablesTest
|
||||||
|
{
|
||||||
|
@Test
|
||||||
|
public void testIsThrowableItself()
|
||||||
|
{
|
||||||
|
Assert.assertTrue(Throwables.isThrowable(new NoClassDefFoundError(), NoClassDefFoundError.class));
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void testIsThrowableNestedThrowable()
|
||||||
|
{
|
||||||
|
Assert.assertTrue(
|
||||||
|
Throwables.isThrowable(new RuntimeException(new NoClassDefFoundError()), NoClassDefFoundError.class)
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void testIsThrowableNonTarget()
|
||||||
|
{
|
||||||
|
Assert.assertFalse(
|
||||||
|
Throwables.isThrowable(new RuntimeException(new ClassNotFoundException()), NoClassDefFoundError.class)
|
||||||
|
);
|
||||||
|
}
|
||||||
|
}
|
|
@ -537,6 +537,25 @@ the [HTTP input source](../ingestion/native-batch.md#http-input-source) and the
|
||||||
|`druid.ingestion.http.allowedProtocols`|List of protocols|Allowed protocols for the HTTP input source and HTTP firehose.|["http", "https"]|
|
|`druid.ingestion.http.allowedProtocols`|List of protocols|Allowed protocols for the HTTP input source and HTTP firehose.|["http", "https"]|
|
||||||
|
|
||||||
|
|
||||||
|
### Ingestion Security Configuration
|
||||||
|
|
||||||
|
#### JDBC Connections to External Databases
|
||||||
|
|
||||||
|
You can use the following properties to specify permissible JDBC options for:
|
||||||
|
- [SQL input source](../ingestion/native-batch.md#sql-input-source)
|
||||||
|
- [SQL firehose](../ingestion/native-batch.md#sqlfirehose),
|
||||||
|
- [globally cached JDBC lookups](../development/extensions-core/lookups-cached-global.md#jdbc-lookup)
|
||||||
|
- [JDBC Data Fetcher for per-lookup caching](../development/extensions-core/druid-lookups.md#data-fetcher-layer).
|
||||||
|
|
||||||
|
These properties do not apply to metadata storage connections.
|
||||||
|
|
||||||
|
|Property|Possible Values|Description|Default|
|
||||||
|
|--------|---------------|-----------|-------|
|
||||||
|
|`druid.access.jdbc.enforceAllowedProperties`|Boolean|When true, Druid applies `druid.access.jdbc.allowedProperties` to JDBC connections starting with `jdbc:postgresql:` or `jdbc:mysql:`. When false, Druid allows any kind of JDBC connections without JDBC property validation. This config is deprecated and will be removed in a future release.|false|
|
||||||
|
|`druid.access.jdbc.allowedProperties`|List of JDBC properties|Defines a list of allowed JDBC properties. Druid always enforces the list for all JDBC connections starting with `jdbc:postgresql:` or `jdbc:mysql:` if `druid.access.jdbc.enforceAllowedProperties` is set to true.<br/><br/>This option is tested against MySQL connector 5.1.48 and PostgreSQL connector 42.2.14. Other connector versions might not work.|["useSSL", "requireSSL", "ssl", "sslmode"]|
|
||||||
|
|`druid.access.jdbc.allowUnknownJdbcUrlFormat`|Boolean|When false, Druid only accepts JDBC connections starting with `jdbc:postgresql:` or `jdbc:mysql:`. When true, Druid allows JDBC connections to any kind of database, but only enforces `druid.access.jdbc.allowedProperties` for PostgreSQL and MySQL.|true|
|
||||||
|
|
||||||
|
|
||||||
### Task Logging
|
### Task Logging
|
||||||
|
|
||||||
If you are running the indexing service in remote mode, the task logs must be stored in S3, Azure Blob Store, Google Cloud Storage or HDFS.
|
If you are running the indexing service in remote mode, the task logs must be stored in S3, Azure Blob Store, Google Cloud Storage or HDFS.
|
||||||
|
|
|
@ -72,7 +72,7 @@ Same for Loading cache, developer can implement a new type of loading cache by i
|
||||||
|
|
||||||
|Field|Type|Description|Required|default|
|
|Field|Type|Description|Required|default|
|
||||||
|-----|----|-----------|--------|-------|
|
|-----|----|-----------|--------|-------|
|
||||||
|dataFetcher|JSON object|Specifies the lookup data fetcher type to use in order to fetch data|yes|null|
|
|dataFetcher|JSON object|Specifies the lookup data fetcher type for fetching data|yes|null|
|
||||||
|cacheFactory|JSON Object|Cache factory implementation|no |onHeapPolling|
|
|cacheFactory|JSON Object|Cache factory implementation|no |onHeapPolling|
|
||||||
|pollPeriod|Period|polling period |no |null (poll once)|
|
|pollPeriod|Period|polling period |no |null (poll once)|
|
||||||
|
|
||||||
|
@ -129,7 +129,7 @@ Guava cache configuration spec.
|
||||||
"type":"loadingLookup",
|
"type":"loadingLookup",
|
||||||
"dataFetcher":{ "type":"jdbcDataFetcher", "connectorConfig":"jdbc://mysql://localhost:3306/my_data_base", "table":"lookup_table_name", "keyColumn":"key_column_name", "valueColumn": "value_column_name"},
|
"dataFetcher":{ "type":"jdbcDataFetcher", "connectorConfig":"jdbc://mysql://localhost:3306/my_data_base", "table":"lookup_table_name", "keyColumn":"key_column_name", "valueColumn": "value_column_name"},
|
||||||
"loadingCacheSpec":{"type":"guava"},
|
"loadingCacheSpec":{"type":"guava"},
|
||||||
"reverseLoadingCacheSpec":{"type":"guava", "maximumSize":500000, "expireAfterAccess":100000, "expireAfterAccess":10000}
|
"reverseLoadingCacheSpec":{"type":"guava", "maximumSize":500000, "expireAfterAccess":100000, "expireAfterWrite":10000}
|
||||||
}
|
}
|
||||||
```
|
```
|
||||||
|
|
||||||
|
@ -150,6 +150,16 @@ Off heap cache is backed by [MapDB](http://www.mapdb.org/) implementation. MapDB
|
||||||
"type":"loadingLookup",
|
"type":"loadingLookup",
|
||||||
"dataFetcher":{ "type":"jdbcDataFetcher", "connectorConfig":"jdbc://mysql://localhost:3306/my_data_base", "table":"lookup_table_name", "keyColumn":"key_column_name", "valueColumn": "value_column_name"},
|
"dataFetcher":{ "type":"jdbcDataFetcher", "connectorConfig":"jdbc://mysql://localhost:3306/my_data_base", "table":"lookup_table_name", "keyColumn":"key_column_name", "valueColumn": "value_column_name"},
|
||||||
"loadingCacheSpec":{"type":"mapDb", "maxEntriesSize":100000},
|
"loadingCacheSpec":{"type":"mapDb", "maxEntriesSize":100000},
|
||||||
"reverseLoadingCacheSpec":{"type":"mapDb", "maxStoreSize":5, "expireAfterAccess":100000, "expireAfterAccess":10000}
|
"reverseLoadingCacheSpec":{"type":"mapDb", "maxStoreSize":5, "expireAfterAccess":100000, "expireAfterWrite":10000}
|
||||||
}
|
}
|
||||||
```
|
```
|
||||||
|
|
||||||
|
### JDBC Data Fetcher
|
||||||
|
|
||||||
|
|Field|Type|Description|Required|default|
|
||||||
|
|-----|----|-----------|--------|-------|
|
||||||
|
|`connectorConfig`|JSON object|Specifies the database connection details. You can set `connectURI`, `user` and `password`. You can selectively allow JDBC properties in `connectURI`. See [JDBC connections security config](../../configuration/index.md#jdbc-connections-to-external-databases) for more details.|yes||
|
||||||
|
|`table`|string|The table name to read from.|yes||
|
||||||
|
|`keyColumn`|string|The column name that contains the lookup key.|yes||
|
||||||
|
|`valueColumn`|string|The column name that contains the lookup value.|yes||
|
||||||
|
|`streamingFetchSize`|int|Fetch size used in JDBC connections.|no|1000|
|
||||||
|
|
|
@ -64,7 +64,6 @@ Globally cached lookups can be specified as part of the [cluster wide config for
|
||||||
"extractionNamespace": {
|
"extractionNamespace": {
|
||||||
"type": "jdbc",
|
"type": "jdbc",
|
||||||
"connectorConfig": {
|
"connectorConfig": {
|
||||||
"createTables": true,
|
|
||||||
"connectURI": "jdbc:mysql:\/\/localhost:3306\/druid",
|
"connectURI": "jdbc:mysql:\/\/localhost:3306\/druid",
|
||||||
"user": "druid",
|
"user": "druid",
|
||||||
"password": "diurd"
|
"password": "diurd"
|
||||||
|
@ -107,7 +106,6 @@ In a simple case where only one [tier](../../querying/lookups.md#dynamic-configu
|
||||||
"extractionNamespace": {
|
"extractionNamespace": {
|
||||||
"type": "jdbc",
|
"type": "jdbc",
|
||||||
"connectorConfig": {
|
"connectorConfig": {
|
||||||
"createTables": true,
|
|
||||||
"connectURI": "jdbc:mysql:\/\/localhost:3306\/druid",
|
"connectURI": "jdbc:mysql:\/\/localhost:3306\/druid",
|
||||||
"user": "druid",
|
"user": "druid",
|
||||||
"password": "diurd"
|
"password": "diurd"
|
||||||
|
@ -136,7 +134,6 @@ Where the Coordinator endpoint `/druid/coordinator/v1/lookups/realtime_customer2
|
||||||
"extractionNamespace": {
|
"extractionNamespace": {
|
||||||
"type": "jdbc",
|
"type": "jdbc",
|
||||||
"connectorConfig": {
|
"connectorConfig": {
|
||||||
"createTables": true,
|
|
||||||
"connectURI": "jdbc:mysql://localhost:3306/druid",
|
"connectURI": "jdbc:mysql://localhost:3306/druid",
|
||||||
"user": "druid",
|
"user": "druid",
|
||||||
"password": "diurd"
|
"password": "diurd"
|
||||||
|
@ -347,7 +344,7 @@ The JDBC lookups will poll a database to populate its local cache. If the `tsCol
|
||||||
|
|
||||||
|Parameter|Description|Required|Default|
|
|Parameter|Description|Required|Default|
|
||||||
|---------|-----------|--------|-------|
|
|---------|-----------|--------|-------|
|
||||||
|`connectorConfig`|The connector config to use|Yes||
|
|`connectorConfig`|The connector config to use. You can set `connectURI`, `user` and `password`. You can selectively allow JDBC properties in `connectURI`. See [JDBC connections security config](../../configuration/index.md#jdbc-connections-to-external-databases) for more details.|Yes||
|
||||||
|`table`|The table which contains the key value pairs|Yes||
|
|`table`|The table which contains the key value pairs|Yes||
|
||||||
|`keyColumn`|The column in `table` which contains the keys|Yes||
|
|`keyColumn`|The column in `table` which contains the keys|Yes||
|
||||||
|`valueColumn`|The column in `table` which contains the values|Yes||
|
|`valueColumn`|The column in `table` which contains the values|Yes||
|
||||||
|
@ -359,7 +356,6 @@ The JDBC lookups will poll a database to populate its local cache. If the `tsCol
|
||||||
{
|
{
|
||||||
"type":"jdbc",
|
"type":"jdbc",
|
||||||
"connectorConfig":{
|
"connectorConfig":{
|
||||||
"createTables":true,
|
|
||||||
"connectURI":"jdbc:mysql://localhost:3306/druid",
|
"connectURI":"jdbc:mysql://localhost:3306/druid",
|
||||||
"user":"druid",
|
"user":"druid",
|
||||||
"password":"diurd"
|
"password":"diurd"
|
||||||
|
|
|
@ -105,7 +105,6 @@ Copy or symlink this file to `extensions/mysql-metadata-storage` under the distr
|
||||||
|`druid.metadata.mysql.ssl.enabledSSLCipherSuites`|Overrides the existing cipher suites with these cipher suites.|none|no|
|
|`druid.metadata.mysql.ssl.enabledSSLCipherSuites`|Overrides the existing cipher suites with these cipher suites.|none|no|
|
||||||
|`druid.metadata.mysql.ssl.enabledTLSProtocols`|Overrides the TLS protocols with these protocols.|none|no|
|
|`druid.metadata.mysql.ssl.enabledTLSProtocols`|Overrides the TLS protocols with these protocols.|none|no|
|
||||||
|
|
||||||
|
|
||||||
### MySQL Firehose
|
### MySQL Firehose
|
||||||
|
|
||||||
The MySQL extension provides an implementation of an [SqlFirehose](../../ingestion/native-batch.md#firehoses-deprecated) which can be used to ingest data into Druid from a MySQL database.
|
The MySQL extension provides an implementation of an [SqlFirehose](../../ingestion/native-batch.md#firehoses-deprecated) which can be used to ingest data into Druid from a MySQL database.
|
||||||
|
|
|
@ -1409,7 +1409,7 @@ Please refer to the Recommended practices section below before using this input
|
||||||
|property|description|required?|
|
|property|description|required?|
|
||||||
|--------|-----------|---------|
|
|--------|-----------|---------|
|
||||||
|type|This should be "sql".|Yes|
|
|type|This should be "sql".|Yes|
|
||||||
|database|Specifies the database connection details. The database type corresponds to the extension that supplies the `connectorConfig` support and this extension must be loaded into Druid. For database types `mysql` and `postgresql`, the `connectorConfig` support is provided by [mysql-metadata-storage](../development/extensions-core/mysql.md) and [postgresql-metadata-storage](../development/extensions-core/postgresql.md) extensions respectively.|Yes|
|
|database|Specifies the database connection details. The database type corresponds to the extension that supplies the `connectorConfig` support. The specified extension must be loaded into Druid:<br/><br/><ul><li>[mysql-metadata-storage](../development/extensions-core/mysql.md) for `mysql`</li><li> [postgresql-metadata-storage](../development/extensions-core/postgresql.md) extension for `postgresql`.</li></ul><br/><br/>You can selectively allow JDBC properties in `connectURI`. See [JDBC connections security config](../configuration/index.md#jdbc-connections-to-external-databases) for more details.|Yes|
|
||||||
|foldCase|Toggle case folding of database column names. This may be enabled in cases where the database returns case insensitive column names in query results.|No|
|
|foldCase|Toggle case folding of database column names. This may be enabled in cases where the database returns case insensitive column names in query results.|No|
|
||||||
|sqls|List of SQL queries where each SQL query would retrieve the data to be indexed.|Yes|
|
|sqls|List of SQL queries where each SQL query would retrieve the data to be indexed.|Yes|
|
||||||
|
|
||||||
|
@ -1763,7 +1763,7 @@ Requires one of the following extensions:
|
||||||
|property|description|default|required?|
|
|property|description|default|required?|
|
||||||
|--------|-----------|-------|---------|
|
|--------|-----------|-------|---------|
|
||||||
|type|This should be "sql".||Yes|
|
|type|This should be "sql".||Yes|
|
||||||
|database|Specifies the database connection details.||Yes|
|
|database|Specifies the database connection details. The database type corresponds to the extension that supplies the `connectorConfig` support. The specified extension must be loaded into Druid:<br/><br/><ul><li>[mysql-metadata-storage](../development/extensions-core/mysql.md) for `mysql`</li><li> [postgresql-metadata-storage](../development/extensions-core/postgresql.md) extension for `postgresql`.</li></ul><br/><br/>You can selectively allow JDBC properties in `connectURI`. See [JDBC connections security config](../configuration/index.md#jdbc-connections-to-external-databases) for more details.||Yes|
|
||||||
|maxCacheCapacityBytes|Maximum size of the cache space in bytes. 0 means disabling cache. Cached files are not removed until the ingestion task completes.|1073741824|No|
|
|maxCacheCapacityBytes|Maximum size of the cache space in bytes. 0 means disabling cache. Cached files are not removed until the ingestion task completes.|1073741824|No|
|
||||||
|maxFetchCapacityBytes|Maximum size of the fetch space in bytes. 0 means disabling prefetch. Prefetched files are removed immediately once they are read.|1073741824|No|
|
|maxFetchCapacityBytes|Maximum size of the fetch space in bytes. 0 means disabling prefetch. Prefetched files are removed immediately once they are read.|1073741824|No|
|
||||||
|prefetchTriggerBytes|Threshold to trigger prefetching SQL result objects.|maxFetchCapacityBytes / 2|No|
|
|prefetchTriggerBytes|Threshold to trigger prefetching SQL result objects.|maxFetchCapacityBytes / 2|No|
|
||||||
|
|
|
@ -110,12 +110,15 @@
|
||||||
<artifactId>jsr311-api</artifactId>
|
<artifactId>jsr311-api</artifactId>
|
||||||
<scope>provided</scope>
|
<scope>provided</scope>
|
||||||
</dependency>
|
</dependency>
|
||||||
|
<dependency>
|
||||||
<!-- Included to improve the out-of-the-box experience for supported JDBC connectors -->
|
<groupId>mysql</groupId>
|
||||||
|
<artifactId>mysql-connector-java</artifactId>
|
||||||
|
<version>${mysql.version}</version>
|
||||||
|
<scope>provided</scope>
|
||||||
|
</dependency>
|
||||||
<dependency>
|
<dependency>
|
||||||
<groupId>org.postgresql</groupId>
|
<groupId>org.postgresql</groupId>
|
||||||
<artifactId>postgresql</artifactId>
|
<artifactId>postgresql</artifactId>
|
||||||
<scope>runtime</scope>
|
|
||||||
</dependency>
|
</dependency>
|
||||||
|
|
||||||
<!-- Tests -->
|
<!-- Tests -->
|
||||||
|
|
|
@ -19,17 +19,28 @@
|
||||||
|
|
||||||
package org.apache.druid.query.lookup.namespace;
|
package org.apache.druid.query.lookup.namespace;
|
||||||
|
|
||||||
|
import com.fasterxml.jackson.annotation.JacksonInject;
|
||||||
import com.fasterxml.jackson.annotation.JsonCreator;
|
import com.fasterxml.jackson.annotation.JsonCreator;
|
||||||
import com.fasterxml.jackson.annotation.JsonProperty;
|
import com.fasterxml.jackson.annotation.JsonProperty;
|
||||||
import com.fasterxml.jackson.annotation.JsonTypeName;
|
import com.fasterxml.jackson.annotation.JsonTypeName;
|
||||||
import com.google.common.base.Preconditions;
|
import com.google.common.base.Preconditions;
|
||||||
|
import com.google.common.collect.Sets;
|
||||||
|
import com.mysql.jdbc.NonRegisteringDriver;
|
||||||
|
import org.apache.druid.java.util.common.IAE;
|
||||||
import org.apache.druid.metadata.MetadataStorageConnectorConfig;
|
import org.apache.druid.metadata.MetadataStorageConnectorConfig;
|
||||||
|
import org.apache.druid.server.initialization.JdbcAccessSecurityConfig;
|
||||||
|
import org.apache.druid.utils.ConnectionUriUtils;
|
||||||
|
import org.apache.druid.utils.Throwables;
|
||||||
import org.joda.time.Period;
|
import org.joda.time.Period;
|
||||||
|
import org.postgresql.Driver;
|
||||||
|
|
||||||
import javax.annotation.Nullable;
|
import javax.annotation.Nullable;
|
||||||
import javax.validation.constraints.Min;
|
import javax.validation.constraints.Min;
|
||||||
import javax.validation.constraints.NotNull;
|
import javax.validation.constraints.NotNull;
|
||||||
|
import java.sql.SQLException;
|
||||||
import java.util.Objects;
|
import java.util.Objects;
|
||||||
|
import java.util.Properties;
|
||||||
|
import java.util.Set;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
*
|
*
|
||||||
|
@ -61,11 +72,15 @@ public class JdbcExtractionNamespace implements ExtractionNamespace
|
||||||
@NotNull @JsonProperty(value = "valueColumn", required = true) final String valueColumn,
|
@NotNull @JsonProperty(value = "valueColumn", required = true) final String valueColumn,
|
||||||
@JsonProperty(value = "tsColumn", required = false) @Nullable final String tsColumn,
|
@JsonProperty(value = "tsColumn", required = false) @Nullable final String tsColumn,
|
||||||
@JsonProperty(value = "filter", required = false) @Nullable final String filter,
|
@JsonProperty(value = "filter", required = false) @Nullable final String filter,
|
||||||
@Min(0) @JsonProperty(value = "pollPeriod", required = false) @Nullable final Period pollPeriod
|
@Min(0) @JsonProperty(value = "pollPeriod", required = false) @Nullable final Period pollPeriod,
|
||||||
|
@JacksonInject JdbcAccessSecurityConfig securityConfig
|
||||||
)
|
)
|
||||||
{
|
{
|
||||||
this.connectorConfig = Preconditions.checkNotNull(connectorConfig, "connectorConfig");
|
this.connectorConfig = Preconditions.checkNotNull(connectorConfig, "connectorConfig");
|
||||||
Preconditions.checkNotNull(connectorConfig.getConnectURI(), "connectorConfig.connectURI");
|
// Check the properties in the connection URL. Note that JdbcExtractionNamespace doesn't use
|
||||||
|
// MetadataStorageConnectorConfig.getDbcpProperties(). If we want to use them,
|
||||||
|
// those DBCP properties should be validated using the same logic.
|
||||||
|
checkConnectionURL(connectorConfig.getConnectURI(), securityConfig);
|
||||||
this.table = Preconditions.checkNotNull(table, "table");
|
this.table = Preconditions.checkNotNull(table, "table");
|
||||||
this.keyColumn = Preconditions.checkNotNull(keyColumn, "keyColumn");
|
this.keyColumn = Preconditions.checkNotNull(keyColumn, "keyColumn");
|
||||||
this.valueColumn = Preconditions.checkNotNull(valueColumn, "valueColumn");
|
this.valueColumn = Preconditions.checkNotNull(valueColumn, "valueColumn");
|
||||||
|
@ -74,6 +89,89 @@ public class JdbcExtractionNamespace implements ExtractionNamespace
|
||||||
this.pollPeriod = pollPeriod == null ? new Period(0L) : pollPeriod;
|
this.pollPeriod = pollPeriod == null ? new Period(0L) : pollPeriod;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Check the given URL whether it contains non-allowed properties.
|
||||||
|
*
|
||||||
|
* This method should be in sync with the following methods:
|
||||||
|
*
|
||||||
|
* - {@code org.apache.druid.server.lookup.jdbc.JdbcDataFetcher.checkConnectionURL()}
|
||||||
|
* - {@code org.apache.druid.firehose.sql.MySQLFirehoseDatabaseConnector.findPropertyKeysFromConnectURL()}
|
||||||
|
* - {@code org.apache.druid.firehose.sql.PostgresqlFirehoseDatabaseConnector.findPropertyKeysFromConnectURL()}
|
||||||
|
*
|
||||||
|
* @see JdbcAccessSecurityConfig#getAllowedProperties()
|
||||||
|
*/
|
||||||
|
private static void checkConnectionURL(String url, JdbcAccessSecurityConfig securityConfig)
|
||||||
|
{
|
||||||
|
Preconditions.checkNotNull(url, "connectorConfig.connectURI");
|
||||||
|
|
||||||
|
if (!securityConfig.isEnforceAllowedProperties()) {
|
||||||
|
// You don't want to do anything with properties.
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Nullable final Properties properties; // null when url has an invalid format
|
||||||
|
|
||||||
|
if (url.startsWith(ConnectionUriUtils.MYSQL_PREFIX)) {
|
||||||
|
try {
|
||||||
|
NonRegisteringDriver driver = new NonRegisteringDriver();
|
||||||
|
properties = driver.parseURL(url, null);
|
||||||
|
}
|
||||||
|
catch (SQLException e) {
|
||||||
|
throw new RuntimeException(e);
|
||||||
|
}
|
||||||
|
catch (Throwable e) {
|
||||||
|
if (Throwables.isThrowable(e, NoClassDefFoundError.class)
|
||||||
|
|| Throwables.isThrowable(e, ClassNotFoundException.class)) {
|
||||||
|
if (e.getMessage().contains("com/mysql/jdbc/NonRegisteringDriver")) {
|
||||||
|
throw new RuntimeException(
|
||||||
|
"Failed to find MySQL driver class. Please check the MySQL connector version 5.1.48 is in the classpath",
|
||||||
|
e
|
||||||
|
);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
throw new RuntimeException(e);
|
||||||
|
}
|
||||||
|
} else if (url.startsWith(ConnectionUriUtils.POSTGRES_PREFIX)) {
|
||||||
|
try {
|
||||||
|
properties = Driver.parseURL(url, null);
|
||||||
|
}
|
||||||
|
catch (Throwable e) {
|
||||||
|
if (Throwables.isThrowable(e, NoClassDefFoundError.class)
|
||||||
|
|| Throwables.isThrowable(e, ClassNotFoundException.class)) {
|
||||||
|
if (e.getMessage().contains("org/postgresql/Driver")) {
|
||||||
|
throw new RuntimeException(
|
||||||
|
"Failed to find PostgreSQL driver class. "
|
||||||
|
+ "Please check the PostgreSQL connector version 42.2.14 is in the classpath",
|
||||||
|
e
|
||||||
|
);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
throw new RuntimeException(e);
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
if (securityConfig.isAllowUnknownJdbcUrlFormat()) {
|
||||||
|
properties = new Properties();
|
||||||
|
} else {
|
||||||
|
// unknown format but it is not allowed
|
||||||
|
throw new IAE("Unknown JDBC connection scheme: %s", url.split(":")[1]);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (properties == null) {
|
||||||
|
// There is something wrong with the URL format.
|
||||||
|
throw new IAE("Invalid URL format [%s]", url);
|
||||||
|
}
|
||||||
|
|
||||||
|
final Set<String> propertyKeys = Sets.newHashSetWithExpectedSize(properties.size());
|
||||||
|
properties.forEach((k, v) -> propertyKeys.add((String) k));
|
||||||
|
|
||||||
|
ConnectionUriUtils.throwIfPropertiesAreNotAllowed(
|
||||||
|
propertyKeys,
|
||||||
|
securityConfig.getSystemPropertyPrefixes(),
|
||||||
|
securityConfig.getAllowedProperties()
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
public MetadataStorageConnectorConfig getConnectorConfig()
|
public MetadataStorageConnectorConfig getConnectorConfig()
|
||||||
{
|
{
|
||||||
return connectorConfig;
|
return connectorConfig;
|
||||||
|
|
|
@ -0,0 +1,431 @@
|
||||||
|
/*
|
||||||
|
* 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.
|
||||||
|
*/
|
||||||
|
|
||||||
|
package org.apache.druid.query.lookup.namespace;
|
||||||
|
|
||||||
|
import com.google.common.collect.ImmutableSet;
|
||||||
|
import org.apache.druid.metadata.MetadataStorageConnectorConfig;
|
||||||
|
import org.apache.druid.server.initialization.JdbcAccessSecurityConfig;
|
||||||
|
import org.joda.time.Period;
|
||||||
|
import org.junit.Rule;
|
||||||
|
import org.junit.Test;
|
||||||
|
import org.junit.experimental.runners.Enclosed;
|
||||||
|
import org.junit.rules.ExpectedException;
|
||||||
|
import org.junit.runner.RunWith;
|
||||||
|
|
||||||
|
import java.util.Set;
|
||||||
|
|
||||||
|
@RunWith(Enclosed.class)
|
||||||
|
public class JdbcExtractionNamespaceUrlCheckTest
|
||||||
|
{
|
||||||
|
private static final String TABLE_NAME = "abstractDbRenameTest";
|
||||||
|
private static final String KEY_NAME = "keyName";
|
||||||
|
private static final String VAL_NAME = "valName";
|
||||||
|
private static final String TS_COLUMN = "tsColumn";
|
||||||
|
|
||||||
|
public static class MySqlTest
|
||||||
|
{
|
||||||
|
@Rule
|
||||||
|
public ExpectedException expectedException = ExpectedException.none();
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void testCreateInstanceWhenUrlHasOnlyAllowedProperties()
|
||||||
|
{
|
||||||
|
new JdbcExtractionNamespace(
|
||||||
|
new MetadataStorageConnectorConfig()
|
||||||
|
{
|
||||||
|
@Override
|
||||||
|
public String getConnectURI()
|
||||||
|
{
|
||||||
|
return "jdbc:mysql://localhost:3306/db?valid_key1=val1&valid_key2=val2";
|
||||||
|
}
|
||||||
|
},
|
||||||
|
TABLE_NAME,
|
||||||
|
KEY_NAME,
|
||||||
|
VAL_NAME,
|
||||||
|
TS_COLUMN,
|
||||||
|
"some filter",
|
||||||
|
new Period(10),
|
||||||
|
new JdbcAccessSecurityConfig()
|
||||||
|
{
|
||||||
|
@Override
|
||||||
|
public Set<String> getAllowedProperties()
|
||||||
|
{
|
||||||
|
return ImmutableSet.of("valid_key1", "valid_key2");
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public boolean isEnforceAllowedProperties()
|
||||||
|
{
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void testThrowWhenUrlHasNonAllowedPropertiesWhenEnforcingAllowedProperties()
|
||||||
|
{
|
||||||
|
expectedException.expect(IllegalArgumentException.class);
|
||||||
|
expectedException.expectMessage("The property [invalid_key1] is not in the allowed list [valid_key1, valid_key2]");
|
||||||
|
new JdbcExtractionNamespace(
|
||||||
|
new MetadataStorageConnectorConfig()
|
||||||
|
{
|
||||||
|
@Override
|
||||||
|
public String getConnectURI()
|
||||||
|
{
|
||||||
|
return "jdbc:mysql://localhost:3306/db?invalid_key1=val1&valid_key2=val2";
|
||||||
|
}
|
||||||
|
},
|
||||||
|
TABLE_NAME,
|
||||||
|
KEY_NAME,
|
||||||
|
VAL_NAME,
|
||||||
|
TS_COLUMN,
|
||||||
|
"some filter",
|
||||||
|
new Period(10),
|
||||||
|
new JdbcAccessSecurityConfig()
|
||||||
|
{
|
||||||
|
@Override
|
||||||
|
public Set<String> getAllowedProperties()
|
||||||
|
{
|
||||||
|
return ImmutableSet.of("valid_key1", "valid_key2");
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public boolean isEnforceAllowedProperties()
|
||||||
|
{
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void testWhenUrlHasNonAllowedPropertiesWhenNotEnforcingAllowedProperties()
|
||||||
|
{
|
||||||
|
new JdbcExtractionNamespace(
|
||||||
|
new MetadataStorageConnectorConfig()
|
||||||
|
{
|
||||||
|
@Override
|
||||||
|
public String getConnectURI()
|
||||||
|
{
|
||||||
|
return "jdbc:mysql://localhost:3306/db?invalid_key1=val1&valid_key2=val2";
|
||||||
|
}
|
||||||
|
},
|
||||||
|
TABLE_NAME,
|
||||||
|
KEY_NAME,
|
||||||
|
VAL_NAME,
|
||||||
|
TS_COLUMN,
|
||||||
|
"some filter",
|
||||||
|
new Period(10),
|
||||||
|
new JdbcAccessSecurityConfig()
|
||||||
|
{
|
||||||
|
@Override
|
||||||
|
public Set<String> getAllowedProperties()
|
||||||
|
{
|
||||||
|
return ImmutableSet.of("valid_key1", "valid_key2");
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public boolean isEnforceAllowedProperties()
|
||||||
|
{
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void testWhenInvalidUrlFormat()
|
||||||
|
{
|
||||||
|
expectedException.expect(IllegalArgumentException.class);
|
||||||
|
expectedException.expectMessage("Invalid URL format [jdbc:mysql:/invalid-url::3006]");
|
||||||
|
new JdbcExtractionNamespace(
|
||||||
|
new MetadataStorageConnectorConfig()
|
||||||
|
{
|
||||||
|
@Override
|
||||||
|
public String getConnectURI()
|
||||||
|
{
|
||||||
|
return "jdbc:mysql:/invalid-url::3006";
|
||||||
|
}
|
||||||
|
},
|
||||||
|
TABLE_NAME,
|
||||||
|
KEY_NAME,
|
||||||
|
VAL_NAME,
|
||||||
|
TS_COLUMN,
|
||||||
|
"some filter",
|
||||||
|
new Period(10),
|
||||||
|
new JdbcAccessSecurityConfig()
|
||||||
|
{
|
||||||
|
@Override
|
||||||
|
public Set<String> getAllowedProperties()
|
||||||
|
{
|
||||||
|
return ImmutableSet.of("valid_key1", "valid_key2");
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public boolean isEnforceAllowedProperties()
|
||||||
|
{
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public static class PostgreSqlTest
|
||||||
|
{
|
||||||
|
@Rule
|
||||||
|
public ExpectedException expectedException = ExpectedException.none();
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void testCreateInstanceWhenUrlHasOnlyAllowedProperties()
|
||||||
|
{
|
||||||
|
new JdbcExtractionNamespace(
|
||||||
|
new MetadataStorageConnectorConfig()
|
||||||
|
{
|
||||||
|
@Override
|
||||||
|
public String getConnectURI()
|
||||||
|
{
|
||||||
|
return "jdbc:postgresql://localhost:5432/db?valid_key1=val1&valid_key2=val2";
|
||||||
|
}
|
||||||
|
},
|
||||||
|
TABLE_NAME,
|
||||||
|
KEY_NAME,
|
||||||
|
VAL_NAME,
|
||||||
|
TS_COLUMN,
|
||||||
|
"some filter",
|
||||||
|
new Period(10),
|
||||||
|
new JdbcAccessSecurityConfig()
|
||||||
|
{
|
||||||
|
@Override
|
||||||
|
public Set<String> getAllowedProperties()
|
||||||
|
{
|
||||||
|
return ImmutableSet.of("valid_key1", "valid_key2");
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public boolean isEnforceAllowedProperties()
|
||||||
|
{
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void testThrowWhenUrlHasNonAllowedPropertiesWhenEnforcingAllowedProperties()
|
||||||
|
{
|
||||||
|
expectedException.expect(IllegalArgumentException.class);
|
||||||
|
expectedException.expectMessage("The property [invalid_key1] is not in the allowed list [valid_key1, valid_key2]");
|
||||||
|
new JdbcExtractionNamespace(
|
||||||
|
new MetadataStorageConnectorConfig()
|
||||||
|
{
|
||||||
|
@Override
|
||||||
|
public String getConnectURI()
|
||||||
|
{
|
||||||
|
return "jdbc:postgresql://localhost:5432/db?invalid_key1=val1&valid_key2=val2";
|
||||||
|
}
|
||||||
|
},
|
||||||
|
TABLE_NAME,
|
||||||
|
KEY_NAME,
|
||||||
|
VAL_NAME,
|
||||||
|
TS_COLUMN,
|
||||||
|
"some filter",
|
||||||
|
new Period(10),
|
||||||
|
new JdbcAccessSecurityConfig()
|
||||||
|
{
|
||||||
|
@Override
|
||||||
|
public Set<String> getAllowedProperties()
|
||||||
|
{
|
||||||
|
return ImmutableSet.of("valid_key1", "valid_key2");
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public boolean isEnforceAllowedProperties()
|
||||||
|
{
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void testWhenUrlHasNonAllowedPropertiesWhenNotEnforcingAllowedProperties()
|
||||||
|
{
|
||||||
|
new JdbcExtractionNamespace(
|
||||||
|
new MetadataStorageConnectorConfig()
|
||||||
|
{
|
||||||
|
@Override
|
||||||
|
public String getConnectURI()
|
||||||
|
{
|
||||||
|
return "jdbc:postgresql://localhost:5432/db?invalid_key1=val1&valid_key2=val2";
|
||||||
|
}
|
||||||
|
},
|
||||||
|
TABLE_NAME,
|
||||||
|
KEY_NAME,
|
||||||
|
VAL_NAME,
|
||||||
|
TS_COLUMN,
|
||||||
|
"some filter",
|
||||||
|
new Period(10),
|
||||||
|
new JdbcAccessSecurityConfig()
|
||||||
|
{
|
||||||
|
@Override
|
||||||
|
public Set<String> getAllowedProperties()
|
||||||
|
{
|
||||||
|
return ImmutableSet.of("valid_key1", "valid_key2");
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public boolean isEnforceAllowedProperties()
|
||||||
|
{
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void testWhenInvalidUrlFormat()
|
||||||
|
{
|
||||||
|
expectedException.expect(IllegalArgumentException.class);
|
||||||
|
expectedException.expectMessage("Invalid URL format [jdbc:postgresql://invalid-url::3006]");
|
||||||
|
new JdbcExtractionNamespace(
|
||||||
|
new MetadataStorageConnectorConfig()
|
||||||
|
{
|
||||||
|
@Override
|
||||||
|
public String getConnectURI()
|
||||||
|
{
|
||||||
|
return "jdbc:postgresql://invalid-url::3006";
|
||||||
|
}
|
||||||
|
},
|
||||||
|
TABLE_NAME,
|
||||||
|
KEY_NAME,
|
||||||
|
VAL_NAME,
|
||||||
|
TS_COLUMN,
|
||||||
|
"some filter",
|
||||||
|
new Period(10),
|
||||||
|
new JdbcAccessSecurityConfig()
|
||||||
|
{
|
||||||
|
@Override
|
||||||
|
public Set<String> getAllowedProperties()
|
||||||
|
{
|
||||||
|
return ImmutableSet.of("valid_key1", "valid_key2");
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public boolean isEnforceAllowedProperties()
|
||||||
|
{
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public static class UnknownSchemeTest
|
||||||
|
{
|
||||||
|
@Rule
|
||||||
|
public ExpectedException expectedException = ExpectedException.none();
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void testThrowWhenUnknownFormatIsNotAllowed()
|
||||||
|
{
|
||||||
|
expectedException.expect(IllegalArgumentException.class);
|
||||||
|
expectedException.expectMessage("Unknown JDBC connection scheme: mydb");
|
||||||
|
new JdbcExtractionNamespace(
|
||||||
|
new MetadataStorageConnectorConfig()
|
||||||
|
{
|
||||||
|
@Override
|
||||||
|
public String getConnectURI()
|
||||||
|
{
|
||||||
|
return "jdbc:mydb://localhost:5432/db?valid_key1=val1&valid_key2=val2";
|
||||||
|
}
|
||||||
|
},
|
||||||
|
TABLE_NAME,
|
||||||
|
KEY_NAME,
|
||||||
|
VAL_NAME,
|
||||||
|
TS_COLUMN,
|
||||||
|
"some filter",
|
||||||
|
new Period(10),
|
||||||
|
new JdbcAccessSecurityConfig()
|
||||||
|
{
|
||||||
|
@Override
|
||||||
|
public Set<String> getAllowedProperties()
|
||||||
|
{
|
||||||
|
return ImmutableSet.of("valid_key1", "valid_key2");
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public boolean isAllowUnknownJdbcUrlFormat()
|
||||||
|
{
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public boolean isEnforceAllowedProperties()
|
||||||
|
{
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void testSkipUrlParsingWhenUnknownFormatIsAllowed()
|
||||||
|
{
|
||||||
|
new JdbcExtractionNamespace(
|
||||||
|
new MetadataStorageConnectorConfig()
|
||||||
|
{
|
||||||
|
@Override
|
||||||
|
public String getConnectURI()
|
||||||
|
{
|
||||||
|
return "jdbc:mydb://localhost:5432/db?valid_key1=val1&valid_key2=val2";
|
||||||
|
}
|
||||||
|
},
|
||||||
|
TABLE_NAME,
|
||||||
|
KEY_NAME,
|
||||||
|
VAL_NAME,
|
||||||
|
TS_COLUMN,
|
||||||
|
"some filter",
|
||||||
|
new Period(10),
|
||||||
|
new JdbcAccessSecurityConfig()
|
||||||
|
{
|
||||||
|
@Override
|
||||||
|
public Set<String> getAllowedProperties()
|
||||||
|
{
|
||||||
|
return ImmutableSet.of("valid_key1", "valid_key2");
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public boolean isAllowUnknownJdbcUrlFormat()
|
||||||
|
{
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public boolean isEnforceAllowedProperties()
|
||||||
|
{
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
|
@ -24,6 +24,7 @@ import org.apache.druid.java.util.common.lifecycle.Lifecycle;
|
||||||
import org.apache.druid.java.util.emitter.service.ServiceEmitter;
|
import org.apache.druid.java.util.emitter.service.ServiceEmitter;
|
||||||
import org.apache.druid.metadata.MetadataStorageConnectorConfig;
|
import org.apache.druid.metadata.MetadataStorageConnectorConfig;
|
||||||
import org.apache.druid.query.lookup.namespace.JdbcExtractionNamespace;
|
import org.apache.druid.query.lookup.namespace.JdbcExtractionNamespace;
|
||||||
|
import org.apache.druid.server.initialization.JdbcAccessSecurityConfig;
|
||||||
import org.apache.druid.server.lookup.namespace.cache.CacheScheduler;
|
import org.apache.druid.server.lookup.namespace.cache.CacheScheduler;
|
||||||
import org.apache.druid.server.lookup.namespace.cache.OnHeapNamespaceExtractionCacheManager;
|
import org.apache.druid.server.lookup.namespace.cache.OnHeapNamespaceExtractionCacheManager;
|
||||||
import org.apache.druid.server.metrics.NoopServiceEmitter;
|
import org.apache.druid.server.metrics.NoopServiceEmitter;
|
||||||
|
@ -42,7 +43,7 @@ import java.util.Collections;
|
||||||
public class JdbcCacheGeneratorTest
|
public class JdbcCacheGeneratorTest
|
||||||
{
|
{
|
||||||
private static final MetadataStorageConnectorConfig MISSING_METADATA_STORAGE_CONNECTOR_CONFIG =
|
private static final MetadataStorageConnectorConfig MISSING_METADATA_STORAGE_CONNECTOR_CONFIG =
|
||||||
createMetadataStorageConnectorConfig("postgresql");
|
createMetadataStorageConnectorConfig("mydb");
|
||||||
|
|
||||||
private static final CacheScheduler.EntryImpl<JdbcExtractionNamespace> KEY =
|
private static final CacheScheduler.EntryImpl<JdbcExtractionNamespace> KEY =
|
||||||
EasyMock.mock(CacheScheduler.EntryImpl.class);
|
EasyMock.mock(CacheScheduler.EntryImpl.class);
|
||||||
|
@ -127,7 +128,8 @@ public class JdbcCacheGeneratorTest
|
||||||
"valueColumn",
|
"valueColumn",
|
||||||
tsColumn,
|
tsColumn,
|
||||||
"filter",
|
"filter",
|
||||||
Period.ZERO
|
Period.ZERO,
|
||||||
|
new JdbcAccessSecurityConfig()
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -19,12 +19,15 @@
|
||||||
|
|
||||||
package org.apache.druid.server.lookup.namespace.cache;
|
package org.apache.druid.server.lookup.namespace.cache;
|
||||||
|
|
||||||
|
import com.fasterxml.jackson.databind.InjectableValues.Std;
|
||||||
|
import com.fasterxml.jackson.databind.ObjectMapper;
|
||||||
import com.google.common.collect.ImmutableList;
|
import com.google.common.collect.ImmutableList;
|
||||||
import com.google.common.collect.ImmutableMap;
|
import com.google.common.collect.ImmutableMap;
|
||||||
import com.google.common.util.concurrent.ListenableFuture;
|
import com.google.common.util.concurrent.ListenableFuture;
|
||||||
import com.google.common.util.concurrent.ListeningExecutorService;
|
import com.google.common.util.concurrent.ListeningExecutorService;
|
||||||
import com.google.common.util.concurrent.MoreExecutors;
|
import com.google.common.util.concurrent.MoreExecutors;
|
||||||
import org.apache.druid.common.config.NullHandling;
|
import org.apache.druid.common.config.NullHandling;
|
||||||
|
import org.apache.druid.jackson.DefaultObjectMapper;
|
||||||
import org.apache.druid.java.util.common.StringUtils;
|
import org.apache.druid.java.util.common.StringUtils;
|
||||||
import org.apache.druid.java.util.common.concurrent.Execs;
|
import org.apache.druid.java.util.common.concurrent.Execs;
|
||||||
import org.apache.druid.java.util.common.io.Closer;
|
import org.apache.druid.java.util.common.io.Closer;
|
||||||
|
@ -34,7 +37,7 @@ import org.apache.druid.metadata.TestDerbyConnector;
|
||||||
import org.apache.druid.query.lookup.namespace.CacheGenerator;
|
import org.apache.druid.query.lookup.namespace.CacheGenerator;
|
||||||
import org.apache.druid.query.lookup.namespace.ExtractionNamespace;
|
import org.apache.druid.query.lookup.namespace.ExtractionNamespace;
|
||||||
import org.apache.druid.query.lookup.namespace.JdbcExtractionNamespace;
|
import org.apache.druid.query.lookup.namespace.JdbcExtractionNamespace;
|
||||||
import org.apache.druid.server.ServerTestHelper;
|
import org.apache.druid.server.initialization.JdbcAccessSecurityConfig;
|
||||||
import org.apache.druid.server.lookup.namespace.JdbcCacheGenerator;
|
import org.apache.druid.server.lookup.namespace.JdbcCacheGenerator;
|
||||||
import org.apache.druid.server.lookup.namespace.NamespaceExtractionConfig;
|
import org.apache.druid.server.lookup.namespace.NamespaceExtractionConfig;
|
||||||
import org.apache.druid.server.metrics.NoopServiceEmitter;
|
import org.apache.druid.server.metrics.NoopServiceEmitter;
|
||||||
|
@ -73,6 +76,7 @@ public class JdbcExtractionNamespaceTest
|
||||||
|
|
||||||
@Rule
|
@Rule
|
||||||
public final TestDerbyConnector.DerbyConnectorRule derbyConnectorRule = new TestDerbyConnector.DerbyConnectorRule();
|
public final TestDerbyConnector.DerbyConnectorRule derbyConnectorRule = new TestDerbyConnector.DerbyConnectorRule();
|
||||||
|
|
||||||
private static final Logger log = new Logger(JdbcExtractionNamespaceTest.class);
|
private static final Logger log = new Logger(JdbcExtractionNamespaceTest.class);
|
||||||
private static final String TABLE_NAME = "abstractDbRenameTest";
|
private static final String TABLE_NAME = "abstractDbRenameTest";
|
||||||
private static final String KEY_NAME = "keyName";
|
private static final String KEY_NAME = "keyName";
|
||||||
|
@ -376,7 +380,8 @@ public class JdbcExtractionNamespaceTest
|
||||||
VAL_NAME,
|
VAL_NAME,
|
||||||
tsColumn,
|
tsColumn,
|
||||||
null,
|
null,
|
||||||
new Period(0)
|
new Period(0),
|
||||||
|
new JdbcAccessSecurityConfig()
|
||||||
);
|
);
|
||||||
try (CacheScheduler.Entry entry = scheduler.schedule(extractionNamespace)) {
|
try (CacheScheduler.Entry entry = scheduler.schedule(extractionNamespace)) {
|
||||||
CacheSchedulerTest.waitFor(entry);
|
CacheSchedulerTest.waitFor(entry);
|
||||||
|
@ -407,7 +412,8 @@ public class JdbcExtractionNamespaceTest
|
||||||
VAL_NAME,
|
VAL_NAME,
|
||||||
tsColumn,
|
tsColumn,
|
||||||
FILTER_COLUMN + "='1'",
|
FILTER_COLUMN + "='1'",
|
||||||
new Period(0)
|
new Period(0),
|
||||||
|
new JdbcAccessSecurityConfig()
|
||||||
);
|
);
|
||||||
try (CacheScheduler.Entry entry = scheduler.schedule(extractionNamespace)) {
|
try (CacheScheduler.Entry entry = scheduler.schedule(extractionNamespace)) {
|
||||||
CacheSchedulerTest.waitFor(entry);
|
CacheSchedulerTest.waitFor(entry);
|
||||||
|
@ -472,6 +478,7 @@ public class JdbcExtractionNamespaceTest
|
||||||
@Test
|
@Test
|
||||||
public void testSerde() throws IOException
|
public void testSerde() throws IOException
|
||||||
{
|
{
|
||||||
|
final JdbcAccessSecurityConfig securityConfig = new JdbcAccessSecurityConfig();
|
||||||
final JdbcExtractionNamespace extractionNamespace = new JdbcExtractionNamespace(
|
final JdbcExtractionNamespace extractionNamespace = new JdbcExtractionNamespace(
|
||||||
derbyConnectorRule.getMetadataConnectorConfig(),
|
derbyConnectorRule.getMetadataConnectorConfig(),
|
||||||
TABLE_NAME,
|
TABLE_NAME,
|
||||||
|
@ -479,11 +486,14 @@ public class JdbcExtractionNamespaceTest
|
||||||
VAL_NAME,
|
VAL_NAME,
|
||||||
tsColumn,
|
tsColumn,
|
||||||
"some filter",
|
"some filter",
|
||||||
new Period(10)
|
new Period(10),
|
||||||
|
securityConfig
|
||||||
);
|
);
|
||||||
|
final ObjectMapper mapper = new DefaultObjectMapper();
|
||||||
|
mapper.setInjectableValues(new Std().addValue(JdbcAccessSecurityConfig.class, securityConfig));
|
||||||
|
|
||||||
final ExtractionNamespace extractionNamespace2 = ServerTestHelper.MAPPER.readValue(
|
final ExtractionNamespace extractionNamespace2 = mapper.readValue(
|
||||||
ServerTestHelper.MAPPER.writeValueAsBytes(extractionNamespace),
|
mapper.writeValueAsBytes(extractionNamespace),
|
||||||
ExtractionNamespace.class
|
ExtractionNamespace.class
|
||||||
);
|
);
|
||||||
|
|
||||||
|
@ -500,7 +510,8 @@ public class JdbcExtractionNamespaceTest
|
||||||
VAL_NAME,
|
VAL_NAME,
|
||||||
tsColumn,
|
tsColumn,
|
||||||
null,
|
null,
|
||||||
new Period(10)
|
new Period(10),
|
||||||
|
new JdbcAccessSecurityConfig()
|
||||||
);
|
);
|
||||||
CacheScheduler.Entry entry = scheduler.schedule(extractionNamespace);
|
CacheScheduler.Entry entry = scheduler.schedule(extractionNamespace);
|
||||||
|
|
||||||
|
|
|
@ -96,12 +96,15 @@
|
||||||
<artifactId>guava</artifactId>
|
<artifactId>guava</artifactId>
|
||||||
<scope>provided</scope>
|
<scope>provided</scope>
|
||||||
</dependency>
|
</dependency>
|
||||||
|
<dependency>
|
||||||
<!-- Included to improve the out-of-the-box experience for supported JDBC connectors -->
|
<groupId>mysql</groupId>
|
||||||
|
<artifactId>mysql-connector-java</artifactId>
|
||||||
|
<version>${mysql.version}</version>
|
||||||
|
<scope>provided</scope>
|
||||||
|
</dependency>
|
||||||
<dependency>
|
<dependency>
|
||||||
<groupId>org.postgresql</groupId>
|
<groupId>org.postgresql</groupId>
|
||||||
<artifactId>postgresql</artifactId>
|
<artifactId>postgresql</artifactId>
|
||||||
<scope>runtime</scope>
|
|
||||||
</dependency>
|
</dependency>
|
||||||
|
|
||||||
<!-- Tests -->
|
<!-- Tests -->
|
||||||
|
|
|
@ -19,15 +19,23 @@
|
||||||
|
|
||||||
package org.apache.druid.server.lookup.jdbc;
|
package org.apache.druid.server.lookup.jdbc;
|
||||||
|
|
||||||
|
import com.fasterxml.jackson.annotation.JacksonInject;
|
||||||
import com.fasterxml.jackson.annotation.JsonProperty;
|
import com.fasterxml.jackson.annotation.JsonProperty;
|
||||||
import com.google.common.base.Preconditions;
|
import com.google.common.base.Preconditions;
|
||||||
import com.google.common.collect.Lists;
|
import com.google.common.collect.Lists;
|
||||||
|
import com.google.common.collect.Sets;
|
||||||
|
import com.mysql.jdbc.NonRegisteringDriver;
|
||||||
import org.apache.druid.common.config.NullHandling;
|
import org.apache.druid.common.config.NullHandling;
|
||||||
|
import org.apache.druid.java.util.common.IAE;
|
||||||
import org.apache.druid.java.util.common.ISE;
|
import org.apache.druid.java.util.common.ISE;
|
||||||
import org.apache.druid.java.util.common.StringUtils;
|
import org.apache.druid.java.util.common.StringUtils;
|
||||||
import org.apache.druid.java.util.common.logger.Logger;
|
import org.apache.druid.java.util.common.logger.Logger;
|
||||||
import org.apache.druid.metadata.MetadataStorageConnectorConfig;
|
import org.apache.druid.metadata.MetadataStorageConnectorConfig;
|
||||||
|
import org.apache.druid.server.initialization.JdbcAccessSecurityConfig;
|
||||||
import org.apache.druid.server.lookup.DataFetcher;
|
import org.apache.druid.server.lookup.DataFetcher;
|
||||||
|
import org.apache.druid.utils.ConnectionUriUtils;
|
||||||
|
import org.apache.druid.utils.Throwables;
|
||||||
|
import org.postgresql.Driver;
|
||||||
import org.skife.jdbi.v2.DBI;
|
import org.skife.jdbi.v2.DBI;
|
||||||
import org.skife.jdbi.v2.TransactionCallback;
|
import org.skife.jdbi.v2.TransactionCallback;
|
||||||
import org.skife.jdbi.v2.exceptions.UnableToObtainConnectionException;
|
import org.skife.jdbi.v2.exceptions.UnableToObtainConnectionException;
|
||||||
|
@ -39,6 +47,8 @@ import java.sql.SQLException;
|
||||||
import java.util.List;
|
import java.util.List;
|
||||||
import java.util.Map;
|
import java.util.Map;
|
||||||
import java.util.Objects;
|
import java.util.Objects;
|
||||||
|
import java.util.Properties;
|
||||||
|
import java.util.Set;
|
||||||
import java.util.function.Supplier;
|
import java.util.function.Supplier;
|
||||||
|
|
||||||
public class JdbcDataFetcher implements DataFetcher<String, String>
|
public class JdbcDataFetcher implements DataFetcher<String, String>
|
||||||
|
@ -71,12 +81,16 @@ public class JdbcDataFetcher implements DataFetcher<String, String>
|
||||||
@JsonProperty("table") String table,
|
@JsonProperty("table") String table,
|
||||||
@JsonProperty("keyColumn") String keyColumn,
|
@JsonProperty("keyColumn") String keyColumn,
|
||||||
@JsonProperty("valueColumn") String valueColumn,
|
@JsonProperty("valueColumn") String valueColumn,
|
||||||
@JsonProperty("streamingFetchSize") @Nullable Integer streamingFetchSize
|
@JsonProperty("streamingFetchSize") @Nullable Integer streamingFetchSize,
|
||||||
|
@JacksonInject JdbcAccessSecurityConfig securityConfig
|
||||||
)
|
)
|
||||||
{
|
{
|
||||||
this.connectorConfig = Preconditions.checkNotNull(connectorConfig, "connectorConfig");
|
this.connectorConfig = Preconditions.checkNotNull(connectorConfig, "connectorConfig");
|
||||||
this.streamingFetchSize = streamingFetchSize == null ? DEFAULT_STREAMING_FETCH_SIZE : streamingFetchSize;
|
this.streamingFetchSize = streamingFetchSize == null ? DEFAULT_STREAMING_FETCH_SIZE : streamingFetchSize;
|
||||||
Preconditions.checkNotNull(connectorConfig.getConnectURI(), "connectorConfig.connectURI");
|
// Check the properties in the connection URL. Note that JdbcDataFetcher doesn't use
|
||||||
|
// MetadataStorageConnectorConfig.getDbcpProperties(). If we want to use them,
|
||||||
|
// those DBCP properties should be validated using the same logic.
|
||||||
|
checkConnectionURL(connectorConfig.getConnectURI(), securityConfig);
|
||||||
this.table = Preconditions.checkNotNull(table, "table");
|
this.table = Preconditions.checkNotNull(table, "table");
|
||||||
this.keyColumn = Preconditions.checkNotNull(keyColumn, "keyColumn");
|
this.keyColumn = Preconditions.checkNotNull(keyColumn, "keyColumn");
|
||||||
this.valueColumn = Preconditions.checkNotNull(valueColumn, "valueColumn");
|
this.valueColumn = Preconditions.checkNotNull(valueColumn, "valueColumn");
|
||||||
|
@ -107,6 +121,89 @@ public class JdbcDataFetcher implements DataFetcher<String, String>
|
||||||
dbi.registerMapper(new KeyValueResultSetMapper(keyColumn, valueColumn));
|
dbi.registerMapper(new KeyValueResultSetMapper(keyColumn, valueColumn));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Check the given URL whether it contains non-allowed properties.
|
||||||
|
*
|
||||||
|
* This method should be in sync with the following methods:
|
||||||
|
*
|
||||||
|
* - {@code org.apache.druid.query.lookup.namespace.JdbcExtractionNamespace.checkConnectionURL()}
|
||||||
|
* - {@code org.apache.druid.firehose.sql.MySQLFirehoseDatabaseConnector.findPropertyKeysFromConnectURL()}
|
||||||
|
* - {@code org.apache.druid.firehose.sql.PostgresqlFirehoseDatabaseConnector.findPropertyKeysFromConnectURL()}
|
||||||
|
*
|
||||||
|
* @see JdbcAccessSecurityConfig#getAllowedProperties()
|
||||||
|
*/
|
||||||
|
private static void checkConnectionURL(String url, JdbcAccessSecurityConfig securityConfig)
|
||||||
|
{
|
||||||
|
Preconditions.checkNotNull(url, "connectorConfig.connectURI");
|
||||||
|
|
||||||
|
if (!securityConfig.isEnforceAllowedProperties()) {
|
||||||
|
// You don't want to do anything with properties.
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Nullable final Properties properties;
|
||||||
|
|
||||||
|
if (url.startsWith(ConnectionUriUtils.MYSQL_PREFIX)) {
|
||||||
|
try {
|
||||||
|
NonRegisteringDriver driver = new NonRegisteringDriver();
|
||||||
|
properties = driver.parseURL(url, null);
|
||||||
|
}
|
||||||
|
catch (SQLException e) {
|
||||||
|
throw new RuntimeException(e);
|
||||||
|
}
|
||||||
|
catch (Throwable e) {
|
||||||
|
if (Throwables.isThrowable(e, NoClassDefFoundError.class)
|
||||||
|
|| Throwables.isThrowable(e, ClassNotFoundException.class)) {
|
||||||
|
if (e.getMessage().contains("com/mysql/jdbc/NonRegisteringDriver")) {
|
||||||
|
throw new RuntimeException(
|
||||||
|
"Failed to find MySQL driver class. Please check the MySQL connector version 5.1.48 is in the classpath",
|
||||||
|
e
|
||||||
|
);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
throw new RuntimeException(e);
|
||||||
|
}
|
||||||
|
} else if (url.startsWith(ConnectionUriUtils.POSTGRES_PREFIX)) {
|
||||||
|
try {
|
||||||
|
properties = Driver.parseURL(url, null);
|
||||||
|
}
|
||||||
|
catch (Throwable e) {
|
||||||
|
if (Throwables.isThrowable(e, NoClassDefFoundError.class)
|
||||||
|
|| Throwables.isThrowable(e, ClassNotFoundException.class)) {
|
||||||
|
if (e.getMessage().contains("org/postgresql/Driver")) {
|
||||||
|
throw new RuntimeException(
|
||||||
|
"Failed to find PostgreSQL driver class. "
|
||||||
|
+ "Please check the PostgreSQL connector version 42.2.14 is in the classpath",
|
||||||
|
e
|
||||||
|
);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
throw new RuntimeException(e);
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
if (securityConfig.isAllowUnknownJdbcUrlFormat()) {
|
||||||
|
properties = new Properties();
|
||||||
|
} else {
|
||||||
|
// unknown format but it is not allowed
|
||||||
|
throw new IAE("Unknown JDBC connection scheme: %s", url.split(":")[1]);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (properties == null) {
|
||||||
|
// There is something wrong with the URL format.
|
||||||
|
throw new IAE("Invalid URL format [%s]", url);
|
||||||
|
}
|
||||||
|
|
||||||
|
final Set<String> propertyKeys = Sets.newHashSetWithExpectedSize(properties.size());
|
||||||
|
properties.forEach((k, v) -> propertyKeys.add((String) k));
|
||||||
|
|
||||||
|
ConnectionUriUtils.throwIfPropertiesAreNotAllowed(
|
||||||
|
propertyKeys,
|
||||||
|
securityConfig.getSystemPropertyPrefixes(),
|
||||||
|
securityConfig.getAllowedProperties()
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public Iterable<Map.Entry<String, String>> fetchAll()
|
public Iterable<Map.Entry<String, String>> fetchAll()
|
||||||
{
|
{
|
||||||
|
@ -232,7 +329,7 @@ public class JdbcDataFetcher implements DataFetcher<String, String>
|
||||||
if (e.getMessage().contains("No suitable driver found")) {
|
if (e.getMessage().contains("No suitable driver found")) {
|
||||||
throw new ISE(
|
throw new ISE(
|
||||||
e,
|
e,
|
||||||
"JDBC driver JAR files missing from extensions/druid-lookups-cached-single directory"
|
"JDBC driver JAR files missing in the classpath"
|
||||||
);
|
);
|
||||||
} else {
|
} else {
|
||||||
throw e;
|
throw e;
|
||||||
|
|
|
@ -19,6 +19,7 @@
|
||||||
|
|
||||||
package org.apache.druid.server.lookup.jdbc;
|
package org.apache.druid.server.lookup.jdbc;
|
||||||
|
|
||||||
|
import com.fasterxml.jackson.databind.InjectableValues.Std;
|
||||||
import com.fasterxml.jackson.databind.ObjectMapper;
|
import com.fasterxml.jackson.databind.ObjectMapper;
|
||||||
import com.google.common.collect.ImmutableMap;
|
import com.google.common.collect.ImmutableMap;
|
||||||
import com.google.common.collect.Sets;
|
import com.google.common.collect.Sets;
|
||||||
|
@ -26,6 +27,7 @@ import org.apache.druid.jackson.DefaultObjectMapper;
|
||||||
import org.apache.druid.java.util.common.StringUtils;
|
import org.apache.druid.java.util.common.StringUtils;
|
||||||
import org.apache.druid.metadata.MetadataStorageConnectorConfig;
|
import org.apache.druid.metadata.MetadataStorageConnectorConfig;
|
||||||
import org.apache.druid.metadata.TestDerbyConnector;
|
import org.apache.druid.metadata.TestDerbyConnector;
|
||||||
|
import org.apache.druid.server.initialization.JdbcAccessSecurityConfig;
|
||||||
import org.apache.druid.server.lookup.DataFetcher;
|
import org.apache.druid.server.lookup.DataFetcher;
|
||||||
import org.apache.druid.testing.InitializedNullHandlingTest;
|
import org.apache.druid.testing.InitializedNullHandlingTest;
|
||||||
import org.junit.After;
|
import org.junit.After;
|
||||||
|
@ -74,7 +76,8 @@ public class JdbcDataFetcherTest extends InitializedNullHandlingTest
|
||||||
"tableName",
|
"tableName",
|
||||||
"keyColumn",
|
"keyColumn",
|
||||||
"valueColumn",
|
"valueColumn",
|
||||||
100
|
100,
|
||||||
|
new JdbcAccessSecurityConfig()
|
||||||
);
|
);
|
||||||
|
|
||||||
handle = derbyConnectorRule.getConnector().getDBI().open();
|
handle = derbyConnectorRule.getConnector().getDBI().open();
|
||||||
|
@ -155,14 +158,17 @@ public class JdbcDataFetcherTest extends InitializedNullHandlingTest
|
||||||
@Test
|
@Test
|
||||||
public void testSerDesr() throws IOException
|
public void testSerDesr() throws IOException
|
||||||
{
|
{
|
||||||
|
final JdbcAccessSecurityConfig securityConfig = new JdbcAccessSecurityConfig();
|
||||||
JdbcDataFetcher jdbcDataFetcher = new JdbcDataFetcher(
|
JdbcDataFetcher jdbcDataFetcher = new JdbcDataFetcher(
|
||||||
new MetadataStorageConnectorConfig(),
|
new MetadataStorageConnectorConfig(),
|
||||||
"table",
|
"table",
|
||||||
"keyColumn",
|
"keyColumn",
|
||||||
"ValueColumn",
|
"ValueColumn",
|
||||||
100
|
100,
|
||||||
|
securityConfig
|
||||||
);
|
);
|
||||||
DefaultObjectMapper mapper = new DefaultObjectMapper();
|
DefaultObjectMapper mapper = new DefaultObjectMapper();
|
||||||
|
mapper.setInjectableValues(new Std().addValue(JdbcAccessSecurityConfig.class, securityConfig));
|
||||||
String jdbcDataFetcherSer = mapper.writeValueAsString(jdbcDataFetcher);
|
String jdbcDataFetcherSer = mapper.writeValueAsString(jdbcDataFetcher);
|
||||||
Assert.assertEquals(jdbcDataFetcher, mapper.readerFor(DataFetcher.class).readValue(jdbcDataFetcherSer));
|
Assert.assertEquals(jdbcDataFetcher, mapper.readerFor(DataFetcher.class).readValue(jdbcDataFetcherSer));
|
||||||
}
|
}
|
||||||
|
@ -213,7 +219,8 @@ public class JdbcDataFetcherTest extends InitializedNullHandlingTest
|
||||||
TABLE_NAME,
|
TABLE_NAME,
|
||||||
KEY_COLUMN,
|
KEY_COLUMN,
|
||||||
VALUE_COLUMN,
|
VALUE_COLUMN,
|
||||||
null
|
null,
|
||||||
|
new JdbcAccessSecurityConfig()
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -244,7 +251,7 @@ public class JdbcDataFetcherTest extends InitializedNullHandlingTest
|
||||||
private void test(Runnable runnable)
|
private void test(Runnable runnable)
|
||||||
{
|
{
|
||||||
exception.expect(IllegalStateException.class);
|
exception.expect(IllegalStateException.class);
|
||||||
exception.expectMessage("JDBC driver JAR files missing from extensions/druid-lookups-cached-single directory");
|
exception.expectMessage("JDBC driver JAR files missing in the classpath");
|
||||||
|
|
||||||
runnable.run();
|
runnable.run();
|
||||||
}
|
}
|
||||||
|
@ -252,8 +259,8 @@ public class JdbcDataFetcherTest extends InitializedNullHandlingTest
|
||||||
@SuppressWarnings("SameParameterValue")
|
@SuppressWarnings("SameParameterValue")
|
||||||
private static MetadataStorageConnectorConfig createMissingMetadataStorageConnectorConfig()
|
private static MetadataStorageConnectorConfig createMissingMetadataStorageConnectorConfig()
|
||||||
{
|
{
|
||||||
String type = "postgresql";
|
String type = "mydb";
|
||||||
String json = "{\"connectURI\":\"jdbc:" + type + "://localhost:5432\"}";
|
String json = "{\"connectURI\":\"jdbc:" + type + "://localhost:3306/\"}";
|
||||||
try {
|
try {
|
||||||
return new ObjectMapper().readValue(json, MetadataStorageConnectorConfig.class);
|
return new ObjectMapper().readValue(json, MetadataStorageConnectorConfig.class);
|
||||||
}
|
}
|
||||||
|
|
|
@ -0,0 +1,409 @@
|
||||||
|
/*
|
||||||
|
* 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.
|
||||||
|
*/
|
||||||
|
|
||||||
|
package org.apache.druid.server.lookup.jdbc;
|
||||||
|
|
||||||
|
import com.google.common.collect.ImmutableSet;
|
||||||
|
import org.apache.druid.metadata.MetadataStorageConnectorConfig;
|
||||||
|
import org.apache.druid.server.initialization.JdbcAccessSecurityConfig;
|
||||||
|
import org.junit.Rule;
|
||||||
|
import org.junit.Test;
|
||||||
|
import org.junit.experimental.runners.Enclosed;
|
||||||
|
import org.junit.rules.ExpectedException;
|
||||||
|
import org.junit.runner.RunWith;
|
||||||
|
|
||||||
|
import java.util.Set;
|
||||||
|
|
||||||
|
@RunWith(Enclosed.class)
|
||||||
|
public class JdbcDataFetcherUrlCheckTest
|
||||||
|
{
|
||||||
|
private static final String TABLE_NAME = "tableName";
|
||||||
|
private static final String KEY_COLUMN = "keyColumn";
|
||||||
|
private static final String VALUE_COLUMN = "valueColumn";
|
||||||
|
|
||||||
|
public static class MySqlTest
|
||||||
|
{
|
||||||
|
@Rule
|
||||||
|
public ExpectedException expectedException = ExpectedException.none();
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void testCreateInstanceWhenUrlHasOnlyAllowedProperties()
|
||||||
|
{
|
||||||
|
new JdbcDataFetcher(
|
||||||
|
new MetadataStorageConnectorConfig()
|
||||||
|
{
|
||||||
|
@Override
|
||||||
|
public String getConnectURI()
|
||||||
|
{
|
||||||
|
return "jdbc:mysql://localhost:3306/db?valid_key1=val1&valid_key2=val2";
|
||||||
|
}
|
||||||
|
},
|
||||||
|
TABLE_NAME,
|
||||||
|
KEY_COLUMN,
|
||||||
|
VALUE_COLUMN,
|
||||||
|
100,
|
||||||
|
new JdbcAccessSecurityConfig()
|
||||||
|
{
|
||||||
|
@Override
|
||||||
|
public Set<String> getAllowedProperties()
|
||||||
|
{
|
||||||
|
return ImmutableSet.of("valid_key1", "valid_key2");
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public boolean isEnforceAllowedProperties()
|
||||||
|
{
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void testThrowWhenUrlHasDisallowedPropertiesWhenEnforcingAllowedProperties()
|
||||||
|
{
|
||||||
|
expectedException.expect(IllegalArgumentException.class);
|
||||||
|
expectedException.expectMessage("The property [invalid_key1] is not in the allowed list [valid_key1, valid_key2]");
|
||||||
|
new JdbcDataFetcher(
|
||||||
|
new MetadataStorageConnectorConfig()
|
||||||
|
{
|
||||||
|
@Override
|
||||||
|
public String getConnectURI()
|
||||||
|
{
|
||||||
|
return "jdbc:mysql://localhost:3306/db?invalid_key1=val1&valid_key2=val2";
|
||||||
|
}
|
||||||
|
},
|
||||||
|
TABLE_NAME,
|
||||||
|
KEY_COLUMN,
|
||||||
|
VALUE_COLUMN,
|
||||||
|
100,
|
||||||
|
new JdbcAccessSecurityConfig()
|
||||||
|
{
|
||||||
|
@Override
|
||||||
|
public Set<String> getAllowedProperties()
|
||||||
|
{
|
||||||
|
return ImmutableSet.of("valid_key1", "valid_key2");
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public boolean isEnforceAllowedProperties()
|
||||||
|
{
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void testWhenUrlHasDisallowedPropertiesWhenNotEnforcingAllowedProperties()
|
||||||
|
{
|
||||||
|
new JdbcDataFetcher(
|
||||||
|
new MetadataStorageConnectorConfig()
|
||||||
|
{
|
||||||
|
@Override
|
||||||
|
public String getConnectURI()
|
||||||
|
{
|
||||||
|
return "jdbc:mysql://localhost:3306/db?invalid_key1=val1&valid_key2=val2";
|
||||||
|
}
|
||||||
|
},
|
||||||
|
TABLE_NAME,
|
||||||
|
KEY_COLUMN,
|
||||||
|
VALUE_COLUMN,
|
||||||
|
100,
|
||||||
|
new JdbcAccessSecurityConfig()
|
||||||
|
{
|
||||||
|
@Override
|
||||||
|
public Set<String> getAllowedProperties()
|
||||||
|
{
|
||||||
|
return ImmutableSet.of("valid_key1", "valid_key2");
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public boolean isEnforceAllowedProperties()
|
||||||
|
{
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void testWhenInvalidUrlFormat()
|
||||||
|
{
|
||||||
|
expectedException.expect(IllegalArgumentException.class);
|
||||||
|
expectedException.expectMessage("Invalid URL format [jdbc:mysql:/invalid-url::3006]");
|
||||||
|
new JdbcDataFetcher(
|
||||||
|
new MetadataStorageConnectorConfig()
|
||||||
|
{
|
||||||
|
@Override
|
||||||
|
public String getConnectURI()
|
||||||
|
{
|
||||||
|
return "jdbc:mysql:/invalid-url::3006";
|
||||||
|
}
|
||||||
|
},
|
||||||
|
TABLE_NAME,
|
||||||
|
KEY_COLUMN,
|
||||||
|
VALUE_COLUMN,
|
||||||
|
100,
|
||||||
|
new JdbcAccessSecurityConfig()
|
||||||
|
{
|
||||||
|
@Override
|
||||||
|
public Set<String> getAllowedProperties()
|
||||||
|
{
|
||||||
|
return ImmutableSet.of("valid_key1", "valid_key2");
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public boolean isEnforceAllowedProperties()
|
||||||
|
{
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public static class PostgreSqlTest
|
||||||
|
{
|
||||||
|
@Rule
|
||||||
|
public ExpectedException expectedException = ExpectedException.none();
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void testCreateInstanceWhenUrlHasOnlyAllowedProperties()
|
||||||
|
{
|
||||||
|
new JdbcDataFetcher(
|
||||||
|
new MetadataStorageConnectorConfig()
|
||||||
|
{
|
||||||
|
@Override
|
||||||
|
public String getConnectURI()
|
||||||
|
{
|
||||||
|
return "jdbc:postgresql://localhost:5432/db?valid_key1=val1&valid_key2=val2";
|
||||||
|
}
|
||||||
|
},
|
||||||
|
TABLE_NAME,
|
||||||
|
KEY_COLUMN,
|
||||||
|
VALUE_COLUMN,
|
||||||
|
100,
|
||||||
|
new JdbcAccessSecurityConfig()
|
||||||
|
{
|
||||||
|
@Override
|
||||||
|
public Set<String> getAllowedProperties()
|
||||||
|
{
|
||||||
|
return ImmutableSet.of("valid_key1", "valid_key2");
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public boolean isEnforceAllowedProperties()
|
||||||
|
{
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void testThrowWhenUrlHasDisallowedPropertiesWhenEnforcingAllowedProperties()
|
||||||
|
{
|
||||||
|
expectedException.expect(IllegalArgumentException.class);
|
||||||
|
expectedException.expectMessage("The property [invalid_key1] is not in the allowed list [valid_key1, valid_key2]");
|
||||||
|
new JdbcDataFetcher(
|
||||||
|
new MetadataStorageConnectorConfig()
|
||||||
|
{
|
||||||
|
@Override
|
||||||
|
public String getConnectURI()
|
||||||
|
{
|
||||||
|
return "jdbc:postgresql://localhost:5432/db?invalid_key1=val1&valid_key2=val2";
|
||||||
|
}
|
||||||
|
},
|
||||||
|
TABLE_NAME,
|
||||||
|
KEY_COLUMN,
|
||||||
|
VALUE_COLUMN,
|
||||||
|
100,
|
||||||
|
new JdbcAccessSecurityConfig()
|
||||||
|
{
|
||||||
|
@Override
|
||||||
|
public Set<String> getAllowedProperties()
|
||||||
|
{
|
||||||
|
return ImmutableSet.of("valid_key1", "valid_key2");
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public boolean isEnforceAllowedProperties()
|
||||||
|
{
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void testWhenUrlHasDisallowedPropertiesWhenNotEnforcingAllowedProperties()
|
||||||
|
{
|
||||||
|
new JdbcDataFetcher(
|
||||||
|
new MetadataStorageConnectorConfig()
|
||||||
|
{
|
||||||
|
@Override
|
||||||
|
public String getConnectURI()
|
||||||
|
{
|
||||||
|
return "jdbc:postgresql://localhost:5432/db?invalid_key1=val1&valid_key2=val2";
|
||||||
|
}
|
||||||
|
},
|
||||||
|
TABLE_NAME,
|
||||||
|
KEY_COLUMN,
|
||||||
|
VALUE_COLUMN,
|
||||||
|
100,
|
||||||
|
new JdbcAccessSecurityConfig()
|
||||||
|
{
|
||||||
|
@Override
|
||||||
|
public Set<String> getAllowedProperties()
|
||||||
|
{
|
||||||
|
return ImmutableSet.of("valid_key1", "valid_key2");
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public boolean isEnforceAllowedProperties()
|
||||||
|
{
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void testWhenInvalidUrlFormat()
|
||||||
|
{
|
||||||
|
expectedException.expect(IllegalArgumentException.class);
|
||||||
|
expectedException.expectMessage("Invalid URL format [jdbc:postgresql://invalid-url::3006]");
|
||||||
|
new JdbcDataFetcher(
|
||||||
|
new MetadataStorageConnectorConfig()
|
||||||
|
{
|
||||||
|
@Override
|
||||||
|
public String getConnectURI()
|
||||||
|
{
|
||||||
|
return "jdbc:postgresql://invalid-url::3006";
|
||||||
|
}
|
||||||
|
},
|
||||||
|
TABLE_NAME,
|
||||||
|
KEY_COLUMN,
|
||||||
|
VALUE_COLUMN,
|
||||||
|
100,
|
||||||
|
new JdbcAccessSecurityConfig()
|
||||||
|
{
|
||||||
|
@Override
|
||||||
|
public Set<String> getAllowedProperties()
|
||||||
|
{
|
||||||
|
return ImmutableSet.of("valid_key1", "valid_key2");
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public boolean isEnforceAllowedProperties()
|
||||||
|
{
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public static class UnknownSchemeTest
|
||||||
|
{
|
||||||
|
@Rule
|
||||||
|
public ExpectedException expectedException = ExpectedException.none();
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void testThrowWhenUnknownFormatIsNotAllowed()
|
||||||
|
{
|
||||||
|
expectedException.expect(IllegalArgumentException.class);
|
||||||
|
expectedException.expectMessage("Unknown JDBC connection scheme: mydb");
|
||||||
|
new JdbcDataFetcher(
|
||||||
|
new MetadataStorageConnectorConfig()
|
||||||
|
{
|
||||||
|
@Override
|
||||||
|
public String getConnectURI()
|
||||||
|
{
|
||||||
|
return "jdbc:mydb://localhost:5432/db?valid_key1=val1&valid_key2=val2";
|
||||||
|
}
|
||||||
|
},
|
||||||
|
TABLE_NAME,
|
||||||
|
KEY_COLUMN,
|
||||||
|
VALUE_COLUMN,
|
||||||
|
100,
|
||||||
|
new JdbcAccessSecurityConfig()
|
||||||
|
{
|
||||||
|
@Override
|
||||||
|
public Set<String> getAllowedProperties()
|
||||||
|
{
|
||||||
|
return ImmutableSet.of("valid_key1", "valid_key2");
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public boolean isAllowUnknownJdbcUrlFormat()
|
||||||
|
{
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public boolean isEnforceAllowedProperties()
|
||||||
|
{
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void testSkipUrlParsingWhenUnknownFormatIsAllowed()
|
||||||
|
{
|
||||||
|
new JdbcDataFetcher(
|
||||||
|
new MetadataStorageConnectorConfig()
|
||||||
|
{
|
||||||
|
@Override
|
||||||
|
public String getConnectURI()
|
||||||
|
{
|
||||||
|
return "jdbc:mydb://localhost:5432/db?valid_key1=val1&valid_key2=val2";
|
||||||
|
}
|
||||||
|
},
|
||||||
|
TABLE_NAME,
|
||||||
|
KEY_COLUMN,
|
||||||
|
VALUE_COLUMN,
|
||||||
|
100,
|
||||||
|
new JdbcAccessSecurityConfig()
|
||||||
|
{
|
||||||
|
@Override
|
||||||
|
public Set<String> getAllowedProperties()
|
||||||
|
{
|
||||||
|
return ImmutableSet.of("valid_key1", "valid_key2");
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public boolean isAllowUnknownJdbcUrlFormat()
|
||||||
|
{
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public boolean isEnforceAllowedProperties()
|
||||||
|
{
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
|
@ -88,6 +88,11 @@
|
||||||
<artifactId>commons-dbcp2</artifactId>
|
<artifactId>commons-dbcp2</artifactId>
|
||||||
<scope>provided</scope>
|
<scope>provided</scope>
|
||||||
</dependency>
|
</dependency>
|
||||||
|
<dependency>
|
||||||
|
<groupId>junit</groupId>
|
||||||
|
<artifactId>junit</artifactId>
|
||||||
|
<scope>test</scope>
|
||||||
|
</dependency>
|
||||||
</dependencies>
|
</dependencies>
|
||||||
|
|
||||||
</project>
|
</project>
|
||||||
|
|
|
@ -19,13 +19,24 @@
|
||||||
|
|
||||||
package org.apache.druid.firehose.sql;
|
package org.apache.druid.firehose.sql;
|
||||||
|
|
||||||
|
import com.fasterxml.jackson.annotation.JacksonInject;
|
||||||
|
import com.fasterxml.jackson.annotation.JsonCreator;
|
||||||
import com.fasterxml.jackson.annotation.JsonProperty;
|
import com.fasterxml.jackson.annotation.JsonProperty;
|
||||||
import com.fasterxml.jackson.annotation.JsonTypeName;
|
import com.fasterxml.jackson.annotation.JsonTypeName;
|
||||||
|
import com.google.common.collect.Sets;
|
||||||
|
import com.mysql.jdbc.NonRegisteringDriver;
|
||||||
import org.apache.commons.dbcp2.BasicDataSource;
|
import org.apache.commons.dbcp2.BasicDataSource;
|
||||||
|
import org.apache.druid.java.util.common.IAE;
|
||||||
import org.apache.druid.metadata.MetadataStorageConnectorConfig;
|
import org.apache.druid.metadata.MetadataStorageConnectorConfig;
|
||||||
import org.apache.druid.metadata.SQLFirehoseDatabaseConnector;
|
import org.apache.druid.metadata.SQLFirehoseDatabaseConnector;
|
||||||
|
import org.apache.druid.server.initialization.JdbcAccessSecurityConfig;
|
||||||
|
import org.apache.druid.utils.Throwables;
|
||||||
import org.skife.jdbi.v2.DBI;
|
import org.skife.jdbi.v2.DBI;
|
||||||
|
|
||||||
|
import java.sql.SQLException;
|
||||||
|
import java.util.Properties;
|
||||||
|
import java.util.Set;
|
||||||
|
|
||||||
|
|
||||||
@JsonTypeName("mysql")
|
@JsonTypeName("mysql")
|
||||||
public class MySQLFirehoseDatabaseConnector extends SQLFirehoseDatabaseConnector
|
public class MySQLFirehoseDatabaseConnector extends SQLFirehoseDatabaseConnector
|
||||||
|
@ -33,12 +44,14 @@ public class MySQLFirehoseDatabaseConnector extends SQLFirehoseDatabaseConnector
|
||||||
private final DBI dbi;
|
private final DBI dbi;
|
||||||
private final MetadataStorageConnectorConfig connectorConfig;
|
private final MetadataStorageConnectorConfig connectorConfig;
|
||||||
|
|
||||||
|
@JsonCreator
|
||||||
public MySQLFirehoseDatabaseConnector(
|
public MySQLFirehoseDatabaseConnector(
|
||||||
@JsonProperty("connectorConfig") MetadataStorageConnectorConfig connectorConfig
|
@JsonProperty("connectorConfig") MetadataStorageConnectorConfig connectorConfig,
|
||||||
|
@JacksonInject JdbcAccessSecurityConfig securityConfig
|
||||||
)
|
)
|
||||||
{
|
{
|
||||||
this.connectorConfig = connectorConfig;
|
this.connectorConfig = connectorConfig;
|
||||||
final BasicDataSource datasource = getDatasource(connectorConfig);
|
final BasicDataSource datasource = getDatasource(connectorConfig, securityConfig);
|
||||||
datasource.setDriverClassLoader(getClass().getClassLoader());
|
datasource.setDriverClassLoader(getClass().getClassLoader());
|
||||||
datasource.setDriverClassName("com.mysql.jdbc.Driver");
|
datasource.setDriverClassName("com.mysql.jdbc.Driver");
|
||||||
this.dbi = new DBI(datasource);
|
this.dbi = new DBI(datasource);
|
||||||
|
@ -55,4 +68,39 @@ public class MySQLFirehoseDatabaseConnector extends SQLFirehoseDatabaseConnector
|
||||||
{
|
{
|
||||||
return dbi;
|
return dbi;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public Set<String> findPropertyKeysFromConnectURL(String connectUrl)
|
||||||
|
{
|
||||||
|
// This method should be in sync with
|
||||||
|
// - org.apache.druid.server.lookup.jdbc.JdbcDataFetcher.checkConnectionURL()
|
||||||
|
// - org.apache.druid.query.lookup.namespace.JdbcExtractionNamespace.checkConnectionURL()
|
||||||
|
Properties properties;
|
||||||
|
try {
|
||||||
|
NonRegisteringDriver driver = new NonRegisteringDriver();
|
||||||
|
properties = driver.parseURL(connectUrl, null);
|
||||||
|
}
|
||||||
|
catch (SQLException e) {
|
||||||
|
throw new RuntimeException(e);
|
||||||
|
}
|
||||||
|
catch (Throwable e) {
|
||||||
|
if (Throwables.isThrowable(e, NoClassDefFoundError.class)
|
||||||
|
|| Throwables.isThrowable(e, ClassNotFoundException.class)) {
|
||||||
|
if (e.getMessage().contains("com/mysql/jdbc/NonRegisteringDriver")) {
|
||||||
|
throw new RuntimeException(
|
||||||
|
"Failed to find MySQL driver class. Please check the MySQL connector version 5.1.48 is in the classpath",
|
||||||
|
e
|
||||||
|
);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
throw new RuntimeException(e);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (properties == null) {
|
||||||
|
throw new IAE("Invalid URL format for MySQL: [%s]", connectUrl);
|
||||||
|
}
|
||||||
|
Set<String> keys = Sets.newHashSetWithExpectedSize(properties.size());
|
||||||
|
properties.forEach((k, v) -> keys.add((String) k));
|
||||||
|
return keys;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -0,0 +1,234 @@
|
||||||
|
/*
|
||||||
|
* 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.
|
||||||
|
*/
|
||||||
|
|
||||||
|
package org.apache.druid.firehose.sql;
|
||||||
|
|
||||||
|
import com.google.common.collect.ImmutableSet;
|
||||||
|
import org.apache.druid.java.util.common.StringUtils;
|
||||||
|
import org.apache.druid.metadata.MetadataStorageConnectorConfig;
|
||||||
|
import org.apache.druid.server.initialization.JdbcAccessSecurityConfig;
|
||||||
|
import org.junit.Rule;
|
||||||
|
import org.junit.Test;
|
||||||
|
import org.junit.rules.ExpectedException;
|
||||||
|
|
||||||
|
import java.util.Set;
|
||||||
|
|
||||||
|
public class MySQLFirehoseDatabaseConnectorTest
|
||||||
|
{
|
||||||
|
@Rule
|
||||||
|
public final ExpectedException expectedException = ExpectedException.none();
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void testSuccessWhenNoPropertyInUriAndNoAllowlist()
|
||||||
|
{
|
||||||
|
MetadataStorageConnectorConfig connectorConfig = new MetadataStorageConnectorConfig()
|
||||||
|
{
|
||||||
|
@Override
|
||||||
|
public String getConnectURI()
|
||||||
|
{
|
||||||
|
return "jdbc:mysql://localhost:3306/test";
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
JdbcAccessSecurityConfig securityConfig = newSecurityConfigEnforcingAllowList(ImmutableSet.of());
|
||||||
|
|
||||||
|
new MySQLFirehoseDatabaseConnector(
|
||||||
|
connectorConfig,
|
||||||
|
securityConfig
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void testSuccessWhenAllowlistAndNoProperty()
|
||||||
|
{
|
||||||
|
MetadataStorageConnectorConfig connectorConfig = new MetadataStorageConnectorConfig()
|
||||||
|
{
|
||||||
|
@Override
|
||||||
|
public String getConnectURI()
|
||||||
|
{
|
||||||
|
return "jdbc:mysql://localhost:3306/test";
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
JdbcAccessSecurityConfig securityConfig = newSecurityConfigEnforcingAllowList(ImmutableSet.of("user"));
|
||||||
|
|
||||||
|
new MySQLFirehoseDatabaseConnector(
|
||||||
|
connectorConfig,
|
||||||
|
securityConfig
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void testFailWhenNoAllowlistAndHaveProperty()
|
||||||
|
{
|
||||||
|
MetadataStorageConnectorConfig connectorConfig = new MetadataStorageConnectorConfig()
|
||||||
|
{
|
||||||
|
@Override
|
||||||
|
public String getConnectURI()
|
||||||
|
{
|
||||||
|
return "jdbc:mysql://localhost:3306/test?user=maytas&password=secret&keyonly";
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
JdbcAccessSecurityConfig securityConfig = newSecurityConfigEnforcingAllowList(ImmutableSet.of(""));
|
||||||
|
|
||||||
|
expectedException.expectMessage("The property [password] is not in the allowed list");
|
||||||
|
expectedException.expect(IllegalArgumentException.class);
|
||||||
|
|
||||||
|
new MySQLFirehoseDatabaseConnector(
|
||||||
|
connectorConfig,
|
||||||
|
securityConfig
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void testSuccessOnlyValidProperty()
|
||||||
|
{
|
||||||
|
MetadataStorageConnectorConfig connectorConfig = new MetadataStorageConnectorConfig()
|
||||||
|
{
|
||||||
|
@Override
|
||||||
|
public String getConnectURI()
|
||||||
|
{
|
||||||
|
return "jdbc:mysql://localhost:3306/test?user=maytas&password=secret&keyonly";
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
JdbcAccessSecurityConfig securityConfig = newSecurityConfigEnforcingAllowList(
|
||||||
|
ImmutableSet.of("user", "password", "keyonly", "etc")
|
||||||
|
);
|
||||||
|
|
||||||
|
new MySQLFirehoseDatabaseConnector(
|
||||||
|
connectorConfig,
|
||||||
|
securityConfig
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void testFailOnlyInvalidProperty()
|
||||||
|
{
|
||||||
|
MetadataStorageConnectorConfig connectorConfig = new MetadataStorageConnectorConfig()
|
||||||
|
{
|
||||||
|
@Override
|
||||||
|
public String getConnectURI()
|
||||||
|
{
|
||||||
|
return "jdbc:mysql://localhost:3306/test?user=maytas&password=secret&keyonly";
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
JdbcAccessSecurityConfig securityConfig = newSecurityConfigEnforcingAllowList(ImmutableSet.of("none", "nonenone"));
|
||||||
|
|
||||||
|
expectedException.expectMessage("The property [password] is not in the allowed list");
|
||||||
|
expectedException.expect(IllegalArgumentException.class);
|
||||||
|
|
||||||
|
new MySQLFirehoseDatabaseConnector(
|
||||||
|
connectorConfig,
|
||||||
|
securityConfig
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void testFailValidAndInvalidProperty()
|
||||||
|
{
|
||||||
|
MetadataStorageConnectorConfig connectorConfig = new MetadataStorageConnectorConfig()
|
||||||
|
{
|
||||||
|
@Override
|
||||||
|
public String getConnectURI()
|
||||||
|
{
|
||||||
|
return "jdbc:mysql://localhost:3306/test?user=maytas&password=secret&keyonly";
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
JdbcAccessSecurityConfig securityConfig = newSecurityConfigEnforcingAllowList(ImmutableSet.of("user", "nonenone"));
|
||||||
|
|
||||||
|
expectedException.expectMessage("The property [password] is not in the allowed list");
|
||||||
|
expectedException.expect(IllegalArgumentException.class);
|
||||||
|
|
||||||
|
new MySQLFirehoseDatabaseConnector(
|
||||||
|
connectorConfig,
|
||||||
|
securityConfig
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void testIgnoreInvalidPropertyWhenNotEnforcingAllowList()
|
||||||
|
{
|
||||||
|
MetadataStorageConnectorConfig connectorConfig = new MetadataStorageConnectorConfig()
|
||||||
|
{
|
||||||
|
@Override
|
||||||
|
public String getConnectURI()
|
||||||
|
{
|
||||||
|
return "jdbc:mysql://localhost:3306/test?user=maytas&password=secret&keyonly";
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
JdbcAccessSecurityConfig securityConfig = new JdbcAccessSecurityConfig()
|
||||||
|
{
|
||||||
|
@Override
|
||||||
|
public Set<String> getAllowedProperties()
|
||||||
|
{
|
||||||
|
return ImmutableSet.of("user", "nonenone");
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
new MySQLFirehoseDatabaseConnector(
|
||||||
|
connectorConfig,
|
||||||
|
securityConfig
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void testFindPropertyKeysFromInvalidConnectUrl()
|
||||||
|
{
|
||||||
|
final String url = "jdbc:mysql:/invalid-url::3006";
|
||||||
|
MetadataStorageConnectorConfig connectorConfig = new MetadataStorageConnectorConfig()
|
||||||
|
{
|
||||||
|
@Override
|
||||||
|
public String getConnectURI()
|
||||||
|
{
|
||||||
|
return url;
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
MySQLFirehoseDatabaseConnector connector = new MySQLFirehoseDatabaseConnector(
|
||||||
|
connectorConfig,
|
||||||
|
new JdbcAccessSecurityConfig()
|
||||||
|
);
|
||||||
|
expectedException.expect(IllegalArgumentException.class);
|
||||||
|
expectedException.expectMessage(StringUtils.format("Invalid URL format for MySQL: [%s]", url));
|
||||||
|
connector.findPropertyKeysFromConnectURL(url);
|
||||||
|
}
|
||||||
|
|
||||||
|
private static JdbcAccessSecurityConfig newSecurityConfigEnforcingAllowList(Set<String> allowedProperties)
|
||||||
|
{
|
||||||
|
return new JdbcAccessSecurityConfig()
|
||||||
|
{
|
||||||
|
@Override
|
||||||
|
public Set<String> getAllowedProperties()
|
||||||
|
{
|
||||||
|
return allowedProperties;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public boolean isEnforceAllowedProperties()
|
||||||
|
{
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
};
|
||||||
|
}
|
||||||
|
}
|
|
@ -19,13 +19,22 @@
|
||||||
|
|
||||||
package org.apache.druid.firehose;
|
package org.apache.druid.firehose;
|
||||||
|
|
||||||
|
import com.fasterxml.jackson.annotation.JacksonInject;
|
||||||
|
import com.fasterxml.jackson.annotation.JsonCreator;
|
||||||
import com.fasterxml.jackson.annotation.JsonProperty;
|
import com.fasterxml.jackson.annotation.JsonProperty;
|
||||||
import com.fasterxml.jackson.annotation.JsonTypeName;
|
import com.fasterxml.jackson.annotation.JsonTypeName;
|
||||||
|
import com.google.common.collect.Sets;
|
||||||
import org.apache.commons.dbcp2.BasicDataSource;
|
import org.apache.commons.dbcp2.BasicDataSource;
|
||||||
|
import org.apache.druid.java.util.common.IAE;
|
||||||
import org.apache.druid.metadata.MetadataStorageConnectorConfig;
|
import org.apache.druid.metadata.MetadataStorageConnectorConfig;
|
||||||
import org.apache.druid.metadata.SQLFirehoseDatabaseConnector;
|
import org.apache.druid.metadata.SQLFirehoseDatabaseConnector;
|
||||||
|
import org.apache.druid.server.initialization.JdbcAccessSecurityConfig;
|
||||||
|
import org.postgresql.Driver;
|
||||||
import org.skife.jdbi.v2.DBI;
|
import org.skife.jdbi.v2.DBI;
|
||||||
|
|
||||||
|
import java.util.Properties;
|
||||||
|
import java.util.Set;
|
||||||
|
|
||||||
|
|
||||||
@JsonTypeName("postgresql")
|
@JsonTypeName("postgresql")
|
||||||
public class PostgresqlFirehoseDatabaseConnector extends SQLFirehoseDatabaseConnector
|
public class PostgresqlFirehoseDatabaseConnector extends SQLFirehoseDatabaseConnector
|
||||||
|
@ -33,12 +42,14 @@ public class PostgresqlFirehoseDatabaseConnector extends SQLFirehoseDatabaseConn
|
||||||
private final DBI dbi;
|
private final DBI dbi;
|
||||||
private final MetadataStorageConnectorConfig connectorConfig;
|
private final MetadataStorageConnectorConfig connectorConfig;
|
||||||
|
|
||||||
|
@JsonCreator
|
||||||
public PostgresqlFirehoseDatabaseConnector(
|
public PostgresqlFirehoseDatabaseConnector(
|
||||||
@JsonProperty("connectorConfig") MetadataStorageConnectorConfig connectorConfig
|
@JsonProperty("connectorConfig") MetadataStorageConnectorConfig connectorConfig,
|
||||||
|
@JacksonInject JdbcAccessSecurityConfig securityConfig
|
||||||
)
|
)
|
||||||
{
|
{
|
||||||
this.connectorConfig = connectorConfig;
|
this.connectorConfig = connectorConfig;
|
||||||
final BasicDataSource datasource = getDatasource(connectorConfig);
|
final BasicDataSource datasource = getDatasource(connectorConfig, securityConfig);
|
||||||
datasource.setDriverClassLoader(getClass().getClassLoader());
|
datasource.setDriverClassLoader(getClass().getClassLoader());
|
||||||
datasource.setDriverClassName("org.postgresql.Driver");
|
datasource.setDriverClassName("org.postgresql.Driver");
|
||||||
this.dbi = new DBI(datasource);
|
this.dbi = new DBI(datasource);
|
||||||
|
@ -55,4 +66,21 @@ public class PostgresqlFirehoseDatabaseConnector extends SQLFirehoseDatabaseConn
|
||||||
{
|
{
|
||||||
return dbi;
|
return dbi;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public Set<String> findPropertyKeysFromConnectURL(String connectUri)
|
||||||
|
{
|
||||||
|
// This method should be in sync with
|
||||||
|
// - org.apache.druid.server.lookup.jdbc.JdbcDataFetcher.checkConnectionURL()
|
||||||
|
// - org.apache.druid.query.lookup.namespace.JdbcExtractionNamespace.checkConnectionURL()
|
||||||
|
|
||||||
|
// Postgresql JDBC driver is embedded and thus must be loaded.
|
||||||
|
Properties properties = Driver.parseURL(connectUri, null);
|
||||||
|
if (properties == null) {
|
||||||
|
throw new IAE("Invalid URL format for PostgreSQL: [%s]", connectUri);
|
||||||
|
}
|
||||||
|
Set<String> keys = Sets.newHashSetWithExpectedSize(properties.size());
|
||||||
|
properties.forEach((k, v) -> keys.add((String) k));
|
||||||
|
return keys;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -57,7 +57,7 @@ public class PostgreSQLConnector extends SQLMetadataConnector
|
||||||
private volatile Boolean canUpsert;
|
private volatile Boolean canUpsert;
|
||||||
|
|
||||||
private final String dbTableSchema;
|
private final String dbTableSchema;
|
||||||
|
|
||||||
@Inject
|
@Inject
|
||||||
public PostgreSQLConnector(
|
public PostgreSQLConnector(
|
||||||
Supplier<MetadataStorageConnectorConfig> config,
|
Supplier<MetadataStorageConnectorConfig> config,
|
||||||
|
|
|
@ -0,0 +1,211 @@
|
||||||
|
/*
|
||||||
|
* 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.
|
||||||
|
*/
|
||||||
|
|
||||||
|
package org.apache.druid.firehose;
|
||||||
|
|
||||||
|
import com.google.common.collect.ImmutableSet;
|
||||||
|
import org.apache.druid.metadata.MetadataStorageConnectorConfig;
|
||||||
|
import org.apache.druid.server.initialization.JdbcAccessSecurityConfig;
|
||||||
|
import org.junit.Rule;
|
||||||
|
import org.junit.Test;
|
||||||
|
import org.junit.rules.ExpectedException;
|
||||||
|
|
||||||
|
import java.util.Set;
|
||||||
|
|
||||||
|
public class PostgresqlFirehoseDatabaseConnectorTest
|
||||||
|
{
|
||||||
|
@Rule
|
||||||
|
public final ExpectedException expectedException = ExpectedException.none();
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void testSuccessWhenNoPropertyInUriAndNoAllowlist()
|
||||||
|
{
|
||||||
|
MetadataStorageConnectorConfig connectorConfig = new MetadataStorageConnectorConfig()
|
||||||
|
{
|
||||||
|
@Override
|
||||||
|
public String getConnectURI()
|
||||||
|
{
|
||||||
|
return "jdbc:postgresql://localhost:3306/test";
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
JdbcAccessSecurityConfig securityConfig = newSecurityConfigEnforcingAllowList(ImmutableSet.of());
|
||||||
|
|
||||||
|
new PostgresqlFirehoseDatabaseConnector(
|
||||||
|
connectorConfig,
|
||||||
|
securityConfig
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void testSuccessWhenAllowlistAndNoProperty()
|
||||||
|
{
|
||||||
|
MetadataStorageConnectorConfig connectorConfig = new MetadataStorageConnectorConfig()
|
||||||
|
{
|
||||||
|
@Override
|
||||||
|
public String getConnectURI()
|
||||||
|
{
|
||||||
|
return "jdbc:postgresql://localhost:3306/test";
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
JdbcAccessSecurityConfig securityConfig = newSecurityConfigEnforcingAllowList(ImmutableSet.of("user"));
|
||||||
|
|
||||||
|
new PostgresqlFirehoseDatabaseConnector(
|
||||||
|
connectorConfig,
|
||||||
|
securityConfig
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void testFailWhenNoAllowlistAndHaveProperty()
|
||||||
|
{
|
||||||
|
MetadataStorageConnectorConfig connectorConfig = new MetadataStorageConnectorConfig()
|
||||||
|
{
|
||||||
|
@Override
|
||||||
|
public String getConnectURI()
|
||||||
|
{
|
||||||
|
return "jdbc:postgresql://localhost:3306/test?user=maytas&password=secret&keyonly";
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
JdbcAccessSecurityConfig securityConfig = newSecurityConfigEnforcingAllowList(ImmutableSet.of(""));
|
||||||
|
|
||||||
|
expectedException.expectMessage("is not in the allowed list");
|
||||||
|
expectedException.expect(IllegalArgumentException.class);
|
||||||
|
|
||||||
|
new PostgresqlFirehoseDatabaseConnector(
|
||||||
|
connectorConfig,
|
||||||
|
securityConfig
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void testSuccessOnlyValidProperty()
|
||||||
|
{
|
||||||
|
MetadataStorageConnectorConfig connectorConfig = new MetadataStorageConnectorConfig()
|
||||||
|
{
|
||||||
|
@Override
|
||||||
|
public String getConnectURI()
|
||||||
|
{
|
||||||
|
return "jdbc:postgresql://localhost:3306/test?user=maytas&password=secret&keyonly";
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
JdbcAccessSecurityConfig securityConfig = newSecurityConfigEnforcingAllowList(
|
||||||
|
ImmutableSet.of("user", "password", "keyonly", "etc")
|
||||||
|
);
|
||||||
|
|
||||||
|
new PostgresqlFirehoseDatabaseConnector(
|
||||||
|
connectorConfig,
|
||||||
|
securityConfig
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void testFailOnlyInvalidProperty()
|
||||||
|
{
|
||||||
|
MetadataStorageConnectorConfig connectorConfig = new MetadataStorageConnectorConfig()
|
||||||
|
{
|
||||||
|
@Override
|
||||||
|
public String getConnectURI()
|
||||||
|
{
|
||||||
|
return "jdbc:postgresql://localhost:3306/test?user=maytas&password=secret&keyonly";
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
JdbcAccessSecurityConfig securityConfig = newSecurityConfigEnforcingAllowList(ImmutableSet.of("none", "nonenone"));
|
||||||
|
|
||||||
|
expectedException.expectMessage("is not in the allowed list");
|
||||||
|
expectedException.expect(IllegalArgumentException.class);
|
||||||
|
|
||||||
|
new PostgresqlFirehoseDatabaseConnector(
|
||||||
|
connectorConfig,
|
||||||
|
securityConfig
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void testFailValidAndInvalidProperty()
|
||||||
|
{
|
||||||
|
MetadataStorageConnectorConfig connectorConfig = new MetadataStorageConnectorConfig()
|
||||||
|
{
|
||||||
|
@Override
|
||||||
|
public String getConnectURI()
|
||||||
|
{
|
||||||
|
return "jdbc:postgresql://localhost:3306/test?user=maytas&password=secret&keyonly";
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
JdbcAccessSecurityConfig securityConfig = newSecurityConfigEnforcingAllowList(ImmutableSet.of("user", "nonenone"));
|
||||||
|
|
||||||
|
expectedException.expectMessage("is not in the allowed list");
|
||||||
|
expectedException.expect(IllegalArgumentException.class);
|
||||||
|
|
||||||
|
new PostgresqlFirehoseDatabaseConnector(
|
||||||
|
connectorConfig,
|
||||||
|
securityConfig
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void testIgnoreInvalidPropertyWhenNotEnforcingAllowList()
|
||||||
|
{
|
||||||
|
MetadataStorageConnectorConfig connectorConfig = new MetadataStorageConnectorConfig()
|
||||||
|
{
|
||||||
|
@Override
|
||||||
|
public String getConnectURI()
|
||||||
|
{
|
||||||
|
return "jdbc:postgresql://localhost:3306/test?user=maytas&password=secret&keyonly";
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
JdbcAccessSecurityConfig securityConfig = new JdbcAccessSecurityConfig()
|
||||||
|
{
|
||||||
|
@Override
|
||||||
|
public Set<String> getAllowedProperties()
|
||||||
|
{
|
||||||
|
return ImmutableSet.of("user", "nonenone");
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
new PostgresqlFirehoseDatabaseConnector(
|
||||||
|
connectorConfig,
|
||||||
|
securityConfig
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
private static JdbcAccessSecurityConfig newSecurityConfigEnforcingAllowList(Set<String> allowedProperties)
|
||||||
|
{
|
||||||
|
return new JdbcAccessSecurityConfig()
|
||||||
|
{
|
||||||
|
@Override
|
||||||
|
public Set<String> getAllowedProperties()
|
||||||
|
{
|
||||||
|
return allowedProperties;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public boolean isEnforceAllowedProperties()
|
||||||
|
{
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
};
|
||||||
|
}
|
||||||
|
}
|
|
@ -63,6 +63,7 @@ import org.apache.druid.segment.writeout.SegmentWriteOutMediumModule;
|
||||||
import org.apache.druid.server.emitter.EmitterModule;
|
import org.apache.druid.server.emitter.EmitterModule;
|
||||||
import org.apache.druid.server.initialization.AuthenticatorMapperModule;
|
import org.apache.druid.server.initialization.AuthenticatorMapperModule;
|
||||||
import org.apache.druid.server.initialization.AuthorizerMapperModule;
|
import org.apache.druid.server.initialization.AuthorizerMapperModule;
|
||||||
|
import org.apache.druid.server.initialization.ExternalStorageAccessSecurityModule;
|
||||||
import org.apache.druid.server.initialization.jetty.JettyServerModule;
|
import org.apache.druid.server.initialization.jetty.JettyServerModule;
|
||||||
import org.apache.druid.server.metrics.MetricsModule;
|
import org.apache.druid.server.metrics.MetricsModule;
|
||||||
import org.apache.druid.server.security.TLSCertificateCheckerModule;
|
import org.apache.druid.server.security.TLSCertificateCheckerModule;
|
||||||
|
@ -411,7 +412,8 @@ public class Initialization
|
||||||
new EscalatorModule(),
|
new EscalatorModule(),
|
||||||
new AuthorizerModule(),
|
new AuthorizerModule(),
|
||||||
new AuthorizerMapperModule(),
|
new AuthorizerMapperModule(),
|
||||||
new StartupLoggingModule()
|
new StartupLoggingModule(),
|
||||||
|
new ExternalStorageAccessSecurityModule()
|
||||||
);
|
);
|
||||||
|
|
||||||
ModuleList actualModules = new ModuleList(baseInjector);
|
ModuleList actualModules = new ModuleList(baseInjector);
|
||||||
|
|
|
@ -42,9 +42,18 @@ public class BasicDataSourceExt extends BasicDataSource
|
||||||
{
|
{
|
||||||
private static final Logger LOGGER = new Logger(BasicDataSourceExt.class);
|
private static final Logger LOGGER = new Logger(BasicDataSourceExt.class);
|
||||||
|
|
||||||
private Properties connectionProperties;
|
|
||||||
private final MetadataStorageConnectorConfig connectorConfig;
|
private final MetadataStorageConnectorConfig connectorConfig;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* The properties that will be used for the JDBC connection.
|
||||||
|
*
|
||||||
|
* Note that these properties are not currently checked against any security configuration such as
|
||||||
|
* an allow list for JDBC properties. Instead, they are supposed to be checked before adding to this class.
|
||||||
|
*
|
||||||
|
* @see SQLFirehoseDatabaseConnector#validateConfigs
|
||||||
|
*/
|
||||||
|
private Properties connectionProperties;
|
||||||
|
|
||||||
public BasicDataSourceExt(MetadataStorageConnectorConfig connectorConfig)
|
public BasicDataSourceExt(MetadataStorageConnectorConfig connectorConfig)
|
||||||
{
|
{
|
||||||
this.connectorConfig = connectorConfig;
|
this.connectorConfig = connectorConfig;
|
||||||
|
|
|
@ -21,8 +21,11 @@ package org.apache.druid.metadata;
|
||||||
|
|
||||||
import com.fasterxml.jackson.annotation.JsonTypeInfo;
|
import com.fasterxml.jackson.annotation.JsonTypeInfo;
|
||||||
import com.google.common.base.Predicate;
|
import com.google.common.base.Predicate;
|
||||||
|
import com.google.common.base.Strings;
|
||||||
import org.apache.commons.dbcp2.BasicDataSource;
|
import org.apache.commons.dbcp2.BasicDataSource;
|
||||||
import org.apache.druid.java.util.common.RetryUtils;
|
import org.apache.druid.java.util.common.RetryUtils;
|
||||||
|
import org.apache.druid.server.initialization.JdbcAccessSecurityConfig;
|
||||||
|
import org.apache.druid.utils.ConnectionUriUtils;
|
||||||
import org.skife.jdbi.v2.DBI;
|
import org.skife.jdbi.v2.DBI;
|
||||||
import org.skife.jdbi.v2.exceptions.DBIException;
|
import org.skife.jdbi.v2.exceptions.DBIException;
|
||||||
import org.skife.jdbi.v2.exceptions.UnableToExecuteStatementException;
|
import org.skife.jdbi.v2.exceptions.UnableToExecuteStatementException;
|
||||||
|
@ -32,6 +35,7 @@ import org.skife.jdbi.v2.tweak.HandleCallback;
|
||||||
import java.sql.SQLException;
|
import java.sql.SQLException;
|
||||||
import java.sql.SQLRecoverableException;
|
import java.sql.SQLRecoverableException;
|
||||||
import java.sql.SQLTransientException;
|
import java.sql.SQLTransientException;
|
||||||
|
import java.util.Set;
|
||||||
|
|
||||||
@JsonTypeInfo(use = JsonTypeInfo.Id.NAME, property = "type")
|
@JsonTypeInfo(use = JsonTypeInfo.Id.NAME, property = "type")
|
||||||
public abstract class SQLFirehoseDatabaseConnector
|
public abstract class SQLFirehoseDatabaseConnector
|
||||||
|
@ -62,8 +66,15 @@ public abstract class SQLFirehoseDatabaseConnector
|
||||||
|| (e instanceof DBIException && isTransientException(e.getCause())));
|
|| (e instanceof DBIException && isTransientException(e.getCause())));
|
||||||
}
|
}
|
||||||
|
|
||||||
protected BasicDataSource getDatasource(MetadataStorageConnectorConfig connectorConfig)
|
protected BasicDataSource getDatasource(
|
||||||
|
MetadataStorageConnectorConfig connectorConfig,
|
||||||
|
JdbcAccessSecurityConfig securityConfig
|
||||||
|
)
|
||||||
{
|
{
|
||||||
|
// We validate only the connection URL here as all properties will be read from only the URL except
|
||||||
|
// users and password. If we want to allow another way to specify user properties such as using
|
||||||
|
// MetadataStorageConnectorConfig.getDbcpProperties(), those properties should be validated as well.
|
||||||
|
validateConfigs(connectorConfig.getConnectURI(), securityConfig);
|
||||||
BasicDataSource dataSource = new BasicDataSourceExt(connectorConfig);
|
BasicDataSource dataSource = new BasicDataSourceExt(connectorConfig);
|
||||||
dataSource.setUsername(connectorConfig.getUser());
|
dataSource.setUsername(connectorConfig.getUser());
|
||||||
dataSource.setPassword(connectorConfig.getPassword());
|
dataSource.setPassword(connectorConfig.getPassword());
|
||||||
|
@ -75,6 +86,23 @@ public abstract class SQLFirehoseDatabaseConnector
|
||||||
return dataSource;
|
return dataSource;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
private void validateConfigs(String urlString, JdbcAccessSecurityConfig securityConfig)
|
||||||
|
{
|
||||||
|
if (Strings.isNullOrEmpty(urlString)) {
|
||||||
|
throw new IllegalArgumentException("connectURI cannot be null or empty");
|
||||||
|
}
|
||||||
|
if (!securityConfig.isEnforceAllowedProperties()) {
|
||||||
|
// You don't want to do anything with properties.
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
final Set<String> propertyKeyFromConnectURL = findPropertyKeysFromConnectURL(urlString);
|
||||||
|
ConnectionUriUtils.throwIfPropertiesAreNotAllowed(
|
||||||
|
propertyKeyFromConnectURL,
|
||||||
|
securityConfig.getSystemPropertyPrefixes(),
|
||||||
|
securityConfig.getAllowedProperties()
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
public String getValidationQuery()
|
public String getValidationQuery()
|
||||||
{
|
{
|
||||||
return "SELECT 1";
|
return "SELECT 1";
|
||||||
|
@ -82,5 +110,8 @@ public abstract class SQLFirehoseDatabaseConnector
|
||||||
|
|
||||||
public abstract DBI getDBI();
|
public abstract DBI getDBI();
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Extract property keys from the given JDBC URL.
|
||||||
|
*/
|
||||||
|
public abstract Set<String> findPropertyKeysFromConnectURL(String connectUri);
|
||||||
}
|
}
|
||||||
|
|
|
@ -0,0 +1,44 @@
|
||||||
|
/*
|
||||||
|
* 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.
|
||||||
|
*/
|
||||||
|
|
||||||
|
package org.apache.druid.server.initialization;
|
||||||
|
|
||||||
|
import com.fasterxml.jackson.databind.Module;
|
||||||
|
import com.google.common.collect.ImmutableList;
|
||||||
|
import com.google.inject.Binder;
|
||||||
|
import org.apache.druid.guice.JsonConfigProvider;
|
||||||
|
import org.apache.druid.initialization.DruidModule;
|
||||||
|
|
||||||
|
import java.util.List;
|
||||||
|
|
||||||
|
public class ExternalStorageAccessSecurityModule implements DruidModule
|
||||||
|
{
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public List<? extends Module> getJacksonModules()
|
||||||
|
{
|
||||||
|
return ImmutableList.of();
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void configure(Binder binder)
|
||||||
|
{
|
||||||
|
JsonConfigProvider.bind(binder, "druid.access.jdbc", JdbcAccessSecurityConfig.class);
|
||||||
|
}
|
||||||
|
}
|
|
@ -0,0 +1,100 @@
|
||||||
|
/*
|
||||||
|
* 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.
|
||||||
|
*/
|
||||||
|
|
||||||
|
package org.apache.druid.server.initialization;
|
||||||
|
|
||||||
|
import com.fasterxml.jackson.annotation.JsonIgnore;
|
||||||
|
import com.fasterxml.jackson.annotation.JsonProperty;
|
||||||
|
import com.google.common.collect.ImmutableSet;
|
||||||
|
|
||||||
|
import java.util.Set;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* A config class that applies to all JDBC connections to other databases.
|
||||||
|
*
|
||||||
|
* @see org.apache.druid.utils.ConnectionUriUtils
|
||||||
|
*/
|
||||||
|
public class JdbcAccessSecurityConfig
|
||||||
|
{
|
||||||
|
static final Set<String> DEFAULT_ALLOWED_PROPERTIES = ImmutableSet.of(
|
||||||
|
// MySQL
|
||||||
|
"useSSL",
|
||||||
|
"requireSSL",
|
||||||
|
|
||||||
|
// PostgreSQL
|
||||||
|
"ssl",
|
||||||
|
"sslmode"
|
||||||
|
);
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Prefixes of the properties that can be added automatically by {@link java.sql.Driver} during
|
||||||
|
* connection URL parsing. Any properties resulted by connection URL parsing are regarded as
|
||||||
|
* system properties if they start with the prefixes in this set.
|
||||||
|
* Only these non-system properties are checkaed against {@link #getAllowedProperties()}.
|
||||||
|
*/
|
||||||
|
private static final Set<String> SYSTEM_PROPERTY_PREFIXES = ImmutableSet.of(
|
||||||
|
// MySQL
|
||||||
|
// There can be multiple host and port properties if multiple addresses are specified.
|
||||||
|
// The pattern of the property name is HOST.i and PORT.i where i is an integer.
|
||||||
|
"HOST",
|
||||||
|
"PORT",
|
||||||
|
"NUM_HOSTS",
|
||||||
|
"DBNAME",
|
||||||
|
|
||||||
|
// PostgreSQL
|
||||||
|
"PGHOST",
|
||||||
|
"PGPORT",
|
||||||
|
"PGDBNAME"
|
||||||
|
);
|
||||||
|
|
||||||
|
@JsonProperty
|
||||||
|
private Set<String> allowedProperties = DEFAULT_ALLOWED_PROPERTIES;
|
||||||
|
|
||||||
|
@JsonProperty
|
||||||
|
private boolean allowUnknownJdbcUrlFormat = true;
|
||||||
|
|
||||||
|
// Enforcing allow list check can break rolling upgrade. This is not good for patch releases
|
||||||
|
// and is why this config is added. However, from the security point of view, this config
|
||||||
|
// should be always enabled in production to secure your cluster. As a result, this config
|
||||||
|
// is deprecated and will be removed in the near future.
|
||||||
|
@Deprecated
|
||||||
|
@JsonProperty
|
||||||
|
private boolean enforceAllowedProperties = false;
|
||||||
|
|
||||||
|
@JsonIgnore
|
||||||
|
public Set<String> getSystemPropertyPrefixes()
|
||||||
|
{
|
||||||
|
return SYSTEM_PROPERTY_PREFIXES;
|
||||||
|
}
|
||||||
|
|
||||||
|
public Set<String> getAllowedProperties()
|
||||||
|
{
|
||||||
|
return allowedProperties;
|
||||||
|
}
|
||||||
|
|
||||||
|
public boolean isAllowUnknownJdbcUrlFormat()
|
||||||
|
{
|
||||||
|
return allowUnknownJdbcUrlFormat;
|
||||||
|
}
|
||||||
|
|
||||||
|
public boolean isEnforceAllowedProperties()
|
||||||
|
{
|
||||||
|
return enforceAllowedProperties;
|
||||||
|
}
|
||||||
|
}
|
|
@ -24,6 +24,7 @@ import com.fasterxml.jackson.annotation.JsonTypeName;
|
||||||
import com.fasterxml.jackson.databind.Module;
|
import com.fasterxml.jackson.databind.Module;
|
||||||
import com.fasterxml.jackson.databind.ObjectMapper;
|
import com.fasterxml.jackson.databind.ObjectMapper;
|
||||||
import com.google.common.collect.ImmutableList;
|
import com.google.common.collect.ImmutableList;
|
||||||
|
import com.google.common.collect.ImmutableSet;
|
||||||
import nl.jqno.equalsverifier.EqualsVerifier;
|
import nl.jqno.equalsverifier.EqualsVerifier;
|
||||||
import org.apache.commons.dbcp2.BasicDataSource;
|
import org.apache.commons.dbcp2.BasicDataSource;
|
||||||
import org.apache.commons.io.FileUtils;
|
import org.apache.commons.io.FileUtils;
|
||||||
|
@ -43,6 +44,7 @@ import org.apache.druid.metadata.MetadataStorageConnectorConfig;
|
||||||
import org.apache.druid.metadata.SQLFirehoseDatabaseConnector;
|
import org.apache.druid.metadata.SQLFirehoseDatabaseConnector;
|
||||||
import org.apache.druid.metadata.TestDerbyConnector;
|
import org.apache.druid.metadata.TestDerbyConnector;
|
||||||
import org.apache.druid.segment.TestHelper;
|
import org.apache.druid.segment.TestHelper;
|
||||||
|
import org.apache.druid.server.initialization.JdbcAccessSecurityConfig;
|
||||||
import org.easymock.EasyMock;
|
import org.easymock.EasyMock;
|
||||||
import org.junit.AfterClass;
|
import org.junit.AfterClass;
|
||||||
import org.junit.Assert;
|
import org.junit.Assert;
|
||||||
|
@ -58,6 +60,7 @@ import java.util.Arrays;
|
||||||
import java.util.Comparator;
|
import java.util.Comparator;
|
||||||
import java.util.List;
|
import java.util.List;
|
||||||
import java.util.Objects;
|
import java.util.Objects;
|
||||||
|
import java.util.Set;
|
||||||
import java.util.stream.Collectors;
|
import java.util.stream.Collectors;
|
||||||
import java.util.stream.Stream;
|
import java.util.stream.Stream;
|
||||||
|
|
||||||
|
@ -246,7 +249,17 @@ public class SqlInputSourceTest
|
||||||
@JsonProperty("connectorConfig") MetadataStorageConnectorConfig metadataStorageConnectorConfig
|
@JsonProperty("connectorConfig") MetadataStorageConnectorConfig metadataStorageConnectorConfig
|
||||||
)
|
)
|
||||||
{
|
{
|
||||||
final BasicDataSource datasource = getDatasource(metadataStorageConnectorConfig);
|
final BasicDataSource datasource = getDatasource(
|
||||||
|
metadataStorageConnectorConfig,
|
||||||
|
new JdbcAccessSecurityConfig()
|
||||||
|
{
|
||||||
|
@Override
|
||||||
|
public Set<String> getAllowedProperties()
|
||||||
|
{
|
||||||
|
return ImmutableSet.of("user", "create");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
);
|
||||||
datasource.setDriverClassLoader(getClass().getClassLoader());
|
datasource.setDriverClassLoader(getClass().getClassLoader());
|
||||||
datasource.setDriverClassName("org.apache.derby.jdbc.ClientDriver");
|
datasource.setDriverClassName("org.apache.derby.jdbc.ClientDriver");
|
||||||
this.dbi = new DBI(datasource);
|
this.dbi = new DBI(datasource);
|
||||||
|
@ -283,5 +296,11 @@ public class SqlInputSourceTest
|
||||||
{
|
{
|
||||||
return dbi;
|
return dbi;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public Set<String> findPropertyKeysFromConnectURL(String connectUri)
|
||||||
|
{
|
||||||
|
return ImmutableSet.of("user", "create");
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -21,16 +21,20 @@ package org.apache.druid.metadata.input;
|
||||||
|
|
||||||
import com.fasterxml.jackson.annotation.JsonProperty;
|
import com.fasterxml.jackson.annotation.JsonProperty;
|
||||||
import com.google.common.collect.ImmutableList;
|
import com.google.common.collect.ImmutableList;
|
||||||
|
import com.google.common.collect.ImmutableSet;
|
||||||
import org.apache.commons.dbcp2.BasicDataSource;
|
import org.apache.commons.dbcp2.BasicDataSource;
|
||||||
import org.apache.druid.java.util.common.StringUtils;
|
import org.apache.druid.java.util.common.StringUtils;
|
||||||
import org.apache.druid.metadata.MetadataStorageConnectorConfig;
|
import org.apache.druid.metadata.MetadataStorageConnectorConfig;
|
||||||
import org.apache.druid.metadata.SQLFirehoseDatabaseConnector;
|
import org.apache.druid.metadata.SQLFirehoseDatabaseConnector;
|
||||||
import org.apache.druid.metadata.TestDerbyConnector;
|
import org.apache.druid.metadata.TestDerbyConnector;
|
||||||
|
import org.apache.druid.server.initialization.JdbcAccessSecurityConfig;
|
||||||
import org.junit.Rule;
|
import org.junit.Rule;
|
||||||
import org.skife.jdbi.v2.Batch;
|
import org.skife.jdbi.v2.Batch;
|
||||||
import org.skife.jdbi.v2.DBI;
|
import org.skife.jdbi.v2.DBI;
|
||||||
import org.skife.jdbi.v2.tweak.HandleCallback;
|
import org.skife.jdbi.v2.tweak.HandleCallback;
|
||||||
|
|
||||||
|
import java.util.Set;
|
||||||
|
|
||||||
public class SqlTestUtils
|
public class SqlTestUtils
|
||||||
{
|
{
|
||||||
@Rule
|
@Rule
|
||||||
|
@ -55,7 +59,17 @@ public class SqlTestUtils
|
||||||
@JsonProperty("connectorConfig") MetadataStorageConnectorConfig metadataStorageConnectorConfig, DBI dbi
|
@JsonProperty("connectorConfig") MetadataStorageConnectorConfig metadataStorageConnectorConfig, DBI dbi
|
||||||
)
|
)
|
||||||
{
|
{
|
||||||
final BasicDataSource datasource = getDatasource(metadataStorageConnectorConfig);
|
final BasicDataSource datasource = getDatasource(
|
||||||
|
metadataStorageConnectorConfig,
|
||||||
|
new JdbcAccessSecurityConfig()
|
||||||
|
{
|
||||||
|
@Override
|
||||||
|
public Set<String> getAllowedProperties()
|
||||||
|
{
|
||||||
|
return ImmutableSet.of("user", "create");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
);
|
||||||
datasource.setDriverClassLoader(getClass().getClassLoader());
|
datasource.setDriverClassLoader(getClass().getClassLoader());
|
||||||
datasource.setDriverClassName("org.apache.derby.jdbc.ClientDriver");
|
datasource.setDriverClassName("org.apache.derby.jdbc.ClientDriver");
|
||||||
this.dbi = dbi;
|
this.dbi = dbi;
|
||||||
|
@ -66,6 +80,12 @@ public class SqlTestUtils
|
||||||
{
|
{
|
||||||
return dbi;
|
return dbi;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public Set<String> findPropertyKeysFromConnectURL(String connectUri)
|
||||||
|
{
|
||||||
|
return ImmutableSet.of("user", "create");
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
public void createAndUpdateTable(final String tableName, int numEntries)
|
public void createAndUpdateTable(final String tableName, int numEntries)
|
||||||
|
|
|
@ -21,7 +21,6 @@ package org.apache.druid.segment.realtime.firehose;
|
||||||
|
|
||||||
import com.fasterxml.jackson.databind.ObjectMapper;
|
import com.fasterxml.jackson.databind.ObjectMapper;
|
||||||
import com.google.common.collect.ImmutableList;
|
import com.google.common.collect.ImmutableList;
|
||||||
import org.apache.commons.dbcp2.BasicDataSource;
|
|
||||||
import org.apache.commons.io.FileUtils;
|
import org.apache.commons.io.FileUtils;
|
||||||
import org.apache.druid.data.input.Firehose;
|
import org.apache.druid.data.input.Firehose;
|
||||||
import org.apache.druid.data.input.Row;
|
import org.apache.druid.data.input.Row;
|
||||||
|
@ -31,8 +30,6 @@ import org.apache.druid.data.input.impl.MapInputRowParser;
|
||||||
import org.apache.druid.data.input.impl.TimeAndDimsParseSpec;
|
import org.apache.druid.data.input.impl.TimeAndDimsParseSpec;
|
||||||
import org.apache.druid.data.input.impl.TimestampSpec;
|
import org.apache.druid.data.input.impl.TimestampSpec;
|
||||||
import org.apache.druid.java.util.common.StringUtils;
|
import org.apache.druid.java.util.common.StringUtils;
|
||||||
import org.apache.druid.metadata.MetadataStorageConnectorConfig;
|
|
||||||
import org.apache.druid.metadata.SQLFirehoseDatabaseConnector;
|
|
||||||
import org.apache.druid.metadata.TestDerbyConnector;
|
import org.apache.druid.metadata.TestDerbyConnector;
|
||||||
import org.apache.druid.metadata.input.SqlTestUtils;
|
import org.apache.druid.metadata.input.SqlTestUtils;
|
||||||
import org.apache.druid.segment.TestHelper;
|
import org.apache.druid.segment.TestHelper;
|
||||||
|
@ -42,7 +39,6 @@ import org.junit.Assert;
|
||||||
import org.junit.BeforeClass;
|
import org.junit.BeforeClass;
|
||||||
import org.junit.Rule;
|
import org.junit.Rule;
|
||||||
import org.junit.Test;
|
import org.junit.Test;
|
||||||
import org.skife.jdbi.v2.DBI;
|
|
||||||
|
|
||||||
import java.io.File;
|
import java.io.File;
|
||||||
import java.io.IOException;
|
import java.io.IOException;
|
||||||
|
@ -236,22 +232,4 @@ public class SqlFirehoseFactoryTest
|
||||||
testUtils.dropTable(TABLE_NAME_2);
|
testUtils.dropTable(TABLE_NAME_2);
|
||||||
|
|
||||||
}
|
}
|
||||||
private static class TestDerbyFirehoseConnector extends SQLFirehoseDatabaseConnector
|
|
||||||
{
|
|
||||||
private final DBI dbi;
|
|
||||||
|
|
||||||
private TestDerbyFirehoseConnector(MetadataStorageConnectorConfig metadataStorageConnectorConfig, DBI dbi)
|
|
||||||
{
|
|
||||||
final BasicDataSource datasource = getDatasource(metadataStorageConnectorConfig);
|
|
||||||
datasource.setDriverClassLoader(getClass().getClassLoader());
|
|
||||||
datasource.setDriverClassName("org.apache.derby.jdbc.ClientDriver");
|
|
||||||
this.dbi = dbi;
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
|
||||||
public DBI getDBI()
|
|
||||||
{
|
|
||||||
return dbi;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
|
@ -0,0 +1,90 @@
|
||||||
|
/*
|
||||||
|
* 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.
|
||||||
|
*/
|
||||||
|
|
||||||
|
package org.apache.druid.server.initialization;
|
||||||
|
|
||||||
|
import com.google.common.collect.ImmutableList;
|
||||||
|
import com.google.common.collect.ImmutableSet;
|
||||||
|
import com.google.inject.Guice;
|
||||||
|
import com.google.inject.Injector;
|
||||||
|
import org.apache.druid.guice.DruidGuiceExtensions;
|
||||||
|
import org.apache.druid.guice.JsonConfigurator;
|
||||||
|
import org.apache.druid.guice.LazySingleton;
|
||||||
|
import org.junit.Assert;
|
||||||
|
import org.junit.Test;
|
||||||
|
|
||||||
|
import javax.validation.Validation;
|
||||||
|
import javax.validation.Validator;
|
||||||
|
import java.util.Properties;
|
||||||
|
|
||||||
|
public class ExternalStorageAccessSecurityModuleTest
|
||||||
|
{
|
||||||
|
@Test
|
||||||
|
public void testSecurityConfigDefault()
|
||||||
|
{
|
||||||
|
JdbcAccessSecurityConfig securityConfig = makeInjectorWithProperties(new Properties()).getInstance(
|
||||||
|
JdbcAccessSecurityConfig.class
|
||||||
|
);
|
||||||
|
Assert.assertNotNull(securityConfig);
|
||||||
|
Assert.assertEquals(
|
||||||
|
JdbcAccessSecurityConfig.DEFAULT_ALLOWED_PROPERTIES,
|
||||||
|
securityConfig.getAllowedProperties()
|
||||||
|
);
|
||||||
|
Assert.assertTrue(securityConfig.isAllowUnknownJdbcUrlFormat());
|
||||||
|
Assert.assertFalse(securityConfig.isEnforceAllowedProperties());
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void testSecurityConfigOverride()
|
||||||
|
{
|
||||||
|
Properties properties = new Properties();
|
||||||
|
properties.setProperty("druid.access.jdbc.allowedProperties", "[\"valid1\", \"valid2\", \"valid3\"]");
|
||||||
|
properties.setProperty("druid.access.jdbc.allowUnknownJdbcUrlFormat", "false");
|
||||||
|
properties.setProperty("druid.access.jdbc.enforceAllowedProperties", "true");
|
||||||
|
JdbcAccessSecurityConfig securityConfig = makeInjectorWithProperties(properties).getInstance(
|
||||||
|
JdbcAccessSecurityConfig.class
|
||||||
|
);
|
||||||
|
Assert.assertNotNull(securityConfig);
|
||||||
|
Assert.assertEquals(
|
||||||
|
ImmutableSet.of(
|
||||||
|
"valid1",
|
||||||
|
"valid2",
|
||||||
|
"valid3"
|
||||||
|
),
|
||||||
|
securityConfig.getAllowedProperties()
|
||||||
|
);
|
||||||
|
Assert.assertFalse(securityConfig.isAllowUnknownJdbcUrlFormat());
|
||||||
|
Assert.assertTrue(securityConfig.isEnforceAllowedProperties());
|
||||||
|
}
|
||||||
|
|
||||||
|
private static Injector makeInjectorWithProperties(final Properties props)
|
||||||
|
{
|
||||||
|
return Guice.createInjector(
|
||||||
|
ImmutableList.of(
|
||||||
|
new DruidGuiceExtensions(),
|
||||||
|
binder -> {
|
||||||
|
binder.bind(Validator.class).toInstance(Validation.buildDefaultValidatorFactory().getValidator());
|
||||||
|
binder.bind(JsonConfigurator.class).in(LazySingleton.class);
|
||||||
|
binder.bind(Properties.class).toInstance(props);
|
||||||
|
},
|
||||||
|
new ExternalStorageAccessSecurityModule()
|
||||||
|
)
|
||||||
|
);
|
||||||
|
}
|
||||||
|
}
|
|
@ -370,6 +370,7 @@ reingest
|
||||||
reingesting
|
reingesting
|
||||||
reingestion
|
reingestion
|
||||||
repo
|
repo
|
||||||
|
requireSSL
|
||||||
rollup
|
rollup
|
||||||
rollups
|
rollups
|
||||||
rsync
|
rsync
|
||||||
|
@ -385,6 +386,8 @@ sharding
|
||||||
skipHeaderRows
|
skipHeaderRows
|
||||||
smooshed
|
smooshed
|
||||||
splittable
|
splittable
|
||||||
|
ssl
|
||||||
|
sslmode
|
||||||
stdout
|
stdout
|
||||||
storages
|
storages
|
||||||
stringified
|
stringified
|
||||||
|
@ -420,6 +423,7 @@ unparsed
|
||||||
unsetting
|
unsetting
|
||||||
untrusted
|
untrusted
|
||||||
useFilterCNF
|
useFilterCNF
|
||||||
|
useSSL
|
||||||
uptime
|
uptime
|
||||||
uris
|
uris
|
||||||
urls
|
urls
|
||||||
|
|
Loading…
Reference in New Issue