2024-05-29 12:50:52 +00:00
|
|
|
from decimal import Decimal
|
2024-06-30 17:15:45 +00:00
|
|
|
from datetime import datetime
|
2024-05-29 12:50:52 +00:00
|
|
|
|
|
|
|
|
2024-06-30 17:15:45 +00:00
|
|
|
def get_latest_month(data, allow_current=False):
|
2024-05-29 13:04:57 +00:00
|
|
|
years = sorted(data.keys())
|
|
|
|
latest_year = years[-1]
|
|
|
|
months = sorted(data[latest_year].keys())
|
|
|
|
latest_month = months[-1]
|
|
|
|
|
2024-06-30 17:15:45 +00:00
|
|
|
if (
|
|
|
|
not allow_current
|
|
|
|
and latest_year == str(datetime.now().year)
|
|
|
|
and latest_month == str(datetime.now().month)
|
|
|
|
):
|
|
|
|
try:
|
|
|
|
latest_month = months[-2]
|
|
|
|
except IndexError:
|
|
|
|
latest_year = years[-2]
|
|
|
|
latest_month = months[-1]
|
|
|
|
|
2024-05-29 13:04:57 +00:00
|
|
|
return int(latest_month), int(latest_year)
|
|
|
|
|
2024-06-03 14:12:21 +00:00
|
|
|
|
2024-07-01 04:57:16 +00:00
|
|
|
def get_transparency_data(data, year=None, month=None, allow_current=False):
|
2024-05-29 12:50:52 +00:00
|
|
|
if year is None:
|
|
|
|
year = max(data.keys())
|
|
|
|
|
|
|
|
if month is None:
|
|
|
|
month = max(data[year].keys())
|
|
|
|
|
2024-07-01 05:04:47 +00:00
|
|
|
year = str(year)
|
2024-10-04 05:33:25 +00:00
|
|
|
month = str(month).zfill(2)
|
2024-07-01 05:04:47 +00:00
|
|
|
|
2024-07-01 04:57:16 +00:00
|
|
|
if (
|
|
|
|
not allow_current
|
|
|
|
and year == str(datetime.now().year)
|
2024-10-04 05:33:25 +00:00
|
|
|
and month == str(datetime.now().month).zfill(2)
|
2024-07-01 04:57:16 +00:00
|
|
|
):
|
|
|
|
try:
|
|
|
|
month = max([m for m in data[year].keys() if m != str(datetime.now().month)])
|
|
|
|
except ValueError:
|
|
|
|
year = str(int(year) - 1)
|
|
|
|
month = max(data[year].keys())
|
|
|
|
|
2024-05-29 12:50:52 +00:00
|
|
|
assert year in data, f"Year {year} not found in data"
|
|
|
|
assert month in data[year], f"Month {month}-{year} not found in data"
|
|
|
|
|
|
|
|
# Initialize balances
|
|
|
|
balances = {}
|
|
|
|
incomes = {}
|
|
|
|
expenses = {}
|
2024-06-03 14:12:21 +00:00
|
|
|
notes = {}
|
2024-05-29 12:50:52 +00:00
|
|
|
|
|
|
|
start_balance = {}
|
|
|
|
end_balance = {}
|
|
|
|
|
|
|
|
for y in sorted(data.keys()):
|
|
|
|
if int(y) > int(year):
|
|
|
|
break
|
|
|
|
|
|
|
|
for m in sorted(data[y].keys()):
|
|
|
|
if int(y) == int(year) and int(m) > int(month):
|
|
|
|
break
|
|
|
|
|
|
|
|
# If the month is the one we are interested in, capture the start balance
|
|
|
|
if int(y) == int(year) and int(m) == int(month):
|
2024-06-03 14:12:21 +00:00
|
|
|
start_balance = {
|
|
|
|
k: Decimal(v) for k, v in balances.items() if k != "Notes"
|
|
|
|
}
|
2024-05-29 12:50:52 +00:00
|
|
|
|
|
|
|
for category in data[y][m]:
|
|
|
|
for currency, amount in data[y][m][category].items():
|
2024-06-03 14:12:21 +00:00
|
|
|
if currency == "Notes":
|
|
|
|
if int(y) == int(year) and int(m) == int(month):
|
|
|
|
notes[category] = amount
|
|
|
|
else:
|
|
|
|
if currency not in balances:
|
|
|
|
balances[currency] = Decimal(0)
|
|
|
|
balances[currency] += Decimal(str(amount))
|
|
|
|
|
|
|
|
# Track incomes and expenses
|
|
|
|
if int(y) == int(year) and int(m) == int(month):
|
|
|
|
if Decimal(str(amount)) > 0:
|
|
|
|
if category not in incomes:
|
|
|
|
incomes[category] = {}
|
|
|
|
if currency not in incomes[category]:
|
|
|
|
incomes[category][currency] = Decimal(0)
|
|
|
|
incomes[category][currency] += Decimal(str(amount))
|
|
|
|
else:
|
|
|
|
if category not in expenses:
|
|
|
|
expenses[category] = {}
|
|
|
|
if currency not in expenses[category]:
|
|
|
|
expenses[category][currency] = Decimal(0)
|
|
|
|
expenses[category][currency] += Decimal(str(amount))
|
2024-05-29 12:50:52 +00:00
|
|
|
|
|
|
|
# If the month is the one we are interested in, capture the end balance
|
|
|
|
if int(y) == int(year) and int(m) == int(month):
|
2024-06-03 14:12:21 +00:00
|
|
|
end_balance = {
|
|
|
|
k: Decimal(v) for k, v in balances.items() if k != "Notes"
|
|
|
|
}
|
2024-05-29 12:50:52 +00:00
|
|
|
|
|
|
|
# Calculate accumulated sums of incomes and expenses
|
|
|
|
accumulated_incomes = {
|
|
|
|
currency: sum(incomes[cat].get(currency, Decimal(0)) for cat in incomes)
|
|
|
|
for currency in balances
|
2024-06-03 14:12:21 +00:00
|
|
|
if currency != "Notes"
|
2024-05-29 12:50:52 +00:00
|
|
|
}
|
|
|
|
accumulated_expenses = {
|
|
|
|
currency: sum(expenses[cat].get(currency, Decimal(0)) for cat in expenses)
|
|
|
|
for currency in balances
|
2024-06-03 14:12:21 +00:00
|
|
|
if currency != "Notes"
|
2024-05-29 12:50:52 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
return {
|
|
|
|
"start_balance": start_balance,
|
|
|
|
"end_balance": end_balance,
|
|
|
|
"incomes": incomes,
|
|
|
|
"expenses": expenses,
|
|
|
|
"accumulated_incomes": accumulated_incomes,
|
|
|
|
"accumulated_expenses": accumulated_expenses,
|
2024-06-03 14:12:21 +00:00
|
|
|
"notes": notes,
|
2024-05-29 12:50:52 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
def generate_transparency_table(result, currencies=None):
|
|
|
|
def extract_currencies(data):
|
|
|
|
return ["EUR"] + (
|
|
|
|
list(
|
|
|
|
set(
|
|
|
|
list(data["start_balance"].keys())
|
|
|
|
+ list(data["end_balance"].keys())
|
|
|
|
+ list(data["accumulated_incomes"].keys())
|
|
|
|
+ list(data["accumulated_expenses"].keys())
|
|
|
|
)
|
2024-06-03 14:12:21 +00:00
|
|
|
- {"EUR", "Notes"}
|
2024-05-29 12:50:52 +00:00
|
|
|
)
|
|
|
|
)
|
|
|
|
|
|
|
|
def format_currency(value, currency):
|
|
|
|
if currency == "EUR":
|
|
|
|
return f"€{value:,.2f}"
|
|
|
|
elif currency in ["BTC", "ETH", "XMR"]:
|
|
|
|
return f"{value:,.9f} {currency}"
|
|
|
|
else:
|
|
|
|
return f"{value} {currency}"
|
|
|
|
|
|
|
|
def format_value(value, currency):
|
|
|
|
if value == 0:
|
|
|
|
return f"{format_currency(value, currency)}"
|
|
|
|
elif value > 0:
|
|
|
|
return f"+ {format_currency(value, currency)}"
|
|
|
|
else:
|
|
|
|
return f"- {format_currency(abs(value), currency)}"
|
|
|
|
|
|
|
|
html = """
|
|
|
|
<table class="table table-bordered table-transparency">
|
|
|
|
<thead class="table-light">
|
|
|
|
<tr>
|
|
|
|
<th scope="col">Category</th>
|
|
|
|
"""
|
|
|
|
|
|
|
|
if currencies is None:
|
|
|
|
currencies = extract_currencies(result)
|
|
|
|
|
|
|
|
# Add currency headers
|
|
|
|
for currency in currencies:
|
|
|
|
if currency == "EUR":
|
|
|
|
html += '<th class="currency-col" scope="col">Euros (€)</th>'
|
|
|
|
elif currency == "BTC":
|
|
|
|
html += '<th class="currency-col" scope="col">Bitcoin (BTC)</th>'
|
|
|
|
elif currency == "ETH":
|
|
|
|
html += '<th class="currency-col" scope="col">Ethereum (ETH)</th>'
|
|
|
|
elif currency == "XMR":
|
|
|
|
html += '<th class="currency-col" scope="col">Monero (XMR)</th>'
|
|
|
|
else:
|
|
|
|
html += f'<th class="currency-col" scope="col">{currency}</th>'
|
|
|
|
|
|
|
|
html += """
|
|
|
|
</tr>
|
|
|
|
</thead>
|
|
|
|
<tbody>
|
|
|
|
"""
|
|
|
|
|
|
|
|
# Add start balance row
|
2024-12-20 12:58:36 +00:00
|
|
|
html += "<tr class=\"transparency-start-balance-row\"><td>Account Balance (start of month)</td>"
|
2024-05-29 12:50:52 +00:00
|
|
|
for currency in currencies:
|
|
|
|
value = result["start_balance"].get(currency, Decimal(0))
|
|
|
|
html += f"<td>{format_value(value, currency)}</td>"
|
|
|
|
html += "</tr>"
|
|
|
|
|
|
|
|
# Add income rows
|
|
|
|
for category, transactions in result["incomes"].items():
|
2024-06-03 14:12:21 +00:00
|
|
|
has_notes = result["notes"].get(category)
|
|
|
|
html += f"<tr><td>{category}{'*' if has_notes else ''}</td>"
|
2024-05-29 12:50:52 +00:00
|
|
|
for currency in currencies:
|
|
|
|
value = transactions.get(currency, "")
|
|
|
|
if value != "":
|
|
|
|
html += f"<td>{format_value(value, currency)}</td>"
|
|
|
|
else:
|
|
|
|
html += "<td></td>"
|
|
|
|
html += "</tr>"
|
|
|
|
|
|
|
|
# Add expense rows
|
|
|
|
for category, transactions in result["expenses"].items():
|
2024-06-03 14:12:21 +00:00
|
|
|
has_notes = result["notes"].get(category)
|
|
|
|
html += f"<tr><td>{category}{'*' if has_notes else ''}</td>"
|
2024-05-29 12:50:52 +00:00
|
|
|
for currency in currencies:
|
|
|
|
value = transactions.get(currency, "")
|
|
|
|
if value != "":
|
|
|
|
html += f"<td>{format_value(value, currency)}</td>"
|
|
|
|
else:
|
|
|
|
html += "<td></td>"
|
|
|
|
html += "</tr>"
|
|
|
|
|
|
|
|
# Add total income row
|
|
|
|
html += '<tr class="table-secondary"><td><b>Total Income</b></td>'
|
|
|
|
for currency in currencies:
|
|
|
|
value = result["accumulated_incomes"].get(currency, Decimal(0))
|
|
|
|
html += f"<td><b>{format_value(value, currency)}</b></td>"
|
|
|
|
html += "</tr>"
|
|
|
|
|
|
|
|
# Add total expenses row
|
|
|
|
html += '<tr class="table-secondary"><td><b>Total Expenses</b></td>'
|
|
|
|
for currency in currencies:
|
|
|
|
value = result["accumulated_expenses"].get(currency, Decimal(0))
|
|
|
|
html += f"<td><b>{format_value(value, currency)}</b></td>"
|
|
|
|
html += "</tr>"
|
|
|
|
|
|
|
|
# Add end balance row
|
|
|
|
html += '<tr class="table-secondary"><td><b>Account Balance (end of month)</b></td>'
|
|
|
|
for currency in currencies:
|
|
|
|
value = result["end_balance"].get(currency, Decimal(0))
|
|
|
|
html += f"<td><b>{format_value(value, currency)}</b></td>"
|
|
|
|
html += "</tr>"
|
|
|
|
|
|
|
|
html += """
|
|
|
|
</tbody>
|
|
|
|
</table>
|
|
|
|
"""
|
|
|
|
|
2024-06-03 14:12:21 +00:00
|
|
|
if result["notes"]:
|
|
|
|
html += "<p><b>Notes:</b></p>"
|
|
|
|
html += "<ul>"
|
|
|
|
for category, footnote in result["notes"].items():
|
|
|
|
html += f"<li>{category}: {footnote}</li>"
|
|
|
|
html += "</ul>"
|
|
|
|
|
2024-05-29 12:50:52 +00:00
|
|
|
return html
|