🚧 Attention: This project is still under construction!
A personal finance dashboard designed to simplify money management by delivering insightful visualizations of investments, tracking financial goals, and optimizing budgets.
I have always wanted a quick yet insightful way to glance at my personal finances while avoiding spending too much time managing them.
Additionally, I strive to work on projects that allow me to explore new technologies or approaches to building software (e.g. event sourcing, CQRS). To address both of these goals, Mankkoo was created.
With Mankkoo, I can effortlessly import transactions from all my bank accounts and investments into one centralized platform for analysis. This enables me to gain insights into their history and distribution. By having an overall view, I can effectively plan my future financial steps.
Below are screenshots of the mockups. Real screenshots will be added soon.
Initially, Mankkoo was developed as a single-runtime Python application, where all data was stored in files. However, due to limitations and a desire to adopt more popular tools, the application was split into two services: a frontend and a backend, each implemented using different technologies. For the persistence layer, a PostgreSQL database was chosen.
graph TD
Frontend[Frontend Service] -->|HTTP API Calls| Backend[Backend Service]
Backend -->|Read/Write| Database[(PostgreSQL Database)]
Frontend <-->|Responses| Backend
A key feature of the backend service is its ability to import transaction files downloaded from bank websites. Each transaction is recorded as an event in an append-only event log, which serves as the source of truth for all operations performed on bank accounts and investments.
Whenever a new batch of events is added to the event log, it triggers updates to multiple projections used to render charts and indicators. This functionality is powered by PostgreSQL’s built-in pg_notify()
and LISTEN
functions, enabling efficient real-time updates (docs).
Here is a schematic diagram of the data structure:
erDiagram
STREAMS {
UUID id PK
TEXT type
BIGINT version
JSONB metadata
}
EVENTS {
UUID id PK
UUID stream_id FK
TEXT type
JSONB data
BIGINT version
TIMESTAMP occured_at
TIMESTAMP added_at
}
VIEWS {
TEXT name PK
JSONB content
TIMESTAMP updated_at
}
STREAMS ||--o{ EVENTS : has
- Frontend: React 18, NextJS, TypeScript, ChartJS
- Backend: Python 3.10, Flask, Pandas, Pytest, Poetry
- Database: PostgreSQL 16
- Prepare data for demo
- Basic view for all streams and events
- Creating and modifying streams on UI
- Investments summary
See the open issues for a full list of proposed features (and known issues).
The best way to get started with this project is by using the defined DevContainers. These encapsulate all the necessary dependencies and include predefined tasks that are useful during development and usage.
- Docker
- VS Code (the IDE with the best support for DevContainers)
If you prefer not to use the DevContainer configuration, make sure you have the following dependencies installed:
- Pip
- Python
- Node.js
- npm
- Taskfile
For more details on each dependency (including required versions), check the .devcontainer directory or the respective service directory.
The DevContainer automatically installs all the necessary libraries for both the backend and frontend services. However, if you prefer to install them manually, follow these steps:
- Navigate to the
services/mankkoo
directory and run:
pip install -r requirements.txt --ignore-installed
- Then, go to the
services/mankkoo-ui
directory and run:
npm install
The easiest way to run database, backend and frontend service is to execute the following task from the Taskfile.yaml
.
To spinup a database and backend service:
task mankkoo:dev
In a separate terminal, start the UI by running:
task mankkoo:ui
If you prefer not to use the predefined commands in the Taskfile.yaml
, you can check their definitions in the relevant files.
To access the website, open: http://localhost:3000/
To access the pgAdmin website, open: http://localhost:5050/ (Username: [email protected], Password: admin)
To list all available tasks defined in the Taskfile.yaml
, run the following command:
task -l
To create a backup, run the following command:
task backup
This will generate a fresh database dump. To restore from the backup, check the name of the backup file, note it, and specify it in the Taskfile.yaml
file under the file
variable. Then, run the following command:
task restore