“Not all treasure is silver and gold, mate.”
This article was published in:
This is a fictional project used for studies in my laboratory using Scala Language, with a focus on micro-services.
Project Owner Requirements:
Create a micro-service that checks and creates a hypothetical bank account.
It must comprise a HTTP Server with two endpoints:
- One to insert a new monetary transaction, money in or out, for a given user;
- One to return a user's current balance.
Requirements:
- It must not be possible to withdraw money for a given user when they don't have enough balance;
- You should take concurrency issues into consideration;
- Store data in memory;
- You should pay homage to Jack Sparrow.
Designer partners and good coding practices:
- Immutability;
- Separation of concerns;
- Unit and integration tests;
- API design;
- Error handling;
- Language idiomatic use;
- Use functional programming paradigm.
The architecture of the proposed solution follows the Hexagonal Architecture concept. The design is based on two books:
Example Diagram of a Hexagonal Architecture:
Organization Application Package Diagram:
Organization of Project Directories:
Directory | Comments |
---|---|
src/main/scala/sparrow/account/domain | Contains class models and business logic |
src/main/scala/sparrow/account/model | Contains template class and interface |
src/main/scala/sparrow/account/interceptors | Classes with templates with errors and exceptions |
src/main/scala/sparrow/account/controller | Account control services |
src/test/scala/sparrow/account/unit | The unit testing of the application |
src/test/scala/sparrow/account/integration | Client http application for integrated testing |
The language used to develop the challenge was Scala. Using the following technologies:
To build a request and response Http Rest Server, Finagle-Finch was used:
Piece of code where the server is used: src/main/scala/sparrow/account/ServerApp.scala
def run(): Unit = {
val app = Http
.server
.withLabel(serverConf.name)
.withAdmissionControl.concurrencyLimit(
maxConcurrentRequests = serverConf.maxConcurrentRequests,
maxWaiters = serverConf.maxWaiters
).serve(s"${serverConf.host}:${serverConf.port}",
(Routes.balanceAccount :+: Routes.fillAccount).toService)
onExit {
app.close()
}
Await.ready(app)
}
The EndPoints available on the server:
Method | EndPoint | Example Parameter |
---|---|---|
POST | /account | {"uuid":"1", "amount":100.50} |
GET | /balance/ | not required |
Piece of code of the routes with the endpoints: src/main/scala/sparrow/account/Routes.scala
final val fillAccount: Endpoint[Account] =
post("account" :: jsonBody[AccountFillRequest]) {req: AccountFillRequest =>
for {
r <- accountService.fillAccount(req.uuid, req.amount)
} yield r match {
case Right(a) => Ok(a)
case Left(m) => BadRequest(m)
}
}
There are two endpoints (fillAccount) that can create an account and deposit a value, as well as can withdraw using the negative value. And the endpoint (balanceAccount) to see the balance available to the user.
The ScalaSTM was used to store the data in memory and control of the concurrency.
Piece of code where atomicity is used: src/main/scala/sparrow/account/controller/AccountController.scala
override def fillAccount(uuid: String, amount: Double): Future[Either[AccountFillException, AccountTransaction]] = Future {
if (accounts.get(uuid).isEmpty) createAccount(uuid, 0)
accounts.get(uuid) match {
case Some(transact) => {
atomic {implicit tx =>
transact() = AccountTransaction(transact().uuid, transact().amount + amount)
displayOperationType(transact().uuid, transact().amount)
if (amountIsNegative(transact().amount)) {
transact() = AccountTransaction(transact().uuid, transact().amount - amount)
}
Right(transact())
}
}
case _ => Left(AccountFillException("Fill account not found."))
}
}
Other tools used in the project are in the order below:
- https://github.com/lightbend/config - Configuration library
- http://www.scalatest.org/ - Tests
First, have the project folder unzipped on your machine and the terminal open.
A Docker image has been generated that can be used for testing, so it is necessary to have the docker installed on your machine.
Simply run the script:
$ ./scripts/create-docker.sh
Note: Please install docker in your Linux distribution.
With docker already installed on your machine, list docker images using the command:
$ docker images
Start docker with the image to do the tests:
$ docker run -p 8080:8080 sparrow-account:v0.1
Docker will run the unit tests and then start the server. Wait until the server loads the message:
*** Stating Jack Sparrow HTTP Server **** Host: 0.0.0.0 Port: 8080
You can do a quick test on the project source code by running the curl scripts.
$ cd sparrow-account
$ ./scripts/curl-test.sh
Note: See more command curl information in item 3.3, to run the integrated tests, see item 3.4. More information on removing and stopping the docker image see item 4.
To build the project on the machine it is necessary to have the programs installed:
Note: The programs above are with the version used to develop this solution.
The following procedures were using a Linux machine running on Ubuntu-64 18.04.2 LTS.
For JDK installation:
$ tar -xvzf jdk-8u212-linux-x64.tar.gz
$ sudo mv jdk-8xxx /opt/jdk1.8.0
$ export JAVA_HOME=/opt/jdk1.8.0
$ export PATH=/opt/jdk1.8.0/bin:${PATH}
For SBT installation:
$ wget https://piccolo.link/sbt-1.2.8.tgz
$ tar xfv sbt-1.2.8.tgz
$ sudo mv sbt /opt/sbt
$ export PATH=/opt/sbt/bin:${PATH}
Note: You can also use OpenJDK, find out how to install OpenJDK in your distribution.
To build the application just run:
$ git clone https://github.com/edersoncorbari/sparrow-account.git
$ sbt compile; sbt test; sbt run
You can also run unit tests like this:
$ sbt clean coverage test coverageReport
Checking the ScalaStyle code:
$ sbt scalastyle
When you receive the message on the terminal after the sbt run:
*** Stating Jack Sparrow HTTP Server **** Host: 0.0.0.0 Port: 8080
On another terminal run the command:
$ cd sparrow-account
$ ./scripts/curl-test.sh
This command uses the curl to make the API endpoint requests. The command create an account for the user jack.sparrow and make the deposit in the amount of 1000.0. Then checks the balance, and then withdraw the value again and the account gets value zero amount.
Creating an account manually via curl:
$ curl -i -H "Content-Type: application/json" -X POST -d '{"uuid":"1", "amount":10.95}' http://127.0.0.1:8080/account
The answer should be:
{"uuid":"1","amount":10.95}
Depositing more than one account:
$ curl -i -H "Content-Type: application/json" -X POST -d '{"uuid":"1", "amount":51.99}' http://127.0.0.1:8080/account
The answer should be:
{"uuid":"1","amount":62.94}
Withdraw account amount:
$ curl -i -H "Content-Type: application/json" -X POST -d '{"uuid":"1", "amount":-30.50}' http://127.0.0.1:8080/account
The answer should be:
{"uuid":"1","amount":32.44}
Withdrawing value where a with no sufficient value in the account:
$ curl -i -H "Content-Type: application/json" -X POST -d '{"uuid":"1", "amount":-300.0}' http://127.0.0.1:8080/account
The answer should be:
{"uuid":"1","amount":32.44}
To check your account balance simply use:
$ curl -i -H "Content-Type: application/json" -X GET http://127.0.0.1:8080/balance/1
The answer should be:
{"uuid":"1","amount":32.44}
If necessary, you can change the host and port configuration of the server. This can be checked at: src/main/resources/application.conf
A small HTTP client server was created to do the integrated test. To run the server it is necessary to compile the code via SBT and it is necessary that the server is running via local compilation or in the docker.
In the project root directory run:
$ sbt "test:runMain sparrow.account.integration.HttpClientSuiteTest"
Stop the docker image:
$ docker container ls
$ docker stop <CONTAINER-ID>
Remove the docker image:
$ docker images
$ docker rmi -f <IMAGE-ID>
Note: This application was developed using a FreeBSD 11.2 machine and using a virtual machine with Ubuntu 18.04.2 LTS for testing.
Enjoy