JDBC-Driver-Cache is JDBC cache which store the result of a SQL query (ResultSet) in files or in memory. The same query requested again will be read from the file, the database is no more requested again.
You may use it to easily mock ResultSets from a database.
JDBC-Driver-Cache is itself a JDBC driver and acts as a wrapper over any third-party JDBC driver.
The library is available on Maven Central.
<dependency>
<groupId>com.qwazr</groupId>
<artifactId>jdbc-cache-driver</artifactId>
<version>1.3</version>
</dependency>
First, you have to initialize the JDBC drivers. In this example we use Apache Derby as backend driver. You can use any compliant JDBC driver.
// Initialize the cache driver
Class.forName("com.qwazr.jdbc.cache.Driver");
// Provide the URL and the Class name of the backend driver
Properties info = new Properties();
info.setProperty("cache.driver.url", "jdbc:derby:memory:myDB;create=true");
info.setProperty("cache.driver.class", "org.apache.derby.jdbc.EmbeddedDriver");
Use the file cache implementation:
// Get your JDBC connection
Connection cnx = DriverManager.getConnection("jdbc:cache:file:/var/jdbc/cache", info);
Or use the in memory cache implementation:
// Get your JDBC connection
Connection cnx = DriverManager.getConnection("jdbc:cache:mem:my-memory-cache", info);
To build a connection you have to provide the URL and some properties. The URL tells the driver where to store the cached ResultSet.
The syntax of the URL can be:
- jdbc:cache:file:{path-to-the-cache-directory} for on disk cache
- jdbc:cache:mem:{name-of-the-cache} for in memory cache
Two possible properties:
- cache.driver.url contains the typical JDBC URL of the backend driver.
- cache.driver.class contains the class name of the backend driver.
The properties are passed to both the cache driver and the backend driver.
You can also disable the cache by setting false to the property cache.driver.active. In this mode, the cache driver is transparent. All the queries and the result handled by the backend-driver.
info.setProperty("cache.driver.active", "false");
Connection cnx = DriverManager.getConnection("jdbc:cache:file:/var/jdbc/cache", info);
When using the JDBC cache driver, it's important to understand how column names are handled:
-
Case Sensitivity: The JDBC cache driver requires exact case matching for column names, even when the underlying database supports case-insensitive column access.
-
Column Name Access:
- Always use the exact case of column names as they are stored in the database
- For H2 databases, column names are typically stored in uppercase by default
- When using quoted identifiers in H2, the case is preserved as specified
-
Example with H2:
// H2 typically stores column names in uppercase
ResultSet rs = stmt.executeQuery("SELECT * FROM employees");
rs.getInt("ID"); // Works
rs.getInt("id"); // May fail
rs.getInt("Id"); // May fail
// With quoted identifiers, case is preserved
ResultSet rs = stmt.executeQuery("SELECT \"userId\", \"firstName\" FROM users");
rs.getInt("userId"); // Works
rs.getInt("USERID"); // May fail
-
Database-Specific Behavior:
- Oracle: Column names are typically uppercase unless quoted
- H2: Column names are uppercase by default unless quoted
- Other databases may have different default behaviors
-
Troubleshooting:
- If you encounter "Column not found" errors, check the actual column names using
ResultSetMetaData
- Print column metadata to verify the exact case of column names:
- If you encounter "Column not found" errors, check the actual column names using
ResultSetMetaData metaData = rs.getMetaData();
for (int i = 1; i <= metaData.getColumnCount(); i++) {
System.out.println("Column " + i + ": " + metaData.getColumnName(i));
}
The JDBC cache driver can be used with Oracle JDBC and HikariCP connection pool. Here's how to set it up:
- Add Required Dependencies:
<dependency>
<groupId>com.qwazr</groupId>
<artifactId>jdbc-cache-driver</artifactId>
<version>1.3</version>
</dependency>
<dependency>
<groupId>com.oracle.database.jdbc</groupId>
<artifactId>ojdbc8</artifactId>
<version>21.1.0.0</version>
</dependency>
<dependency>
<groupId>com.zaxxer</groupId>
<artifactId>HikariCP</artifactId>
<version>4.0.3</version>
</dependency>
- Configure HikariCP with JDBC Cache:
// Create HikariCP configuration
HikariConfig config = new HikariConfig();
config.setJdbcUrl("jdbc:cache:file:/path/to/cache/directory");
config.setUsername("your_username");
config.setPassword("your_password");
// Set the JDBC cache driver class
config.setDriverClassName("com.qwazr.jdbc.cache.Driver");
// Configure cache properties
Properties cacheProps = new Properties();
cacheProps.setProperty("cache.driver.url", "jdbc:oracle:thin:@localhost:1521:ORCL");
cacheProps.setProperty("cache.driver.class", "oracle.jdbc.OracleDriver");
cacheProps.setProperty("cache.ttl", "3600"); // Cache TTL in seconds
config.setDataSourceProperties(cacheProps);
// Create the connection pool
HikariDataSource dataSource = new HikariDataSource(config);
- Using the Connection Pool:
try (Connection conn = dataSource.getConnection();
Statement stmt = conn.createStatement();
ResultSet rs = stmt.executeQuery("SELECT * FROM employees")) {
while (rs.next()) {
// Access data using column names
int id = rs.getInt("ID");
String name = rs.getString("NAME");
double salary = rs.getDouble("SALARY");
}
} catch (SQLException e) {
e.printStackTrace();
}
-
Important Notes:
- The cache directory must exist and be writable
- Set appropriate TTL (Time To Live) for cached results
- Oracle column names are typically uppercase unless quoted
- The cache driver will reuse cached results for identical queries within the TTL period
-
Example with Error Handling:
try (Connection conn = dataSource.getConnection()) {
// First query - will hit the database
try (Statement stmt = conn.createStatement();
ResultSet rs = stmt.executeQuery("SELECT * FROM employees")) {
while (rs.next()) {
System.out.println("First query result: " + rs.getString("NAME"));
}
}
// Second query - will use cache if within TTL
try (Statement stmt = conn.createStatement();
ResultSet rs = stmt.executeQuery("SELECT * FROM employees")) {
while (rs.next()) {
System.out.println("Cached result: " + rs.getString("NAME"));
}
}
} catch (SQLException e) {
System.err.println("Database error: " + e.getMessage());
e.printStackTrace();
} finally {
// Don't close the dataSource here - it should be closed when the application shuts down
}
- Performance Considerations:
- HikariCP's connection pooling helps manage database connections efficiently
- The cache driver reduces database load by serving cached results
- Monitor cache directory size and adjust TTL as needed
- Consider using different cache directories for different types of queries
JDBC-Driver-Cache is open source and is licensed under the Apache 2.0 License.
Report any issue here: https://github.com/qwazr/jdbc-cache-driver/issues