Hi 👋 Thanks for checking out my repo.
EasyLegal.app is case management software that helps attorneys keep track of clients' cases, court dates and financial accounts. In the course of representing criminal defense clients, attorneys must keep track of criminal charges, court dates, meetings with clients and others, tasks to be completed, and maintain notes for billing purposes. EasyLegal.app lets them do all this in one place. Feel free to visit and create a temporary evaluation account at demo.EasyLegal.app.
On the frontend, EasyLegal.app is a javascript application built using React/Redux, Redux Toolkit, RTK Query, GraphQL and Tailwind CSS. The frontend code is located in the /src
directory of the project and the GraphQL schema is located here.
React is an open-source component library, originally created by FaceBook, for creating user interfaces (UI). React makes it easy for developers to breakdown views into their constituent pieces and then reassemble them into the final view. This makes it much easier to develop, test, style and reuse frontend components. See /src/components
for the frontend components reused throughout this project.
Redux is an open-source library used to manage frontend application state and is designed to be predictable and easy to test and debug. Redux Toolkit is an associated library that significantly reduces the amount of boilerplate code required by Redux-managed applications. EasyLegal.app uses Redux (with Redux Toolkit) to keep track of client IDs which prevents the need for excessive prop drilling on case tab views (discussed in the Resource CRUD
section below). RTK Query is used for fetching data from the GraphQL API and then caching it. Redux is configured in /src/app/store.js
and RTK Query is used in all the /src/features/[resources]/[resources]Slice.js
files.
GraphQL is a query language that you can include in your frontend code to request exactly what information the view needs from the API. EasyLegal.app uses an AWS-managed GraphQL API to fulfill these requests. This too is further discussed in the Resource CRUD
section below.
Tailwind CSS is a library of CSS utility classes that makes it easy to style UI components without adding custom CSS files.
On the backend, EasyLegal.app is a serverless application, built on a number of Amazon Web Services (AWS) mananged services. The middleware (APIs, auth, lambdas) for interacting with the AWS managed services is located in the /amplify
directory. Amplify is a framework offered by AWS that makes it relatively easy to setup different services, get them configured and working with one another. I used Amplify to setup and configure about half of the backend services used in this project (authentication, lambda, the GraphQL API) while the other half (DNS, message queues, scheduling and sending of emails) I had to setup and configure myself.
EasyLegal.app uses AWS Cognito to authenticate and authorize users. It integrates automatically with our backend datastore, DynamoDB, so it was a natural choice.
The app schema is located in schema.graphql. This section provides a brief, high-level explanation of what's going on from a workflow perspective.
When a user signs up, the user submitted data (name and email address) are sent to the SignupCommand lambda where an AWS Cognito user is created and then the data are sent onward to 2 queues: the CreateAppUser queue and the AdminDisableUserCommand queue (both maintained in AWS console). The user only actually needs to wait on the creation of a cognito user. By offloading the provisioning responsiblities of creating an app user and disabling the cognito user to queues running in the background, we're able to reduce user wait time from about 3.5 seconds down to about 1.5 seconds on initial signup.
A Cognito user is just a mapping from email address to UUID (GUID). It exists solely for interacting with the datastore (DynamoDB). Cognito users are enabled by default, before their emails are verified, so the AdminDisableUserCommand lambda (which is triggered by the AdminDisableUserCommand queue) immediately disables them. The CreateAppUser lambda (triggered by the CreateAppUser queue) does 3 things:
- It creates a mapping between the Cognito user ID and an EasyLegal.app ID. This allows for future decoupling from the Cognito auth service. (See
What I would do next
below for more information.) - It creates a record of consent to the site's terms and conditions
- It creates a temporary record of user data to be used later when the user confirms her email address
Cognito uses a standard email confirmation workflow. When a user successfully enters a confirmation code that has been sent to the email address they've provided, the ConfirmationFlow lambda re-enables their account and marks it as email verified. The user's info is then again forwarded to another queue (CreateLawyer) to be processed in the background as the user logs in for the first time, again saving the user a couple seconds of wait time.
The CreateLawyer lambda (triggered by the CreateLawyer queue) creates a laywer, a practice and adds the lawyer to the practice. This is also discussed in the What I would do next
section below.
There are some Jest unit tests here and some Playwright integration tests here. Playwright is cool because it can help surface accessibility issues. I actually redesigned my list views based on the results of preliminary Playwright results.
The GraphQL queries need to be trimmed up. Right now I'm just using the default autogenerated queries which ask for all attributes a few layers deep. This would be WAY TOO MUCH data to send over the wire in a production app. The queries need to be modified to only ask for the data that's needed.
I would add terraform to replace Amplify and manage all the infrastructure. Amplify is nice just for getting started quickly, but it leaves something to be desired in terms of best practices, e.g., different environments' infrastructure should reside in different AWS accounts. Additionally, Amplify doesn't manage queues - I had to setup all queues by hand. Theses should be managed by Terraform instead. Implementing Terraform would have the additional benefit of making it easier to decouple from AWS.
I would move duplicate lambda code to a lambda layer. Right now createRequest
and graphqlOperation
are defined in 7 different files. These (along with SES) should be moved to AWS Lambda Layers.
The first iteration of the app presented here is designed for the desktop. The next thing to do would be to make the app mobile friendly. This can easily be done with more Tailwind classes.