Reguli text per cont (operation_text_rules), resolve_prestatii cu param aditiv text_rules + precedenta stricta, threadat pe toate cele 6 callsite-uri + valid_codes + seam classify_prezentare. UI Mapari: sectiune reguli + preview pre-salvare + overlap + telemetrie text_rule_hit. UX tabel: cod_rar sub operatie, pill eticheta scurta, fara scroll orizontal (scopat .tabel-trimiteri + carduri <768px), detaliu inline expandabil (a11y + pauza poll). code-review: reparat regula auto_send=0 care trimitea automat la RAR in loc sa tina randul pentru review. 814 passed. Co-Authored-By: Claude Opus 4.8 (1M context) <noreply@anthropic.com>
181 lines
6.4 KiB
Python
181 lines
6.4 KiB
Python
"""Teste US-009 (PRD 5.8) — preview pre-salvare regula text.
|
|
|
|
POST /mapari/reguli-text/preview primeste `pattern`, normalizeaza cu
|
|
normalize_for_match, numara operatiile DISTINCTE nemapate ale contului
|
|
(needs_mapping, reuse pending_unmapped) al caror text contine pattern-ul si
|
|
intoarce pana la 3 exemple. NU salveaza nimic (zero scriere DB). Pattern gol ->
|
|
fragment gol (nu numara „tot"). Scoped pe contul sesiunii (require_login + CSRF).
|
|
"""
|
|
|
|
from __future__ import annotations
|
|
|
|
import json
|
|
import os
|
|
import re
|
|
import tempfile
|
|
|
|
import pytest
|
|
from starlette.testclient import TestClient
|
|
|
|
|
|
def _create_account_user(email: str, name: str = "Service", password: str = "parolasecreta10"):
|
|
from app.accounts import create_account
|
|
from app.users import create_user
|
|
from app.db import get_connection
|
|
|
|
conn = get_connection()
|
|
try:
|
|
acct_id = create_account(conn, name, active=True)
|
|
create_user(conn, acct_id, email, password)
|
|
return acct_id
|
|
finally:
|
|
conn.close()
|
|
|
|
|
|
def _login(client, email: str, password: str = "parolasecreta10") -> None:
|
|
resp = client.get("/login")
|
|
m = re.search(r'name="csrf_token"\s+value="([^"]+)"', resp.text)
|
|
assert m
|
|
resp = client.post("/login", data={"email": email, "parola": password, "csrf_token": m.group(1)})
|
|
assert resp.status_code == 303
|
|
|
|
|
|
def _csrf(client) -> str:
|
|
resp = client.get("/?tab=mapari")
|
|
m = re.search(r'name="csrf_token"\s+value="([^"]+)"', resp.text)
|
|
assert m, "csrf_token negasit"
|
|
return m.group(1)
|
|
|
|
|
|
def _seed_needs_mapping(acct: int, *, op: str, denumire: str | None = None) -> int:
|
|
"""Submission needs_mapping pe canal API (batch_id NULL) cu o operatie nemapata."""
|
|
from app.db import get_connection
|
|
conn = get_connection()
|
|
try:
|
|
cur = conn.execute(
|
|
"INSERT INTO submissions (idempotency_key, account_id, status, payload_json, rar_error) "
|
|
"VALUES (?, ?, 'needs_mapping', ?, ?)",
|
|
(
|
|
f"k-{os.urandom(6).hex()}",
|
|
acct,
|
|
json.dumps({
|
|
"vin": "WVWZZZ1JZXW000111",
|
|
"nr_inmatriculare": "B11AAA",
|
|
"data_prestatie": "2026-06-18",
|
|
"odometru_final": "12345",
|
|
"prestatii": [{"cod_prestatie": None, "cod_op_service": op, "denumire": denumire or op}],
|
|
}),
|
|
json.dumps({"unmapped": [{"cod_op_service": op}]}),
|
|
),
|
|
)
|
|
conn.commit()
|
|
return int(cur.lastrowid)
|
|
finally:
|
|
conn.close()
|
|
|
|
|
|
def _count_submissions() -> int:
|
|
from app.db import get_connection
|
|
conn = get_connection()
|
|
try:
|
|
return int(conn.execute("SELECT COUNT(*) AS c FROM submissions").fetchone()["c"])
|
|
finally:
|
|
conn.close()
|
|
|
|
|
|
def _count_text_rules() -> int:
|
|
from app.db import get_connection
|
|
conn = get_connection()
|
|
try:
|
|
return int(conn.execute("SELECT COUNT(*) AS c FROM operation_text_rules").fetchone()["c"])
|
|
finally:
|
|
conn.close()
|
|
|
|
|
|
@pytest.fixture()
|
|
def client(monkeypatch):
|
|
tmp = tempfile.mkdtemp()
|
|
monkeypatch.setenv("AUTOPASS_DB_PATH", os.path.join(tmp, "preview_regula.db"))
|
|
monkeypatch.setenv("AUTOPASS_WEB_AUTH_REQUIRED", "true")
|
|
from app.config import get_settings
|
|
get_settings.cache_clear()
|
|
from app.web import ratelimit
|
|
ratelimit._hits.clear()
|
|
from app.main import app
|
|
with TestClient(app, follow_redirects=False) as c:
|
|
yield c
|
|
ratelimit._hits.clear()
|
|
get_settings.cache_clear()
|
|
|
|
|
|
def test_preview_numara_potriviri(client):
|
|
"""Numara operatiile DISTINCTE nemapate al caror text contine pattern-ul."""
|
|
acct = _create_account_user("p1@test.com")
|
|
_seed_needs_mapping(acct, op="VERIFICARE FARURI", denumire="Verificare faruri")
|
|
_seed_needs_mapping(acct, op="VERIFICARE FRANE", denumire="Verificare frane")
|
|
_seed_needs_mapping(acct, op="SCHIMB ULEI", denumire="Schimb ulei")
|
|
|
|
_login(client, "p1@test.com")
|
|
csrf = _csrf(client)
|
|
before = _count_submissions()
|
|
resp = client.post("/mapari/reguli-text/preview", data={
|
|
"pattern": "verificare", "csrf_token": csrf,
|
|
})
|
|
assert resp.status_code == 200
|
|
# Cele doua operatii „verificare", nu si „schimb ulei".
|
|
assert "2" in resp.text
|
|
assert "potriveste" in resp.text.lower()
|
|
# NU salveaza nimic: nici submission, nici regula.
|
|
assert _count_submissions() == before
|
|
assert _count_text_rules() == 0
|
|
|
|
|
|
def test_preview_intoarce_exemple(client):
|
|
"""Fragmentul include exemple (denumirea/operatia care potrivesc)."""
|
|
acct = _create_account_user("p2@test.com")
|
|
_seed_needs_mapping(acct, op="VERIFICARE FARURI", denumire="Verificare faruri")
|
|
_seed_needs_mapping(acct, op="VERIFICARE FRANE", denumire="Verificare frane")
|
|
|
|
_login(client, "p2@test.com")
|
|
csrf = _csrf(client)
|
|
resp = client.post("/mapari/reguli-text/preview", data={
|
|
"pattern": "verificare", "csrf_token": csrf,
|
|
})
|
|
assert resp.status_code == 200
|
|
text = resp.text.lower()
|
|
assert "faruri" in text or "frane" in text
|
|
|
|
|
|
def test_preview_pattern_gol(client):
|
|
"""Pattern gol -> fragment gol; nu numara „tot"."""
|
|
acct = _create_account_user("p3@test.com")
|
|
_seed_needs_mapping(acct, op="VERIFICARE FARURI", denumire="Verificare faruri")
|
|
|
|
_login(client, "p3@test.com")
|
|
csrf = _csrf(client)
|
|
resp = client.post("/mapari/reguli-text/preview", data={
|
|
"pattern": " ", "csrf_token": csrf,
|
|
})
|
|
assert resp.status_code == 200
|
|
# Nu raporteaza nicio potrivire numerica pentru pattern gol.
|
|
assert "potriveste" not in resp.text.lower()
|
|
assert resp.text.strip() == "" or "1" not in resp.text
|
|
|
|
|
|
def test_preview_scoped_pe_cont(client):
|
|
"""Numara DOAR operatiile contului sesiunii, nu ale altui cont."""
|
|
acct_a = _create_account_user("a@test.com", name="Cont A")
|
|
acct_b = _create_account_user("b@test.com", name="Cont B")
|
|
# Contul A are 2 potriviri, contul B niciuna.
|
|
_seed_needs_mapping(acct_a, op="VERIFICARE FARURI", denumire="Verificare faruri")
|
|
_seed_needs_mapping(acct_a, op="VERIFICARE FRANE", denumire="Verificare frane")
|
|
|
|
_login(client, "b@test.com")
|
|
csrf = _csrf(client)
|
|
resp = client.post("/mapari/reguli-text/preview", data={
|
|
"pattern": "verificare", "csrf_token": csrf,
|
|
})
|
|
assert resp.status_code == 200
|
|
# Contul B nu are nicio operatie nemapata -> nicio potrivire.
|
|
assert "Nicio potrivire" in resp.text
|