routes.py 13 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401
  1. import io
  2. from dataclasses import dataclass
  3. from datetime import datetime
  4. from pathlib import Path
  5. from urllib.parse import unquote, urlencode
  6. import pandas as pd
  7. from docx import Document
  8. from docxtpl import DocxTemplate
  9. from fastapi import APIRouter, Depends, Form, HTTPException, Request, Response
  10. from fastapi.responses import (
  11. FileResponse,
  12. HTMLResponse,
  13. RedirectResponse,
  14. StreamingResponse,
  15. )
  16. from fastapi.templating import Jinja2Templates
  17. from sqlalchemy import text
  18. from sqlalchemy.orm import Session
  19. from .auth import ldap_authenticate
  20. from .db import get_session
  21. from .models import Bemerkung, Forderung
  22. from .schemas import BemerkungIn
  23. router = APIRouter()
  24. templates = Jinja2Templates(directory="templates")
  25. @router.get("/", response_class=HTMLResponse)
  26. def index(request: Request):
  27. return RedirectResponse(url="/login")
  28. @router.get("/login", response_class=HTMLResponse)
  29. def login_get(request: Request):
  30. return templates.TemplateResponse(request, "base/login.html", {"request": request})
  31. @router.get("/select", response_class=HTMLResponse)
  32. def select_get(request: Request):
  33. return templates.TemplateResponse(request, "base/select.html", {"request": request})
  34. @router.post("/login")
  35. def login_post(username: str = Form(...), password: str = Form(...)):
  36. user = ldap_authenticate(username, password)
  37. if not user:
  38. raise HTTPException(status_code=401, detail="Invalid credentials")
  39. # For a simple demo we set a cookie (not secure) -- replace with real session in production
  40. response = RedirectResponse(url="/forderungen", status_code=302)
  41. response.set_cookie("user", user["username"])
  42. return response
  43. @router.get("/forderungen", response_class=HTMLResponse)
  44. def forderungsliste(request: Request, db: Session = Depends(get_session), limit: int = 100):
  45. q = db.query(Forderung).order_by(Forderung.faelligkeit.asc()).limit(limit).all()
  46. return templates.TemplateResponse(request, "base/list.html", {"request": request, "forderungen": q})
  47. def number_format(input: float) -> str:
  48. return format(input, "0,.2f").replace(".", ":").replace(",", ".").replace(":", ",")
  49. def date_format(input: datetime) -> str:
  50. if input is None:
  51. return ""
  52. return input.strftime("%d.%m.%Y")
  53. templates.env.filters["number_format"] = number_format
  54. templates.env.filters["date_format"] = date_format
  55. def single_quote(text: str):
  56. return "'" + unquote(text) + "'"
  57. @dataclass
  58. class FilterConfig:
  59. name: str
  60. text: str
  61. filter_type: str
  62. width: str
  63. key_column: str
  64. value_column: str
  65. visible: bool = True
  66. default_value: str = ""
  67. options: list[dict[str, str]] | None = None
  68. current_value: str | None = None
  69. forderung_filter_config = [
  70. FilterConfig("Hauptbetrieb", "Hauptbetrieb", "select", "2", "Client_DB", "Hauptbetrieb_Name"),
  71. FilterConfig("Standort", "Standort", "select", "3", "Standort_ID", "Standort_Name"),
  72. FilterConfig("Bereich", "Bereich", "select", "3", "Bereich", "Bereich"),
  73. FilterConfig("Verursacher", "Verursacher", "select", "4", "Verursacher", "Verursacher"),
  74. FilterConfig("Rechnungsnummer", "Rechnungsnummer", "text", "3", "Document_No", "Document_No"),
  75. FilterConfig(
  76. "RechnungsdatumVon",
  77. "Rechnungsdatum von",
  78. "date",
  79. "2",
  80. "Invoice_Date",
  81. "Invoice_Date",
  82. default_value="2000-01-01T00:00:00",
  83. ),
  84. FilterConfig(
  85. "RechnungsdatumBis",
  86. "Rechnungsdatum bis",
  87. "date",
  88. "2",
  89. "Invoice_Date",
  90. "Invoice_Date",
  91. default_value="2027-01-01T00:00:00",
  92. ),
  93. FilterConfig("Kunde", "Kunde", "text", "5", "Kunde", "Kunde"),
  94. FilterConfig(
  95. "WiedervorlageVon",
  96. "Wiedervorlage von",
  97. "date",
  98. "2",
  99. "Wiedervorlage",
  100. "Wiedervorlage",
  101. default_value="2000-01-01T00:00:00",
  102. ),
  103. FilterConfig(
  104. "WiedervorlageBis",
  105. "Wiedervorlage bis",
  106. "date",
  107. "2",
  108. "Wiedervorlage",
  109. "Wiedervorlage",
  110. default_value="2027-01-01T00:00:00",
  111. ),
  112. FilterConfig("Fahrzeug", "Fahrzeug", "select", "4", "VIN", "VIN"),
  113. FilterConfig("Staffel", "Staffel", "select", "2", "Staffel", "Staffel"),
  114. FilterConfig("Mahnstufe", "Mahnstufe", "select", "2", "Mahnstufe", "Mahnstufe"),
  115. FilterConfig("BenutzerSelect", "Benutzer", "hidden", "2", "", "", visible=False, default_value="winter"),
  116. ]
  117. filter_type_option = {
  118. "text": "text",
  119. "select": "selectedOptionValue",
  120. "date": "selectedDate",
  121. "hidden": "selectedOptionValue",
  122. }
  123. @router.get("/app/forderungen/liste", response_class=HTMLResponse)
  124. def forderungen_liste(request: Request, db: Session = Depends(get_session), limit: int = 100, page: int = 0):
  125. params = {
  126. f.name: request.query_params[f.name]
  127. for f in forderung_filter_config
  128. if request.query_params.get(f.name, "") != ""
  129. }
  130. if page == 0 and len(params) < len(request.query_params):
  131. return RedirectResponse(url="/app/forderungen/liste?" + urlencode(params))
  132. context = {}
  133. for f in forderung_filter_config:
  134. context[f.name] = {filter_type_option[f.filter_type]: single_quote(params.get(f.name, f.default_value))}
  135. query = templates.TemplateResponse(request, "forderungen/queries/forderungen_liste.sql", context).body.decode(
  136. "utf-8"
  137. )
  138. # print(query)
  139. # q = db.execute(text("SELECT * FROM [dbo].[Forderungen]"))
  140. q = db.execute(text(query)).fetchall()
  141. col_names = list(q[0]._asdict().keys())
  142. filters = {}
  143. for f in forderung_filter_config:
  144. if f.key_column == "":
  145. continue
  146. filters[f.name] = {row[col_names.index(f.key_column)]: row[col_names.index(f.value_column)] for row in q}
  147. f.options = filters[f.name]
  148. summary = {
  149. "offen": sum([r[col_names.index("offen")] for r in q]),
  150. "Anzahl": len(q),
  151. }
  152. q_limit = q[page * limit : (page + 1) * limit]
  153. defaults = {}
  154. for f in forderung_filter_config:
  155. defaults[f.name] = unquote(request.query_params.get(f.name, f.default_value))
  156. f.current_value = defaults[f.name]
  157. if len(q_limit) == 0:
  158. return None
  159. template = "forderungen/liste.html"
  160. if page > 0:
  161. template = "forderungen/liste_tabelle.html"
  162. return templates.TemplateResponse(
  163. request,
  164. template,
  165. {
  166. "request": request,
  167. "forderungen_liste": q_limit,
  168. "filter": filters,
  169. "defaults": defaults,
  170. "summary": summary,
  171. "page": page,
  172. "filter_config": forderung_filter_config,
  173. },
  174. )
  175. @router.get("/app/forderungen/details/{client_db}_{document_no}", response_class=HTMLResponse)
  176. def forderungen_details(
  177. request: Request, client_db: str, document_no: str, db: Session = Depends(get_session), limit: int = 100
  178. ):
  179. context = {
  180. "appsmith": {
  181. "URL": {
  182. "queryParams": {
  183. "Client_DB": "'" + client_db + "'",
  184. "Document_No": "'" + document_no + "'",
  185. }
  186. }
  187. },
  188. "BelegeFilter": {"selectedOptionValue": "D"},
  189. "BenutzerSelect": {"selectedOptionValue": "winter"},
  190. }
  191. template_context = {}
  192. for filename in [
  193. "auftrag_positionen",
  194. "forderung_belege",
  195. "forderung_kommentar",
  196. "forderung_kopf",
  197. "forderung_mahnung",
  198. ]:
  199. query = templates.TemplateResponse(request, f"forderungen/queries/{filename}.sql", context).body.decode("utf-8")
  200. template_context[filename] = db.execute(text(query)).fetchall()
  201. template_context["forderung"] = {
  202. "id": "12345",
  203. "rechnungsnummer": "WERE123445",
  204. "kunde": "Burghard",
  205. "betrag": "120,00",
  206. "faelligkeit": "23.12.1983",
  207. }
  208. fileserver = "C:\\Projekte\\Reisacher-Fileserver"
  209. files = [f.relative_to(fileserver) for f in Path(f"{fileserver}\\{client_db}\\{document_no}").glob("*")]
  210. template_context["files"] = files
  211. return templates.TemplateResponse(request, "forderungen/details.html", template_context)
  212. @router.get("/app/forderungen/dashboard", response_class=HTMLResponse)
  213. def forderungen_dashboard(request: Request, db: Session = Depends(get_session)):
  214. context = {"summary": False}
  215. return templates.TemplateResponse(request, "forderungen/dashboard.html", context)
  216. @router.get("/chat", response_class=HTMLResponse)
  217. def chat(request: Request):
  218. # q = db.query(Forderung).order_by(Forderung.faelligkeit.asc()).limit(limit).all()
  219. return templates.TemplateResponse(request, "base/chat.html")
  220. @router.get("/export/csv")
  221. def export_csv(db: Session = Depends(get_session)):
  222. q = db.query(Forderung).all()
  223. rows = []
  224. for f in q:
  225. rows.append(
  226. {
  227. "id": f.id,
  228. "rechnungsnummer": f.rechnungsnummer,
  229. "kunde": f.kunde.name if f.kunde else "",
  230. "betrag": float(f.betrag),
  231. "faelligkeit": f.faelligkeit.isoformat() if f.faelligkeit else "",
  232. }
  233. )
  234. df = pd.DataFrame(rows)
  235. buf = io.StringIO()
  236. df.to_csv(buf, index=False)
  237. buf.seek(0)
  238. return Response(
  239. content=buf.getvalue(),
  240. media_type="text/csv",
  241. headers={"Content-Disposition": "attachment; filename=forderungen.csv"},
  242. )
  243. @router.get("/export/xlsx")
  244. def export_xlsx(db: Session = Depends(get_session)):
  245. q = db.query(Forderung).all()
  246. rows = []
  247. for f in q:
  248. rows.append(
  249. {
  250. "id": f.id,
  251. "rechnungsnummer": f.rechnungsnummer,
  252. "kunde": f.kunde.name if f.kunde else "",
  253. "betrag": float(f.betrag),
  254. "faelligkeit": f.faelligkeit.isoformat() if f.faelligkeit else "",
  255. }
  256. )
  257. df = pd.DataFrame(rows)
  258. buf = io.BytesIO()
  259. with pd.ExcelWriter(buf, engine="openpyxl") as writer:
  260. df.to_excel(writer, index=False, sheet_name="Forderungen")
  261. buf.seek(0)
  262. return StreamingResponse(
  263. buf,
  264. media_type="application/vnd.openxmlformats-officedocument.spreadsheetml.sheet",
  265. headers={"Content-Disposition": "attachment; filename=forderungen.xlsx"},
  266. )
  267. @router.get("/detail/{id}", response_class=HTMLResponse)
  268. def detail(request: Request, id: int, db: Session = Depends(get_session)):
  269. f = db.query(Forderung).filter(Forderung.id == id).first()
  270. if not f:
  271. raise HTTPException(status_code=404)
  272. return templates.TemplateResponse(request, "base/detail.html", {"request": request, "forderung": f})
  273. @router.post("/detail/{id}/bemerkung")
  274. def save_bemerkung(id: int, data: BemerkungIn, request: Request, db: Session = Depends(get_session)):
  275. user = request.cookies.get("user") or "anonymous"
  276. b = Bemerkung(
  277. forderung_id=id, benutzer=user, bemerkung=data.bemerkung, wiedervorlage_datum=data.wiedervorlage_datum
  278. )
  279. db.add(b)
  280. db.commit()
  281. return RedirectResponse(url=f"/detail/{id}", status_code=302)
  282. @router.get("/detail/{id}/export/docx")
  283. def export_docx(id: int, db: Session = Depends(get_session)):
  284. f = db.query(Forderung).filter(Forderung.id == id).first()
  285. if not f:
  286. raise HTTPException(status_code=404)
  287. doc = Document()
  288. doc.add_heading(f"Rechnung {f.rechnungsnummer}", level=1)
  289. doc.add_paragraph(f'Kunde: {f.kunde.name if f.kunde else ""}')
  290. doc.add_paragraph(f"Betrag: {float(f.betrag)}")
  291. doc.add_paragraph(f"Fälligkeit: {f.faelligkeit}")
  292. buf = io.BytesIO()
  293. doc.save(buf)
  294. buf.seek(0)
  295. return StreamingResponse(
  296. buf,
  297. media_type="application/vnd.openxmlformats-officedocument.wordprocessingml.document",
  298. headers={"Content-Disposition": f"attachment; filename=rechnung_{f.rechnungsnummer}.docx"},
  299. )
  300. @router.get("/app/forderungen/export/{client_db}_{document_no}/")
  301. def export_docx2(request: Request, client_db: str, document_no: str, db: Session = Depends(get_session)):
  302. context = {
  303. "appsmith": {
  304. "URL": {
  305. "queryParams": {
  306. "Client_DB": "'" + client_db + "'",
  307. "Document_No": "'" + document_no + "'",
  308. }
  309. }
  310. },
  311. }
  312. query = templates.TemplateResponse(request, "forderungen/queries/forderung_kopf.sql", context).body.decode("utf-8")
  313. q = db.execute(text(query)).fetchone()
  314. doc = DocxTemplate("templates\\forderungen\\docs\\Mahnung_AHR.docx")
  315. filename = Path(
  316. f"C:\\Projekte\\Reisacher-Fileserver\\{client_db}\\{document_no}\\Mahnung_{client_db}_{document_no}.docx"
  317. )
  318. filename.parent.mkdir(parents=True, exist_ok=True)
  319. context = {
  320. "Kunde_Name": q.Kunde,
  321. "Kunde_Adresse": "Im Waldhof 14a",
  322. "Kunde_PLZ": "61476",
  323. "Kunde_Ort": "Kronberg",
  324. "Datum_heute": datetime.now().strftime("%d.%m.%Y"),
  325. "Kunde_Nr": q.Kunde.split("-")[-1].strip(),
  326. "Fahrzeug_Kennzeichen": "OF-RB 512",
  327. "Rechnung_Nr": q.Document_No,
  328. "Betrag_SB": "250,00",
  329. "Betrag_USt": "380,00",
  330. }
  331. doc.render(context)
  332. doc.save(filename)
  333. return FileResponse(filename, media_type="application/octet-stream", filename=filename.name)
  334. @router.get("/files/")
  335. def get_files(request: Request):
  336. name = request.query_params.get("name")
  337. filename = Path(f"C:\\Projekte\\Reisacher-Fileserver\\{unquote(name)}")
  338. return FileResponse(filename, media_type="application/octet-stream", filename=filename.name)