## ๐ค AI Agent Integration
This fork includes an AI Portfolio Assistant powered by Claude + LangGraph.
See [AGENT_README.md](./AGENT_README.md) for setup, demo, and architecture.
---
[

](https://ghostfol.io)
# Ghostfolio
**Open Source Wealth Management Software**
[**Ghostfol.io**](https://ghostfol.io) | [**Live Demo**](https://ghostfol.io/en/demo) | [**Ghostfolio Premium**](https://ghostfol.io/en/pricing) | [**FAQ**](https://ghostfol.io/en/faq) |
[**Blog**](https://ghostfol.io/en/blog) | [**LinkedIn**](https://www.linkedin.com/company/ghostfolio) | [**Slack**](https://join.slack.com/t/ghostfolio/shared_invite/zt-vsaan64h-F_I0fEo5M0P88lP9ibCxFg) | [**X**](https://x.com/ghostfolio_)
[](https://www.buymeacoffee.com/ghostfolio)
[](#contributing) [](https://hub.docker.com/r/ghostfolio/ghostfolio)
[](https://www.gnu.org/licenses/agpl-3.0)
**Ghostfolio** is an open source wealth management software built with web technology. The application empowers busy people to keep track of stocks, ETFs or cryptocurrencies and make solid, data-driven investment decisions. The software is designed for personal use in continuous operation.
[

](https://www.youtube.com/watch?v=yY6ObSQVJZk)
## Ghostfolio Premium
Our official **[Ghostfolio Premium](https://ghostfol.io/en/pricing)** cloud offering is the easiest way to get started. Due to the time it saves, this will be the best option for most people. Revenue is used to cover operational costs for the hosting infrastructure and professional data providers, and to fund ongoing development.
If you prefer to run Ghostfolio on your own infrastructure, please find further instructions in the [Self-hosting](#self-hosting) section.
## Why Ghostfolio?
Ghostfolio is for you if you are...
- ๐ผ trading stocks, ETFs or cryptocurrencies on multiple platforms
- ๐ฆ pursuing a buy & hold strategy
- ๐ฏ interested in getting insights of your portfolio composition
- ๐ป valuing privacy and data ownership
- ๐ง into minimalism
- ๐งบ caring about diversifying your financial resources
- ๐ interested in financial independence
- ๐
saying no to spreadsheets
- ๐ still reading this list
## Features
- โ
Create, update and delete transactions
- โ
Multi account management
- โ
Portfolio performance: Return on Average Investment (ROAI) for `Today`, `WTD`, `MTD`, `YTD`, `1Y`, `5Y`, `Max`
- โ
Various charts
- โ
Static analysis to identify potential risks in your portfolio
- โ
Import and export transactions
- โ
Dark Mode
- โ
Zen Mode
- โ
Progressive Web App (PWA) with a mobile-first design
## Technology Stack
Ghostfolio is a modern web application written in [TypeScript](https://www.typescriptlang.org) and organized as an [Nx](https://nx.dev) workspace.
### Backend
The backend is based on [NestJS](https://nestjs.com) using [PostgreSQL](https://www.postgresql.org) as a database together with [Prisma](https://www.prisma.io) and [Redis](https://redis.io) for caching.
### Frontend
The frontend is built with [Angular](https://angular.dev) and uses [Angular Material](https://material.angular.io) with utility classes from [Bootstrap](https://getbootstrap.com).
## Self-hosting
We provide official container images hosted on [Docker Hub](https://hub.docker.com/r/ghostfolio/ghostfolio) for `linux/amd64`, `linux/arm/v7` and `linux/arm64`.
[

](https://www.buymeacoffee.com/ghostfolio)
### Supported Environment Variables
| Name | Type | Default Value | Description |
| --------------------------- | --------------------- | --------------------- | ----------------------------------------------------------------------------------------------------------------------------------- |
| `ACCESS_TOKEN_SALT` | `string` | | A random string used as salt for access tokens |
| `API_KEY_COINGECKO_DEMO` | `string` (optional) | ย | The _CoinGecko_ Demo API key |
| `API_KEY_COINGECKO_PRO` | `string` (optional) | | The _CoinGecko_ Pro API key |
| `DATABASE_URL` | `string` | | The database connection URL, e.g. `postgresql://${POSTGRES_USER}:${POSTGRES_PASSWORD}@localhost:5432/${POSTGRES_DB}?sslmode=prefer` |
| `ENABLE_FEATURE_AUTH_TOKEN` | `boolean` (optional) | `true` | Enables authentication via security token |
| `HOST` | `string` (optional) | `0.0.0.0` | The host where the Ghostfolio application will run on |
| `JWT_SECRET_KEY` | `string` | | A random string used for _JSON Web Tokens_ (JWT) |
| `LOG_LEVELS` | `string[]` (optional) | | The logging levels for the Ghostfolio application, e.g. `["debug","error","log","warn"]` |
| `PORT` | `number` (optional) | `3333` | The port where the Ghostfolio application will run on |
| `POSTGRES_DB` | `string` | | The name of the _PostgreSQL_ database |
| `POSTGRES_PASSWORD` | `string` | | The password of the _PostgreSQL_ database |
| `POSTGRES_USER` | `string` | | The user of the _PostgreSQL_ database |
| `REDIS_DB` | `number` (optional) | `0` | The database index of _Redis_ |
| `REDIS_HOST` | `string` | | The host where _Redis_ is running |
| `REDIS_PASSWORD` | `string` | | The password of _Redis_ |
| `REDIS_PORT` | `number` | | The port where _Redis_ is running |
| `REQUEST_TIMEOUT` | `number` (optional) | `2000` | The timeout of network requests to data providers in milliseconds |
| `ROOT_URL` | `string` (optional) | `http://0.0.0.0:3333` | The root URL of the Ghostfolio application, used for generating callback URLs and external links. |
#### OpenID Connect OIDC (Experimental)
| Name | Type | Default Value | Description |
| -------------------------- | --------------------- | ------------------------------------ | ---------------------------------------------------------------------------------------------------- |
| `ENABLE_FEATURE_AUTH_OIDC` | `boolean` (optional) | `false` | Enables authentication via _OpenID Connect_ |
| `OIDC_AUTHORIZATION_URL` | `string` (optional) | | Manual override for the OIDC authorization endpoint (falls back to the discovery from the issuer) |
| `OIDC_CALLBACK_URL` | `string` (optional) | `${ROOT_URL}/api/auth/oidc/callback` | The OIDC callback URL |
| `OIDC_CLIENT_ID` | `string` | | The OIDC client ID |
| `OIDC_CLIENT_SECRET` | `string` | | The OIDC client secret |
| `OIDC_ISSUER` | `string` | | The OIDC issuer URL, used to discover the OIDC configuration via `/.well-known/openid-configuration` |
| `OIDC_SCOPE` | `string[]` (optional) | `["openid"]` | The OIDC scope to request, e.g. `["email","openid","profile"]` |
| `OIDC_TOKEN_URL` | `string` (optional) | | Manual override for the OIDC token endpoint (falls back to the discovery from the issuer) |
| `OIDC_USER_INFO_URL` | `string` (optional) | | Manual override for the OIDC user info endpoint (falls back to the discovery from the issuer) |
### Run with Docker Compose
#### Prerequisites
- Basic knowledge of Docker
- Installation of [Docker](https://www.docker.com/products/docker-desktop)
- Create a local copy of this Git repository (clone)
- Copy the file `.env.example` to `.env` and populate it with your data (`cp .env.example .env`)
#### a. Run environment
Run the following command to start the Docker images from [Docker Hub](https://hub.docker.com/r/ghostfolio/ghostfolio):
```bash
docker compose -f docker/docker-compose.yml up -d
```
#### b. Build and run environment
Run the following commands to build and start the Docker images:
```bash
docker compose -f docker/docker-compose.build.yml build
docker compose -f docker/docker-compose.build.yml up -d
```
#### Setup
1. Open http://localhost:3333 in your browser
1. Create a new user via _Get Started_ (this first user will get the role `ADMIN`)
#### Upgrade Version
1. Update the _Ghostfolio_ Docker image
- Increase the version of the `ghostfolio/ghostfolio` Docker image in `docker/docker-compose.yml`
- Run the following command if `ghostfolio:latest` is set:
```bash
docker compose -f docker/docker-compose.yml pull
```
1. Run the following command to start the new Docker image:
```bash
docker compose -f docker/docker-compose.yml up -d
```
The container will automatically apply any required database schema migrations during startup.
### Home Server Systems (Community)
Ghostfolio is available for various home server systems, including [CasaOS](https://github.com/bigbeartechworld/big-bear-casaos), [Home Assistant](https://github.com/lildude/ha-addon-ghostfolio), [Runtipi](https://www.runtipi.io/docs/apps-available), [TrueCharts](https://truecharts.org/charts/stable/ghostfolio), [Umbrel](https://apps.umbrel.com/app/ghostfolio), and [Unraid](https://unraid.net/community/apps?q=ghostfolio).
## Development
For detailed information on the environment setup and development process, please refer to [DEVELOPMENT.md](./DEVELOPMENT.md).
## Public API
### Authorization: Bearer Token
Set the header for each request as follows:
```
"Authorization": "Bearer eyJh..."
```
You can get the _Bearer Token_ via `POST http://localhost:3333/api/v1/auth/anonymous` (Body: `{ "accessToken": "" }`)
Deprecated: `GET http://localhost:3333/api/v1/auth/anonymous/` or `curl -s http://localhost:3333/api/v1/auth/anonymous/`.
### Health Check (experimental)
#### Request
`GET http://localhost:3333/api/v1/health`
**Info:** No Bearer Token is required for health check
#### Response
##### Success
`200 OK`
```
{
"status": "OK"
}
```
### Import Activities
#### Prerequisites
[Bearer Token](#authorization-bearer-token) for authorization
#### Request
`POST http://localhost:3333/api/v1/import`
#### Body
```
{
"activities": [
{
"currency": "USD",
"dataSource": "YAHOO",
"date": "2021-09-15T00:00:00.000Z",
"fee": 19,
"quantity": 5,
"symbol": "MSFT",
"type": "BUY",
"unitPrice": 298.58
}
]
}
```
| Field | Type | Description |
| ------------ | ------------------- | ------------------------------------------------------------------- |
| `accountId` | `string` (optional) | Id of the account |
| `comment` | `string` (optional) | Comment of the activity |
| `currency` | `string` | `CHF` \| `EUR` \| `USD` etc. |
| `dataSource` | `string` | `COINGECKO` \| `GHOSTFOLIO` [^1] \| `MANUAL` \| `YAHOO` |
| `date` | `string` | Date in the format `ISO-8601` |
| `fee` | `number` | Fee of the activity |
| `quantity` | `number` | Quantity of the activity |
| `symbol` | `string` | Symbol of the activity (suitable for `dataSource`) |
| `type` | `string` | `BUY` \| `DIVIDEND` \| `FEE` \| `INTEREST` \| `LIABILITY` \| `SELL` |
| `unitPrice` | `number` | Price per unit of the activity |
#### Response
##### Success
`201 Created`
##### Error
`400 Bad Request`
```
{
"error": "Bad Request",
"message": [
"activities.1 is a duplicate activity"
]
}
```
### Portfolio (experimental)
#### Prerequisites
Grant access of type _Public_ in the _Access_ tab of _My Ghostfolio_.
#### Request
`GET http://localhost:3333/api/v1/public//portfolio`
**Info:** No Bearer Token is required for authorization
#### Response
##### Success
```
{
"performance": {
"1d": {
"relativeChange": 0 // normalized from -1 to 1
};
"ytd": {
"relativeChange": 0 // normalized from -1 to 1
},
"max": {
"relativeChange": 0 // normalized from -1 to 1
}
}
}
```
## ๐ Real Estate / Housing Integration (AgentForge)
This project extends the Ghostfolio AI Agent with a **Housing & Real Estate** capability โ an example of brownfield AgentForge integration where a new tool is added to an existing LangGraph agent without touching core portfolio functionality.
### Feature Flag
The feature is **off by default** and must be explicitly enabled:
```bash
# agent/.env
ENABLE_REAL_ESTATE=true # activate
ENABLE_REAL_ESTATE=false # deactivate (default) โ zero behavior change
```
When `false`, all real estate keywords are ignored and the agent behaves exactly as it did before this feature was added.
### What it adds
Three new agent capabilities accessible through the existing AI chat interface:
| Capability | Example prompt |
| ----------------------- | --------------------------------------------------------------- |
| Neighborhood snapshot | `"What's the housing market like in Austin?"` |
| Listing search | `"Show me listings in Seattle"` |
| Neighborhood comparison | `"Compare Austin vs Denver for affordability and rental yield"` |
| Cashflow analysis | `"Estimate rental cashflow for Chicago real estate"` |
### 2-Minute Demo (Friday)
**Step 1 โ start the agent** (if not already running):
```bash
cd ghostfolio/agent
source venv/bin/activate
uvicorn main:app --reload --port 8000
```
**Step 2 โ confirm the flag is on**:
Open `agent/.env` and verify `ENABLE_REAL_ESTATE=true`.
**Step 3 โ open the AI chat** in the Ghostfolio app (the floating chat FAB button).
**Step 4 โ run these demo prompts in order:**
```
1. "What's the housing market like in Austin?"
โ Shows: median price, price/sqft, DOM, YoY change, inventory, rental yield
2. "Show me listings in Seattle"
โ Shows: 2 active listings with price, beds/baths, sqft, cap rate
3. "Compare Austin vs Chicago for real estate investment"
โ Shows: side-by-side metrics + which market wins on affordability, yield, walkability
4. "Estimate rental cashflow for a Chicago investment property"
โ Agent synthesizes cap rate, rent estimate, and mortgage estimate
```
**Expected output**: Conversational responses with specific data points, no raw JSON, citations included.
### Architecture (how it integrates)
```
graph.py classify_node โโโบ detects real estate keywords
โโโบ routes to query_type = real_estate_snapshot | real_estate_search | real_estate_compare
graph.py tools_node โโโบ calls tools/real_estate.py functions
(get_neighborhood_snapshot / search_listings / compare_neighborhoods)
tools/real_estate.py โโโบ MockProvider with realistic 2024 data for 10 US cities
TTL cache (5 min) to avoid repeat computation
Feature flag guard on every function
tools/__init__.py โโโบ 3 new entries in TOOL_REGISTRY (append-only)
```
**Critical paths untouched**: `state.py`, `main.py` endpoints, existing tool files, Angular routes, Ghostfolio NestJS API โ none of these files were modified.
### Running the tests
```bash
cd ghostfolio/agent
source venv/bin/activate
python -m pytest evals/test_real_estate.py -v
# โ 5 passed in < 1s
```
### Rollback
1. **Instant off**: Set `ENABLE_REAL_ESTATE=false` in `.env` โ restart agent. Zero UI or behavior change.
2. **Full revert**: `git checkout main` โ removes all real estate code.
3. **Partial revert**: `git revert` the specific commits tagged `feat(real-estate)`.
### Supported Cities (Mock Provider)
Austin TX ยท San Francisco CA ยท New York NY ยท Denver CO ยท Seattle WA ยท
Miami FL ยท Chicago IL ยท Phoenix AZ ยท Nashville TN ยท Dallas TX
### Next Steps (real provider)
To swap in live data, replace `MockProvider` calls in `tools/real_estate.py` with an HTTP call to:
- **Attom Data API** (comprehensive, paid) โ `REAL_ESTATE_API_KEY` env var
- **RealtyMole / Realty in US** (RapidAPI) โ free tier available
- **HUD Fair Market Rents API** (free, government data, rent estimates)
The normalized return schema is identical โ no changes to `graph.py` needed.
---
## Community Projects
Discover a variety of community projects for Ghostfolio: https://github.com/topics/ghostfolio
Are you building your own project? Add the `ghostfolio` topic to your _GitHub_ repository to get listed as well. [Learn more โ](https://docs.github.com/en/articles/classifying-your-repository-with-topics)
## Contributing
Ghostfolio is **100% free** and **open source**. We encourage and support an active and healthy community that accepts contributions from the public - including you.
Not sure what to work on? We have [some ideas](https://github.com/ghostfolio/ghostfolio/issues?q=is%3Aissue+is%3Aopen+label%3A%22help+wanted%22%20no%3Aassignee), even for [newcomers](https://github.com/ghostfolio/ghostfolio/issues?q=is%3Aissue+is%3Aopen+label%3A%22good+first+issue%22%20no%3Aassignee). Please join the Ghostfolio [Slack](https://join.slack.com/t/ghostfolio/shared_invite/zt-vsaan64h-F_I0fEo5M0P88lP9ibCxFg) channel or post to [@ghostfolio\_](https://x.com/ghostfolio_) on _X_. We would love to hear from you.
If you like to support this project, become a [**Sponsor**](https://github.com/sponsors/ghostfolio), get [**Ghostfolio Premium**](https://ghostfol.io/en/pricing) or [**Buy me a coffee**](https://www.buymeacoffee.com/ghostfolio).
## Sponsors
## Analytics

## License
ยฉ 2021 - 2026 [Ghostfolio](https://ghostfol.io)
Licensed under the [AGPLv3 License](https://www.gnu.org/licenses/agpl-3.0.html).
[^1]: Available with [**Ghostfolio Premium**](https://ghostfol.io/en/pricing).