feat(pitch-deck): Finanzplan-Export nach Excel mit Live-Formeln und Charts
Build pitch-deck / build-push-deploy (push) Failing after 24s
CI / go-lint (push) Has been skipped
CI / python-lint (push) Has been skipped
CI / nodejs-lint (push) Has been skipped
CI / test-go-consent (push) Successful in 5m28s
CI / test-python-voice (push) Successful in 4m0s
CI / test-bqas (push) Successful in 32s

Generiert pro Szenario (Wandeldarlehen 200k/Bear/Bull, 1 Mio Base/Bear/Bull)
ein .xlsx mit 10 Tabs (Dashboard, Kunden, Umsatzerlöse, Personalkosten,
Investitionen, Materialaufwand, Betriebliche Aufwendungen, Liquidität, GuV,
Formelübersicht). Editierbare Eingaben bleiben rohe Werte; abgeleitete Zellen
werden zu echten Excel-Formeln über Tabs hinweg, sodass das Bearbeiten von
Inputs Personal/Opex/Liquidität/GuV neu berechnet.

Dashboard-Tab fasst Jahres-KPIs zusammen und enthält fünf Charts
(Umsatz/Material/Personal/EBIT YoY, Jahresüberschuss YoY, Liquidität,
Headcount, Personalkosten monatlich).

Run: PG_CONN=... pitch-deck/scripts/export-finanzplan.sh

Co-Authored-By: Claude Opus 4.7 (1M context) <noreply@anthropic.com>
This commit is contained in:
Sharang Parnerkar
2026-05-16 00:08:27 +02:00
parent 9382d2a7a4
commit 77993d0ea0
12 changed files with 2496 additions and 0 deletions
+3
View File
@@ -62,3 +62,6 @@ consent-service/server
# Coverage # Coverage
coverage/ coverage/
*.coverage *.coverage
# Allow Finanzplan exports (generated by pitch-deck/scripts/export-finanzplan.sh)
!pitch-deck/exports/*.xlsx
Binary file not shown.
Binary file not shown.
Binary file not shown.
+1077
View File
File diff suppressed because it is too large Load Diff
+1
View File
@@ -32,6 +32,7 @@
"@types/react-dom": "^18.3.5", "@types/react-dom": "^18.3.5",
"@vitest/expect": "^4.1.2", "@vitest/expect": "^4.1.2",
"autoprefixer": "^10.4.20", "autoprefixer": "^10.4.20",
"exceljs": "^4.4.0",
"postcss": "^8.4.49", "postcss": "^8.4.49",
"tailwindcss": "^3.4.16", "tailwindcss": "^3.4.16",
"tsx": "^4.21.0", "tsx": "^4.21.0",
+145
View File
@@ -0,0 +1,145 @@
"""
Post-process Finanzplan Excel exports: attach charts to the Dashboard sheet
and move Dashboard to be the first tab.
Run after export-finanzplan-excel.ts:
python3 scripts/add-charts.py
"""
from __future__ import annotations
import sys
from pathlib import Path
from openpyxl import load_workbook
from openpyxl.chart import BarChart, LineChart, Reference
from openpyxl.chart.label import DataLabelList
from openpyxl.utils import get_column_letter
EXPORTS_DIR = Path(__file__).resolve().parent.parent / "exports"
def column_count_until_empty(ws, start_col: int, header_row: int) -> int:
"""Count consecutive non-empty cells starting at (header_row, start_col)."""
c = start_col
while ws.cell(row=header_row, column=c).value not in (None, ""):
c += 1
return c - start_col
def add_charts_to_workbook(path: Path) -> None:
wb = load_workbook(path)
if "Dashboard" not in wb.sheetnames:
print(f" skip {path.name}: no Dashboard tab")
return
ws = wb["Dashboard"]
# --- Chart 1: YoY Revenue / Material / Personnel / EBIT ---
# Source: rows 3..9 (3=year headers, rows 4=Umsatz, 5=Material, 6=Personal, 9=EBIT)
# categories: B3:F3 (years), data: rows 4,5,6,9
cat_ref = Reference(ws, min_col=2, max_col=6, min_row=3, max_row=3)
chart1 = BarChart()
chart1.type = "col"
chart1.style = 11
chart1.title = "Umsatz / Material / Personal / EBIT — YoY"
chart1.y_axis.title = "EUR"
chart1.x_axis.title = "Jahr"
for r in (4, 5, 6, 9):
data_ref = Reference(ws, min_col=1, max_col=6, min_row=r, max_row=r)
chart1.add_data(data_ref, titles_from_data=True, from_rows=True)
chart1.set_categories(cat_ref)
chart1.height = 9
chart1.width = 18
ws.add_chart(chart1, "H3")
# --- Chart 2: Jahresueberschuss YoY (row 11) ---
chart2 = BarChart()
chart2.type = "col"
chart2.style = 13
chart2.title = "Jahresüberschuss — YoY"
chart2.y_axis.title = "EUR"
chart2.x_axis.title = "Jahr"
data_ref = Reference(ws, min_col=1, max_col=6, min_row=11, max_row=11)
chart2.add_data(data_ref, titles_from_data=True, from_rows=True)
chart2.set_categories(cat_ref)
chart2.height = 9
chart2.width = 18
chart2.dataLabels = DataLabelList(showVal=True)
ws.add_chart(chart2, "H22")
# --- Chart 3: Liquidity monthly (row 16, months from col B) ---
# Determine the last month column by scanning row 15 (month labels)
n_months = column_count_until_empty(ws, 2, 15)
last_col = 1 + n_months # col 2..(1+n_months)
chart3 = LineChart()
chart3.title = "Liquidität (monatlich)"
chart3.y_axis.title = "EUR"
chart3.x_axis.title = "Monat"
chart3.style = 12
cat_months = Reference(ws, min_col=2, max_col=last_col, min_row=15, max_row=15)
data_liq = Reference(ws, min_col=1, max_col=last_col, min_row=16, max_row=16)
chart3.add_data(data_liq, titles_from_data=True, from_rows=True)
chart3.set_categories(cat_months)
chart3.height = 9
chart3.width = 24
ws.add_chart(chart3, "H41")
# --- Chart 4: Headcount (row 20) ---
chart4 = LineChart()
chart4.title = "Headcount (monatlich)"
chart4.y_axis.title = "Personen"
chart4.x_axis.title = "Monat"
chart4.style = 10
data_hc = Reference(ws, min_col=1, max_col=last_col, min_row=20, max_row=20)
chart4.add_data(data_hc, titles_from_data=True, from_rows=True)
chart4.set_categories(cat_months)
chart4.height = 9
chart4.width = 24
ws.add_chart(chart4, "H60")
# --- Chart 5: Personalkosten total monthly (row 24) ---
chart5 = LineChart()
chart5.title = "Personalkosten total (monatlich)"
chart5.y_axis.title = "EUR"
chart5.x_axis.title = "Monat"
chart5.style = 13
data_pers = Reference(ws, min_col=1, max_col=last_col, min_row=24, max_row=24)
chart5.add_data(data_pers, titles_from_data=True, from_rows=True)
chart5.set_categories(cat_months)
chart5.height = 9
chart5.width = 24
ws.add_chart(chart5, "H79")
# --- Move Dashboard to be the first sheet ---
idx = wb.sheetnames.index("Dashboard")
if idx != 0:
# openpyxl uses _sheets internal list; reorder by index.
wb._sheets.insert(0, wb._sheets.pop(idx))
# Also put the Formelübersicht (docs) tab at the end if present
for docs_name in ("Formelübersicht", "Formulas"):
if docs_name in wb.sheetnames:
fidx = wb.sheetnames.index(docs_name)
wb._sheets.append(wb._sheets.pop(fidx))
break
# Make Dashboard the active sheet on open
wb.active = 0
wb.save(path)
print(f" charts added to {path.name}")
def main() -> None:
if not EXPORTS_DIR.exists():
print(f"no exports dir at {EXPORTS_DIR}")
sys.exit(1)
files = sorted(EXPORTS_DIR.glob("Finanzplan-*.xlsx"))
if not files:
print("no Finanzplan-*.xlsx files found in exports/")
sys.exit(1)
for f in files:
print(f"Processing {f.name}")
add_charts_to_workbook(f)
if __name__ == "__main__":
main()
File diff suppressed because it is too large Load Diff
+16
View File
@@ -0,0 +1,16 @@
#!/usr/bin/env bash
# Generate Finanzplan Excel exports with formulas + charts.
# Step 1: TS script writes data + formulas via exceljs
# Step 2: Python script adds charts via openpyxl
#
# Requires PG_CONN env var pointing at the breakpilot_db postgres instance.
set -e
cd "$(dirname "$0")/.."
if [[ -z "${PG_CONN:-}" ]]; then
echo "PG_CONN env var is required (postgresql://user:pass@host:port/breakpilot_db)" >&2
exit 1
fi
npx tsx scripts/export-finanzplan-excel.ts
python3 scripts/add-charts.py
echo
echo "Done. Files in $(pwd)/exports/"