Skip to content

Commit d182fe3

Browse files
committed
fix compatibility with Flyway 10.17.1 and above
1 parent 7e31fea commit d182fe3

File tree

7 files changed

+89
-12
lines changed

7 files changed

+89
-12
lines changed

build.gradle

+1-1
Original file line numberDiff line numberDiff line change
@@ -323,7 +323,7 @@ project(':embedded-database-spring-test') {
323323
"testRuntimeClasspath_${suite.name}_${version.name}" {
324324
extendsFrom testRuntimeClasspath
325325

326-
if (version.flyway != null && version.flyway.startsWith('10.')) {
326+
if (version.flyway != null && (version.flyway.startsWith('10.') || version.flyway.startsWith('11.'))) {
327327
dependencies.add(project.dependencies.create("org.flywaydb:flyway-database-postgresql:${version.flyway}"))
328328
}
329329

embedded-database-spring-test/src/main/java/io/zonky/test/db/flyway/FlywayWrapper.java

+18-3
Original file line numberDiff line numberDiff line change
@@ -17,6 +17,7 @@
1717
package io.zonky.test.db.flyway;
1818

1919
import com.google.common.collect.ImmutableList;
20+
import org.aopalliance.intercept.Interceptor;
2021
import org.aopalliance.intercept.MethodInterceptor;
2122
import org.flywaydb.core.Flyway;
2223
import org.flywaydb.core.api.resolver.MigrationResolver;
@@ -113,8 +114,7 @@ public Collection<ResolvedMigration> getMigrations() {
113114
if (flywayVersion.isGreaterThanOrEqualTo("9")) {
114115
return invokeMethod(resolver, "resolveMigrations", config);
115116
} else if (flywayVersion.isGreaterThanOrEqualTo("5.2")) {
116-
Class<?> contextType = ClassUtils.forName("org.flywaydb.core.api.resolver.Context", classLoader);
117-
Object contextInstance = ProxyFactory.getProxy(contextType, (MethodInterceptor) invocation ->
117+
Object contextInstance = createMock("org.flywaydb.core.api.resolver.Context", (MethodInterceptor) invocation ->
118118
"getConfiguration".equals(invocation.getMethod().getName()) ? config : invocation.proceed());
119119
return invokeMethod(resolver, "resolveMigrations", contextInstance);
120120
} else {
@@ -126,7 +126,17 @@ public Collection<ResolvedMigration> getMigrations() {
126126
}
127127

128128
private MigrationResolver createMigrationResolver(Flyway flyway) throws ClassNotFoundException {
129-
if (flywayVersion.isGreaterThanOrEqualTo("8")) {
129+
if (flywayVersion.isGreaterThanOrEqualTo("10.17.1")) {
130+
Object executor = getField(flyway, "flywayExecutor");
131+
Object providers = invokeMethod(executor, "createResourceAndClassProviders", true);
132+
Object resourceProvider = getField(providers, "left");
133+
Object classProvider = getField(providers, "right");
134+
Object sqlScript = createMock("org.flywaydb.core.internal.sqlscript.SqlScript", (MethodInterceptor) invocation -> false);
135+
Object sqlScriptFactory = createMock("org.flywaydb.core.internal.sqlscript.SqlScriptFactory", (MethodInterceptor) invocation -> sqlScript);
136+
Object sqlScriptExecutorFactory = createMock("org.flywaydb.core.internal.sqlscript.SqlScriptExecutorFactory");
137+
Object parsingContext = invokeConstructor("org.flywaydb.core.internal.parser.ParsingContext");
138+
return invokeMethod(executor, "createMigrationResolver", resourceProvider, classProvider, sqlScriptExecutorFactory, sqlScriptFactory, parsingContext, null);
139+
} else if (flywayVersion.isGreaterThanOrEqualTo("8")) {
130140
Object executor = getField(flyway, "flywayExecutor");
131141
Object providers = invokeMethod(executor, "createResourceAndClassProviders", true);
132142
Object resourceProvider = getField(providers, "left");
@@ -493,4 +503,9 @@ private static Object createMock(String className) throws ClassNotFoundException
493503
Class<?> proxyInterface = ClassUtils.forName(className, classLoader);
494504
return ProxyFactory.getProxy(proxyInterface, (MethodInterceptor) invocation -> null);
495505
}
506+
507+
private static Object createMock(String className, Interceptor interceptor) throws ClassNotFoundException {
508+
Class<?> proxyInterface = ClassUtils.forName(className, classLoader);
509+
return ProxyFactory.getProxy(proxyInterface, interceptor);
510+
}
496511
}

embedded-database-spring-test/src/test/java/io/zonky/test/db/provider/OpenTableProviderWithConfigurationIntegrationTest.java

+2-2
Original file line numberDiff line numberDiff line change
@@ -19,6 +19,7 @@
1919
import com.opentable.db.postgres.embedded.EmbeddedPostgres;
2020
import io.zonky.test.category.PostgresTestSuite;
2121
import io.zonky.test.db.AutoConfigureEmbeddedDatabase;
22+
import io.zonky.test.support.TestSocketUtils;
2223
import org.junit.Test;
2324
import org.junit.experimental.categories.Category;
2425
import org.junit.runner.RunWith;
@@ -28,7 +29,6 @@
2829
import org.springframework.context.annotation.Configuration;
2930
import org.springframework.test.context.ContextConfiguration;
3031
import org.springframework.test.context.junit4.SpringRunner;
31-
import org.springframework.util.SocketUtils;
3232

3333
import javax.sql.DataSource;
3434
import java.sql.SQLException;
@@ -49,7 +49,7 @@ static class Config {
4949

5050
@Bean
5151
public Integer randomPort() {
52-
return SocketUtils.findAvailableTcpPort();
52+
return TestSocketUtils.findAvailableTcpPort();
5353
}
5454

5555
@Bean

embedded-database-spring-test/src/test/java/io/zonky/test/db/provider/ZonkyProviderWithConfigurationIntegrationTest.java

+2-2
Original file line numberDiff line numberDiff line change
@@ -19,6 +19,7 @@
1919
import io.zonky.test.category.PostgresTestSuite;
2020
import io.zonky.test.db.AutoConfigureEmbeddedDatabase;
2121
import io.zonky.test.db.postgres.embedded.EmbeddedPostgres;
22+
import io.zonky.test.support.TestSocketUtils;
2223
import org.junit.Test;
2324
import org.junit.experimental.categories.Category;
2425
import org.junit.runner.RunWith;
@@ -28,7 +29,6 @@
2829
import org.springframework.context.annotation.Configuration;
2930
import org.springframework.test.context.ContextConfiguration;
3031
import org.springframework.test.context.junit4.SpringRunner;
31-
import org.springframework.util.SocketUtils;
3232

3333
import javax.sql.DataSource;
3434
import java.sql.SQLException;
@@ -49,7 +49,7 @@ static class Config {
4949

5050
@Bean
5151
public Integer randomPort() {
52-
return SocketUtils.findAvailableTcpPort();
52+
return TestSocketUtils.findAvailableTcpPort();
5353
}
5454

5555
@Bean

embedded-database-spring-test/src/test/java/io/zonky/test/db/provider/postgres/OpenTablePostgresDatabaseProviderTest.java

+2-2
Original file line numberDiff line numberDiff line change
@@ -20,6 +20,7 @@
2020
import io.zonky.test.db.preparer.DatabasePreparer;
2121
import io.zonky.test.db.provider.support.BlockingDatabaseWrapper;
2222
import io.zonky.test.db.support.TestDatabasePreparer;
23+
import io.zonky.test.support.TestSocketUtils;
2324
import org.junit.Before;
2425
import org.junit.Test;
2526
import org.junit.runner.RunWith;
@@ -29,7 +30,6 @@
2930
import org.springframework.beans.factory.ObjectProvider;
3031
import org.springframework.jdbc.core.JdbcTemplate;
3132
import org.springframework.mock.env.MockEnvironment;
32-
import org.springframework.util.SocketUtils;
3333

3434
import javax.sql.DataSource;
3535
import java.sql.SQLException;
@@ -93,7 +93,7 @@ public void testGetDatabase() throws Exception {
9393

9494
@Test
9595
public void testDatabaseCustomizers() throws Exception {
96-
int randomPort = SocketUtils.findAvailableTcpPort();
96+
int randomPort = TestSocketUtils.findAvailableTcpPort();
9797
when(databaseCustomizers.getIfAvailable()).thenReturn(Collections.singletonList(builder -> builder.setPort(randomPort)));
9898

9999
DatabasePreparer preparer = TestDatabasePreparer.empty();

embedded-database-spring-test/src/test/java/io/zonky/test/db/provider/postgres/ZonkyPostgresDatabaseProviderTest.java

+2-2
Original file line numberDiff line numberDiff line change
@@ -20,6 +20,7 @@
2020
import io.zonky.test.db.preparer.DatabasePreparer;
2121
import io.zonky.test.db.provider.support.BlockingDatabaseWrapper;
2222
import io.zonky.test.db.support.TestDatabasePreparer;
23+
import io.zonky.test.support.TestSocketUtils;
2324
import org.junit.Before;
2425
import org.junit.Test;
2526
import org.junit.runner.RunWith;
@@ -29,7 +30,6 @@
2930
import org.springframework.beans.factory.ObjectProvider;
3031
import org.springframework.jdbc.core.JdbcTemplate;
3132
import org.springframework.mock.env.MockEnvironment;
32-
import org.springframework.util.SocketUtils;
3333

3434
import javax.sql.DataSource;
3535
import java.sql.SQLException;
@@ -93,7 +93,7 @@ public void testGetDatabase() throws Exception {
9393

9494
@Test
9595
public void testDatabaseCustomizers() throws Exception {
96-
int randomPort = SocketUtils.findAvailableTcpPort();
96+
int randomPort = TestSocketUtils.findAvailableTcpPort();
9797
when(databaseCustomizers.getIfAvailable()).thenReturn(Collections.singletonList(builder -> builder.setPort(randomPort)));
9898

9999
DatabasePreparer preparer = TestDatabasePreparer.empty();
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,62 @@
1+
/*
2+
* Copyright 2024 the original author or authors.
3+
*
4+
* Licensed under the Apache License, Version 2.0 (the "License");
5+
* you may not use this file except in compliance with the License.
6+
* You may obtain a copy of the License at
7+
*
8+
* http://www.apache.org/licenses/LICENSE-2.0
9+
*
10+
* Unless required by applicable law or agreed to in writing, software
11+
* distributed under the License is distributed on an "AS IS" BASIS,
12+
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13+
* See the License for the specific language governing permissions and
14+
* limitations under the License.
15+
*/
16+
17+
package io.zonky.test.support;
18+
19+
import org.springframework.util.Assert;
20+
21+
import javax.net.ServerSocketFactory;
22+
import java.net.InetAddress;
23+
import java.net.ServerSocket;
24+
import java.util.Random;
25+
26+
public class TestSocketUtils {
27+
28+
private static final int PORT_RANGE_MIN = 1024;
29+
private static final int PORT_RANGE_MAX = 65535;
30+
private static final int PORT_RANGE_PLUS_ONE = PORT_RANGE_MAX - PORT_RANGE_MIN + 1;
31+
private static final int MAX_ATTEMPTS = 1_000;
32+
33+
private static final Random random = new Random(System.nanoTime());
34+
35+
private TestSocketUtils() {}
36+
37+
public static int findAvailableTcpPort() {
38+
int candidatePort;
39+
int searchCounter = 0;
40+
do {
41+
Assert.state(++searchCounter <= MAX_ATTEMPTS, () -> String.format(
42+
"Could not find an available TCP port in the range [%d, %d] after %d attempts",
43+
PORT_RANGE_MIN, PORT_RANGE_MAX, MAX_ATTEMPTS));
44+
candidatePort = PORT_RANGE_MIN + random.nextInt(PORT_RANGE_PLUS_ONE);
45+
}
46+
while (!isPortAvailable(candidatePort));
47+
48+
return candidatePort;
49+
}
50+
51+
private static boolean isPortAvailable(int port) {
52+
try {
53+
ServerSocket serverSocket = ServerSocketFactory.getDefault()
54+
.createServerSocket(port, 1, InetAddress.getByName("localhost"));
55+
serverSocket.close();
56+
return true;
57+
}
58+
catch (Exception ex) {
59+
return false;
60+
}
61+
}
62+
}

0 commit comments

Comments
 (0)