This directory provides the source code for a Cloud Spanner sample financial application. It consists of 2 components:
- A gRPC server that provides application level functionality (see: service.proto) using CloudSpanner as the storage backend, found in the server directory
- A workload generator for the above, found in the workload directory
The goal of the sample is to provide a simple working example to jump-start exploring Cloud Spanner.
This is how everything fits together:
The FinAppServer is the backend of the application. It provides a gRPC interface with application-level operations and uses Cloud Spanner as the storage backend. We provide two alternative implementations for connecting to Cloud Spanner for educational purposes:
Clients, including the workload generator, a potential frontend component, or tools like grpc_cli can connect to the FinAppServer using gRPC.
The application models the following entities:
- Customer: End-user of the application. A customer can have one or more accounts.
- Account: A financial account, keeps track of balance. Multiple customers could access a single account with different roles.
- Transaction: A monetary transfer associated with one or more accounts.
The backend should support the following operations:
- CreateCustomer
- CreateAccount
- CreateCustomerRole: Associate a customer with a role for a specified account
- CreateTransactionForAccount: Monetary transfer between an account and an external entity
- MoveAccountBalance: Monetary transfer between two accounts
- GetRecentTransactionsForAccount: Returns transactions for an account within a specified time window, ordered by descending timestamp (more recent first). Should support pagination.
Good schema design can unlock Cloud Spanner capabilities for scale-out with essentially no limits by automatically sharding data based on load and size. On the other hand, schema anti-patterns, like hotspotting, can cause bottlenecks and seriously handicap performance. Developers using Cloud Spanner should familiarize themselves early with Cloud Spanner's data model and schema best practices. This section gives examples of how schema best practices apply to the finance application, for the full schema see schema.sdl.
Using a column whose value monotonically increases, or decreases, as the first key of a high write rate table or index should be avoided as it can cause hot-spots. For this reason the sample uses UUIDv4 for all primary keys (e.g: CustomerId, AccountId).
Interleaving is a good choice for many parent-child (1-to-N) relationships. It allows co-location of child rows with their parent rows and can significantly improve performance, for details see: interleaved tables, database splits, comparison with foreign keys.
The Account to TransactionHistory relation is a good example for using interleaving. It is a 1-to-N relationship. Moreover, all financial transfers would need to update both tables (add a row to TransactionHistory and update the Account balance for the same AccountId). Interleaving TransactionHistory under Account results in co-locating relevant rows and as a result allows such updates to be performant.
The Customer to Account relation, is an M-to-N and requires a separate table: CustomerRole. CustomerRole could use foreign keys for both CustomerId and AccountId, or be interleaved in one of Customer, Account and use foreign key for the other. We decided to interleave under Customer assuming that future operations might take advantage of co-location, but any solution would work equally well for the target operations.
Tables (or indexes) that have some type of history keyed by timestamp are common, TransactionHistory is a good example. In such cases it is important to:
- Not use the timestamp as the first key part (see hot-spotting)
- Use descending order for timestamps
Indeed, for TransactionHistory we use primary key:
AccountId, EventTimestamp DESC
which is optimal for the
GetRecentTransactionsForAccount
operation. Note that all operations described
create at most one new row in TransactionHistory. In such cases it is safe
to use the commit timestamp as a key.
NOTE: Requires bash, gcloud, mvn, grpc_cli installed.
NOTE: The PostgreSQL Interface will only work with a Cloud Spanner instance
-
Bring up the FinAppServer hosting a grpc service.
$ bash run.sh server java \ --spanner_project_id=test-project --spanner_instance_id=test-instance \ --spanner_database_id=test-database
NOTE: To run the application using the JDBC implementation, in the command above, substitute
java
withjdbc
.NOTE: To run the application using the PG Interface implementation, in the command above, substitute
java
withpg
. -
Call RPCs using grpc_cli.
$ grpc_cli call localhost:8080 CreateCustomer \ "name: 'google' address: 'amphitheatre pkwy'" --channel_creds_type=insecure
NOTE: The PostgreSQL Interface will not work with the Spanner emulator
-
Create a database locally using cloud-spanner-emulator and export spanner host for client libraries to work.
$ mvn clean install -Dmaven.test.skip=true $ bash run.sh emulator $ export SPANNER_EMULATOR_HOST="localhost:9010"
-
Bring up the FinAppServer hosting a grpc service.
$ bash run.sh server java \ --spanner_project_id=test-project --spanner_instance_id=test-instance \ --spanner_database_id=test-database
NOTE: To run the application using the JDBC implementation, in the command above, substitute
java
withjdbc
. -
Call RPCs using grpc_cli.
$ grpc_cli call localhost:8080 CreateCustomer \ "name: 'google' address: 'amphitheatre pkwy'" --channel_creds_type=insecure
-
Bring up the finapp server using steps described above.
-
In a separate terminal, bring up the workload using the following command:
$ bash run.sh workload \ --address-name localhost --port 8080 --num-accounts 200
- Set up the emulator as described in #1 above.
mvn integration-test
tests the Java client implementationmvn integration-test -DSPANNER_USE_JDBC=true
tests the JDBC implementation- Testing the PG Interface is not yet available