Skip to content

Commit 7c0d985

Browse files
authored
Fix TestDatabasePool (#1131)
TestDatabasePool was: 1. Catching the wrong exception (`PersistenceException` instead of `SQLException`) 2. Not releasing database names for re-use to the correct pool (instead they would be released to a brand new pool) This change fixes both of these issues and adds an integration test that works on a real MySQL database, and a regression test for database name releases.
1 parent e02b880 commit 7c0d985

File tree

3 files changed

+57
-6
lines changed

3 files changed

+57
-6
lines changed

misk-hibernate-testing/src/main/kotlin/misk/jdbc/MySqlTestDatabasePoolBackend.kt

+1-1
Original file line numberDiff line numberDiff line change
@@ -16,7 +16,7 @@ import javax.sql.DataSource
1616
internal class MySqlTestDatabasePoolBackend @Inject constructor(
1717
val config: DataSourceConfig
1818
) : TestDatabasePool.Backend {
19-
private val connection: Connection by lazy {
19+
internal val connection: Connection by lazy {
2020
try {
2121
DriverDataSource(
2222
config.buildJdbcUrl(Environment.TESTING),

misk-hibernate-testing/src/main/kotlin/misk/jdbc/TestDatabasePool.kt

+15-4
Original file line numberDiff line numberDiff line change
@@ -1,6 +1,7 @@
11
package misk.jdbc
22

33
import misk.logging.getLogger
4+
import java.sql.SQLException
45
import java.time.Clock
56
import java.time.Duration
67
import java.time.LocalDate
@@ -66,7 +67,13 @@ class TestDatabasePool(
6667

6768
/** Return a config-specific pool, creating it if necessary. */
6869
internal fun getPool(config: DataSourceConfig): ConfigSpecificPool {
69-
val key = config.database!!
70+
var key = config.database!!
71+
for (pool in poolsByKey.values) {
72+
if (key.startsWith(pool.key)) {
73+
key = pool.key
74+
break
75+
}
76+
}
7077
return poolsByKey.computeIfAbsent(key) { ConfigSpecificPool(key, config.type) }
7178
}
7279

@@ -80,7 +87,7 @@ class TestDatabasePool(
8087
private val formatter = DateTimeFormatter.BASIC_ISO_DATE
8188

8289
/** Database names that are available. */
83-
private val pool = LinkedBlockingDeque<String>()
90+
internal val pool = LinkedBlockingDeque<String>()
8491

8592
/** Decodes a database name string. */
8693
fun decode(databaseName: String): DatabaseName? {
@@ -127,7 +134,7 @@ class TestDatabasePool(
127134
try {
128135
backend.createDatabase(databaseName.toString())
129136
break
130-
} catch (_: PersistenceException) {
137+
} catch (_: SQLException) {
131138
logger.info { "Lost a race trying to allocate a test database $databaseName" }
132139
databaseName = databaseName.copy(version = databaseName.version + 1)
133140
}
@@ -140,7 +147,11 @@ class TestDatabasePool(
140147
val evictBefore = LocalDate.now(clock).minus(Period.ofDays(retention.toDays().toInt()))
141148
val evictBeforeYearMonthDay = evictBefore.format(formatter).toLong()
142149

143-
val oldDatabases = getDatabases().filter { it.yearMonthDay < evictBeforeYearMonthDay }
150+
val oldDatabases = if (retention.isZero) {
151+
getDatabases()
152+
} else {
153+
getDatabases().filter { it.yearMonthDay < evictBeforeYearMonthDay }
154+
}
144155

145156
for (database in oldDatabases) {
146157
backend.dropDatabase("$database")

misk-hibernate-testing/src/test/kotlin/misk/jdbc/TestDatabasePoolTest.kt

+41-1
Original file line numberDiff line numberDiff line change
@@ -10,7 +10,10 @@ import misk.testing.MiskTestModule
1010
import org.assertj.core.api.Assertions.assertThat
1111
import org.junit.jupiter.api.BeforeEach
1212
import org.junit.jupiter.api.Test
13+
import org.junit.jupiter.api.assertThrows
14+
import java.sql.SQLException
1315
import java.time.Clock
16+
import java.time.Duration
1417
import javax.inject.Inject
1518
import javax.persistence.PersistenceException
1619

@@ -28,7 +31,12 @@ class TestDatabasePoolTest {
2831
@Inject private lateinit var clock: Clock
2932
@Inject private lateinit var backend: FakeDatabaseBackend
3033

31-
private val config = DataSourceConfig(type = DataSourceType.MYSQL, database = "test")
34+
private val config = DataSourceConfig(
35+
type = DataSourceType.MYSQL,
36+
database = "test",
37+
username = "root",
38+
password = ""
39+
)
3240

3341
private lateinit var testDatabasePool: TestDatabasePool
3442

@@ -103,6 +111,38 @@ class TestDatabasePoolTest {
103111
)
104112
}
105113

114+
@Test fun releasesDatabaseNamesForReuse() {
115+
assertThat(testDatabasePool.takeDatabase(config).database).isEqualTo("test__20180101__1")
116+
assertThat(testDatabasePool.takeDatabase(config).database).isEqualTo("test__20180101__2")
117+
testDatabasePool.releaseDatabase(config.copy(database = "test__20180101__1"))
118+
assertThat(testDatabasePool.takeDatabase(config).database).isEqualTo("test__20180101__1")
119+
}
120+
121+
@Test fun integrationTest() {
122+
val realDatabasePool = TestDatabasePool(MySqlTestDatabasePoolBackend(config), clock)
123+
realDatabasePool.getPool(config)
124+
realDatabasePool.pruneOldDatabases(retention = Duration.ZERO)
125+
126+
// Create a database. This will be the first one that the pool tries to create.
127+
realDatabasePool.backend.createDatabase("test__20180101__1")
128+
129+
// Demonstrate that attempting to create an existing database throws.
130+
assertThrows<SQLException> {
131+
realDatabasePool.backend.createDatabase("test__20180101__1")
132+
}
133+
134+
// Assert that the database pool does not yet know about any database names.
135+
assertThat(realDatabasePool.getPool(config).pool).isEmpty()
136+
137+
// At this point, the pool has tried to allocate "test__20180101__1", failed, and bumped the
138+
// sequence number.
139+
assertThat(realDatabasePool.takeDatabase(config).database).isEqualTo("test__20180101__2")
140+
141+
// If we return a name to the database pool, it will be the next database to be taken.
142+
realDatabasePool.releaseDatabase(config.copy(database = "test__20180101__1"))
143+
assertThat(realDatabasePool.takeDatabase(config).database).isEqualTo("test__20180101__1")
144+
}
145+
106146
private fun backendHasDatabases() {
107147
val todaysDatabases = listOf(
108148
"test__20180101__1",

0 commit comments

Comments
 (0)