[composer] Connector factory and JAR discovery utilities (#2662)
parent
7da3eaef77
commit
98c929e6e4
@ -0,0 +1,88 @@
|
|||||||
|
/*
|
||||||
|
* Copyright 2023 Ververica Inc.
|
||||||
|
*
|
||||||
|
* Licensed 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 com.ververica.cdc.composer.utils;
|
||||||
|
|
||||||
|
import com.ververica.cdc.common.factories.Factory;
|
||||||
|
import org.slf4j.Logger;
|
||||||
|
import org.slf4j.LoggerFactory;
|
||||||
|
|
||||||
|
import java.net.URL;
|
||||||
|
import java.util.ArrayList;
|
||||||
|
import java.util.List;
|
||||||
|
import java.util.ServiceLoader;
|
||||||
|
import java.util.stream.Collectors;
|
||||||
|
import java.util.stream.StreamSupport;
|
||||||
|
|
||||||
|
/** Discovery utilities for {@link Factory}. */
|
||||||
|
public class FactoryDiscoveryUtils {
|
||||||
|
|
||||||
|
private static final Logger LOG = LoggerFactory.getLogger(FactoryDiscoveryUtils.class);
|
||||||
|
|
||||||
|
private FactoryDiscoveryUtils() {}
|
||||||
|
|
||||||
|
/** Returns the {@link Factory} for the given identifier. */
|
||||||
|
@SuppressWarnings("unchecked")
|
||||||
|
public static <T extends Factory> T getFactoryByIdentifier(
|
||||||
|
String identifier, Class<T> factoryClass) {
|
||||||
|
|
||||||
|
final ServiceLoader<Factory> loader = ServiceLoader.load(Factory.class);
|
||||||
|
final List<Factory> factoryList = new ArrayList<>();
|
||||||
|
|
||||||
|
for (Factory factory : loader) {
|
||||||
|
if (factory != null
|
||||||
|
&& factory.identifier().equals(identifier)
|
||||||
|
&& factoryClass.isAssignableFrom(factory.getClass())) {
|
||||||
|
factoryList.add(factory);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (factoryList.isEmpty()) {
|
||||||
|
throw new RuntimeException(
|
||||||
|
String.format(
|
||||||
|
"No factory found in the classpath.\n\n"
|
||||||
|
+ "Available factory classes are:\n\n"
|
||||||
|
+ "%s",
|
||||||
|
StreamSupport.stream(loader.spliterator(), false)
|
||||||
|
.map(f -> f.getClass().getName())
|
||||||
|
.sorted()
|
||||||
|
.collect(Collectors.joining("\n"))));
|
||||||
|
}
|
||||||
|
|
||||||
|
if (factoryList.size() > 1) {
|
||||||
|
throw new RuntimeException(
|
||||||
|
String.format(
|
||||||
|
"Multiple factories found in the classpath.\n\n"
|
||||||
|
+ "Ambiguous factory classes are:\n\n"
|
||||||
|
+ "%s",
|
||||||
|
factoryList.stream()
|
||||||
|
.map(f -> f.getClass().getName())
|
||||||
|
.sorted()
|
||||||
|
.collect(Collectors.joining("\n"))));
|
||||||
|
}
|
||||||
|
|
||||||
|
return (T) factoryList.get(0);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Return the path of the jar file that contains the {@link Factory} for the given identifier.
|
||||||
|
*/
|
||||||
|
public static <T extends Factory> URL getJarPathByIdentifier(
|
||||||
|
String identifier, Class<T> factoryClass) {
|
||||||
|
Factory factory = getFactoryByIdentifier(identifier, factoryClass);
|
||||||
|
return factory.getClass().getProtectionDomain().getCodeSource().getLocation();
|
||||||
|
}
|
||||||
|
}
|
@ -0,0 +1,65 @@
|
|||||||
|
/*
|
||||||
|
* Copyright 2023 Ververica Inc.
|
||||||
|
*
|
||||||
|
* Licensed 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 com.ververica.cdc.composer.utils;
|
||||||
|
|
||||||
|
import com.ververica.cdc.common.factories.Factory;
|
||||||
|
import com.ververica.cdc.composer.utils.factory.DataSinkFactory1;
|
||||||
|
import com.ververica.cdc.composer.utils.factory.DataSourceFactory1;
|
||||||
|
import org.junit.jupiter.api.Test;
|
||||||
|
|
||||||
|
import static org.assertj.core.api.Assertions.assertThat;
|
||||||
|
import static org.assertj.core.api.Assertions.assertThatThrownBy;
|
||||||
|
|
||||||
|
/** Test for {@link FactoryDiscoveryUtils}. */
|
||||||
|
class FactoryDiscoveryUtilsTest {
|
||||||
|
|
||||||
|
@Test
|
||||||
|
void getFactoryByIdentifier() {
|
||||||
|
assertThat(
|
||||||
|
FactoryDiscoveryUtils.getFactoryByIdentifier(
|
||||||
|
"data-source-factory-1", Factory.class))
|
||||||
|
.isInstanceOf(DataSourceFactory1.class);
|
||||||
|
|
||||||
|
assertThat(
|
||||||
|
FactoryDiscoveryUtils.getFactoryByIdentifier(
|
||||||
|
"data-sink-factory-1", Factory.class))
|
||||||
|
.isInstanceOf(DataSinkFactory1.class);
|
||||||
|
|
||||||
|
assertThatThrownBy(
|
||||||
|
() ->
|
||||||
|
FactoryDiscoveryUtils.getFactoryByIdentifier(
|
||||||
|
"data-sink-factory-3", Factory.class))
|
||||||
|
.hasMessage(
|
||||||
|
"No factory found in the classpath.\n"
|
||||||
|
+ "\n"
|
||||||
|
+ "Available factory classes are:\n"
|
||||||
|
+ "\n"
|
||||||
|
+ "com.ververica.cdc.composer.utils.factory.DataSinkFactory1\n"
|
||||||
|
+ "com.ververica.cdc.composer.utils.factory.DataSinkFactory2\n"
|
||||||
|
+ "com.ververica.cdc.composer.utils.factory.DataSourceFactory1\n"
|
||||||
|
+ "com.ververica.cdc.composer.utils.factory.DataSourceFactory2");
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
void getJarPathByIdentifier() {
|
||||||
|
assertThat(
|
||||||
|
FactoryDiscoveryUtils.getJarPathByIdentifier(
|
||||||
|
"data-source-factory-1", Factory.class)
|
||||||
|
.getPath())
|
||||||
|
.endsWith("/flink-cdc" + "-composer/target/test-classes/");
|
||||||
|
}
|
||||||
|
}
|
@ -0,0 +1,60 @@
|
|||||||
|
/*
|
||||||
|
* Copyright 2023 Ververica Inc.
|
||||||
|
*
|
||||||
|
* Licensed 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 com.ververica.cdc.composer.utils.factory;
|
||||||
|
|
||||||
|
import org.apache.flink.configuration.ConfigOption;
|
||||||
|
|
||||||
|
import com.ververica.cdc.common.factories.DataSinkFactory;
|
||||||
|
import com.ververica.cdc.common.sink.DataSink;
|
||||||
|
import com.ververica.cdc.common.sink.EventSinkProvider;
|
||||||
|
import com.ververica.cdc.common.sink.MetadataApplier;
|
||||||
|
|
||||||
|
import java.util.HashSet;
|
||||||
|
import java.util.Set;
|
||||||
|
|
||||||
|
/** A dummy {@link DataSinkFactory} for testing. */
|
||||||
|
public class DataSinkFactory1 implements DataSinkFactory {
|
||||||
|
@Override
|
||||||
|
public DataSink createDataSink() {
|
||||||
|
return new DataSink() {
|
||||||
|
@Override
|
||||||
|
public EventSinkProvider getEventSinkProvider() {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public MetadataApplier getMetadataApplier() {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public String identifier() {
|
||||||
|
return "data-sink-factory-1";
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public Set<ConfigOption<?>> requiredOptions() {
|
||||||
|
return new HashSet<>();
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public Set<ConfigOption<?>> optionalOptions() {
|
||||||
|
return new HashSet<>();
|
||||||
|
}
|
||||||
|
}
|
@ -0,0 +1,60 @@
|
|||||||
|
/*
|
||||||
|
* Copyright 2023 Ververica Inc.
|
||||||
|
*
|
||||||
|
* Licensed 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 com.ververica.cdc.composer.utils.factory;
|
||||||
|
|
||||||
|
import org.apache.flink.configuration.ConfigOption;
|
||||||
|
|
||||||
|
import com.ververica.cdc.common.factories.DataSinkFactory;
|
||||||
|
import com.ververica.cdc.common.sink.DataSink;
|
||||||
|
import com.ververica.cdc.common.sink.EventSinkProvider;
|
||||||
|
import com.ververica.cdc.common.sink.MetadataApplier;
|
||||||
|
|
||||||
|
import java.util.HashSet;
|
||||||
|
import java.util.Set;
|
||||||
|
|
||||||
|
/** A dummy {@link DataSinkFactory} for testing. */
|
||||||
|
public class DataSinkFactory2 implements DataSinkFactory {
|
||||||
|
@Override
|
||||||
|
public DataSink createDataSink() {
|
||||||
|
return new DataSink() {
|
||||||
|
@Override
|
||||||
|
public EventSinkProvider getEventSinkProvider() {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public MetadataApplier getMetadataApplier() {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public String identifier() {
|
||||||
|
return "data-source-factory-2";
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public Set<ConfigOption<?>> requiredOptions() {
|
||||||
|
return new HashSet<>();
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public Set<ConfigOption<?>> optionalOptions() {
|
||||||
|
return new HashSet<>();
|
||||||
|
}
|
||||||
|
}
|
@ -0,0 +1,60 @@
|
|||||||
|
/*
|
||||||
|
* Copyright 2023 Ververica Inc.
|
||||||
|
*
|
||||||
|
* Licensed 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 com.ververica.cdc.composer.utils.factory;
|
||||||
|
|
||||||
|
import org.apache.flink.configuration.ConfigOption;
|
||||||
|
|
||||||
|
import com.ververica.cdc.common.factories.DataSourceFactory;
|
||||||
|
import com.ververica.cdc.common.source.DataSource;
|
||||||
|
import com.ververica.cdc.common.source.EventSourceProvider;
|
||||||
|
import com.ververica.cdc.common.source.MetadataAccessor;
|
||||||
|
|
||||||
|
import java.util.HashSet;
|
||||||
|
import java.util.Set;
|
||||||
|
|
||||||
|
/** A dummy {@link DataSourceFactory} for testing. */
|
||||||
|
public class DataSourceFactory1 implements DataSourceFactory {
|
||||||
|
@Override
|
||||||
|
public DataSource createDataSource() {
|
||||||
|
return new DataSource() {
|
||||||
|
@Override
|
||||||
|
public EventSourceProvider getEventSourceProvider() {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public MetadataAccessor getMetadataAccessor() {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public String identifier() {
|
||||||
|
return "data-source-factory-1";
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public Set<ConfigOption<?>> requiredOptions() {
|
||||||
|
return new HashSet<>();
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public Set<ConfigOption<?>> optionalOptions() {
|
||||||
|
return new HashSet<>();
|
||||||
|
}
|
||||||
|
}
|
@ -0,0 +1,61 @@
|
|||||||
|
/*
|
||||||
|
* Copyright 2023 Ververica Inc.
|
||||||
|
*
|
||||||
|
* Licensed 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 com.ververica.cdc.composer.utils.factory;
|
||||||
|
|
||||||
|
import org.apache.flink.configuration.ConfigOption;
|
||||||
|
|
||||||
|
import com.ververica.cdc.common.factories.DataSourceFactory;
|
||||||
|
import com.ververica.cdc.common.source.DataSource;
|
||||||
|
import com.ververica.cdc.common.source.EventSourceProvider;
|
||||||
|
import com.ververica.cdc.common.source.MetadataAccessor;
|
||||||
|
|
||||||
|
import java.util.HashSet;
|
||||||
|
import java.util.Set;
|
||||||
|
|
||||||
|
/** A dummy {@link DataSourceFactory} for testing. */
|
||||||
|
public class DataSourceFactory2 implements DataSourceFactory {
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public DataSource createDataSource() {
|
||||||
|
return new DataSource() {
|
||||||
|
@Override
|
||||||
|
public EventSourceProvider getEventSourceProvider() {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public MetadataAccessor getMetadataAccessor() {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public String identifier() {
|
||||||
|
return "data-sink-factory-2";
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public Set<ConfigOption<?>> requiredOptions() {
|
||||||
|
return new HashSet<>();
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public Set<ConfigOption<?>> optionalOptions() {
|
||||||
|
return new HashSet<>();
|
||||||
|
}
|
||||||
|
}
|
@ -0,0 +1,18 @@
|
|||||||
|
# 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.
|
||||||
|
com.ververica.cdc.composer.utils.factory.DataSinkFactory1
|
||||||
|
com.ververica.cdc.composer.utils.factory.DataSinkFactory2
|
||||||
|
com.ververica.cdc.composer.utils.factory.DataSourceFactory1
|
||||||
|
com.ververica.cdc.composer.utils.factory.DataSourceFactory2
|
@ -0,0 +1,26 @@
|
|||||||
|
################################################################################
|
||||||
|
# Copyright 2023 Ververica Inc.
|
||||||
|
#
|
||||||
|
# Licensed 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.
|
||||||
|
################################################################################
|
||||||
|
|
||||||
|
# Set root logger level to OFF to not flood build logs
|
||||||
|
# set manually to INFO for debugging purposes
|
||||||
|
rootLogger.level=INFO
|
||||||
|
rootLogger.appenderRef.test.ref = TestLogger
|
||||||
|
|
||||||
|
appender.testlogger.name = TestLogger
|
||||||
|
appender.testlogger.type = CONSOLE
|
||||||
|
appender.testlogger.target = SYSTEM_ERR
|
||||||
|
appender.testlogger.layout.type = PatternLayout
|
||||||
|
appender.testlogger.layout.pattern = %-4r [%t] %-5p %c - %m%n
|
Loading…
Reference in New Issue