routes.py 12 KB

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