Getting Started with ScalarDB JDBC
This document briefly explains how you can get started with ScalarDB JDBC with a simple electronic money application. Here we assume Oracle JDK 8 and the underlying storage/database such as Cassandra are properly configured. Please see Getting Started with ScalarDB for the details of how to configure the underlying storage/database.
From this point forward, we will assume that scalardb.properties that holds the configuration for ScalarDB exists.
Let's move to the getting-started-with-jdbc
directory so that we can avoid too much copy-and-paste.
cd docs/getting-started-with-jdbc
Set up database schema
First, you need to define how the data will be organized (a.k.a database schema) in the application with ScalarDB database schema. You can do that with the command line tool for ScalarDB SQL. Download the command line tool that matches with the version you use from ScalarDB SQL releases, and run the following command to start it:
java -jar scalardb-sql-cli-<version>-all.jar --config scalardb.properties
And then, you need to create the namespace emoney
and the table account
in the namespace as follows:
0: scalardb> CREATE NAMESPACE emoney;
0: scalardb> CREATE TABLE emoney.account (id TEXT PRIMARY KEY, balance INT);
Also, you need to create coordinator tables that are used in transactions as follows:
0: scalardb> CREATE COORDINATOR TABLES;
Store & retrieve data
ElectronicMoney.java
is a simple electronic money application.
(Be careful: it is simplified for ease of reading and far from practical and is certainly not production-ready.)
public class ElectronicMoney {
private static final String SCALARDB_PROPERTIES =
System.getProperty("user.dir") + File.separator + "scalardb.properties";
private static final String JDBC_CONNECTION_URL = "jdbc:scalardb:" + SCALARDB_PROPERTIES;
private static final String FULL_TABLENAME = "emoney.account";
private static final String ID = "id";
private static final String BALANCE = "balance";
public void charge(String id, int amount) throws SQLException {
try (Connection connection = DriverManager.getConnection(JDBC_CONNECTION_URL)) {
// Turn off autocommit
connection.setAutoCommit(false);
try {
// Retrieve the current balance for id
Optional<Integer> currentBalance;
try (PreparedStatement preparedStatement = connection.prepareStatement(
"SELECT " + BALANCE + " FROM " + FULL_TABLENAME + " WHERE " + ID + "=?")) {
preparedStatement.setString(1, id);
try (ResultSet resultSet = preparedStatement.executeQuery()) {
if (resultSet.next()) {
currentBalance = Optional.of(resultSet.getInt(BALANCE));
} else {
currentBalance = Optional.empty();
}
}
}
// Calculate the balance
int balance = amount;
if (currentBalance.isPresent()) {
int current = currentBalance.get();
balance += current;
}
// Update the balance
try (PreparedStatement preparedStatement = connection.prepareStatement(
"UPDATE " + FULL_TABLENAME + " SET " + BALANCE + "=?" + " WHERE " + ID + "=?")) {
preparedStatement.setInt(1, balance);
preparedStatement.setString(2, id);
preparedStatement.executeUpdate();
}
// Commit the transaction (records are automatically recovered in case of failure)
connection.commit();
} catch (Exception e) {
connection.rollback();
throw e;
}
}
}
public void pay(String fromId, String toId, int amount) throws SQLException {
try (Connection connection = DriverManager.getConnection(JDBC_CONNECTION_URL)) {
// Turn off autocommit
connection.setAutoCommit(false);
try {
// Retrieve the current balances for ids (it assumes that both accounts exist)
int fromBalance;
int toBalance;
try (PreparedStatement preparedStatement = connection.prepareStatement(
"SELECT " + BALANCE + " FROM " + FULL_TABLENAME + " WHERE " + ID + "=?")) {
preparedStatement.setString(1, fromId);
try (ResultSet resultSet = preparedStatement.executeQuery()) {
resultSet.next();
fromBalance = resultSet.getInt(BALANCE);
}
preparedStatement.clearParameters();
preparedStatement.setString(1, toId);
try (ResultSet resultSet = preparedStatement.executeQuery()) {
resultSet.next();
toBalance = resultSet.getInt(BALANCE);
}
}
// Calculate the balances
int newFromBalance = fromBalance - amount;
int newToBalance = toBalance + amount;
if (newFromBalance < 0) {
throw new RuntimeException(fromId + " doesn't have enough balance.");
}
// Update the balances
try (PreparedStatement preparedStatement = connection.prepareStatement(
"UPDATE " + FULL_TABLENAME + " SET " + BALANCE + "=?" + " WHERE " + ID + "=?")) {
preparedStatement.setInt(1, newFromBalance);
preparedStatement.setString(2, fromId);
preparedStatement.executeUpdate();
preparedStatement.clearParameters();
preparedStatement.setInt(1, newToBalance);
preparedStatement.setString(2, toId);
preparedStatement.executeUpdate();
}
// Commit the transaction (records are automatically recovered in case of failure)
connection.commit();
} catch (Exception e) {
connection.rollback();
throw e;
}
}
}
public int getBalance(String id) throws SQLException {
try (Connection connection = DriverManager.getConnection(JDBC_CONNECTION_URL)) {
// Turn off autocommit
connection.setAutoCommit(false);
try {
// Retrieve the current balance for id
Optional<Integer> currentBalance;
try (PreparedStatement preparedStatement = connection.prepareStatement(
"SELECT " + BALANCE + " FROM " + FULL_TABLENAME + " WHERE " + ID + "=?")) {
preparedStatement.setString(1, id);
try (ResultSet resultSet = preparedStatement.executeQuery()) {
if (resultSet.next()) {
currentBalance = Optional.of(resultSet.getInt(BALANCE));
} else {
currentBalance = Optional.empty();
}
}
}
// Commit the transaction
connection.commit();
return currentBalance.orElse(-1);
} catch (Exception e) {
connection.rollback();
throw e;
}
}
}
}
Before you run the application, you need to specify your GitHub username and your personal access token to access our private Maven repository. One of the ways to specify them is using environment variables as follows (See build.gradle):
export GPR_USERNAME=<your GitHub username>
export GPR_PASSWORD=<your personal access token>
Now we can run the application.
- Charge
1000
touser1
:
./gradlew run --args="-action charge -amount 1000 -to user1"
- Charge
0
tomerchant1
(Just create an account formerchant1
):
./gradlew run --args="-action charge -amount 0 -to merchant1"
- Pay
100
fromuser1
tomerchant1
:
./gradlew run --args="-action pay -amount 100 -from user1 -to merchant1"