Matthew M Dalby

Inventory Tracking System

Architecture

The core requirements for an API are pretty standard here, and listed as follows:

  • Logging
  • HTTP call handler
  • Error handling/reporting
  • Middleware
  • Environment based configuration
  • Persistent data storage
  • Messaging
  • Build configuration

System Overview

Keep in mind, the point of the project is to implement all API calls in both Java and Node stacks in order to illustrate the common concepts and how they are applied to each stack.

API instances are intended to be deployed onto a cloud environment. For scaleability purposes, we will be supporting the ability to

General Design Decisions

Authentication will be minimally addressed in this project. For most most part we are just conceearned with the presence of a header with an auth token which just establishes the fact the user has completed some prior authentication step, and allows us to pull up user details and permissions. The core concept here is that checks are performed in a central location, and applying to all incomming requests prior to servicing the requests.

1Sample source code....
1Sample source code....

Exception Reporting

In a similar manner to the approach for authentication, we also centralize exception handling. The goal is to not only standarding exception reporting in the respone payloads, but also to provide for exception reporting to a third party application for more enhanced alerting. From a Node perspective, this is applied via middleware. As for the Spring implementation, we make use of the @ControllerAdvice annotation, however the concept is similar across both stacks.

1Sample source code....
1Sample source code....

Server Side Aspects

Logging Approach

From a Java perspective, we will be working with the Log4j framework, fairly standard stuff. From a Node perspective, there are a few more options, however winston appears to lead the pack, it is what I have used in the path, so it seemed like a natural fit.

In both cases, I will be wrapping each of the libraries into wrappers. The idea here is to support the ability to pass a transaction id along with all messages so all events may be correlated with the same unique transaction id.

In addition to standard logging operations, support for logging and structuring certain log related information for certain calculations for auditing purposes. I had previously shared an article on this approach which can be located at: https://www.matthewdalby.dev/articles/software-engineering/business-process-auditing

Frameworks

For the Java stack I will be using the Spring framework, specifically Spring Boot, which is almost a defacto standard. In the Node ecosystem I will be using the Express framework. Widely supported, and more than capable of meeting the requirements for this effort.

Exception Handling

Java API Implementations

Node API Implementations

Environment based configurations

Persistence

For the Spring based stack, we will be using JPA (Java Persistace API). This is an formal API specification. The API provides an impressive set of functionality.

From a Node perspective, things become a bit more complicated. Unlike the walled garded in a Spring stack, there are a few more options from which Sequelize and TypeORM. I ultimated decided on this implementation to select TypeORM. Given the fact that it is typescript based, and supports the use of decorators for defining mapping rules, it seemed like the natural choice. I am a big fan of annotations/decorators and am exited to see them finally beging to materialize in the Javaascript ecosystem.

In both cases we will &pos;mostly' be working with the data via the ORM layer, however we will be making exceptions in cases where applying performance operations make sense.

Messaging

In both stacks we will be using the same messaging solution, which is RabbitMQ. I decided to use this as I have worked with it in the past, and have no strong requirement to use another package. AmazonMQ is another potential option, that would makes sense since we are deploying to an AWS environment. In either case, the application will be architected in such a manner that it is decoupled from the actual messaging implementation.

1Sample source code....
1Sample source code....

Build Configuration

Java API Implementations

Node API Implementations

UI Aspects

UI stack

I decided to use React for the frontend as this is where I spend most of my time over the past several years.

API Calls

For all API interactions, I will be performing the calls via the Tankstack useQuery hook. I am a big fan of this approach, and have shared an previous article on this approach (Performing API calls with Tanstack).

CSS Frameworks

After carefull review of options, I decided to implement an CSS framework in place of hand rolling the entire effort.

State Management

State management was an interesting decision point. I have worked with Redux on many projects in the past. Due to the fact we are using the Tanstack approach to retrieving data, I am taking a different approach towards persisting data as the library has it&s own global store for caching purposes.