z230
This commit is contained in:
@@ -0,0 +1 @@
|
|||||||
|
OPENAI_API_KEY=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
|
||||||
@@ -0,0 +1,50 @@
|
|||||||
|
# OrdinaceProjekt
|
||||||
|
|
||||||
|
## DŮLEŽITÉ — pracovní adresář
|
||||||
|
|
||||||
|
Hlavní projekt je **adresář obsahující tento soubor AGENTS.md** (kořen projektu OrdinaceProjekt).
|
||||||
|
Výsledné soubory (skripty, knihovny, data) vždy ukládej do tohoto kořenového adresáře nebo jeho podadresářů.
|
||||||
|
|
||||||
|
Worktree (`.Codex/worktrees/*`) slouží jen pro interní práci Codex, ne jako výstup.
|
||||||
|
|
||||||
|
## Přečti na začátku každé konverzace
|
||||||
|
|
||||||
|
Každý adresář se skriptem má vlastní `NOTES.md` s technickými detaily. Přečti relevantní NOTES.md podle toho, čeho se konverzace týká.
|
||||||
|
|
||||||
|
## Anthropic API klíč
|
||||||
|
|
||||||
|
Uložen v `Medevio/.env` jako `ANTHROPIC_API_KEY=sk-ant-...`.
|
||||||
|
Skripty, které volají Codex API, si ho načítají samy — vzor:
|
||||||
|
|
||||||
|
```python
|
||||||
|
def _load_env():
|
||||||
|
env_path = Path(__file__).resolve().parent.parent / "Medevio" / ".env"
|
||||||
|
if env_path.exists():
|
||||||
|
for line in env_path.read_text(encoding="utf-8").splitlines():
|
||||||
|
line = line.strip()
|
||||||
|
if "=" in line and not line.startswith("#"):
|
||||||
|
k, v = line.split("=", 1)
|
||||||
|
os.environ[k.strip()] = v.strip()
|
||||||
|
|
||||||
|
_load_env()
|
||||||
|
```
|
||||||
|
|
||||||
|
## Sdílené knihovny (`Knihovny/`)
|
||||||
|
|
||||||
|
Před psaním nového kódu vždy zkontroluj, zda existuje vhodná sdílená funkce.
|
||||||
|
Import vždy přes `sys.path` na kořen projektu nebo přímou cestou.
|
||||||
|
|
||||||
|
| Modul | Klíčová funkce / třída | Popis |
|
||||||
|
|-------|------------------------|-------|
|
||||||
|
| `najdi_dropbox.py` | `get_dropbox_root() → str` | Zjistí cestu k Dropboxu z registru nebo info.json — **používej místo pevných cest** |
|
||||||
|
| `EmailMessagingGraph.py` | — | Odesílání e-mailů přes Microsoft Graph API |
|
||||||
|
| `mysql_db.py` | — | Připojení a operace s MySQL databází |
|
||||||
|
| `medicus_db.py` | — | Připojení k databázi Medicus (Firebird) |
|
||||||
|
| `vzpb2b_client.py` | — | Klient pro VZP B2B API (stav pojištění) |
|
||||||
|
|
||||||
|
## Přehled skriptů
|
||||||
|
|
||||||
|
| Skript | Adresář | Popis |
|
||||||
|
|--------|---------|-------|
|
||||||
|
| `stahni_str8ts.py` | `SběrDatRůzné/DailyStr8ts/` | Stahuje daily Str8ts puzzle jako PDF, odesílá emailem — viz [NOTES.md](SběrDatRůzné/DailyStr8ts/NOTES.md) |
|
||||||
|
| `10_StahnoutXML.py`, `11_ParseXML.py` | `Recepty/NačteníPředpisuWithClaude/` | Pipeline pro stahování detailů receptů z eRecept SÚKL — viz [NacistPredpis_DOKUMENTACE.md](Recepty/NačteníPředpisuWithClaude/NacistPredpis_DOKUMENTACE.md) |
|
||||||
@@ -0,0 +1,377 @@
|
|||||||
|
# FakturyRename.py
|
||||||
|
# Verze: 1.2
|
||||||
|
# Datum: 05JUN2026
|
||||||
|
# Autor: Vladimír Buzalka
|
||||||
|
#
|
||||||
|
# Popis:
|
||||||
|
# Projde PDF faktury a doklady přímo ve vstupním adresáři, pošle je do
|
||||||
|
# OpenAI Responses API k vytěžení údajů a navrhne jednotný název souboru.
|
||||||
|
# Výsledný formát názvu:
|
||||||
|
# YYYY-MM-DD Typ Dodavatel ČÍSLO [popis] [částka MĚNA].pdf
|
||||||
|
#
|
||||||
|
# Při DRY_RUN = False skript soubor přejmenuje a přesune do podadresáře
|
||||||
|
# NamedInvoicesbyOpenAI, aby další běh zpracovával jen nové dokumenty.
|
||||||
|
# Loguje původní název, návrh, tokeny, odhad ceny a stav zpracování.
|
||||||
|
|
||||||
|
import os
|
||||||
|
from dotenv import load_dotenv
|
||||||
|
import base64
|
||||||
|
import json
|
||||||
|
import re
|
||||||
|
import time
|
||||||
|
from pathlib import Path
|
||||||
|
|
||||||
|
from openai import OpenAI
|
||||||
|
|
||||||
|
|
||||||
|
# =========================
|
||||||
|
# CENA API
|
||||||
|
# =========================
|
||||||
|
|
||||||
|
USD_TO_CZK = 25.0
|
||||||
|
|
||||||
|
# # Ceny nastav ručně podle modelu, který používáš.
|
||||||
|
# # Zde počítáme:
|
||||||
|
# # input = 5 USD / 1M tokenů
|
||||||
|
# # output = 30 USD / 1M tokenů
|
||||||
|
# PRICE_INPUT_USD_PER_1M = 5.00
|
||||||
|
# PRICE_OUTPUT_USD_PER_1M = 30.00
|
||||||
|
# # Model s podporou PDF / vision vstupu.
|
||||||
|
# MODEL = "gpt-5.5"
|
||||||
|
|
||||||
|
MODEL = "gpt-5.4-mini"
|
||||||
|
|
||||||
|
PRICE_INPUT_USD_PER_1M = 0.75
|
||||||
|
PRICE_OUTPUT_USD_PER_1M = 4.50
|
||||||
|
|
||||||
|
# =========================
|
||||||
|
# NASTAVENÍ
|
||||||
|
# =========================
|
||||||
|
|
||||||
|
FOLDER = Path(
|
||||||
|
r"u:\Dropbox\Ordinace\!!MUDr. Michaela Buzalková s.r.o\Prosek\#040 Faktury přijaté"
|
||||||
|
)
|
||||||
|
|
||||||
|
PROCESSED_FOLDER = FOLDER / "NamedInvoicesbyOpenAI"
|
||||||
|
|
||||||
|
# Pro test na 3 fakturách nech DRY_RUN = True.
|
||||||
|
# Skript jen vypíše a zaloguje návrhy názvů, ale soubory nepřejmenuje.
|
||||||
|
DRY_RUN = False
|
||||||
|
|
||||||
|
# Nepůjde do podadresářů, protože používáme glob(), ne rglob().
|
||||||
|
PDF_PATTERN = "*.pdf"
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
LOG_FILE = FOLDER / "_rename_log_invoices.txt"
|
||||||
|
|
||||||
|
ENV_FILE = Path(r"U:\ordinaceprojekt\.env")
|
||||||
|
|
||||||
|
|
||||||
|
# =========================
|
||||||
|
# PRAVIDLA PRO POJMENOVÁNÍ
|
||||||
|
# =========================
|
||||||
|
|
||||||
|
NAMING_RULES = """
|
||||||
|
Jsi pomocník pro pojmenování naskenovaných PDF dokladů MUDr. Michaely Buzalkové.
|
||||||
|
|
||||||
|
ÚKOL:
|
||||||
|
Z PDF faktury/dokladu vytěž datum, typ dokladu, dodavatele, číslo dokladu, stručný popis, částku a měnu.
|
||||||
|
Vrať pouze JSON s polem "filename".
|
||||||
|
|
||||||
|
CÍLOVÝ FORMÁT:
|
||||||
|
YYYY-MM-DD Typ Dodavatel ČÍSLO [popis] [částka MĚNA].pdf
|
||||||
|
|
||||||
|
PŘÍKLADY:
|
||||||
|
2026-06-01 Faktura ASKER 261103225 [kontejner Yannick 1.5 l] [339.00 CZK].pdf
|
||||||
|
2026-06-01 Faktura MEDIPOS 10195703 [CRP, kapiláry, písty, rukavice, nádoba] [5578.97 CZK].pdf
|
||||||
|
2026-05-29 Faktura Ptáček 202604570 [vakcíny Adacel, Vaqta, Havrix] [9235.20 CZK].pdf
|
||||||
|
2026-05-29 Faktura Poliklinika Prosek 91260763 [lékárna] [16165.40 CZK].pdf
|
||||||
|
2026-06-01 Dodací list QuickSeal 200609058 [VivaDiag Hydroxyvitamin D3] [2620.00 CZK].pdf
|
||||||
|
|
||||||
|
DŮLEŽITÁ PRAVIDLA:
|
||||||
|
1. Prefix [POHODA] nikdy nepřidávej.
|
||||||
|
2. Používej datum vystavení dokladu, ne datum splatnosti.
|
||||||
|
3. Typ dokladu vyber podle dokumentu:
|
||||||
|
- Faktura
|
||||||
|
- Dobropis
|
||||||
|
- Paragon
|
||||||
|
- Dodací list
|
||||||
|
- Zálohová faktura
|
||||||
|
- Smlouva
|
||||||
|
- Platba
|
||||||
|
- Poplatek
|
||||||
|
- Výdajový pokladní doklad
|
||||||
|
4. Pokud je v dokumentu napsáno "Dodací list není daňový doklad - nehraďte", typ musí být "Dodací list", ne "Faktura".
|
||||||
|
5. Dodavatel zapisuj krátce a konzistentně:
|
||||||
|
- MEDIPOS
|
||||||
|
- MEDEVIO
|
||||||
|
- MEDATRON
|
||||||
|
- ASKER
|
||||||
|
- QuickSeal
|
||||||
|
- Poliklinika Prosek
|
||||||
|
- Alza
|
||||||
|
- Microsoft
|
||||||
|
- OpenAI
|
||||||
|
- Ptáček
|
||||||
|
6. SPECIÁLNÍ PRAVIDLO: pokud je dodavatel/firma "Distribuce CZ", v názvu souboru použij dodavatele "Ptáček".
|
||||||
|
7. SPECIÁLNÍ PRAVIDLO: u faktur MEDIPOS použij jako číslo dokladu variabilní symbol nebo hlavní číslo faktury bez mezer, například 10195703. Nepoužívej interní evidenční číslo typu FV-5703/2026.
|
||||||
|
8. Částku piš vždy s desetinnou tečkou a měnou, například [5578.97 CZK].
|
||||||
|
9. Když je částka v Kč, měna je CZK.
|
||||||
|
10. Popis drž krátký, praktický a česky.
|
||||||
|
11. Popis dávej do hranatých závorek.
|
||||||
|
12. Nepoužívej dvojtečky, lomítka, uvozovky ani znaky nevhodné pro Windows názvy souborů.
|
||||||
|
13. Pokud jde jen o dodací list bez daňového dokladu, částku můžeš uvést, ale typ musí zůstat Dodací list.
|
||||||
|
14. Pokud si nejsi jistý popisem, použij obecný popis typu [materiál do ordinace], [lékárna], [vakcíny], [testy].
|
||||||
|
15. Výstup musí být pouze validní JSON, nic jiného.
|
||||||
|
|
||||||
|
JSON FORMÁT:
|
||||||
|
{
|
||||||
|
"filename": "YYYY-MM-DD Faktura Dodavatel 123456 [popis] [123.45 CZK].pdf"
|
||||||
|
}
|
||||||
|
"""
|
||||||
|
|
||||||
|
|
||||||
|
# =========================
|
||||||
|
# POMOCNÉ FUNKCE
|
||||||
|
# =========================
|
||||||
|
|
||||||
|
def pdf_to_base64_data_url(path: Path) -> str:
|
||||||
|
data = path.read_bytes()
|
||||||
|
b64 = base64.b64encode(data).decode("utf-8")
|
||||||
|
return f"data:application/pdf;base64,{b64}"
|
||||||
|
|
||||||
|
|
||||||
|
def sanitize_windows_filename(name: str) -> str:
|
||||||
|
"""
|
||||||
|
Očistí název souboru pro Windows.
|
||||||
|
"""
|
||||||
|
# Zakázané znaky ve Windows: < > : " / \ | ? *
|
||||||
|
name = re.sub(r'[<>:"/\\|?*]', " ", name)
|
||||||
|
|
||||||
|
# Sjednocení mezer
|
||||||
|
name = re.sub(r"\s+", " ", name).strip()
|
||||||
|
|
||||||
|
# Windows nemá rád tečku nebo mezeru na konci
|
||||||
|
name = name.rstrip(" .")
|
||||||
|
|
||||||
|
if not name.lower().endswith(".pdf"):
|
||||||
|
name += ".pdf"
|
||||||
|
|
||||||
|
return name
|
||||||
|
|
||||||
|
|
||||||
|
def unique_path(target: Path) -> Path:
|
||||||
|
"""
|
||||||
|
Pokud cílový soubor existuje, přidá (2), (3), ...
|
||||||
|
"""
|
||||||
|
if not target.exists():
|
||||||
|
return target
|
||||||
|
|
||||||
|
stem = target.stem
|
||||||
|
suffix = target.suffix
|
||||||
|
parent = target.parent
|
||||||
|
|
||||||
|
i = 2
|
||||||
|
while True:
|
||||||
|
candidate = parent / f"{stem} ({i}){suffix}"
|
||||||
|
if not candidate.exists():
|
||||||
|
return candidate
|
||||||
|
i += 1
|
||||||
|
|
||||||
|
|
||||||
|
def extract_json_object(text: str) -> dict:
|
||||||
|
"""
|
||||||
|
Kdyby model náhodou vrátil něco kolem JSONu, zkusí vytáhnout první JSON objekt.
|
||||||
|
"""
|
||||||
|
text = text.strip()
|
||||||
|
|
||||||
|
try:
|
||||||
|
return json.loads(text)
|
||||||
|
except json.JSONDecodeError:
|
||||||
|
pass
|
||||||
|
|
||||||
|
match = re.search(r"\{.*\}", text, flags=re.DOTALL)
|
||||||
|
if not match:
|
||||||
|
raise ValueError(f"Model nevrátil JSON:\n{text}")
|
||||||
|
|
||||||
|
return json.loads(match.group(0))
|
||||||
|
|
||||||
|
|
||||||
|
def get_usage_value(usage, key: str) -> int:
|
||||||
|
"""
|
||||||
|
Bezpečně přečte usage hodnotu.
|
||||||
|
Funguje pro objekt i dict.
|
||||||
|
"""
|
||||||
|
if usage is None:
|
||||||
|
return 0
|
||||||
|
|
||||||
|
if isinstance(usage, dict):
|
||||||
|
return usage.get(key, 0) or 0
|
||||||
|
|
||||||
|
return getattr(usage, key, 0) or 0
|
||||||
|
|
||||||
|
|
||||||
|
def calculate_cost_from_usage(usage) -> dict:
|
||||||
|
"""
|
||||||
|
Spočítá odhad ceny z response.usage.
|
||||||
|
"""
|
||||||
|
input_tokens = get_usage_value(usage, "input_tokens")
|
||||||
|
output_tokens = get_usage_value(usage, "output_tokens")
|
||||||
|
total_tokens = get_usage_value(usage, "total_tokens")
|
||||||
|
|
||||||
|
if not total_tokens:
|
||||||
|
total_tokens = input_tokens + output_tokens
|
||||||
|
|
||||||
|
input_cost_usd = input_tokens / 1_000_000 * PRICE_INPUT_USD_PER_1M
|
||||||
|
output_cost_usd = output_tokens / 1_000_000 * PRICE_OUTPUT_USD_PER_1M
|
||||||
|
total_cost_usd = input_cost_usd + output_cost_usd
|
||||||
|
total_cost_czk = total_cost_usd * USD_TO_CZK
|
||||||
|
|
||||||
|
return {
|
||||||
|
"input_tokens": input_tokens,
|
||||||
|
"output_tokens": output_tokens,
|
||||||
|
"total_tokens": total_tokens,
|
||||||
|
"input_cost_usd": input_cost_usd,
|
||||||
|
"output_cost_usd": output_cost_usd,
|
||||||
|
"total_cost_usd": total_cost_usd,
|
||||||
|
"total_cost_czk": total_cost_czk,
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
def ask_openai_for_filename(client: OpenAI, pdf_path: Path) -> tuple[str, dict]:
|
||||||
|
file_data = pdf_to_base64_data_url(pdf_path)
|
||||||
|
|
||||||
|
response = client.responses.create(
|
||||||
|
model=MODEL,
|
||||||
|
input=[
|
||||||
|
{
|
||||||
|
"role": "user",
|
||||||
|
"content": [
|
||||||
|
{
|
||||||
|
"type": "input_file",
|
||||||
|
"filename": pdf_path.name,
|
||||||
|
"file_data": file_data,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"type": "input_text",
|
||||||
|
"text": NAMING_RULES,
|
||||||
|
},
|
||||||
|
],
|
||||||
|
}
|
||||||
|
],
|
||||||
|
)
|
||||||
|
|
||||||
|
text = response.output_text.strip()
|
||||||
|
obj = extract_json_object(text)
|
||||||
|
|
||||||
|
filename = obj.get("filename", "").strip()
|
||||||
|
if not filename:
|
||||||
|
raise ValueError(f"JSON neobsahuje filename:\n{text}")
|
||||||
|
|
||||||
|
cost = calculate_cost_from_usage(response.usage)
|
||||||
|
|
||||||
|
return sanitize_windows_filename(filename), cost
|
||||||
|
|
||||||
|
|
||||||
|
def log_line(text: str) -> None:
|
||||||
|
print(text)
|
||||||
|
with LOG_FILE.open("a", encoding="utf-8") as f:
|
||||||
|
f.write(text + "\n")
|
||||||
|
|
||||||
|
|
||||||
|
# =========================
|
||||||
|
# HLAVNÍ BĚH
|
||||||
|
# =========================
|
||||||
|
|
||||||
|
def main() -> None:
|
||||||
|
if not FOLDER.exists():
|
||||||
|
raise FileNotFoundError(f"Adresář neexistuje: {FOLDER}")
|
||||||
|
|
||||||
|
# Načtení API klíče z .env
|
||||||
|
load_dotenv(ENV_FILE)
|
||||||
|
|
||||||
|
if not os.getenv("OPENAI_API_KEY"):
|
||||||
|
raise RuntimeError(
|
||||||
|
f"Chybí OPENAI_API_KEY. Zkontroluj soubor {ENV_FILE}"
|
||||||
|
)
|
||||||
|
|
||||||
|
client = OpenAI()
|
||||||
|
|
||||||
|
pdfs = sorted(FOLDER.glob(PDF_PATTERN))
|
||||||
|
pdfs = [p for p in pdfs if p.is_file() and p.suffix.lower() == ".pdf"]
|
||||||
|
|
||||||
|
if not pdfs:
|
||||||
|
print("Nenalezeno žádné PDF.")
|
||||||
|
return
|
||||||
|
|
||||||
|
total_input_tokens = 0
|
||||||
|
total_output_tokens = 0
|
||||||
|
total_tokens = 0
|
||||||
|
total_cost_usd = 0.0
|
||||||
|
total_cost_czk = 0.0
|
||||||
|
|
||||||
|
log_line("")
|
||||||
|
log_line("=" * 80)
|
||||||
|
log_line(f"START: {time.strftime('%Y-%m-%d %H:%M:%S')}")
|
||||||
|
log_line(f"Adresář: {FOLDER}")
|
||||||
|
log_line(f"Hotové faktury: {PROCESSED_FOLDER}")
|
||||||
|
log_line(f"Počet PDF: {len(pdfs)}")
|
||||||
|
log_line(f"DRY_RUN: {DRY_RUN}")
|
||||||
|
log_line(f"MODEL: {MODEL}")
|
||||||
|
log_line(f"Kurz: 1 USD = {USD_TO_CZK:.2f} CZK")
|
||||||
|
log_line("=" * 80)
|
||||||
|
|
||||||
|
for i, pdf in enumerate(pdfs, start=1):
|
||||||
|
log_line(f"\n[{i}/{len(pdfs)}] Původní název: {pdf.name}")
|
||||||
|
|
||||||
|
try:
|
||||||
|
new_name, cost = ask_openai_for_filename(client, pdf)
|
||||||
|
|
||||||
|
total_input_tokens += cost["input_tokens"]
|
||||||
|
total_output_tokens += cost["output_tokens"]
|
||||||
|
total_tokens += cost["total_tokens"]
|
||||||
|
total_cost_usd += cost["total_cost_usd"]
|
||||||
|
total_cost_czk += cost["total_cost_czk"]
|
||||||
|
|
||||||
|
log_line(f" Návrh: {new_name}")
|
||||||
|
log_line(
|
||||||
|
f" Tokeny: input={cost['input_tokens']}, "
|
||||||
|
f"output={cost['output_tokens']}, "
|
||||||
|
f"total={cost['total_tokens']}"
|
||||||
|
)
|
||||||
|
log_line(
|
||||||
|
f" Cena volání: ${cost['total_cost_usd']:.6f} "
|
||||||
|
f"≈ {cost['total_cost_czk']:.2f} Kč"
|
||||||
|
)
|
||||||
|
|
||||||
|
target = unique_path(PROCESSED_FOLDER / new_name)
|
||||||
|
if target.name != new_name:
|
||||||
|
log_line(f" Cíl po vyřešení konfliktu: {target.name}")
|
||||||
|
|
||||||
|
if DRY_RUN:
|
||||||
|
log_line(f" Cíl: {target}")
|
||||||
|
log_line(" Stav: DRY-RUN, nepřejmenováno/nepřesunuto")
|
||||||
|
else:
|
||||||
|
PROCESSED_FOLDER.mkdir(exist_ok=True)
|
||||||
|
pdf.rename(target)
|
||||||
|
if pdf.name == new_name:
|
||||||
|
log_line(" Stav: PŘESUNUTO")
|
||||||
|
else:
|
||||||
|
log_line(" Stav: PŘEJMENOVÁNO A PŘESUNUTO")
|
||||||
|
|
||||||
|
except Exception as e:
|
||||||
|
log_line(f" CHYBA: {type(e).__name__}: {e}")
|
||||||
|
|
||||||
|
log_line("")
|
||||||
|
log_line("=" * 80)
|
||||||
|
log_line("SOUHRN CENY")
|
||||||
|
log_line(f"Tokeny celkem: input={total_input_tokens}, output={total_output_tokens}, total={total_tokens}")
|
||||||
|
log_line(f"Cena celkem: ${total_cost_usd:.6f} ≈ {total_cost_czk:.2f} Kč")
|
||||||
|
log_line("=" * 80)
|
||||||
|
|
||||||
|
log_line("\nHOTOVO")
|
||||||
|
|
||||||
|
|
||||||
|
if __name__ == "__main__":
|
||||||
|
main()
|
||||||
@@ -1,161 +0,0 @@
|
|||||||
#!/usr/bin/env python3
|
|
||||||
# -*- coding: utf-8 -*-
|
|
||||||
"""
|
|
||||||
Jednorázový skript: znovu stáhne přílohy pro 1c935d36-c9df-46a1-9ef2-b7f327f376c7 (Šmídová).
|
|
||||||
Přeskočí přílohy, které jsou již v medevio_downloads. Po úspěchu označí požadavek jako zpracovaný.
|
|
||||||
"""
|
|
||||||
|
|
||||||
import requests
|
|
||||||
import pymysql
|
|
||||||
import time
|
|
||||||
from pathlib import Path
|
|
||||||
|
|
||||||
TARGET_REQUEST_ID = "1c935d36-c9df-46a1-9ef2-b7f327f376c7"
|
|
||||||
RETRY_ATTEMPTS = 5
|
|
||||||
RETRY_DELAY = 3 # sekund mezi pokusy
|
|
||||||
|
|
||||||
TOKEN_PATH = Path(__file__).resolve().parent.parent / "token.txt"
|
|
||||||
|
|
||||||
DB_CONFIG = {
|
|
||||||
"host": "192.168.1.76",
|
|
||||||
"port": 3306,
|
|
||||||
"user": "root",
|
|
||||||
"password": "Vlado9674+",
|
|
||||||
"database": "medevio",
|
|
||||||
"charset": "utf8mb4",
|
|
||||||
"cursorclass": pymysql.cursors.DictCursor,
|
|
||||||
}
|
|
||||||
|
|
||||||
GRAPHQL_QUERY = r"""
|
|
||||||
query ClinicRequestDetail_GetPatientRequest2($requestId: UUID!) {
|
|
||||||
patientRequestMedicalRecords: listMedicalRecordsForPatientRequest(
|
|
||||||
attachmentTypes: [ECRF_FILL_ATTACHMENT, MESSAGE_ATTACHMENT, PATIENT_REQUEST_ATTACHMENT]
|
|
||||||
patientRequestId: $requestId
|
|
||||||
pageInfo: {first: 100, offset: 0}
|
|
||||||
) {
|
|
||||||
attachmentType
|
|
||||||
id
|
|
||||||
medicalRecord {
|
|
||||||
contentType
|
|
||||||
description
|
|
||||||
downloadUrl
|
|
||||||
id
|
|
||||||
url
|
|
||||||
visibleToPatient
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
"""
|
|
||||||
|
|
||||||
|
|
||||||
def extract_filename_from_url(url: str) -> str:
|
|
||||||
try:
|
|
||||||
return url.split("/")[-1].split("?")[0]
|
|
||||||
except Exception:
|
|
||||||
return "unknown_filename"
|
|
||||||
|
|
||||||
|
|
||||||
def read_token(p: Path) -> str:
|
|
||||||
tok = p.read_text(encoding="utf-8").strip()
|
|
||||||
return tok.split(" ", 1)[1] if tok.startswith("Bearer ") else tok
|
|
||||||
|
|
||||||
|
|
||||||
def download_with_retry(url: str, attempts: int, delay: int) -> bytes:
|
|
||||||
for attempt in range(1, attempts + 1):
|
|
||||||
try:
|
|
||||||
r = requests.get(url, timeout=60)
|
|
||||||
if r.status_code == 200:
|
|
||||||
return r.content
|
|
||||||
print(f" ⚠️ HTTP {r.status_code}, pokus {attempt}/{attempts}")
|
|
||||||
except Exception as e:
|
|
||||||
print(f" ⚠️ Chyba stahování (pokus {attempt}/{attempts}): {e}")
|
|
||||||
if attempt < attempts:
|
|
||||||
time.sleep(delay)
|
|
||||||
raise RuntimeError(f"Stahování selhalo po {attempts} pokusech: {url}")
|
|
||||||
|
|
||||||
|
|
||||||
def main():
|
|
||||||
token = read_token(TOKEN_PATH)
|
|
||||||
headers = {"Authorization": f"Bearer {token}", "Content-Type": "application/json"}
|
|
||||||
|
|
||||||
conn = pymysql.connect(**DB_CONFIG)
|
|
||||||
|
|
||||||
# Načíst již stažené attachment_id pro tento požadavek
|
|
||||||
with conn.cursor() as cur:
|
|
||||||
cur.execute("SELECT attachment_id FROM medevio_downloads WHERE request_id = %s", (TARGET_REQUEST_ID,))
|
|
||||||
existing_ids = {row["attachment_id"] for row in cur.fetchall()}
|
|
||||||
|
|
||||||
print(f"🔍 Zpracovávám požadavek {TARGET_REQUEST_ID}")
|
|
||||||
print(f" Již staženo příloh: {len(existing_ids)}")
|
|
||||||
|
|
||||||
# GraphQL dotaz
|
|
||||||
payload = {
|
|
||||||
"operationName": "ClinicRequestDetail_GetPatientRequest2",
|
|
||||||
"query": GRAPHQL_QUERY,
|
|
||||||
"variables": {"requestId": TARGET_REQUEST_ID},
|
|
||||||
}
|
|
||||||
r = requests.post("https://api.medevio.cz/graphql", json=payload, headers=headers, timeout=30)
|
|
||||||
attachments = r.json().get("data", {}).get("patientRequestMedicalRecords", [])
|
|
||||||
print(f" Nalezeno příloh celkem: {len(attachments)}")
|
|
||||||
|
|
||||||
# Načíst createdAt pro INSERT
|
|
||||||
with conn.cursor() as cur:
|
|
||||||
cur.execute("SELECT createdAt FROM pozadavky WHERE id = %s", (TARGET_REQUEST_ID,))
|
|
||||||
row = cur.fetchone()
|
|
||||||
created_date = row["createdAt"] if row else None
|
|
||||||
|
|
||||||
saved = 0
|
|
||||||
skipped = 0
|
|
||||||
errors = 0
|
|
||||||
|
|
||||||
with conn.cursor() as cur:
|
|
||||||
for a in attachments:
|
|
||||||
m = a.get("medicalRecord") or {}
|
|
||||||
att_id = a.get("id")
|
|
||||||
|
|
||||||
if att_id in existing_ids:
|
|
||||||
print(f" ⏭️ Přeskočeno (již existuje): {att_id}")
|
|
||||||
skipped += 1
|
|
||||||
continue
|
|
||||||
|
|
||||||
url = m.get("downloadUrl")
|
|
||||||
if not url:
|
|
||||||
print(f" ⚠️ Příloha {att_id} nemá downloadUrl, přeskakuji")
|
|
||||||
skipped += 1
|
|
||||||
continue
|
|
||||||
|
|
||||||
try:
|
|
||||||
content = download_with_retry(url, RETRY_ATTEMPTS, RETRY_DELAY)
|
|
||||||
filename = extract_filename_from_url(url)
|
|
||||||
cur.execute("""
|
|
||||||
INSERT INTO medevio_downloads (
|
|
||||||
request_id, attachment_id, attachment_type,
|
|
||||||
filename, content_type, file_size,
|
|
||||||
created_at, file_content
|
|
||||||
) VALUES (%s,%s,%s,%s,%s,%s,%s,%s)
|
|
||||||
""", (TARGET_REQUEST_ID, att_id, a.get("attachmentType"), filename,
|
|
||||||
m.get("contentType"), len(content), created_date, content))
|
|
||||||
existing_ids.add(att_id)
|
|
||||||
print(f" 💾 Uloženo: {filename} ({len(content) / 1024:.1f} kB)")
|
|
||||||
saved += 1
|
|
||||||
time.sleep(0.5)
|
|
||||||
except Exception as e:
|
|
||||||
print(f" ❌ Chyba: {e}")
|
|
||||||
errors += 1
|
|
||||||
|
|
||||||
conn.commit()
|
|
||||||
|
|
||||||
if errors == 0:
|
|
||||||
with conn.cursor() as cur:
|
|
||||||
cur.execute("UPDATE pozadavky SET attachmentsProcessed = NOW() WHERE id = %s", (TARGET_REQUEST_ID,))
|
|
||||||
conn.commit()
|
|
||||||
print(f"\n✅ Hotovo. Uloženo: {saved}, přeskočeno: {skipped}. Požadavek označen jako zpracovaný.")
|
|
||||||
else:
|
|
||||||
print(f"\n⚠️ Hotovo s chybami. Uloženo: {saved}, přeskočeno: {skipped}, chyby: {errors}.")
|
|
||||||
print(" Požadavek NEBYL označen jako zpracovaný (opakujte po kontrole).")
|
|
||||||
|
|
||||||
conn.close()
|
|
||||||
|
|
||||||
|
|
||||||
if __name__ == "__main__":
|
|
||||||
main()
|
|
||||||
File diff suppressed because it is too large
Load Diff
@@ -0,0 +1,65 @@
|
|||||||
|
# Pravidla pro přejmenování souborů
|
||||||
|
|
||||||
|
Tato pravidla platí vždy při generování polí `poznamka` a `nazev_souboru`.
|
||||||
|
|
||||||
|
1. Název souboru má vždy tvar: `RODNECISLO YYYY-MM-DD Příjmení, Jméno [TYP ODBORNOST] [popis].pdf`
|
||||||
|
- TYP je vždy buď `LZ` (lékařská zpráva / ambulantní zpráva) nebo `PZ` (propouštěcí zpráva z hospitalizace).
|
||||||
|
- Jiné typy dokumentů (Laboratoř, CT, MRI, kolonoskopie, poukaz FT apod.) nemají TYP prefix — píší se celým názvem: `[Laboratoř]`, `[CT břicha]` atd.
|
||||||
|
- Příklady: `[LZ chirurgie]`, `[PZ interna]`, `[Laboratoř]`, `[CT břicha]`
|
||||||
|
|
||||||
|
2. Když je typ dokumentu PZ (propouštěcí zpráva), umísti do druhé závorky jako první věc data hospitalizace ve tvaru `DDMMMYYYY–DDMMMYYYY` (měsíc třemi písmeny anglicky, velká, bez mezer), za pomlčkou pak popis.
|
||||||
|
- Příklad: `[PZ interna] [12–15APR2026 srdeční selhání]`
|
||||||
|
- Pokud je datum přijetí a propuštění ve stejném měsíci, stačí: `[12–15APR2026 ...]`
|
||||||
|
- Pokud datum hospitalizace nelze určit, druhou závorku napiš bez datumu.
|
||||||
|
|
||||||
|
3. Když je dokument typ "Laboratoř", do `poznamka` uváděj POUZE hodnoty mimo normu (patologické nálezy) — hodnoty v normě vynech. **Osmolalitu séra (Osmolalita, Osm, osmolality) NIKDY nezmiňuj — ani když je mimo normu, ani v jakékoli zkratce.** Toto je absolutní výjimka: osmolalita se do názvu souboru ani do poznámky nepíše nikdy za žádných okolností. Chybně: `C_Osmolalita 293 (↑)` — správně: tuto hodnotu zcela vynech.
|
||||||
|
4. Pokud laboratorní výsledky obsahují glomerulární filtraci — bývá označena jako eGFR, CKD-EPI nebo CK-EPI — do `poznamka` nikdy nepiš číselnou hodnotu eGFR. Místo toho uveď pouze klasifikaci dle stadií CHRIG1–CHRIG5.
|
||||||
|
- **Jednotka:** Nejprve zkontroluj jednotku uvedenou v laboratoři:
|
||||||
|
- Pokud je hodnota v **ml/s** nebo **ml/sec** (typicky malá čísla jako 0.8, 1.14, 1.5…), přenásob ×60 pro převod na ml/min.
|
||||||
|
- Pokud je hodnota v **ml/min** nebo **ml/min/1.73m²** (typicky velká čísla jako 55, 68, 90…), použij přímo.
|
||||||
|
- **Klasifikace** (v ml/min/1.73m²): ≥ 90 → CHRIG1, 60–89 → CHRIG2, 45–59 → CHRIG3a, 30–44 → CHRIG3b, 15–29 → CHRIG4, < 15 → CHRIG5.
|
||||||
|
- Prahové hodnoty pro orientaci při jednotce ml/s: ≥ 1.50 → G1, 1.00–1.49 → G2, 0.75–0.99 → G3a, 0.50–0.74 → G3b, 0.25–0.49 → G4, < 0.25 → G5.
|
||||||
|
- Klasifikaci uváděj pouze pokud je CHRIG2 nebo horší (tj. eGFR < 90 ml/min nebo < 1.50 ml/s) — CHRIG1 je v normě, nezmiňuj ho.
|
||||||
|
5. Když je dokument typ "Laboratoř" a zpráva obsahuje diagnózu (dg., dg:, diagnóza), umísti ji do `nazev_souboru` jako první část druhé závorky, tedy: `[Laboratoř] [dg. XY00 - stručná poznamka]`.
|
||||||
|
6. Zkratky a pojmenování: slovo „sono" (sonografie/ultrazvuk) piš vždy malými písmeny — `sono břicha`, `sono ŠŽ`, nikoli `SONO`. Štítnou žlázu označuj vždy zkratkou `ŠŽ`. Sonografii prsu/prsů (sono mamm., sono mamografie, sono mamma apod.) piš vždy jako `sono prsů`. Denzitometrii (DEXA, DXA, denzitometrie) piš vždy pouze jako `[DXA]` — bez prefixu LZ. Algologii piš vždy jako `[LZ léčba bolesti]`. Dermatovenerologii (dermatologie, dermatovenerologie, kožní) piš vždy jako `[LZ kožní]`. Angiologii piš vždy jako `[LZ cévní]`.
|
||||||
|
7. V číselných hodnotách VŽDY používej desetinnou tečku, nikoli desetinnou čárku. Toto pravidlo platí absolutně pro všechna čísla v `poznamka` i `nazev_souboru` — např. `TG 4.73`, nikoli `TG 4,73`.
|
||||||
|
|
||||||
|
8. Rozpoznávání vzorců — sideropenická anémie: Pokud laboratorní výsledky splňují typický obraz sideropenické (železo-deficitní) anémie, přidej diagnózu jako první část druhé závorky ve tvaru `[sideropenická anémie, ...]`.
|
||||||
|
Typický obraz (stačí kombinace několika z těchto nálezů):
|
||||||
|
- Krevní obraz: ↓ Hb, ↓ Htk, ↓ MCV (mikrocytóza), ↓ MCH nebo ↓ MCHC (hypochromie), ↑ RDW (anisocytóza)
|
||||||
|
- Metabolismus železa: ↓ sérové Fe (železo), ↓ ferritin, ↑ transferrin (nebo TIBC), ↓ saturace transferrinu
|
||||||
|
- Diagnózu uveď pouze pokud je obraz dostatečně přesvědčivý (alespoň ↓ Hb + ↓ MCV nebo ↓ Fe/ferritin).
|
||||||
|
- Příklad výsledného názvu: `[Laboratoř] [sideropenická anémie, Hb 98, MCV 71, Fe 5.2]`
|
||||||
|
|
||||||
|
9. Jaterní enzymy (ALT, AST, GGT, ALP, LD/LDH) a bilirubin — hodnoty pod dolní hranicí normy (snížené) nezmiňuj v `poznamka` ani v `nazev_souboru`. Uváděj pouze hodnoty nad normu (zvýšené).
|
||||||
|
|
||||||
|
10. Druhá závorka pro LZ a PZ — obsah a pořadí: Pro dokumenty typu LZ (lékařská zpráva) a PZ (propouštěcí zpráva) tvoří druhou závorku tyto části v tomto pořadí (oddělené čárkou):
|
||||||
|
a) **Typ návštěvy** — uveď pouze pokud je explicitně rozpoznatelný ze zprávy:
|
||||||
|
- `kontrola` — plánovaná kontrola (např. „plánovaná kontrola", „přichází na kontrolu")
|
||||||
|
- `neplánovaná kontrola` — pokud je výslovně uvedeno, že kontrola nebyla plánovaná
|
||||||
|
- `akutní` — pacient přichází do akutní ambulance nebo cestou RZS/záchranné služby
|
||||||
|
- Pokud typ návštěvy není ve zprávě uveden, tuto část zcela vynech (nepsat žádný fallback).
|
||||||
|
b) **Hlavní diagnóza** — získej z části „Diagnózy", „Závěr" nebo „Dg." — uveď první (hlavní) diagnózu, která je obvykle důvodem návštěvy. Stručně, výstižně.
|
||||||
|
c) **Termín příští plánované kontroly** — pokud je na konci dokumentu uveden konkrétní plánovaný termín příští kontroly (např. „jaro 2027", „za 3 měsíce", „ročně"), umísti ho jako **poslední část druhé závorky**.
|
||||||
|
- Uváděj pouze explicitně naplánované termíny — formát: `ko` + termín bez mezery, např. `ko jaro2027`, `ko za6m`, `ko ročně`.
|
||||||
|
- **Nezahrn** podmíněné návštěvy jako „dle obtíží", „při zhoršení", „při hematurii ihned" apod. — ty jsou samozřejmé a do názvu nepatří.
|
||||||
|
- Pokud dokument žádný plánovaný termín neobsahuje, tuto část vynech.
|
||||||
|
- Příklad (s typem návštěvy): `[LZ kardiologie] [kontrola, ICHS, ko za3m]`
|
||||||
|
- Příklad (bez typu návštěvy): `[LZ neurologie] [migréna, pokračovat v léčbě]`
|
||||||
|
- Příklad akutní: `[LZ interna] [akutní, dekompenzovaná hypertenze, hospitalizace]`
|
||||||
|
- Příklad s termínem kontroly: `[LZ urologie] [kontrola, hematurie microsc., angiomyolipoma renis, ko jaro2027]`
|
||||||
|
- Pro PZ zůstává datum hospitalizace jako první (před typem návštěvy), viz pravidlo 2.
|
||||||
|
|
||||||
|
11. Datum v názvu souboru nesmí být v budoucnosti: Pokud datum nalezené na zprávě a navrhované pro název souboru je pozdější než dnešní datum, je to chyba (např. špatně rozpoznané číslo). Hledej na zprávě jiné datum. Pokud žádné vhodné datum nenajdeš, použij dnešní datum.
|
||||||
|
|
||||||
|
12. Poukaz domácí péče (DP): Dokument nadepsaný „POUKAZ NA VYŠETŘENÍ / OŠETŘENÍ DP" nebo „poukaz domácí péče" se pojmenovává takto:
|
||||||
|
- První závorka: vždy `[domácí péče]` (bez prefixu LZ/PZ).
|
||||||
|
- Datum souboru: pole „Datum" na poukazu (datum vystavení), ve formátu YYYY-MM-DD.
|
||||||
|
- Druhá závorka obsahuje v tomto pořadí, odděleno čárkou:
|
||||||
|
a) **Číslo poukazu** — pole „Pořadové číslo poukazu" (celé číslo, např. `1`).
|
||||||
|
b) **Platnost** — „do DDMMMYYYY" kde datum je z pole „Platnost do" (měsíc třemi velkými písmeny anglicky, bez mezer), např. `do 30JUN2026`.
|
||||||
|
c) **Výkony** — každý výkon (kód ze sloupce „Požadováno") se uvede jako:
|
||||||
|
- `{kód} ad hoc` — pokud je u výkonu uvedeno **0x týdně** (bez ohledu na četnost denně); znamená to výkon pouze dle potřeby, ne na pravidelné bázi.
|
||||||
|
- `{kód} {N}xd{M}xt` — pokud je týdenní četnost M > 0; N = četnost denně, M = četnost týdně. Např. pro 1x denně 3x týdně: `06313 1xd3xt`.
|
||||||
|
- Příklad (oba výkony ad hoc): `[domácí péče] [1 do 30JUN2026, 06313 ad hoc, 06323 ad hoc]`
|
||||||
|
- Příklad (pravidelné výkony): `[domácí péče] [2 do 31AUG2026, 06313 1xd5xt, 06321 2xd7xt]`
|
||||||
@@ -1818,5 +1818,69 @@
|
|||||||
{
|
{
|
||||||
"original": "0760245079 2026-06-03 [EKG] [bez hodnocení].pdf",
|
"original": "0760245079 2026-06-03 [EKG] [bez hodnocení].pdf",
|
||||||
"corrected": "0760245079 Ryšavá, Denisa 2026-06-03 [EKG] [bez hodnocení].pdf"
|
"corrected": "0760245079 Ryšavá, Denisa 2026-06-03 [EKG] [bez hodnocení].pdf"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"original": "366103079 2026-06-01 Čížkovská, Jaroslava [domácí péče] [1 do 30JUN2026].pdf",
|
||||||
|
"corrected": "366103079 2026-06-01 Čížkovská, Jaroslava [K od psychiatra] [že máme zařídit domácí péči, kterou pacientka domů nepustí].pdf"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"original": "435520110 2026-06-01 Nechodomová, Marie [LZ gastroenterologie] [Inkompetence kardie, pseudopolyp subkardiálně, biliární reflux, atrofie fornixu, hyperémie antra].pdf",
|
||||||
|
"corrected": "435520110 2026-06-01 Nechodomová, Marie [LZ gastroenterologie] [gastroskopie Inkompetence kardie, pseudopolyp subkardiálně, biliární reflux, atrofie fornixu, hyperémie antra].pdf"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"original": "5404211967 2014-11-19 Zich, Jiří [operační protokol neurochirurgie] [Stenóza L4/5, miniinvazivní over-the-top dekomprese L4/5 zleva].pdf",
|
||||||
|
"corrected": "5404211967 2014-11-19 Zich, Jiří [operační protokol neurochirurgie] [Stenóza L45, miniinvazivní over-the-top dekomprese L45 zleva].pdf"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"original": "5404211967 2014-11-24 Zich, Jiří [PZ neurochirurgie] [18–24NOV2014 degenerativní stenóza L4/5, miniinvazivní over-the-top dekomprese L4/5 zleva].pdf",
|
||||||
|
"corrected": "5404211967 2014-11-24 Zich, Jiří [PZ neurochirurgie] [18–24NOV2014 degenerativní stenóza L45, miniinvazivní over-the-top dekomprese L45 zleva].pdf"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"original": "5404211967 Zich, Jiří [dopis pacientovi] [žádost o typ kovové výztuhy bérce z r. 2001 před op. kolena].pdf",
|
||||||
|
"corrected": "5404211967 2026-06-05 Zich, Jiří [dopis pacienta] [žádost o typ kovové výztuhy bérce z r. 2001 před op. kolena].pdf"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"original": "5404211967 2022-06-22 Zich, Jiří [PZ ortopedicko-traumatologická] [20–22JUN2022 extrakce OS hřebu bérce vlevo, gonartróza L kolena, TEP plánována].pdf",
|
||||||
|
"corrected": "5404211967 2022-06-22 Zich, Jiří [PZ ortopedie] [20–22JUN2022 extrakce OS hřebu bérce vlevo, gonartróza L kolena, TEP plánována].pdf"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"original": "5404211967 2026-06-01 Zich, Jiří [řidičský průkaz zdravotní způsobilost] [zdravotně způsobilý sk. B, platnost do 01.06.2028].pdf",
|
||||||
|
"corrected": "5404211967 2026-06-01 Zich, Jiří [Posudek ŘP] [zdravotně způsobilý sk. B, platnost do 01.06.2028].pdf"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"original": "5404211967 2026-06-01 Zich, Jiří [Prohlášení zdravotní způsobilosti řidiče] [sk. B, cítí se zdráv, užívá Tamsulosin na prostatu].pdf",
|
||||||
|
"corrected": "5404211967 2026-06-01 Zich, Jiří [Prohlášení ŘP] [sk. B, cítí se zdráv, užívá Tamsulosin na prostatu].pdf"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"original": "7857103232 2014-07-14 Dubová, Zita [LZ neurologie] [noční můry od dětství, RLS, spánková anamnéza, ESS 15/24].pdf",
|
||||||
|
"corrected": "7857103232 2014-07-14 Dubová, Zita [LZ neurologie] [noční můry od dětství, RLS, spánková anamnéza, ESS 1524].pdf"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"original": "8157220159 2007-03-27 Vrňáková, Lucie [LZ hematologie] [kontrola, kompletní remise HL, gonadotox, subklinická hypothyreóza, diskrétní plicní tox].pdf",
|
||||||
|
"corrected": "8157220159 2007-06-27 Vrňáková, Lucie [LZ hematologie] [kontrola, kompletní remise HL, gonadotox, subklinická hypothyreóza, diskrétní plicní tox].pdf"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"original": "8157220159 2009-06-30 Vrňáková, Lucie [LZ hematologie] [kontrola, kompletní remise 7 let po terapii, incip. poradiační hypothyreóza, ko endokrinologie].pdf",
|
||||||
|
"corrected": "8157220159 2009-07-01 Vrňáková, Lucie [LZ hematologie] [kontrola, kompletní remise 7 let po terapii, incip. poradiační hypothyreóza, ko endokrinologie].pdf"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"original": "8157220159 2022-03-16 Vrňáková, Lucie [LZ kardiologie] [kontrola, stp. CHT a AR 2001, EF 73%, norm. diastol. fce, stopová MR/TR/PR, bez zn. plicní HTN, ko za 3-5 let].pdf",
|
||||||
|
"corrected": "8157220159 2022-03-16 Vrňáková, Lucie [LZ kardiologie] [kontrola, stp. CHT a AR 2001, EF 73%, norm. diastol. fce, stopová MRTRPR, bez zn. plicní HTN, ko za 3-5 let].pdf"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"original": "8812310408 2026-06-04 Sekrt, Zdeněk [EKG] [bez hodnocení].pdf",
|
||||||
|
"corrected": "8812310408 2026-06-04 Sekrt, Zdeněk [EKG] [bez hodnocení].pdf"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"original": "5459051862 2026-06-04 Vortelová, Eva [EKG] [bez hodnocení].pdf",
|
||||||
|
"corrected": "5459051862 2026-06-04 Vortelová, Eva [EKG] [bez hodnocení].pdf"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"original": "5962050149 2026-06-04 Jelínková, Eva [EKG] [bez hodnocení].pdf",
|
||||||
|
"corrected": "5962050149 2026-06-04 Jelínková, Eva [EKG] [bez hodnocení].pdf"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"original": "405712023 2026-06-05 Pilná, Marta [EKG] [bez hodnocení].pdf",
|
||||||
|
"corrected": "405712023 2026-06-05 Pilná, Marta [EKG] [bez hodnocení].pdf"
|
||||||
}
|
}
|
||||||
]
|
]
|
||||||
@@ -0,0 +1,30 @@
|
|||||||
|
Stack trace:
|
||||||
|
Frame Function Args
|
||||||
|
0007FFFFB540 00021005FE8E (000210285F68, 00021026AB6E, 0007FFFFB540, 0007FFFFA440) msys-2.0.dll+0x1FE8E
|
||||||
|
0007FFFFB540 0002100467F9 (000000000000, 000000000000, 000000000000, 0007FFFFB818) msys-2.0.dll+0x67F9
|
||||||
|
0007FFFFB540 000210046832 (000210286019, 0007FFFFB3F8, 0007FFFFB540, 000000000000) msys-2.0.dll+0x6832
|
||||||
|
0007FFFFB540 000210068CF6 (000000000000, 000000000000, 000000000000, 000000000000) msys-2.0.dll+0x28CF6
|
||||||
|
0007FFFFB540 000210068E24 (0007FFFFB550, 000000000000, 000000000000, 000000000000) msys-2.0.dll+0x28E24
|
||||||
|
0007FFFFB820 00021006A225 (0007FFFFB550, 000000000000, 000000000000, 000000000000) msys-2.0.dll+0x2A225
|
||||||
|
End of stack trace
|
||||||
|
Loaded modules:
|
||||||
|
000100400000 bash.exe
|
||||||
|
7FFD0F4F0000 ntdll.dll
|
||||||
|
7FFD0D6F0000 KERNEL32.DLL
|
||||||
|
7FFD0CD00000 KERNELBASE.dll
|
||||||
|
7FFD08A10000 apphelp.dll
|
||||||
|
7FFD0F260000 USER32.dll
|
||||||
|
7FFD0D320000 win32u.dll
|
||||||
|
7FFD0E6D0000 GDI32.dll
|
||||||
|
000210040000 msys-2.0.dll
|
||||||
|
7FFD0CBE0000 gdi32full.dll
|
||||||
|
7FFD0D350000 msvcp_win.dll
|
||||||
|
7FFD0D3F0000 ucrtbase.dll
|
||||||
|
7FFD0D510000 advapi32.dll
|
||||||
|
7FFD0F090000 msvcrt.dll
|
||||||
|
7FFD0F410000 sechost.dll
|
||||||
|
7FFD0E780000 RPCRT4.dll
|
||||||
|
7FFD0CB60000 bcrypt.dll
|
||||||
|
7FFD0C410000 CRYPTBASE.DLL
|
||||||
|
7FFD0D1E0000 bcryptPrimitives.dll
|
||||||
|
7FFD0D7C0000 IMM32.DLL
|
||||||
Reference in New Issue
Block a user